Capturing stdout when calling Runtime.exec

后端 未结 8 1177
终归单人心
终归单人心 2020-11-21 23:26

When experiencing networking problems on client machines, I\'d like to be able to run a few command lines and email the results of them to myself.

I\'ve found Runtim

相关标签:
8条回答
  • 2020-11-21 23:37

    Use Plexus Utils, it is used by Maven to execut all external processes.

    Commandline commandLine = new Commandline();
    commandLine.setExecutable(executable.getAbsolutePath());
    
    Collection<String> args = getArguments();
    
    for (String arg : args) {
        Arg _arg = commandLine.createArg();
        _arg.setValue(arg);
    }
    
    WriterStreamConsumer systemOut = new WriterStreamConsumer(console);
    WriterStreamConsumer systemErr = new WriterStreamConsumer(console);
    
    returnCode = CommandLineUtils.executeCommandLine(commandLine, systemOut, systemErr, 10);
    if (returnCode != 0) {
        // bad
    } else {
        // good
    }
    
    0 讨论(0)
  • 2020-11-21 23:37

    Using Runtime.exec gives you a process. You can these use getInputStream to get the stdout of this process, and put this input stream into a String, through a StringBuffer for example.

    0 讨论(0)
  • 2020-11-21 23:44

    This is my helper class been using for years. One small class. It has JavaWorld streamgobbler class to fix JVM resource leaks. Don't know if still valid for JVM6 and JVM7 but does not hurt. Helper can read output buffer for later use.

    import java.io.*;
    
    /**
     * Execute external process and optionally read output buffer.
     */
    public class ShellExec {
        private int exitCode;
        private boolean readOutput, readError;
        private StreamGobbler errorGobbler, outputGobbler;
    
        public ShellExec() { 
            this(false, false);
        }
    
        public ShellExec(boolean readOutput, boolean readError) {
            this.readOutput = readOutput;
            this.readError = readError;
        }
    
        /**
         * Execute a command.
         * @param command   command ("c:/some/folder/script.bat" or "some/folder/script.sh")
         * @param workdir   working directory or NULL to use command folder
         * @param wait  wait for process to end
         * @param args  0..n command line arguments
         * @return  process exit code
         */
        public int execute(String command, String workdir, boolean wait, String...args) throws IOException {
            String[] cmdArr;
            if (args != null && args.length > 0) {
                cmdArr = new String[1+args.length];
                cmdArr[0] = command;
                System.arraycopy(args, 0, cmdArr, 1, args.length);
            } else {
                cmdArr = new String[] { command };
            }
    
            ProcessBuilder pb =  new ProcessBuilder(cmdArr);
            File workingDir = (workdir==null ? new File(command).getParentFile() : new File(workdir) );
            pb.directory(workingDir);
    
            Process process = pb.start();
    
            // Consume streams, older jvm's had a memory leak if streams were not read,
            // some other jvm+OS combinations may block unless streams are consumed.
            errorGobbler  = new StreamGobbler(process.getErrorStream(), readError);
            outputGobbler = new StreamGobbler(process.getInputStream(), readOutput);
            errorGobbler.start();
            outputGobbler.start();
    
            exitCode = 0;
            if (wait) {
                try { 
                    process.waitFor();
                    exitCode = process.exitValue();                 
                } catch (InterruptedException ex) { }
            }
            return exitCode;
        }   
    
        public int getExitCode() {
            return exitCode;
        }
    
        public boolean isOutputCompleted() {
            return (outputGobbler != null ? outputGobbler.isCompleted() : false);
        }
    
        public boolean isErrorCompleted() {
            return (errorGobbler != null ? errorGobbler.isCompleted() : false);
        }
    
        public String getOutput() {
            return (outputGobbler != null ? outputGobbler.getOutput() : null);        
        }
    
        public String getError() {
            return (errorGobbler != null ? errorGobbler.getOutput() : null);        
        }
    
    //********************************************
    //********************************************    
    
        /**
         * StreamGobbler reads inputstream to "gobble" it.
         * This is used by Executor class when running 
         * a commandline applications. Gobblers must read/purge
         * INSTR and ERRSTR process streams.
         * http://www.javaworld.com/javaworld/jw-12-2000/jw-1229-traps.html?page=4
         */
        private class StreamGobbler extends Thread {
            private InputStream is;
            private StringBuilder output;
            private volatile boolean completed; // mark volatile to guarantee a thread safety
    
            public StreamGobbler(InputStream is, boolean readStream) {
                this.is = is;
                this.output = (readStream ? new StringBuilder(256) : null);
            }
    
            public void run() {
                completed = false;
                try {
                    String NL = System.getProperty("line.separator", "\r\n");
    
                    InputStreamReader isr = new InputStreamReader(is);
                    BufferedReader br = new BufferedReader(isr);
                    String line;
                    while ( (line = br.readLine()) != null) {
                        if (output != null)
                            output.append(line + NL); 
                    }
                } catch (IOException ex) {
                    // ex.printStackTrace();
                }
                completed = true;
            }
    
            /**
             * Get inputstream buffer or null if stream
             * was not consumed.
             * @return
             */
            public String getOutput() {
                return (output != null ? output.toString() : null);
            }
    
            /**
             * Is input stream completed.
             * @return
             */
            public boolean isCompleted() {
                return completed;
            }
    
        }
    
    }
    

    Here is an example reading output from .vbs script but similar works for linux sh scripts.

       ShellExec exec = new ShellExec(true, false);
       exec.execute("cscript.exe", null, true,
          "//Nologo",
          "//B",            // batch mode, no prompts
          "//T:320",        // timeout seconds
          "c:/my/script/test1.vbs",  // unix path delim works for script.exe
          "script arg 1",
          "script arg 2",
       );
       System.out.println(exec.getOutput());
    
    0 讨论(0)
  • 2020-11-21 23:45

    VerboseProcess utility class from jcabi-log can help you:

    String output = new VerboseProcess(
      new ProcessBuilder("executable with output")
    ).stdout();
    

    The only dependency you need:

    <dependency>
      <groupId>com.jcabi</groupId>
      <artifactId>jcabi-log</artifactId>
      <version>0.7.5</version>
    </dependency>
    
    0 讨论(0)
  • 2020-11-21 23:46

    Use ProcessBuilder. After calling start() you'll get a Process object from which you can get the stderr and stdout streams.

    UPDATE: ProcessBuilder gives you more control; You don't have to use it but I find it easier in the long run. Especially the ability to redirect stderr to stdout which means you only have to suck down one stream.

    0 讨论(0)
  • 2020-11-21 23:49

    Runtime.exec() returns a Process object, from which you can extract the output of whatever command you ran.

    0 讨论(0)
提交回复
热议问题