Development Class Java

import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
 * Launches a process, redirecting the output of that sub-process to the output
 * of this (the parent) process.
 * 
 * @author Klaas Waslander
 */
public final class ProcessLauncher {
  /** the logger for this class */
  private final static Logger LOGGER = Logger.getLogger(ProcessLauncher.class.getName());
  private String commandLine;
  private String[] commandArray;
  private File baseDir;
  private ArrayList listeners = new ArrayList(1);
  private Process subProcess;
  private boolean finished = false;
  StringBuffer out = new StringBuffer();
  StringBuffer err = new StringBuffer();
  /**
   * Constructs new process launcher with the given command line.
   */
  public ProcessLauncher(String commandLine) {
    this(commandLine, null);
  }
  public ProcessLauncher(String commandLine, File baseDir) {
    this.commandLine = commandLine;
    this.baseDir = baseDir;
  }
  /**
   * Constructs new process launcher with the given command array.
   */
  public ProcessLauncher(String[] commandArray) {
    this(commandArray, null);
  }
  public ProcessLauncher(String[] commandArray, File baseDir) {
    this.commandArray = commandArray;
    this.baseDir = baseDir;
  }
  /**
   * Constructs new process launcher with the given command element list.
   */
  public ProcessLauncher(ArrayList commandList) {
    this(commandList, null);
  }
  public ProcessLauncher(ArrayList commandList, File baseDir) {
    this(toStringArray(commandList), baseDir);
  }
  private static String[] toStringArray(ArrayList list) {
    String[] result = new String[list.size()];
    Iterator iter = list.iterator();
    int arrayIndex = 0;
    while (iter.hasNext()) {
      result[arrayIndex++] = iter.next().toString();
    }
    return result;
  }
  /**
   * Classes implementing this interface can receive output generated by
   * processes launched using the ProcessLauncher.
   */
  public interface OutputListener {
    public void standardOutput(char[] output);
    public void errorOutput(char[] output);
  }
  /**
   * Add a listener for output from the to-be-launched process.
   */
  public void addOutputListener(OutputListener listener) {
    this.listeners.add(listener);
  }
  /** fire error output event */
  private void fireErr(char[] err) {
    if (this.listeners.isEmpty()) {
      this.err.append(out);
    }
    Iterator iter = this.listeners.iterator();
    while (iter.hasNext()) {
      ((OutputListener) iter.next()).errorOutput(err);
    }
  }
  /** fire standard output event */
  private void fireOut(char[] out) {
    if (this.listeners.isEmpty()) {
      this.out.append(out);
    }
    Iterator iter = this.listeners.iterator();
    while (iter.hasNext()) {
      ((OutputListener) iter.next()).standardOutput(out);
    }
  }
  /**
   * Get standard output, in case no listeners were registered - never returns
   * null.
   */
  public String getStandardOutput() {
    if (!this.listeners.isEmpty()) {
      throw new IllegalStateException(
          "Cannot get standard output, because outputlisteners have been registered.");
    }
    return this.out.toString();
  }
  /**
   * Get error output, in case no listeners were registered - never returns
   * null.
   */
  public String getErrorOutput() {
    if (!this.listeners.isEmpty()) {
      throw new IllegalStateException(
          "Cannot get error output, because outputlisteners have been registered.");
    }
    return this.err.toString();
  }
  /**
   * Get the commandline that is used to launch the process.
   */
  public String getCommandLine() {
    String usedCommand = this.commandLine;
    if (this.commandLine == null && this.commandArray != null) {
      usedCommand = "";
      for (int i = 0; i < this.commandArray.length; i++) {
        if (i > 0) {
          usedCommand += " ";
        }
        usedCommand += this.commandArray[i];
      }
    }
    return usedCommand;
  }
  /**
   * Check whether execution has finished.
   */
  public boolean hasFinished() {
    return finished;
  }
  /**
   * Launches the process, and blocks until that process completes execution.
   * 
   * @throws CommandNotExistsException
   *           If the command could not be executed because it does not exist
   */
  public int launch() throws CommandNotExistsException {
    this.err.setLength(0);
    this.out.setLength(0);
    BackgroundPrinter stdout = null;
    BackgroundPrinter stderr = null;
    try {
      if (this.commandArray != null) {
        this.subProcess = Runtime.getRuntime().exec(this.commandArray, null, this.baseDir);
      } else {
        this.subProcess = Runtime.getRuntime().exec(this.commandLine, null, this.baseDir);
      }
      stdout = new BackgroundPrinter(subProcess.getInputStream(), false);
      stderr = new BackgroundPrinter(subProcess.getErrorStream(), true);
      stdout.start();
      stderr.start();
      // kill process and wait max 10 seconds for output to complete
      int exitValue = this.subProcess.waitFor();
      stdout.join(10000);
      stderr.join(10000);
      /*
       * if (exitValue != 0) { LOGGER.fine("WARNING: exit value " + exitValue + "
       * for command \"" + getCommandLine() + "\""); }
       */
      return exitValue;
    } catch (IOException ioe) {
      // usually caused if the command does not exist at all
      throw new CommandNotExistsException("Command probably does not exist: " + ioe);
    } catch (Exception e) {
      LOGGER
          .log(Level.SEVERE, "Exception while running/launching \"" + getCommandLine() + "\".", e);
    } finally {
      if (this.subProcess != null) {
        this.subProcess.destroy();
        this.subProcess = null;
      }
      if (stdout != null) {
        stdout.close();
      }
      if (stderr != null) {
        stderr.close();
      }
      this.finished = true;
    }
    return -1;
  }
  /**
   * Tries to abort the currently running process.
   */
  public void abort() {
    if (this.subProcess != null) {
      this.subProcess.destroy();
      this.subProcess = null;
    }
  }
  /**
   * Catches output from a "java.lang.Process" and writes it to either
   * System.err or System.out.
   * 
   * @author Klaas Waslander - Sun Java Center
   */
  private class BackgroundPrinter extends Thread {
    private InputStream in;
    boolean isErrorOutput;
    public BackgroundPrinter(InputStream in, boolean isErrorOutput) {
      this.in = in;
      this.isErrorOutput = isErrorOutput;
    }
    public void run() {
      try {
        BufferedReader reader = new BufferedReader(new InputStreamReader(this.in));
        // read buffer
        char[] buf = new char[1024];
        // write data to target, until no more data is left to read
        int numberOfReadBytes;
        while ((numberOfReadBytes = reader.read(buf)) != -1) {
          char[] clearedbuf = new char[numberOfReadBytes];
          System.arraycopy(buf, 0, clearedbuf, 0, numberOfReadBytes);
          if (this.isErrorOutput) {
            fireErr(clearedbuf);
          } else {
            fireOut(clearedbuf);
          }
        }
        /*
         * } catch (IOException ioe) { // ignore this: process has ended,
         * causing IOException } catch (NullPointerException ioe) { // ignore
         * this: there was no resulting output
         */
      } catch (Exception e) {
        LOGGER.log(Level.FINE, "Exception while reading from stream from subprocess.", e);
      }
    }
    public void close() {
      try {
        this.in.close();
      } catch (Exception e) {
        LOGGER.log(Level.WARNING,
            "Closing background stream for launched process caused exception.", e);
      }
    }
  }
  /**
   * Exception that is thrown when a command could not be executed because it
   * (probably) does not exist at all.
   * 
   * @author Klaas Waslander
   */
  public static class CommandNotExistsException extends RuntimeException {
    /**
     * Construct a new exception for a command that does not exist.
     * 
     * @param msg
     *          The message for this exception.
     */
    public CommandNotExistsException(String msg) {
      super(msg);
    }
  }
}