jdk/src/java.base/share/classes/java/lang/ProcessBuilder.java
author thartmann
Tue, 03 Nov 2015 09:42:11 +0100
changeset 33663 2cd62a4bd471
parent 32763 c11c2b9b45a5
child 33826 a9e5f1b8ea57
permissions -rw-r--r--
8141132: JEP 254: Compact Strings Summary: Adopt a more space-efficient internal representation for strings. Reviewed-by: alanb, bdelsart, coleenp, iklam, jiangli, jrose, kevinw, naoto, pliden, roland, smarks, twisti Contributed-by: Brent Christian <brent.christian@oracle.com>, Vivek Deshpande <vivek.r.deshpande@intel.com>, Tobias Hartmann <tobias.hartmann@oracle.com>, Charlie Hunt <charlie.hunt@oracle.com>, Vladimir Kozlov <vladimir.kozlov@oracle.com>, Roger Riggs <roger.riggs@oracle.com>, Xueming Shen <xueming.shen@oracle.com>, Aleksey Shipilev <aleksey.shipilev@oracle.com>, Sandhya Viswanathan <sandhya.viswanathan@intel.com>
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
32763
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
     2
 * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5186
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5186
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5186
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5186
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 5186
diff changeset
    23
 * questions.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
package java.lang;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import java.io.File;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import java.io.IOException;
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    30
import java.io.InputStream;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    31
import java.io.OutputStream;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    32
import java.util.Arrays;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
import java.util.ArrayList;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
import java.util.List;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    35
import java.util.Map;
32763
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
    36
import java.security.AccessController;
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
    37
