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
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#[cfg(doc)]
use crate::Producer;
use {
crate::{
io::{Reader, Writer},
ConsumeFailure, ConsumeFault, Consumer,
},
conventus::{AssembleFrom, DisassembleInto},
core::{cell::RefCell, fmt::Debug},
fehler::throws,
std::process::{Child, Command, ExitStatus, Stdio},
};
#[derive(Debug)]
pub struct Process<I: DisassembleInto<u8>, O: AssembleFrom<u8>, E: AssembleFrom<u8>> {
command_str: String,
child: RefCell<Child>,
input: Writer<I>,
output: Reader<O>,
error: Reader<E>,
}
impl<I: DisassembleInto<u8>, O: AssembleFrom<u8>, E: AssembleFrom<u8>> Process<I, O, E> {
#[allow(clippy::unwrap_in_result)]
#[inline]
#[throws(CreateProcessError)]
pub fn new(mut command: Command) -> Self {
let command_str = format!("{:?}", command);
let mut child = command
.stdin(Stdio::piped())
.stdout(Stdio::piped())
.stderr(Stdio::piped())
.spawn()
.map_err(|error| CreateProcessError {
command: command_str.clone(),
error,
})?;
#[allow(clippy::unwrap_used)]
Self {
input: Writer::new(format!("{} IN", command_str), child.stdin.take().unwrap()),
output: Reader::new(format!("{} OUT", command_str), child.stdout.take().unwrap()),
error: Reader::new(format!("{} ERR", command_str), child.stderr.take().unwrap()),
child: RefCell::new(child),
command_str,
}
}
#[inline]
pub fn input(&self) -> &Writer<I> {
&self.input
}
#[inline]
pub fn output(&self) -> &Reader<O> {
&self.output
}
#[inline]
pub fn error(&self) -> &Reader<E> {
&self.error
}
}
impl<I: DisassembleInto<u8>, O: AssembleFrom<u8>, E: AssembleFrom<u8>> Consumer
for Process<I, O, E>
{
type Good = ExitStatus;
type Failure = ConsumeFailure<WaitFault>;
#[inline]
#[throws(Self::Failure)]
fn consume(&self) -> Self::Good {
let status = self
.child
.borrow_mut()
.try_wait()
.map_err(|error| WaitFault {
command: self.command_str.clone(),
error,
})?
.ok_or(ConsumeFailure::EmptyStock)?;
self.input.cancel();
self.output.cancel();
self.error.cancel();
status
}
}
#[derive(Debug, thiserror::Error)]
#[error("Failed to create `{command}`: {error}")]
pub struct CreateProcessError {
command: String,
error: std::io::Error,
}
#[derive(Debug, ConsumeFault, thiserror::Error)]
#[error("Failed to wait for `{command}`: {error}")]
pub struct WaitFault {
command: String,
error: std::io::Error,
}