blob: f48e01674ad8a9ab1b8ed9b3c65264030dbd7484 [file] [log] [blame]
package aQute.libg.command;
import java.io.*;
import java.util.*;
import java.util.Map.Entry;
import java.util.concurrent.*;
import java.util.concurrent.atomic.*;
import aQute.lib.io.*;
import aQute.service.reporter.*;
public class Command {
boolean trace;
Reporter reporter;
List<String> arguments = new ArrayList<String>();
Map<String,String> variables = new LinkedHashMap<String,String>();
long timeout = 0;
File cwd = new File("").getAbsoluteFile();
static Timer timer = new Timer(Command.class.getName(), true);
Process process;
volatile boolean timedout;
String fullCommand;
public Command(String fullCommand) {
this.fullCommand = fullCommand;
}
public Command() {}
public int execute(Appendable stdout, Appendable stderr) throws Exception {
return execute((InputStream) null, stdout, stderr);
}
public int execute(String input, Appendable stdout, Appendable stderr) throws Exception {
InputStream in = new ByteArrayInputStream(input.getBytes("UTF-8"));
return execute(in, stdout, stderr);
}
public static boolean needsWindowsQuoting(String s) {
int len = s.length();
if (len == 0) // empty string have to be quoted
return true;
for (int i = 0; i < len; i++) {
switch (s.charAt(i)) {
case ' ' :
case '\t' :
case '\\' :
case '"' :
return true;
}
}
return false;
}
public static String windowsQuote(String s) {
if (!needsWindowsQuoting(s))
return s;
s = s.replaceAll("([\\\\]*)\"", "$1$1\\\\\"");
s = s.replaceAll("([\\\\]*)\\z", "$1$1");
return "\"" + s + "\"";
}
public int execute(final InputStream in, Appendable stdout, Appendable stderr) throws Exception {
if (reporter != null) {
reporter.trace("executing cmd: %s", arguments);
}
ProcessBuilder p;
if (fullCommand != null) {
p = new ProcessBuilder(fullCommand);
} else {
//[cs] Arguments on windows aren't processed correctly. Thus the below junk
// http://bugs.sun.com/bugdatabase/view_bug.do?bug_id=6511002
if (System.getProperty("os.name").startsWith("Windows")) {
List<String> adjustedStrings = new LinkedList<String>();
for (String a : arguments) {
adjustedStrings.add(windowsQuote(a));
}
p = new ProcessBuilder(adjustedStrings);
} else {
p = new ProcessBuilder(arguments);
}
}
Map<String, String> env = p.environment();
for (Entry<String,String> s : variables.entrySet()) {
env.put(s.getKey(), s.getValue());
}
p.directory(cwd);
process = p.start();
// Make sure the command will not linger when we go
Runnable r = new Runnable() {
public void run() {
process.destroy();
}
};
Thread hook = new Thread(r, arguments.toString());
Runtime.getRuntime().addShutdownHook(hook);
TimerTask timer = null;
final OutputStream stdin = process.getOutputStream();
Thread rdInThread = null;
if (timeout != 0) {
timer = new TimerTask() {
//@Override TODO why did this not work? TimerTask implements Runnable
public void run() {
timedout = true;
process.destroy();
}
};
Command.timer.schedule(timer, timeout);
}
final AtomicBoolean finished = new AtomicBoolean(false);
InputStream out = process.getInputStream();
try {
InputStream err = process.getErrorStream();
try {
Collector cout = new Collector(out, stdout);
cout.start();
Collector cerr = new Collector(err, stderr);
cerr.start();
if (in != null) {
if (in == System.in) {
rdInThread = new Thread("Read Input Thread") {
@Override
public void run() {
try {
while (!finished.get()) {
int n = in.available();
if (n == 0) {
sleep(100);
} else {
int c = in.read();
if (c < 0) {
stdin.close();
return;
}
stdin.write(c);
if (c == '\n')
stdin.flush();
}
}
}
catch (InterruptedIOException e) {
// Ignore here
}
catch (Exception e) {
// Who cares?
}
finally {
IO.close(stdin);
}
}
};
rdInThread.setDaemon(true);
rdInThread.start();
} else {
IO.copy(in, stdin);
stdin.close();
}
}
if (reporter != null)
reporter.trace("exited process");
cerr.join();
cout.join();
if (reporter != null)
reporter.trace("stdout/stderr streams have finished");
}
finally {
err.close();
}
}
finally {
out.close();
if (timer != null)
timer.cancel();
Runtime.getRuntime().removeShutdownHook(hook);
}
byte exitValue = (byte) process.waitFor();
finished.set(true);
if (rdInThread != null) {
if (in != null)
IO.close(in);
rdInThread.interrupt();
}
if (reporter != null)
reporter.trace("cmd %s executed with result=%d, result: %s/%s, timedout=%s", arguments, exitValue, stdout,
stderr, timedout);
if (timedout)
return Integer.MIN_VALUE;
return exitValue;
}
public void add(String... args) {
for (String arg : args)
arguments.add(arg);
}
public void addAll(Collection<String> args) {
arguments.addAll(args);
}
public void setTimeout(long duration, TimeUnit unit) {
timeout = unit.toMillis(duration);
}
public void setTrace() {
this.trace = true;
}
public void setReporter(Reporter reporter) {
this.reporter = reporter;
}
public void setCwd(File dir) {
if (!dir.isDirectory())
throw new IllegalArgumentException("Working directory must be a directory: " + dir);
this.cwd = dir;
}
public void cancel() {
process.destroy();
}
class Collector extends Thread {
final InputStream in;
final Appendable sb;
Collector(InputStream inputStream, Appendable sb) {
this.in = inputStream;
this.sb = sb;
setDaemon(true);
}
@Override
public void run() {
try {
int c = in.read();
while (c >= 0) {
sb.append((char) c);
c = in.read();
}
}
catch (IOException e) {
// We assume the socket is closed
}
catch (Exception e) {
try {
sb.append("\n**************************************\n");
sb.append(e.toString());
sb.append("\n**************************************\n");
}
catch (IOException e1) {}
if (reporter != null) {
reporter.trace("cmd exec: %s", e);
}
}
}
}
public Command var(String name, String value) {
variables.put(name, value);
return this;
}
public Command arg(String... args) {
add(args);
return this;
}
public Command full(String full) {
fullCommand = full;
return this;
}
public void inherit() {
ProcessBuilder pb = new ProcessBuilder();
for (Entry<String,String> e : pb.environment().entrySet()) {
var(e.getKey(), e.getValue());
}
}
public String var(String name) {
return variables.get(name);
}
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
String del = "";
for (String argument : arguments) {
sb.append(del);
sb.append(argument);
del = " ";
}
return sb.toString();
}
}