import java.security.PrivilegedAction;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
 * This class is used to create operating system processes.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
 *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    41
 * <p>Each {@code ProcessBuilder} instance manages a collection
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
 * of process attributes.  The {@link #start()} method creates a new
90ce3da70b43 Initial load
duke
parents:
diff changeset
    43
 * {@link Process} instance with those attributes.  The {@link
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
 * #start()} method can be invoked repeatedly from the same instance
90ce3da70b43 Initial load
duke
parents:
diff changeset
    45
 * to create new subprocesses with identical or related attributes.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
 * <p>Each process builder manages these process attributes:
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
 * <ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
 * <li>a <i>command</i>, a list of strings which signifies the
90ce3da70b43 Initial load
duke
parents:
diff changeset
    52
 * external program file to be invoked and its arguments, if any.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
 * Which string lists represent a valid operating system command is
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
 * system-dependent.  For example, it is common for each conceptual
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
 * argument to be an element in this list, but there are operating
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
 * systems where programs are expected to tokenize command line
90ce3da70b43 Initial load
duke
parents:
diff changeset
    57
 * strings themselves - on such a system a Java implementation might
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
 * require commands to contain exactly two elements.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
 * <li>an <i>environment</i>, which is a system-dependent mapping from
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
 * <i>variables</i> to <i>values</i>.  The initial value is a copy of
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
 * the environment of the current process (see {@link System#getenv()}).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    63
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
 * <li>a <i>working directory</i>.  The default value is the current
90ce3da70b43 Initial load
duke
parents:
diff changeset
    65
 * working directory of the current process, usually the directory
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    66
 * named by the system property {@code user.dir}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    67
 *
18776
c17100862d86 8019862: Fix doclint errors in java.lang.*.
bpb
parents: 17467
diff changeset
    68
 * <li><a name="redirect-input">a source of <i>standard input</i></a>.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    69
 * By default, the subprocess reads input from a pipe.  Java code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    70
 * can access this pipe via the output stream returned by
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    71
 * {@link Process#getOutputStream()}.  However, standard input may
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    72
 * be redirected to another source using
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    73
 * {@link #redirectInput(Redirect) redirectInput}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    74
 * In this case, {@link Process#getOutputStream()} will return a
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    75
 * <i>null output stream</i>, for which:
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    76
 *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    77
 * <ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    78
 * <li>the {@link OutputStream#write(int) write} methods always
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    79
 * throw {@code IOException}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    80
 * <li>the {@link OutputStream#close() close} method does nothing
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    81
 * </ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    82
 *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    83
 * <li><a name="redirect-output">a destination for <i>standard output</i>
18776
c17100862d86 8019862: Fix doclint errors in java.lang.*.
bpb
parents: 17467
diff changeset
    84
 * and <i>standard error</i></a>.  By default, the subprocess writes standard
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    85
 * output and standard error to pipes.  Java code can access these pipes
29601
d6e9fbd8497a 8073220: A typo in the documentation for class ProcessBuilder
rriggs
parents: 29600
diff changeset
    86
 * via the input streams returned by {@link Process#getOutputStream()} and
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    87
 * {@link Process#getErrorStream()}.  However, standard output and
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    88
 * standard error may be redirected to other destinations using
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    89
 * {@link #redirectOutput(Redirect) redirectOutput} and
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    90
 * {@link #redirectError(Redirect) redirectError}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    91
 * In this case, {@link Process#getInputStream()} and/or
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    92
 * {@link Process#getErrorStream()} will return a <i>null input
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    93
 * stream</i>, for which:
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    94
 *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    95
 * <ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    96
 * <li>the {@link InputStream#read() read} methods always return
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    97
 * {@code -1}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    98
 * <li>the {@link InputStream#available() available} method always returns
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
    99
 * {@code 0}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   100
 * <li>the {@link InputStream#close() close} method does nothing
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   101
 * </ul>
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
 * <li>a <i>redirectErrorStream</i> property.  Initially, this property
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   104
 * is {@code false}, meaning that the standard output and error
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   105
 * output of a subprocess are sent to two separate streams, which can
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
 * be accessed using the {@link Process#getInputStream()} and {@link
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   107
 * Process#getErrorStream()} methods.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   108
 *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   109
 * <p>If the value is set to {@code true}, then:
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   110
 *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   111
 * <ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   112
 * <li>standard error is merged with the standard output and always sent
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   113
 * to the same destination (this makes it easier to correlate error
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   114
 * messages with the corresponding output)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   115
 * <li>the common destination of standard error and standard output can be
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   116
 * redirected using
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   117
 * {@link #redirectOutput(Redirect) redirectOutput}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   118
 * <li>any redirection set by the
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   119
 * {@link #redirectError(Redirect) redirectError}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   120
 * method is ignored when creating a subprocess
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   121
 * <li>the stream returned from {@link Process#getErrorStream()} will
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   122
 * always be a <a href="#redirect-output">null input stream</a>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   123
 * </ul>
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
 * </ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
 * <p>Modifying a process builder's attributes will affect processes
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
 * subsequently started by that object's {@link #start()} method, but
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
 * will never affect previously started processes or the Java process
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
 * itself.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
 * <p>Most error checking is performed by the {@link #start()} method.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
 * It is possible to modify the state of an object so that {@link
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
 * #start()} will fail.  For example, setting the command attribute to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
 * an empty list will not throw an exception unless {@link #start()}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
 * is invoked.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   138
 * <p><strong>Note that this class is not synchronized.</strong>
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   139
 * If multiple threads access a {@code ProcessBuilder} instance
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
 * concurrently, and at least one of the threads modifies one of the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
 * attributes structurally, it <i>must</i> be synchronized externally.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
 * <p>Starting a new process which uses the default working directory
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
 * and environment is easy:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
 *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   146
 * <pre> {@code
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
 * Process p = new ProcessBuilder("myCommand", "myArg").start();
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   148
 * }</pre>
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
 * <p>Here is an example that starts a process with a modified working
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   151
 * directory and environment, and redirects standard output and error
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   152
 * to be appended to a log file:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
 *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   154
 * <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   155
 * ProcessBuilder pb =
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   156
 *   new ProcessBuilder("myCommand", "myArg1", "myArg2");
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   157
 * Map<String, String> env = pb.environment();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
 * env.put("VAR1", "myValue");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
 * env.remove("OTHERVAR");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
 * env.put("VAR2", env.get("VAR1") + "suffix");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
 * pb.directory(new File("myDir"));
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   162
 * File log = new File("log");
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   163
 * pb.redirectErrorStream(true);
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   164
 * pb.redirectOutput(Redirect.appendTo(log));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
 * Process p = pb.start();
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   166
 * assert pb.redirectInput() == Redirect.PIPE;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   167
 * assert pb.redirectOutput().file() == log;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   168
 * assert p.getInputStream().read() == -1;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   169
 * }</pre>
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
 * <p>To start a process with an explicit set of environment
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
 * variables, first call {@link java.util.Map#clear() Map.clear()}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
 * before adding environment variables.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
 *
29381
6d4ed10391a6 8058464: (process spec) ProcessBuilder.redirectXXX throws unspecified NPE
rriggs
parents: 28872
diff changeset
   175
 * <p>
6d4ed10391a6 8058464: (process spec) ProcessBuilder.redirectXXX throws unspecified NPE
rriggs
parents: 28872
diff changeset
   176
 * Unless otherwise noted, passing a {@code null} argument to a constructor
6d4ed10391a6 8058464: (process spec) ProcessBuilder.redirectXXX throws unspecified NPE
rriggs
parents: 28872
diff changeset
   177
 * or method in this class will cause a {@link NullPointerException} to be
6d4ed10391a6 8058464: (process spec) ProcessBuilder.redirectXXX throws unspecified NPE
rriggs
parents: 28872
diff changeset
   178
 * thrown.
6d4ed10391a6 8058464: (process spec) ProcessBuilder.redirectXXX throws unspecified NPE
rriggs
parents: 28872
diff changeset
   179
 *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   180
 * @author Martin Buchholz
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
 * @since 1.5
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
public final class ProcessBuilder
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
{
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
    private List<String> command;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
    private File directory;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
    private Map<String,String> environment;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
    private boolean redirectErrorStream;
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   190
    private Redirect[] redirects;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
     * Constructs a process builder with the specified operating
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
     * system program and arguments.  This constructor does <i>not</i>
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   195
     * make a copy of the {@code command} list.  Subsequent
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
     * updates to the list will be reflected in the state of the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
     * process builder.  It is not checked whether
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   198
     * {@code command} corresponds to a valid operating system
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   199
     * command.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   201
     * @param  command the list containing the program and its arguments
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
    public ProcessBuilder(List<String> command) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
        if (command == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
            throw new NullPointerException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
        this.command = command;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   207
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   208
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
     * Constructs a process builder with the specified operating
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
     * system program and arguments.  This is a convenience
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
     * constructor that sets the process builder's command to a string
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   213
     * list containing the same strings as the {@code command}
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
     * array, in the same order.  It is not checked whether
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   215
     * {@code command} corresponds to a valid operating system
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   216
     * command.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   218
     * @param command a string array containing the program and its arguments
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   219
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
    public ProcessBuilder(String... command) {
7803
56bc97d69d93 6880112: Project Coin: Port JDK core library code to use diamond operator
smarks
parents: 7515
diff changeset
   221
        this.command = new ArrayList<>(command.length);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
        for (String arg : command)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
            this.command.add(arg);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
     * Sets this process builder's operating system program and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
     * arguments.  This method does <i>not</i> make a copy of the
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   229
     * {@code command} list.  Subsequent updates to the list will
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
     * be reflected in the state of the process builder.  It is not
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   231
     * checked whether {@code command} corresponds to a valid
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   232
     * operating system command.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   234
     * @param  command the list containing the program and its arguments
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   235
     * @return this process builder
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
    public ProcessBuilder command(List<String> command) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
        if (command == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
            throw new NullPointerException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
        this.command = command;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
        return this;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
     * Sets this process builder's operating system program and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
     * arguments.  This is a convenience method that sets the command
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
     * to a string list containing the same strings as the
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   248
     * {@code command} array, in the same order.  It is not
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   249
     * checked whether {@code command} corresponds to a valid
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   250
     * operating system command.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   252
     * @param  command a string array containing the program and its arguments
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   253
     * @return this process builder
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
    public ProcessBuilder command(String... command) {
7803
56bc97d69d93 6880112: Project Coin: Port JDK core library code to use diamond operator
smarks
parents: 7515
diff changeset
   256
        this.command = new ArrayList<>(command.length);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
        for (String arg : command)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
            this.command.add(arg);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
        return this;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   263
     * Returns this process builder's operating system program and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
     * arguments.  The returned list is <i>not</i> a copy.  Subsequent
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
     * updates to the list will be reflected in the state of this
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   266
     * process builder.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   267
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   268
     * @return this process builder's program and its arguments
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
    public List<String> command() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
        return command;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
     * Returns a string map view of this process builder's environment.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
     * Whenever a process builder is created, the environment is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
     * initialized to a copy of the current process environment (see
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
     * {@link System#getenv()}).  Subprocesses subsequently started by
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
     * this object's {@link #start()} method will use this map as
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
     * their environment.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
     * <p>The returned object may be modified using ordinary {@link
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
     * java.util.Map Map} operations.  These modifications will be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
     * visible to subprocesses started via the {@link #start()}
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   286
     * method.  Two {@code ProcessBuilder} instances always
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   287
     * contain independent process environments, so changes to the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   288
     * returned map will never be reflected in any other
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   289
     * {@code ProcessBuilder} instance or the values returned by
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
     * {@link System#getenv System.getenv}.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
     * <p>If the system does not support environment variables, an
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
     * empty map is returned.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   295
     * <p>The returned map does not permit null keys or values.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   296
     * Attempting to insert or query the presence of a null key or
90ce3da70b43 Initial load
duke
parents:
diff changeset
   297
     * value will throw a {@link NullPointerException}.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   298
     * Attempting to query the presence of a key or value which is not
90ce3da70b43 Initial load
duke
parents:
diff changeset
   299
     * of type {@link String} will throw a {@link ClassCastException}.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   300
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   301
     * <p>The behavior of the returned map is system-dependent.  A
90ce3da70b43 Initial load
duke
parents:
diff changeset
   302
     * system may not allow modifications to environment variables or
90ce3da70b43 Initial load
duke
parents:
diff changeset
   303
     * may forbid certain variable names or values.  For this reason,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   304
     * attempts to modify the map may fail with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   305
     * {@link UnsupportedOperationException} or
90ce3da70b43 Initial load
duke
parents:
diff changeset
   306
     * {@link IllegalArgumentException}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   307
     * if the modification is not permitted by the operating system.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   308
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
     * <p>Since the external format of environment variable names and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
     * values is system-dependent, there may not be a one-to-one
90ce3da70b43 Initial load
duke
parents:
diff changeset
   311
     * mapping between them and Java's Unicode strings.  Nevertheless,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
     * the map is implemented in such a way that environment variables
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
     * which are not modified by Java code will have an unmodified
90ce3da70b43 Initial load
duke
parents:
diff changeset
   314
     * native representation in the subprocess.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
     * <p>The returned map and its collection views may not obey the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   317
     * general contract of the {@link Object#equals} and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
     * {@link Object#hashCode} methods.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   319
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   320
     * <p>The returned map is typically case-sensitive on all platforms.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
     * <p>If a security manager exists, its
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   323
     * {@link SecurityManager#checkPermission checkPermission} method
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   324
     * is called with a
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   325
     * {@link RuntimePermission}{@code ("getenv.*")} permission.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   326
     * This may result in a {@link SecurityException} being thrown.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
     * <p>When passing information to a Java subprocess,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
     * <a href=System.html#EnvironmentVSSystemProperties>system properties</a>
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   330
     * are generally preferred over environment variables.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   332
     * @return this process builder's environment
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   333
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   334
     * @throws SecurityException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   335
     *         if a security manager exists and its
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   336
     *         {@link SecurityManager#checkPermission checkPermission}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   337
     *         method doesn't allow access to the process environment
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   339
     * @see    Runtime#exec(String[],String[],java.io.File)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   340
     * @see    System#getenv()
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
    public Map<String,String> environment() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
        SecurityManager security = System.getSecurityManager();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   344
        if (security != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   345
            security.checkPermission(new RuntimePermission("getenv.*"));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   346
90ce3da70b43 Initial load
duke
parents:
diff changeset
   347
        if (environment == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   348
            environment = ProcessEnvironment.environment();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   349
90ce3da70b43 Initial load
duke
parents:
diff changeset
   350
        assert environment != null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   351
90ce3da70b43 Initial load
duke
parents:
diff changeset
   352
        return environment;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   353
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   354
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
    // Only for use by Runtime.exec(...envp...)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
    ProcessBuilder environment(String[] envp) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   357
        assert environment == null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
        if (envp != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
            environment = ProcessEnvironment.emptyEnvironment(envp.length);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
            assert environment != null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
            for (String envstring : envp) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
                // Before 1.5, we blindly passed invalid envstrings
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
                // to the child process.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
                // We would like to throw an exception, but do not,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
                // for compatibility with old broken code.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
                // Silently discard any trailing junk.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
                if (envstring.indexOf((int) '\u0000') != -1)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
                    envstring = envstring.replaceFirst("\u0000.*", "");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
                int eqlsign =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
                    envstring.indexOf('=', ProcessEnvironment.MIN_NAME_LENGTH);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
                // Silently ignore envstrings lacking the required `='.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
                if (eqlsign != -1)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
                    environment.put(envstring.substring(0,eqlsign),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
                                    envstring.substring(eqlsign+1));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
        return this;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
     * Returns this process builder's working directory.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
     * Subprocesses subsequently started by this object's {@link
90ce3da70b43 Initial load
duke
parents:
diff changeset
   387
     * #start()} method will use this as their working directory.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   388
     * The returned value may be {@code null} -- this means to use
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
     * the working directory of the current Java process, usually the
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   390
     * directory named by the system property {@code user.dir},
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   391
     * as the working directory of the child process.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   393
     * @return this process builder's working directory
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
    public File directory() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   396
        return directory;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
     * Sets this process builder's working directory.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
     * Subprocesses subsequently started by this object's {@link
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
     * #start()} method will use this as their working directory.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   404
     * The argument may be {@code null} -- this means to use the
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
     * working directory of the current Java process, usually the
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   406
     * directory named by the system property {@code user.dir},
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   407
     * as the working directory of the child process.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   408
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   409
     * @param  directory the new working directory
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   410
     * @return this process builder
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   412
    public ProcessBuilder directory(File directory) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   413
        this.directory = directory;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
        return this;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   417
    // ---------------- I/O Redirection ----------------
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   418
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   419
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   420
     * Implements a <a href="#redirect-output">null input stream</a>.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   421
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   422
    static class NullInputStream extends InputStream {
5786
f60ef38202e7 6944584: Improvements to subprocess handling on Unix
martin
parents: 5506
diff changeset
   423
        static final NullInputStream INSTANCE = new NullInputStream();
f60ef38202e7 6944584: Improvements to subprocess handling on Unix
martin
parents: 5506
diff changeset
   424
        private NullInputStream() {}
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   425
        public int read()      { return -1; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   426
        public int available() { return 0; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   427
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   428
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   429
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   430
     * Implements a <a href="#redirect-input">null output stream</a>.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   431
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   432
    static class NullOutputStream extends OutputStream {
5786
f60ef38202e7 6944584: Improvements to subprocess handling on Unix
martin
parents: 5506
diff changeset
   433
        static final NullOutputStream INSTANCE = new NullOutputStream();
f60ef38202e7 6944584: Improvements to subprocess handling on Unix
martin
parents: 5506
diff changeset
   434
        private NullOutputStream() {}
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   435
        public void write(int b) throws IOException {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   436
            throw new IOException("Stream closed");
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   437
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   438
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   439
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   440
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   441
     * Represents a source of subprocess input or a destination of
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   442
     * subprocess output.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   443
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   444
     * Each {@code Redirect} instance is one of the following:
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   445
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   446
     * <ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   447
     * <li>the special value {@link #PIPE Redirect.PIPE}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   448
     * <li>the special value {@link #INHERIT Redirect.INHERIT}
32763
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   449
     * <li>the special value {@link #DISCARD Redirect.DISCARD}
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   450
     * <li>a redirection to read from a file, created by an invocation of
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   451
     *     {@link Redirect#from Redirect.from(File)}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   452
     * <li>a redirection to write to a file,  created by an invocation of
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   453
     *     {@link Redirect#to Redirect.to(File)}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   454
     * <li>a redirection to append to a file, created by an invocation of
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   455
     *     {@link Redirect#appendTo Redirect.appendTo(File)}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   456
     * </ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   457
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   458
     * <p>Each of the above categories has an associated unique
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   459
     * {@link Type Type}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   460
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   461
     * @since 1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   462
     */
32649
2ee9017c7597 8136583: Core libraries should use blessed modifier order
martin
parents: 29601
diff changeset
   463
    public abstract static class Redirect {
32763
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   464
        private static final File NULL_FILE = AccessController.doPrivileged(
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   465
                (PrivilegedAction<File>) () -> {
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   466
                    return new File((System.getProperty("os.name")
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   467
                            .startsWith("Windows") ? "NUL" : "/dev/null"));
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   468
                }
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   469
        );
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   470
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   471
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   472
         * The type of a {@link Redirect}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   473
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   474
        public enum Type {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   475
            /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   476
             * The type of {@link Redirect#PIPE Redirect.PIPE}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   477
             */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   478
            PIPE,
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   479
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   480
            /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   481
             * The type of {@link Redirect#INHERIT Redirect.INHERIT}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   482
             */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   483
            INHERIT,
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   484
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   485
            /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   486
             * The type of redirects returned from
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   487
             * {@link Redirect#from Redirect.from(File)}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   488
             */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   489
            READ,
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   490
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   491
            /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   492
             * The type of redirects returned from
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   493
             * {@link Redirect#to Redirect.to(File)}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   494
             */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   495
            WRITE,
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   496
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   497
            /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   498
             * The type of redirects returned from
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   499
             * {@link Redirect#appendTo Redirect.appendTo(File)}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   500
             */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   501
            APPEND
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   502
        };
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   503
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   504
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   505
         * Returns the type of this {@code Redirect}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   506
         * @return the type of this {@code Redirect}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   507
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   508
        public abstract Type type();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   509
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   510
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   511
         * Indicates that subprocess I/O will be connected to the
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   512
         * current Java process over a pipe.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   513
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   514
         * This is the default handling of subprocess standard I/O.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   515
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   516
         * <p>It will always be true that
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   517
         *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   518
         * Redirect.PIPE.file() == null &&
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   519
         * Redirect.PIPE.type() == Redirect.Type.PIPE
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   520
         * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   521
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   522
        public static final Redirect PIPE = new Redirect() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   523
                public Type type() { return Type.PIPE; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   524
                public String toString() { return type().toString(); }};
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   525
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   526
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   527
         * Indicates that subprocess I/O source or destination will be the
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   528
         * same as those of the current process.  This is the normal
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   529
         * behavior of most operating system command interpreters (shells).
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   530
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   531
         * <p>It will always be true that
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   532
         *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   533
         * Redirect.INHERIT.file() == null &&
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   534
         * Redirect.INHERIT.type() == Redirect.Type.INHERIT
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   535
         * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   536
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   537
        public static final Redirect INHERIT = new Redirect() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   538
                public Type type() { return Type.INHERIT; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   539
                public String toString() { return type().toString(); }};
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   540
32763
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   541
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   542
        /**
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   543
         * Indicates that subprocess output will be discarded.
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   544
         * A typical implementation discards the output by writing to
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   545
         * an operating system specific "null file".
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   546
         *
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   547
         * <p>It will always be true that
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   548
         * <pre> {@code
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   549
         * Redirect.DISCARD.file() the filename appropriate for the operating system
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   550
         * and may be null &&
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   551
         * Redirect.DISCARD.type() == Redirect.Type.WRITE &&
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   552
         * Redirect.DISCARD.append() == false
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   553
         * }</pre>
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   554
         * @since 9
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   555
         */
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   556
        public static final Redirect DISCARD = new Redirect() {
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   557
                public Type type() { return Type.WRITE; }
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   558
                public String toString() { return type().toString(); }
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   559
                public File file() { return NULL_FILE; }
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   560
                boolean append() { return false; }
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   561
        };
c11c2b9b45a5 8132541: (process) ProcessBuilder support for redirection to discard output
rriggs
parents: 32649
diff changeset
   562
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   563
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   564
         * Returns the {@link File} source or destination associated
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   565
         * with this redirect, or {@code null} if there is no such file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   566
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   567
         * @return the file associated with this redirect,
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   568
         *         or {@code null} if there is no such file
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   569
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   570
        public File file() { return null; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   571
7515
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   572
        /**
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   573
         * When redirected to a destination file, indicates if the output
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   574
         * is to be written to the end of the file.
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   575
         */
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   576
        boolean append() {
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   577
            throw new UnsupportedOperationException();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   578
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   579
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   580
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   581
         * Returns a redirect to read from the specified file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   582
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   583
         * <p>It will always be true that
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   584
         *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   585
         * Redirect.from(file).file() == file &&
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   586
         * Redirect.from(file).type() == Redirect.Type.READ
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   587
         * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   588
         *
18776
c17100862d86 8019862: Fix doclint errors in java.lang.*.
bpb
parents: 17467
diff changeset
   589
         * @param file The {@code File} for the {@code Redirect}.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   590
         * @return a redirect to read from the specified file
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   591
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   592
        public static Redirect from(final File file) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   593
            if (file == null)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   594
                throw new NullPointerException();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   595
            return new Redirect() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   596
                    public Type type() { return Type.READ; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   597
                    public File file() { return file; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   598
                    public String toString() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   599
                        return "redirect to read from file \"" + file + "\"";
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   600
                    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   601
                };
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   602
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   603
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   604
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   605
         * Returns a redirect to write to the specified file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   606
         * If the specified file exists when the subprocess is started,
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   607
         * its previous contents will be discarded.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   608
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   609
         * <p>It will always be true that
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   610
         *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   611
         * Redirect.to(file).file() == file &&
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   612
         * Redirect.to(file).type() == Redirect.Type.WRITE
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   613
         * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   614
         *
18776
c17100862d86 8019862: Fix doclint errors in java.lang.*.
bpb
parents: 17467
diff changeset
   615
         * @param file The {@code File} for the {@code Redirect}.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   616
         * @return a redirect to write to the specified file
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   617
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   618
        public static Redirect to(final File file) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   619
            if (file == null)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   620
                throw new NullPointerException();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   621
            return new Redirect() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   622
                    public Type type() { return Type.WRITE; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   623
                    public File file() { return file; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   624
                    public String toString() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   625
                        return "redirect to write to file \"" + file + "\"";
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   626
                    }
7515
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   627
                    boolean append() { return false; }
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   628
                };
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   629
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   630
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   631
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   632
         * Returns a redirect to append to the specified file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   633
         * Each write operation first advances the position to the
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   634
         * end of the file and then writes the requested data.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   635
         * Whether the advancement of the position and the writing
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   636
         * of the data are done in a single atomic operation is
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   637
         * system-dependent and therefore unspecified.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   638
         *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   639
         * <p>It will always be true that
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   640
         *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   641
         * Redirect.appendTo(file).file() == file &&
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   642
         * Redirect.appendTo(file).type() == Redirect.Type.APPEND
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   643
         * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   644
         *
18776
c17100862d86 8019862: Fix doclint errors in java.lang.*.
bpb
parents: 17467
diff changeset
   645
         * @param file The {@code File} for the {@code Redirect}.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   646
         * @return a redirect to append to the specified file
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   647
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   648
        public static Redirect appendTo(final File file) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   649
            if (file == null)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   650
                throw new NullPointerException();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   651
            return new Redirect() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   652
                    public Type type() { return Type.APPEND; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   653
                    public File file() { return file; }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   654
                    public String toString() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   655
                        return "redirect to append to file \"" + file + "\"";
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   656
                    }
7515
43202796198e 6709457: (fc) lock/tryLock() throws IOException "Access is denied" when file opened for append [win]
alanb
parents: 5786
diff changeset
   657
                    boolean append() { return true; }
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   658
                };
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   659
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   660
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   661
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   662
         * Compares the specified object with this {@code Redirect} for
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   663
         * equality.  Returns {@code true} if and only if the two
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   664
         * objects are identical or both objects are {@code Redirect}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   665
         * instances of the same type associated with non-null equal
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   666
         * {@code File} instances.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   667
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   668
        public boolean equals(Object obj) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   669
            if (obj == this)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   670
                return true;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   671
            if (! (obj instanceof Redirect))
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   672
                return false;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   673
            Redirect r = (Redirect) obj;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   674
            if (r.type() != this.type())
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   675
                return false;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   676
            assert this.file() != null;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   677
            return this.file().equals(r.file());
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   678
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   679
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   680
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   681
         * Returns a hash code value for this {@code Redirect}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   682
         * @return a hash code value for this {@code Redirect}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   683
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   684
        public int hashCode() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   685
            File file = file();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   686
            if (file == null)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   687
                return super.hashCode();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   688
            else
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   689
                return file.hashCode();
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   690
        }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   691
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   692
        /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   693
         * No public constructors.  Clients must use predefined
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   694
         * static {@code Redirect} instances or factory methods.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   695
         */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   696
        private Redirect() {}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   697
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   698
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   699
    private Redirect[] redirects() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   700
        if (redirects == null)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   701
            redirects = new Redirect[] {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   702
                Redirect.PIPE, Redirect.PIPE, Redirect.PIPE
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   703
            };
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   704
        return redirects;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   705
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   706
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   707
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   708
     * Sets this process builder's standard input source.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   709
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   710
     * Subprocesses subsequently started by this object's {@link #start()}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   711
     * method obtain their standard input from this source.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   712
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   713
     * <p>If the source is {@link Redirect#PIPE Redirect.PIPE}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   714
     * (the initial value), then the standard input of a
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   715
     * subprocess can be written to using the output stream
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   716
     * returned by {@link Process#getOutputStream()}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   717
     * If the source is set to any other value, then
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   718
     * {@link Process#getOutputStream()} will return a
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   719
     * <a href="#redirect-input">null output stream</a>.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   720
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   721
     * @param  source the new standard input source
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   722
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   723
     * @throws IllegalArgumentException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   724
     *         if the redirect does not correspond to a valid source
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   725
     *         of data, that is, has type
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   726
     *         {@link Redirect.Type#WRITE WRITE} or
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   727
     *         {@link Redirect.Type#APPEND APPEND}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   728
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   729
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   730
    public ProcessBuilder redirectInput(Redirect source) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   731
        if (source.type() == Redirect.Type.WRITE ||
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   732
            source.type() == Redirect.Type.APPEND)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   733
            throw new IllegalArgumentException(
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   734
                "Redirect invalid for reading: " + source);
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   735
        redirects()[0] = source;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   736
        return this;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   737
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   738
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   739
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   740
     * Sets this process builder's standard output destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   741
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   742
     * Subprocesses subsequently started by this object's {@link #start()}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   743
     * method send their standard output to this destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   744
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   745
     * <p>If the destination is {@link Redirect#PIPE Redirect.PIPE}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   746
     * (the initial value), then the standard output of a subprocess
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   747
     * can be read using the input stream returned by {@link
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   748
     * Process#getInputStream()}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   749
     * If the destination is set to any other value, then
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   750
     * {@link Process#getInputStream()} will return a
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   751
     * <a href="#redirect-output">null input stream</a>.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   752
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   753
     * @param  destination the new standard output destination
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   754
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   755
     * @throws IllegalArgumentException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   756
     *         if the redirect does not correspond to a valid
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   757
     *         destination of data, that is, has type
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   758
     *         {@link Redirect.Type#READ READ}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   759
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   760
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   761
    public ProcessBuilder redirectOutput(Redirect destination) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   762
        if (destination.type() == Redirect.Type.READ)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   763
            throw new IllegalArgumentException(
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   764
                "Redirect invalid for writing: " + destination);
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   765
        redirects()[1] = destination;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   766
        return this;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   767
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   768
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   769
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   770
     * Sets this process builder's standard error destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   771
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   772
     * Subprocesses subsequently started by this object's {@link #start()}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   773
     * method send their standard error to this destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   774
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   775
     * <p>If the destination is {@link Redirect#PIPE Redirect.PIPE}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   776
     * (the initial value), then the error output of a subprocess
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   777
     * can be read using the input stream returned by {@link
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   778
     * Process#getErrorStream()}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   779
     * If the destination is set to any other value, then
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   780
     * {@link Process#getErrorStream()} will return a
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   781
     * <a href="#redirect-output">null input stream</a>.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   782
     *
28058
87940c838900 8056238: (process) ProcessBuilder.redirectError spec has a broken link
rriggs
parents: 25859
diff changeset
   783
     * <p>If the {@link #redirectErrorStream() redirectErrorStream}
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   784
     * attribute has been set {@code true}, then the redirection set
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   785
     * by this method has no effect.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   786
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   787
     * @param  destination the new standard error destination
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   788
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   789
     * @throws IllegalArgumentException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   790
     *         if the redirect does not correspond to a valid
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   791
     *         destination of data, that is, has type
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   792
     *         {@link Redirect.Type#READ READ}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   793
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   794
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   795
    public ProcessBuilder redirectError(Redirect destination) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   796
        if (destination.type() == Redirect.Type.READ)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   797
            throw new IllegalArgumentException(
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   798
                "Redirect invalid for writing: " + destination);
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   799
        redirects()[2] = destination;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   800
        return this;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   801
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   802
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   803
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   804
     * Sets this process builder's standard input source to a file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   805
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   806
     * <p>This is a convenience method.  An invocation of the form
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   807
     * {@code redirectInput(file)}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   808
     * behaves in exactly the same way as the invocation
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   809
     * {@link #redirectInput(Redirect) redirectInput}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   810
     * {@code (Redirect.from(file))}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   811
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   812
     * @param  file the new standard input source
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   813
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   814
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   815
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   816
    public ProcessBuilder redirectInput(File file) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   817
        return redirectInput(Redirect.from(file));
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   818
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   819
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   820
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   821
     * Sets this process builder's standard output destination to a file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   822
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   823
     * <p>This is a convenience method.  An invocation of the form
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   824
     * {@code redirectOutput(file)}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   825
     * behaves in exactly the same way as the invocation
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   826
     * {@link #redirectOutput(Redirect) redirectOutput}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   827
     * {@code (Redirect.to(file))}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   828
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   829
     * @param  file the new standard output destination
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   830
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   831
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   832
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   833
    public ProcessBuilder redirectOutput(File file) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   834
        return redirectOutput(Redirect.to(file));
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   835
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   836
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   837
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   838
     * Sets this process builder's standard error destination to a file.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   839
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   840
     * <p>This is a convenience method.  An invocation of the form
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   841
     * {@code redirectError(file)}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   842
     * behaves in exactly the same way as the invocation
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   843
     * {@link #redirectError(Redirect) redirectError}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   844
     * {@code (Redirect.to(file))}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   845
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   846
     * @param  file the new standard error destination
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   847
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   848
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   849
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   850
    public ProcessBuilder redirectError(File file) {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   851
        return redirectError(Redirect.to(file));
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   852
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   853
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   854
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   855
     * Returns this process builder's standard input source.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   856
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   857
     * Subprocesses subsequently started by this object's {@link #start()}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   858
     * method obtain their standard input from this source.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   859
     * The initial value is {@link Redirect#PIPE Redirect.PIPE}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   860
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   861
     * @return this process builder's standard input source
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   862
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   863
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   864
    public Redirect redirectInput() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   865
        return (redirects == null) ? Redirect.PIPE : redirects[0];
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   866
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   867
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   868
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   869
     * Returns this process builder's standard output destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   870
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   871
     * Subprocesses subsequently started by this object's {@link #start()}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   872
     * method redirect their standard output to this destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   873
     * The initial value is {@link Redirect#PIPE Redirect.PIPE}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   874
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   875
     * @return this process builder's standard output destination
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   876
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   877
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   878
    public Redirect redirectOutput() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   879
        return (redirects == null) ? Redirect.PIPE : redirects[1];
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   880
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   881
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   882
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   883
     * Returns this process builder's standard error destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   884
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   885
     * Subprocesses subsequently started by this object's {@link #start()}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   886
     * method redirect their standard error to this destination.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   887
     * The initial value is {@link Redirect#PIPE Redirect.PIPE}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   888
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   889
     * @return this process builder's standard error destination
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   890
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   891
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   892
    public Redirect redirectError() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   893
        return (redirects == null) ? Redirect.PIPE : redirects[2];
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   894
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   895
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   896
    /**
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   897
     * Sets the source and destination for subprocess standard I/O
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   898
     * to be the same as those of the current Java process.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   899
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   900
     * <p>This is a convenience method.  An invocation of the form
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   901
     *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   902
     * pb.inheritIO()
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   903
     * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   904
     * behaves in exactly the same way as the invocation
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   905
     *  <pre> {@code
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   906
     * pb.redirectInput(Redirect.INHERIT)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   907
     *   .redirectOutput(Redirect.INHERIT)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   908
     *   .redirectError(Redirect.INHERIT)
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   909
     * }</pre>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   910
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   911
     * This gives behavior equivalent to most operating system
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   912
     * command interpreters, or the standard C library function
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   913
     * {@code system()}.
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   914
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   915
     * @return this process builder
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   916
     * @since  1.7
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   917
     */
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   918
    public ProcessBuilder inheritIO() {
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   919
        Arrays.fill(redirects(), Redirect.INHERIT);
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   920
        return this;
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   921
    }
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   922
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   923
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   924
     * Tells whether this process builder merges standard error and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   925
     * standard output.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   926
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   927
     * <p>If this property is {@code true}, then any error output
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   928
     * generated by subprocesses subsequently started by this object's
90ce3da70b43 Initial load
duke
parents:
diff changeset
   929
     * {@link #start()} method will be merged with the standard
90ce3da70b43 Initial load
duke
parents:
diff changeset
   930
     * output, so that both can be read using the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   931
     * {@link Process#getInputStream()} method.  This makes it easier
90ce3da70b43 Initial load
duke
parents:
diff changeset
   932
     * to correlate error messages with the corresponding output.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   933
     * The initial value is {@code false}.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   934
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   935
     * @return this process builder's {@code redirectErrorStream} property
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   936
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   937
    public boolean redirectErrorStream() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   938
        return redirectErrorStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   939
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   940
90ce3da70b43 Initial load
duke
parents:
diff changeset
   941
    /**
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   942
     * Sets this process builder's {@code redirectErrorStream} property.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   943
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   944
     * <p>If this property is {@code true}, then any error output
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   945
     * generated by subprocesses subsequently started by this object's
90ce3da70b43 Initial load
duke
parents:
diff changeset
   946
     * {@link #start()} method will be merged with the standard
90ce3da70b43 Initial load
duke
parents:
diff changeset
   947
     * output, so that both can be read using the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   948
     * {@link Process#getInputStream()} method.  This makes it easier
90ce3da70b43 Initial load
duke
parents:
diff changeset
   949
     * to correlate error messages with the corresponding output.
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   950
     * The initial value is {@code false}.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   951
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   952
     * @param  redirectErrorStream the new property value
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   953
     * @return this process builder
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   954
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   955
    public ProcessBuilder redirectErrorStream(boolean redirectErrorStream) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   956
        this.redirectErrorStream = redirectErrorStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   957
        return this;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   958
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   959
90ce3da70b43 Initial load
duke
parents:
diff changeset
   960
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   961
     * Starts a new process using the attributes of this process builder.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   962
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   963
     * <p>The new process will
90ce3da70b43 Initial load
duke
parents:
diff changeset
   964
     * invoke the command and arguments given by {@link #command()},
90ce3da70b43 Initial load
duke
parents:
diff changeset
   965
     * in a working directory as given by {@link #directory()},
90ce3da70b43 Initial load
duke
parents:
diff changeset
   966
     * with a process environment as given by {@link #environment()}.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   967
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   968
     * <p>This method checks that the command is a valid operating
90ce3da70b43 Initial load
duke
parents:
diff changeset
   969
     * system command.  Which commands are valid is system-dependent,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   970
     * but at the very least the command must be a non-empty list of
90ce3da70b43 Initial load
duke
parents:
diff changeset
   971
     * non-null strings.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   972
     *
9500
268f823d9e1c 7034570: java.lang.Runtime.exec(String[] cmd, String[] env) can not work properly if SystemRoot not inherited
michaelm
parents: 7816
diff changeset
   973
     * <p>A minimal set of system dependent environment variables may
268f823d9e1c 7034570: java.lang.Runtime.exec(String[] cmd, String[] env) can not work properly if SystemRoot not inherited
michaelm
parents: 7816
diff changeset
   974
     * be required to start a process on some operating systems.
268f823d9e1c 7034570: java.lang.Runtime.exec(String[] cmd, String[] env) can not work properly if SystemRoot not inherited
michaelm
parents: 7816
diff changeset
   975
     * As a result, the subprocess may inherit additional environment variable
268f823d9e1c 7034570: java.lang.Runtime.exec(String[] cmd, String[] env) can not work properly if SystemRoot not inherited
michaelm
parents: 7816
diff changeset
   976
     * settings beyond those in the process builder's {@link #environment()}.
268f823d9e1c 7034570: java.lang.Runtime.exec(String[] cmd, String[] env) can not work properly if SystemRoot not inherited
michaelm
parents: 7816
diff changeset
   977
     *
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   978
     * <p>If there is a security manager, its
90ce3da70b43 Initial load
duke
parents:
diff changeset
   979
     * {@link SecurityManager#checkExec checkExec}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   980
     * method is called with the first component of this object's
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
   981
     * {@code command} array as its argument. This may result in
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   982
     * a {@link SecurityException} being thrown.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   983
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   984
     * <p>Starting an operating system process is highly system-dependent.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   985
     * Among the many things that can go wrong are:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   986
     * <ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   987
     * <li>The operating system program file was not found.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   988
     * <li>Access to the program file was denied.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   989
     * <li>The working directory does not exist.
25539
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
   990
     * <li>Invalid character in command argument, such as NUL.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   991
     * </ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   992
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   993
     * <p>In such cases an exception will be thrown.  The exact nature
90ce3da70b43 Initial load
duke
parents:
diff changeset
   994
     * of the exception is system-dependent, but it will always be a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   995
     * subclass of {@link IOException}.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   996
     *
28872
82a09f5a7ed6 8072034: (process) ProcessBuilder.start and Runtime.exec UnsupportedOperationException editorial cleanup
rriggs
parents: 28750
diff changeset
   997
     * <p>If the operating system does not support the creation of
82a09f5a7ed6 8072034: (process) ProcessBuilder.start and Runtime.exec UnsupportedOperationException editorial cleanup
rriggs
parents: 28750
diff changeset
   998
     * processes, an {@link UnsupportedOperationException} will be thrown.
82a09f5a7ed6 8072034: (process) ProcessBuilder.start and Runtime.exec UnsupportedOperationException editorial cleanup
rriggs
parents: 28750
diff changeset
   999
     *
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1000
     * <p>Subsequent modifications to this process builder will not
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1001
     * affect the returned {@link Process}.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1002
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1003
     * @return a new {@link Process} object for managing the subprocess
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1004
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1005
     * @throws NullPointerException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1006
     *         if an element of the command list is null
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1007
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1008
     * @throws IndexOutOfBoundsException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1009
     *         if the command is an empty list (has size {@code 0})
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1010
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1011
     * @throws SecurityException
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1012
     *         if a security manager exists and
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1013
     *         <ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1014
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1015
     *         <li>its
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1016
     *         {@link SecurityManager#checkExec checkExec}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1017
     *         method doesn't allow creation of the subprocess, or
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1018
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1019
     *         <li>the standard input to the subprocess was
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1020
     *         {@linkplain #redirectInput redirected from a file}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1021
     *         and the security manager's
29600
3c042ba9fd42 8071480: (process spec) ProcessBuilder.start spec linked to the wrong checkRead and checkWrite methods
rriggs
parents: 29381
diff changeset
  1022
     *         {@link SecurityManager#checkRead(String) checkRead} method
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1023
     *         denies read access to the file, or
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1024
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1025
     *         <li>the standard output or standard error of the
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1026
     *         subprocess was
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1027
     *         {@linkplain #redirectOutput redirected to a file}
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1028
     *         and the security manager's
29600
3c042ba9fd42 8071480: (process spec) ProcessBuilder.start spec linked to the wrong checkRead and checkWrite methods
rriggs
parents: 29381
diff changeset
  1029
     *         {@link SecurityManager#checkWrite(String) checkWrite} method
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1030
     *         denies write access to the file
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1031
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1032
     *         </ul>
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1033
     *
28750
a1dae439cdab 8055330: (process spec) ProcessBuilder.start and Runtime.exec should throw UnsupportedOperationException on platforms that don't support
rriggs
parents: 28058
diff changeset
  1034
     * @throws  UnsupportedOperationException
a1dae439cdab 8055330: (process spec) ProcessBuilder.start and Runtime.exec should throw UnsupportedOperationException on platforms that don't support
rriggs
parents: 28058
diff changeset
  1035
     *          If the operating system does not support the creation of processes.
a1dae439cdab 8055330: (process spec) ProcessBuilder.start and Runtime.exec should throw UnsupportedOperationException on platforms that don't support
rriggs
parents: 28058
diff changeset
  1036
     *
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1037
     * @throws IOException if an I/O error occurs
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1038
     *
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1039
     * @see Runtime#exec(String[], String[], java.io.File)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1040
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1041
    public Process start() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1042
        // Must convert to array first -- a malicious user-supplied
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1043
        // list might try to circumvent the security check.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1044
        String[] cmdarray = command.toArray(new String[command.size()]);
5186
1bbb82130b38 6910590: Application can modify command array, in ProcessBuilder
michaelm
parents: 48
diff changeset
  1045
        cmdarray = cmdarray.clone();
1bbb82130b38 6910590: Application can modify command array, in ProcessBuilder
michaelm
parents: 48
diff changeset
  1046
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1047
        for (String arg : cmdarray)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1048
            if (arg == null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1049
                throw new NullPointerException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1050
        // Throws IndexOutOfBoundsException if command is empty
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1051
        String prog = cmdarray[0];
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1052
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1053
        SecurityManager security = System.getSecurityManager();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1054
        if (security != null)
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1055
            security.checkExec(prog);
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1056
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1057
        String dir = directory == null ? null : directory.toString();
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1058
25539
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
  1059
        for (int i = 1; i < cmdarray.length; i++) {
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
  1060
            if (cmdarray[i].indexOf('\u0000') >= 0) {
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
  1061
                throw new IOException("invalid null character in command");
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
  1062
            }
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
  1063
        }
9d290547d266 8036571: (process) Process process arguments carefully
rriggs
parents: 23010
diff changeset
  1064
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1065
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1066
            return ProcessImpl.start(cmdarray,
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1067
                                     environment,
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1068
                                     dir,
48
dc5744ca15ea 4960438: (process) Need IO redirection API for subprocesses
martin
parents: 2
diff changeset
  1069
                                     redirects,
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1070
                                     redirectErrorStream);
17467
374c1cceefff 8012453: (process) Runtime.exec(String) fails if command contains spaces [win]
uta
parents: 16877
diff changeset
  1071
        } catch (IOException | IllegalArgumentException e) {
16877
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1072
            String exceptionInfo = ": " + e.getMessage();
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1073
            Throwable cause = e;
17467
374c1cceefff 8012453: (process) Runtime.exec(String) fails if command contains spaces [win]
uta
parents: 16877
diff changeset
  1074
            if ((e instanceof IOException) && security != null) {
16877
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1075
                // Can not disclose the fail reason for read-protected files.
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1076
                try {
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1077
                    security.checkRead(prog);
19606
6c846d61ba2f 8023717: (process) ProcessBuilder should catch SecurityException rather than AccessControlException
alanb
parents: 18776
diff changeset
  1078
                } catch (SecurityException se) {
16877
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1079
                    exceptionInfo = "";
19606
6c846d61ba2f 8023717: (process) ProcessBuilder should catch SecurityException rather than AccessControlException
alanb
parents: 18776
diff changeset
  1080
                    cause = se;
16877
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1081
                }
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1082
            }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1083
            // It's much easier for us to create a high-quality error
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1084
            // message than the low-level C code which found the problem.
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1085
            throw new IOException(
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1086
                "Cannot run program \"" + prog + "\""
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1087
                + (dir == null ? "" : " (in directory \"" + dir + "\")")
16877
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1088
                + exceptionInfo,
d12c06e6e2ba 8005942: (process) Improved Runtime.exec
uta
parents: 14342
diff changeset
  1089
                cause);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1090
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1091
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
  1092
}