2013-09-16 15:28:56 -07:00
|
|
|
// Copyright 2013 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
//! Bindings for executing child processes
|
|
|
|
|
|
|
|
use prelude::*;
|
|
|
|
|
|
|
|
use libc;
|
2013-11-10 22:46:32 -08:00
|
|
|
use io;
|
2014-01-29 16:33:57 -08:00
|
|
|
use io::IoResult;
|
2013-12-05 17:25:48 -08:00
|
|
|
use rt::rtio::{RtioProcess, IoFactory, LocalIo};
|
2013-09-16 15:28:56 -07:00
|
|
|
|
2013-11-12 11:37:14 +10:00
|
|
|
use fmt;
|
|
|
|
|
2013-10-06 13:22:18 -07:00
|
|
|
// windows values don't matter as long as they're at least one of unix's
|
|
|
|
// TERM/KILL/INT signals
|
|
|
|
#[cfg(windows)] pub static PleaseExitSignal: int = 15;
|
|
|
|
#[cfg(windows)] pub static MustDieSignal: int = 9;
|
|
|
|
#[cfg(not(windows))] pub static PleaseExitSignal: int = libc::SIGTERM as int;
|
|
|
|
#[cfg(not(windows))] pub static MustDieSignal: int = libc::SIGKILL as int;
|
|
|
|
|
2013-09-16 15:28:56 -07:00
|
|
|
pub struct Process {
|
2013-10-16 14:48:05 -07:00
|
|
|
priv handle: ~RtioProcess,
|
2013-09-16 15:28:56 -07:00
|
|
|
io: ~[Option<io::PipeStream>],
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This configuration describes how a new process should be spawned. This is
|
|
|
|
/// translated to libuv's own configuration
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct ProcessConfig<'a> {
|
2013-09-16 15:28:56 -07:00
|
|
|
/// Path to the program to run
|
2013-12-09 23:16:18 -08:00
|
|
|
program: &'a str,
|
2013-09-16 15:28:56 -07:00
|
|
|
|
|
|
|
/// Arguments to pass to the program (doesn't include the program itself)
|
2013-12-09 23:16:18 -08:00
|
|
|
args: &'a [~str],
|
2013-09-16 15:28:56 -07:00
|
|
|
|
|
|
|
/// Optional environment to specify for the program. If this is None, then
|
|
|
|
/// it will inherit the current process's environment.
|
2013-12-09 23:16:18 -08:00
|
|
|
env: Option<&'a [(~str, ~str)]>,
|
2013-09-16 15:28:56 -07:00
|
|
|
|
|
|
|
/// Optional working directory for the new process. If this is None, then
|
|
|
|
/// the current directory of the running process is inherited.
|
2013-12-09 23:16:18 -08:00
|
|
|
cwd: Option<&'a str>,
|
2013-09-16 15:28:56 -07:00
|
|
|
|
|
|
|
/// Any number of streams/file descriptors/pipes may be attached to this
|
|
|
|
/// process. This list enumerates the file descriptors and such for the
|
|
|
|
/// process to be spawned, and the file descriptors inherited will start at
|
|
|
|
/// 0 and go to the length of this array.
|
|
|
|
///
|
|
|
|
/// Standard file descriptors are:
|
|
|
|
///
|
|
|
|
/// 0 - stdin
|
|
|
|
/// 1 - stdout
|
|
|
|
/// 2 - stderr
|
2014-02-06 22:38:05 -08:00
|
|
|
io: &'a [StdioContainer],
|
|
|
|
|
|
|
|
/// Sets the child process's user id. This translates to a `setuid` call in
|
|
|
|
/// the child process. Setting this value on windows will cause the spawn to
|
|
|
|
/// fail. Failure in the `setuid` call on unix will also cause the spawn to
|
|
|
|
/// fail.
|
|
|
|
uid: Option<uint>,
|
|
|
|
|
|
|
|
/// Similar to `uid`, but sets the group id of the child process. This has
|
|
|
|
/// the same semantics as the `uid` field.
|
|
|
|
gid: Option<uint>,
|
|
|
|
|
|
|
|
/// If true, the child process is spawned in a detached state. On unix, this
|
|
|
|
/// means that the child is the leader of a new process group.
|
|
|
|
detach: bool,
|
2013-09-16 15:28:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Describes what to do with a standard io stream for a child process.
|
|
|
|
pub enum StdioContainer {
|
|
|
|
/// This stream will be ignored. This is the equivalent of attaching the
|
|
|
|
/// stream to `/dev/null`
|
|
|
|
Ignored,
|
|
|
|
|
|
|
|
/// The specified file descriptor is inherited for the stream which it is
|
|
|
|
/// specified for.
|
|
|
|
InheritFd(libc::c_int),
|
|
|
|
|
2013-10-16 11:39:51 -07:00
|
|
|
/// Creates a pipe for the specified file descriptor which will be created
|
|
|
|
/// when the process is spawned.
|
2013-09-16 15:28:56 -07:00
|
|
|
///
|
|
|
|
/// The first boolean argument is whether the pipe is readable, and the
|
|
|
|
/// second is whether it is writable. These properties are from the view of
|
|
|
|
/// the *child* process, not the parent process.
|
2013-10-16 11:39:51 -07:00
|
|
|
CreatePipe(bool /* readable */, bool /* writable */),
|
2013-09-16 15:28:56 -07:00
|
|
|
}
|
|
|
|
|
2013-11-12 11:37:14 +10:00
|
|
|
/// Describes the result of a process after it has terminated.
|
2013-11-14 02:58:19 +09:00
|
|
|
/// Note that Windows have no signals, so the result is usually ExitStatus.
|
2013-11-12 11:37:14 +10:00
|
|
|
#[deriving(Eq)]
|
|
|
|
pub enum ProcessExit {
|
|
|
|
/// Normal termination with an exit status.
|
|
|
|
ExitStatus(int),
|
|
|
|
|
|
|
|
/// Termination by signal, with the signal number.
|
|
|
|
ExitSignal(int),
|
|
|
|
}
|
|
|
|
|
2014-01-30 00:46:37 +11:00
|
|
|
impl fmt::Show for ProcessExit {
|
2013-11-12 11:37:14 +10:00
|
|
|
/// Format a ProcessExit enum, to nicely present the information.
|
2014-02-05 23:55:13 +11:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match *self {
|
2013-11-12 11:37:14 +10:00
|
|
|
ExitStatus(code) => write!(f.buf, "exit code: {}", code),
|
|
|
|
ExitSignal(code) => write!(f.buf, "signal: {}", code),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ProcessExit {
|
|
|
|
/// Was termination successful? Signal termination not considered a success,
|
|
|
|
/// and success is defined as a zero exit status.
|
|
|
|
pub fn success(&self) -> bool {
|
|
|
|
return self.matches_exit_status(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Checks whether this ProcessExit matches the given exit status.
|
|
|
|
/// Termination by signal will never match an exit code.
|
|
|
|
pub fn matches_exit_status(&self, wanted: int) -> bool {
|
|
|
|
*self == ExitStatus(wanted)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-06 22:38:05 -08:00
|
|
|
impl<'a> ProcessConfig<'a> {
|
|
|
|
/// Creates a new configuration with blanks as all of the defaults. This is
|
|
|
|
/// useful when using functional struct updates:
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use std::io::process::{ProcessConfig, Process};
|
|
|
|
///
|
|
|
|
/// let config = ProcessConfig {
|
|
|
|
/// program: "/bin/sh",
|
|
|
|
/// args: &'static [~"-c", ~"echo hello"],
|
|
|
|
/// .. ProcessConfig::new()
|
|
|
|
/// };
|
|
|
|
///
|
|
|
|
/// let p = Process::new(config);
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
pub fn new() -> ProcessConfig<'static> {
|
|
|
|
ProcessConfig {
|
|
|
|
program: "",
|
|
|
|
args: &'static [],
|
|
|
|
env: None,
|
|
|
|
cwd: None,
|
|
|
|
io: &'static [],
|
|
|
|
uid: None,
|
|
|
|
gid: None,
|
|
|
|
detach: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-16 15:28:56 -07:00
|
|
|
impl Process {
|
|
|
|
/// Creates a new pipe initialized, but not bound to any particular
|
|
|
|
/// source/destination
|
2014-01-29 16:33:57 -08:00
|
|
|
pub fn new(config: ProcessConfig) -> IoResult<Process> {
|
2013-12-12 17:30:41 -08:00
|
|
|
let mut config = Some(config);
|
|
|
|
LocalIo::maybe_raise(|io| {
|
|
|
|
io.spawn(config.take_unwrap()).map(|(p, io)| {
|
|
|
|
Process {
|
|
|
|
handle: p,
|
|
|
|
io: io.move_iter().map(|p| {
|
|
|
|
p.map(|p| io::PipeStream::new(p))
|
|
|
|
}).collect()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
2013-09-16 15:28:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the process id of this child process
|
|
|
|
pub fn id(&self) -> libc::pid_t { self.handle.id() }
|
|
|
|
|
|
|
|
/// Sends the specified signal to the child process, returning whether the
|
|
|
|
/// signal could be delivered or not.
|
|
|
|
///
|
|
|
|
/// Note that this is purely a wrapper around libuv's `uv_process_kill`
|
|
|
|
/// function.
|
|
|
|
///
|
2014-01-30 16:55:20 -08:00
|
|
|
/// If the signal delivery fails, the corresponding error is returned.
|
2014-01-29 16:33:57 -08:00
|
|
|
pub fn signal(&mut self, signal: int) -> IoResult<()> {
|
|
|
|
self.handle.kill(signal)
|
2013-09-16 15:28:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Wait for the child to exit completely, returning the status that it
|
|
|
|
/// exited with. This function will continue to have the same return value
|
|
|
|
/// after it has been called at least once.
|
2013-11-12 11:37:14 +10:00
|
|
|
pub fn wait(&mut self) -> ProcessExit { self.handle.wait() }
|
2013-09-16 15:28:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Process {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
// Close all I/O before exiting to ensure that the child doesn't wait
|
|
|
|
// forever to print some text or something similar.
|
2013-12-23 16:20:52 +01:00
|
|
|
loop {
|
|
|
|
match self.io.pop() {
|
|
|
|
Some(_) => (),
|
|
|
|
None => break,
|
|
|
|
}
|
2013-09-16 15:28:56 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
self.wait();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-26 18:28:24 -08:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use io::process::{ProcessConfig, Process};
|
|
|
|
use prelude::*;
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn smoke() {
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
2014-01-15 14:39:08 -05:00
|
|
|
args: &[~"-c", ~"true"],
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
|
|
|
let p = Process::new(args);
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(p.is_ok());
|
2013-12-26 18:28:24 -08:00
|
|
|
let mut p = p.unwrap();
|
|
|
|
assert!(p.wait().success());
|
|
|
|
})
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn smoke_failure() {
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "if-this-is-a-binary-then-the-world-has-ended",
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
2014-01-30 14:10:53 -08:00
|
|
|
match Process::new(args) {
|
2013-12-26 18:28:24 -08:00
|
|
|
Ok(..) => fail!(),
|
|
|
|
Err(..) => {}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn exit_reported_right() {
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
2014-01-15 14:39:08 -05:00
|
|
|
args: &[~"-c", ~"exit 1"],
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
|
|
|
let p = Process::new(args);
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(p.is_ok());
|
2013-12-26 18:28:24 -08:00
|
|
|
let mut p = p.unwrap();
|
|
|
|
assert!(p.wait().matches_exit_status(1));
|
|
|
|
})
|
|
|
|
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn signal_reported_right() {
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
2014-01-15 14:39:08 -05:00
|
|
|
args: &[~"-c", ~"kill -1 $$"],
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
|
|
|
let p = Process::new(args);
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(p.is_ok());
|
2013-12-26 18:28:24 -08:00
|
|
|
let mut p = p.unwrap();
|
|
|
|
match p.wait() {
|
|
|
|
process::ExitSignal(1) => {},
|
|
|
|
result => fail!("not terminated by signal 1 (instead, {})", result),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
pub fn read_all(input: &mut Reader) -> ~str {
|
2014-01-30 14:10:53 -08:00
|
|
|
input.read_to_str().unwrap()
|
2013-12-26 18:28:24 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn run_output(args: ProcessConfig) -> ~str {
|
|
|
|
let p = Process::new(args);
|
2014-01-30 14:10:53 -08:00
|
|
|
assert!(p.is_ok());
|
2013-12-26 18:28:24 -08:00
|
|
|
let mut p = p.unwrap();
|
|
|
|
assert!(p.io[0].is_none());
|
|
|
|
assert!(p.io[1].is_some());
|
|
|
|
let ret = read_all(p.io[1].get_mut_ref() as &mut Reader);
|
|
|
|
assert!(p.wait().success());
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn stdout_works() {
|
|
|
|
let io = ~[Ignored, CreatePipe(false, true)];
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
2014-01-15 14:39:08 -05:00
|
|
|
args: &[~"-c", ~"echo foobar"],
|
2013-12-26 18:28:24 -08:00
|
|
|
io: io,
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
|
|
|
assert_eq!(run_output(args), ~"foobar\n");
|
|
|
|
})
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn set_cwd_works() {
|
|
|
|
let io = ~[Ignored, CreatePipe(false, true)];
|
|
|
|
let cwd = Some("/");
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
2014-01-15 14:39:08 -05:00
|
|
|
args: &[~"-c", ~"pwd"],
|
2013-12-26 18:28:24 -08:00
|
|
|
cwd: cwd,
|
|
|
|
io: io,
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
|
|
|
assert_eq!(run_output(args), ~"/\n");
|
|
|
|
})
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn stdin_works() {
|
|
|
|
let io = ~[CreatePipe(true, false),
|
|
|
|
CreatePipe(false, true)];
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
2014-01-15 14:39:08 -05:00
|
|
|
args: &[~"-c", ~"read line; echo $line"],
|
2013-12-26 18:28:24 -08:00
|
|
|
io: io,
|
2014-02-06 22:38:05 -08:00
|
|
|
.. ProcessConfig::new()
|
2013-12-26 18:28:24 -08:00
|
|
|
};
|
2014-01-30 14:10:53 -08:00
|
|
|
let mut p = Process::new(args).unwrap();
|
|
|
|
p.io[0].get_mut_ref().write("foobar".as_bytes()).unwrap();
|
2013-12-26 18:28:24 -08:00
|
|
|
p.io[0] = None; // close stdin;
|
|
|
|
let out = read_all(p.io[1].get_mut_ref() as &mut Reader);
|
|
|
|
assert!(p.wait().success());
|
|
|
|
assert_eq!(out, ~"foobar\n");
|
|
|
|
})
|
|
|
|
|
2014-02-06 22:38:05 -08:00
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn detach_works() {
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
|
|
|
args: &[~"-c", ~"true"],
|
|
|
|
detach: true,
|
|
|
|
.. ProcessConfig::new()
|
|
|
|
};
|
|
|
|
let mut p = Process::new(args).unwrap();
|
|
|
|
assert!(p.wait().success());
|
|
|
|
})
|
|
|
|
|
|
|
|
#[cfg(windows)]
|
|
|
|
iotest!(fn uid_fails_on_windows() {
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "test",
|
|
|
|
uid: Some(10),
|
|
|
|
.. ProcessConfig::new()
|
|
|
|
};
|
|
|
|
assert!(Process::new(args).is_err());
|
|
|
|
})
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn uid_works() {
|
|
|
|
use libc;
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/sh",
|
|
|
|
args: &[~"-c", ~"true"],
|
|
|
|
uid: Some(unsafe { libc::getuid() as uint }),
|
|
|
|
gid: Some(unsafe { libc::getgid() as uint }),
|
|
|
|
.. ProcessConfig::new()
|
|
|
|
};
|
|
|
|
let mut p = Process::new(args).unwrap();
|
|
|
|
assert!(p.wait().success());
|
|
|
|
})
|
|
|
|
|
|
|
|
// FIXME(#10380)
|
|
|
|
#[cfg(unix, not(target_os="android"))]
|
|
|
|
iotest!(fn uid_to_root_fails() {
|
|
|
|
use libc;
|
|
|
|
|
|
|
|
// if we're already root, this isn't a valid test. Most of the bots run
|
|
|
|
// as non-root though (android is an exception).
|
|
|
|
if unsafe { libc::getuid() == 0 } { return }
|
|
|
|
let args = ProcessConfig {
|
|
|
|
program: "/bin/ls",
|
|
|
|
uid: Some(0),
|
|
|
|
gid: Some(0),
|
|
|
|
.. ProcessConfig::new()
|
|
|
|
};
|
|
|
|
assert!(Process::new(args).is_err());
|
|
|
|
})
|
2013-12-26 18:28:24 -08:00
|
|
|
}
|