1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
//! Implementation of process group extensions for the //! [standard library `Command` type](std::process::Command). use std::{ io::Result, process::{ExitStatus, Output}, }; use crate::GroupChild; #[cfg(target_family = "windows")] mod windows; #[cfg(target_family = "unix")] mod unix; /// Extensions for [`Command`](std::process::Command) adding support for process groups. pub trait CommandGroup { /// Executes the command as a child process group, returning a handle to it. /// /// By default, stdin, stdout and stderr are inherited from the parent. /// /// On Windows, this creates a job object instead of a POSIX process group. /// /// # Examples /// /// Basic usage: /// /// ```no_run /// use std::process::Command; /// use command_group::CommandGroup; /// /// Command::new("ls") /// .group_spawn() /// .expect("ls command failed to start"); /// ``` fn group_spawn(&mut self) -> Result<GroupChild>; /// Executes the command as a child process group, waiting for it to finish and /// collecting all of its output. /// /// By default, stdout and stderr are captured (and used to provide the /// resulting output). Stdin is not inherited from the parent and any /// attempt by the child process to read from the stdin stream will result /// in the stream immediately closing. /// /// On Windows, this creates a job object instead of a POSIX process group. /// /// # Examples /// /// ```should_panic /// use std::process::Command; /// use std::io::{self, Write}; /// use command_group::CommandGroup; /// /// let output = Command::new("/bin/cat") /// .arg("file.txt") /// .group_output() /// .expect("failed to execute process"); /// /// println!("status: {}", output.status); /// io::stdout().write_all(&output.stdout).unwrap(); /// io::stderr().write_all(&output.stderr).unwrap(); /// /// assert!(output.status.success()); /// ``` fn group_output(&mut self) -> Result<Output> { self.group_spawn() .and_then(|child| child.wait_with_output()) } /// Executes a command as a child process group, waiting for it to finish and /// collecting its status. /// /// By default, stdin, stdout and stderr are inherited from the parent. /// /// On Windows, this creates a job object instead of a POSIX process group. /// /// # Examples /// /// ```should_panic /// use std::process::Command; /// use command_group::CommandGroup; /// /// let status = Command::new("/bin/cat") /// .arg("file.txt") /// .group_status() /// .expect("failed to execute process"); /// /// println!("process finished with: {}", status); /// /// assert!(status.success()); /// ``` fn group_status(&mut self) -> Result<ExitStatus> { self.group_spawn().and_then(|mut child| child.wait()) } }