Package org.flatland.drip

Source Code of org.flatland.drip.Main

package org.flatland.drip;
import java.lang.reflect.Method;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.io.*;
import java.util.Map;
import java.util.HashMap;
import java.util.Scanner;
import java.util.LinkedList;
import java.util.List;
import java.util.Arrays;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

public class Main {
  private List<Switchable> lazyStreams;
  private String mainClass;
  private File dir;
  private SwitchableOutputStream err;
  private SwitchableOutputStream out;
  private SwitchableInputStream  in;

  public Main(String mainClass, String dir) {
    this.mainClass = mainClass;
    this.dir = new File(dir);
  }

  private void killAfterTimeout() {
    String idleTimeStr = System.getenv("DRIP_SHUTDOWN"); // in minutes
    int idleTime;
    if (idleTimeStr == null) {
      idleTime = 4 * 60; // four hours
    } else {
      idleTime = Integer.parseInt(idleTimeStr);
    }

    try {
      Thread.sleep(idleTime * 60 * 1000); // convert minutes to ms
    } catch (InterruptedException e) {
      System.err.println("drip: Interrupted??");
      return; // I guess someone wanted to kill the timeout thread?
    }

    File lockDir = new File(dir, "lock");
    if (lockDir.mkdir()) {
      System.exit(0);
    } else {
      // someone is already connected; let the process finish
    }
  }

  private void startIdleKiller() {
    Thread idleKiller = new Thread() {
        public void run() {
          killAfterTimeout();
        }
      };

    idleKiller.setDaemon(true);
    idleKiller.start();
  }

  public static void main(String[] args) throws Exception {
    new Main(args[0], args[1]).start();
  }

  public void start() throws Exception {
    reopenStreams();

    Method main = mainMethod(mainClass);   
    Method init = mainMethod(System.getenv("DRIP_INIT_CLASS"));
    String initArgs = System.getenv("DRIP_INIT");
    initArgs = initArgs == null ? "" : initArgs;
    invoke(init == null ? main : init, split(initArgs, "\n"));
    startIdleKiller();

    Scanner fromBash = new Scanner(new File(dir, "control"));
    String mainArgs    = readString(fromBash);
    String runtimeArgs = readString(fromBash);
    String environment = readString(fromBash);
    fromBash.close();

    mergeEnv(parseEnv(environment));
    setProperties(runtimeArgs);
    switchStreams();

    invoke(main, split(mainArgs, "\u0000"));
  }

  private Method mainMethod(String className)
    throws ClassNotFoundException, NoSuchMethodException {
    if (className == null || className.equals("")) {
      return null;
    } else {
      return Class.forName(className, true, ClassLoader.getSystemClassLoader())
        .getMethod("main", String[].class);
    }
  }

  private String[] split(String str, String delim) {
    if (str.length() == 0) {
      return new String[0];
    } else {
      Scanner s = new Scanner(str);
      s.useDelimiter(delim);

      LinkedList<String> list = new LinkedList<String>();
      while (s.hasNext()) {
        list.add(s.next());
      }
      return list.toArray(new String[0]);
    }
  }

  private void invoke(Method main, String[] args) throws Exception {
    main.invoke(null, (Object)args);
  }

  private void setProperties(String runtimeArgs) {
    Matcher m = Pattern.compile("-D([^=]+)=([^\u0000]+)").matcher(runtimeArgs);

    while (m.find()) {
      System.setProperty(m.group(1), m.group(2));
    }
  }

  private Map<String, String> parseEnv(String str) {
    Map<String, String> env = new HashMap<String, String>();

    for (String line: split(str, "\u0000")) {
      String[] var = line.split("=", 2);
      env.put(var[0], var[1]);
    }
    return env;
  }

  @SuppressWarnings("unchecked")
  private void mergeEnv(Map<String, String> newEnv)
    throws NoSuchFieldException, IllegalAccessException {
    Map<String, String> env = System.getenv();
    Class<?> classToHack = env.getClass();
    if (!(classToHack.getName().equals("java.util.Collections$UnmodifiableMap"))) {
      throw new RuntimeException("Don't know how to hack " + classToHack);
    }

    Field field = classToHack.getDeclaredField("m");
    field.setAccessible(true);
    ((Map<String,String>)field.get(env)).putAll(newEnv);
    field.setAccessible(false);
  }

  @SuppressWarnings("unchecked")
  static void replaceFileDescriptor(FileDescriptor a, FileDescriptor b)
    throws NoSuchFieldException, IllegalAccessException {
    Field field = FileDescriptor.class.getDeclaredField("fd");
    field.setAccessible(true);
    field.set(a, field.get(b));
    field.setAccessible(false);
  }

  private void flip(Switchable s) throws IllegalStateException, IOException {
    while (! s.path().exists()) {
      try {
        Thread.sleep(50);
      } catch (InterruptedException e) {
      }
    }
    s.flip();
  }

  private void reopenStreams() throws FileNotFoundException, IOException {
    this.in  = new SwitchableInputStream(System.in, new File(dir, "in"));
    this.out = new SwitchableOutputStream(System.out, new File(dir, "out"));
    this.err = new SwitchableOutputStream(System.err, new File(dir, "err"));

    System.setIn(new BufferedInputStream(in));
    System.setOut(new PrintStream(out));
    System.setErr(new PrintStream(err));
  }

  private void switchStreams() throws Exception {
    flip(in);
    flip(out);
    flip(err);

    replaceFileDescriptor(FileDescriptor.in,  this.in.getFD());
    replaceFileDescriptor(FileDescriptor.out, this.out.getFD());
    replaceFileDescriptor(FileDescriptor.err, this.err.getFD());
  }

  private static final Pattern EVERYTHING = Pattern.compile(".+", Pattern.DOTALL);
  private String readString(Scanner s) throws IOException {
    s.useDelimiter(":");
    int numChars = s.nextInt();
    s.skip(":");

    String arg;
    if (numChars == 0) { // horizon treats 0 as "unbounded"
      arg = "";
    } else {
      arg = s.findWithinHorizon(EVERYTHING, numChars);
      if (arg.length() != numChars) {
        throw new IOException("Expected " + numChars + " characters but found only " + arg.length() + " in string: \"" + arg + "\"");
      }
    }

    String terminator = s.findWithinHorizon(",", 1);
    if (!(terminator.equals(","))) {
      throw new IOException("Instead of comma terminator after \"" + arg + "\", found " + terminator);
    }
    return arg;
  }
}
TOP

Related Classes of org.flatland.drip.Main

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.