2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2024-11-01 10:09:57 +00:00
pict-rs/src/process.rs

344 lines
9.4 KiB
Rust
Raw Normal View History

2021-10-14 00:06:53 +00:00
use actix_web::web::Bytes;
use flume::r#async::RecvFut;
use std::{
ffi::OsStr,
future::Future,
pin::Pin,
2023-07-10 20:29:41 +00:00
process::{ExitStatus, Stdio},
task::{Context, Poll},
2023-08-05 17:41:06 +00:00
time::{Duration, Instant},
};
2021-09-14 01:22:42 +00:00
use tokio::{
io::{AsyncRead, AsyncWriteExt, ReadBuf},
2023-08-05 17:41:06 +00:00
process::{Child, ChildStdin, ChildStdout, Command},
task::JoinHandle,
2021-09-14 01:22:42 +00:00
};
use tracing::{Instrument, Span};
2023-09-05 02:58:57 +00:00
use crate::{error_code::ErrorCode, future::WithTimeout};
2023-09-02 01:50:10 +00:00
2023-07-22 21:47:59 +00:00
struct MetricsGuard {
start: Instant,
armed: bool,
command: String,
}
impl MetricsGuard {
fn guard(command: String) -> Self {
2023-07-23 02:11:28 +00:00
metrics::increment_counter!("pict-rs.process.start", "command" => command.clone());
2023-07-22 21:47:59 +00:00
Self {
start: Instant::now(),
armed: true,
command,
}
}
fn disarm(mut self) {
self.armed = false;
}
}
impl Drop for MetricsGuard {
fn drop(&mut self) {
metrics::histogram!(
"pict-rs.process.duration",
self.start.elapsed().as_secs_f64(),
"command" => self.command.clone(),
"completed" => (!self.armed).to_string(),
);
2023-07-23 02:11:28 +00:00
metrics::increment_counter!("pict-rs.process.end", "completed" => (!self.armed).to_string() , "command" => self.command.clone());
2023-07-22 21:47:59 +00:00
}
}
#[derive(Debug)]
2023-07-14 00:21:28 +00:00
struct StatusError(ExitStatus);
pub(crate) struct Process {
command: String,
2021-09-14 01:22:42 +00:00
child: Child,
2023-07-22 21:47:59 +00:00
guard: MetricsGuard,
2023-08-05 17:41:06 +00:00
timeout: Duration,
2022-04-07 02:40:49 +00:00
}
impl std::fmt::Debug for Process {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Process").field("child", &"Child").finish()
}
}
2021-10-20 23:58:32 +00:00
struct DropHandle {
inner: JoinHandle<()>,
}
2023-08-05 17:41:06 +00:00
pub(crate) struct ProcessRead<I> {
inner: I,
err_recv: RecvFut<'static, std::io::Error>,
2023-08-05 17:41:06 +00:00
err_closed: bool,
#[allow(dead_code)]
handle: DropHandle,
eof: bool,
sleep: Pin<Box<tokio::time::Sleep>>,
}
2023-07-10 20:29:41 +00:00
#[derive(Debug, thiserror::Error)]
pub(crate) enum ProcessError {
2023-07-17 02:51:14 +00:00
#[error("Required command {0} not found, make sure it exists in pict-rs' $PATH")]
2023-07-10 20:29:41 +00:00
NotFound(String),
2023-07-17 02:51:14 +00:00
#[error("Cannot run command {0} due to invalid permissions on binary, make sure the pict-rs user has permission to run it")]
PermissionDenied(String),
2023-07-10 20:29:41 +00:00
#[error("Reached process spawn limit")]
LimitReached,
2023-08-05 17:41:06 +00:00
#[error("{0} timed out")]
Timeout(String),
#[error("{0} Failed with {1}")]
Status(String, ExitStatus),
2023-07-10 20:29:41 +00:00
#[error("Unknown process error")]
Other(#[source] std::io::Error),
}
2023-09-02 01:50:10 +00:00
impl ProcessError {
pub(crate) const fn error_code(&self) -> ErrorCode {
match self {
Self::NotFound(_) => ErrorCode::COMMAND_NOT_FOUND,
Self::PermissionDenied(_) => ErrorCode::COMMAND_PERMISSION_DENIED,
Self::LimitReached | Self::Other(_) => ErrorCode::COMMAND_ERROR,
Self::Timeout(_) => ErrorCode::COMMAND_TIMEOUT,
Self::Status(_, _) => ErrorCode::COMMAND_FAILURE,
}
}
}
impl Process {
pub(crate) fn run<T>(
command: &str,
args: &[T],
envs: &[(&str, &OsStr)],
timeout: u64,
) -> Result<Self, ProcessError>
where
T: AsRef<OsStr>,
{
let res = tracing::trace_span!(parent: None, "Create command", %command).in_scope(|| {
Self::spawn(
command,
2023-11-10 00:26:57 +00:00
Command::new(command).args(args).envs(envs.iter().copied()),
timeout,
)
});
2023-07-10 20:29:41 +00:00
match res {
Ok(this) => Ok(this),
Err(e) => match e.kind() {
std::io::ErrorKind::NotFound => Err(ProcessError::NotFound(command.to_string())),
2023-07-17 02:51:14 +00:00
std::io::ErrorKind::PermissionDenied => {
Err(ProcessError::PermissionDenied(command.to_string()))
}
2023-07-10 20:29:41 +00:00
std::io::ErrorKind::WouldBlock => Err(ProcessError::LimitReached),
_ => Err(ProcessError::Other(e)),
},
}
2021-09-14 01:22:42 +00:00
}
2023-08-05 17:41:06 +00:00
fn spawn(command: &str, cmd: &mut Command, timeout: u64) -> std::io::Result<Self> {
tracing::trace_span!(parent: None, "Spawn command", %command).in_scope(|| {
2023-07-22 21:47:59 +00:00
let guard = MetricsGuard::guard(command.into());
2023-07-10 20:29:41 +00:00
let cmd = cmd
.stdin(Stdio::piped())
.stdout(Stdio::piped())
.kill_on_drop(true);
cmd.spawn().map(|child| Process {
child,
command: String::from(command),
2023-07-22 21:47:59 +00:00
guard,
2023-08-05 17:41:06 +00:00
timeout: Duration::from_secs(timeout),
})
2022-04-07 17:56:40 +00:00
})
}
#[tracing::instrument(skip(self))]
2023-08-05 17:41:06 +00:00
pub(crate) async fn wait(self) -> Result<(), ProcessError> {
let Process {
command,
mut child,
guard,
timeout,
} = self;
2023-09-05 02:58:57 +00:00
let res = child.wait().with_timeout(timeout).await;
2023-07-10 20:29:41 +00:00
match res {
2023-08-05 17:41:06 +00:00
Ok(Ok(status)) if status.success() => {
guard.disarm();
2023-07-22 21:47:59 +00:00
Ok(())
}
2023-08-05 17:41:06 +00:00
Ok(Ok(status)) => Err(ProcessError::Status(command, status)),
Ok(Err(e)) => Err(ProcessError::Other(e)),
Err(_) => {
child.kill().await.map_err(ProcessError::Other)?;
Err(ProcessError::Timeout(command))
}
2021-10-23 19:14:12 +00:00
}
}
2023-08-05 17:41:06 +00:00
pub(crate) fn bytes_read(self, input: Bytes) -> ProcessRead<ChildStdout> {
self.spawn_fn(move |mut stdin| {
2022-09-25 22:35:52 +00:00
let mut input = input;
async move { stdin.write_all_buf(&mut input).await }
})
}
2023-08-05 17:41:06 +00:00
pub(crate) fn read(self) -> ProcessRead<ChildStdout> {
self.spawn_fn(|_| async { Ok(()) })
}
#[allow(unknown_lints)]
#[allow(clippy::let_with_type_underscore)]
#[tracing::instrument(level = "trace", skip_all)]
2023-08-05 17:41:06 +00:00
fn spawn_fn<F, Fut>(self, f: F) -> ProcessRead<ChildStdout>
2022-09-25 22:35:52 +00:00
where
F: FnOnce(ChildStdin) -> Fut + 'static,
Fut: Future<Output = std::io::Result<()>>,
{
2023-08-05 17:41:06 +00:00
let Process {
command,
mut child,
guard,
timeout,
} = self;
2023-08-05 17:41:06 +00:00
let stdin = child.stdin.take().expect("stdin exists");
let stdout = child.stdout.take().expect("stdout exists");
let (tx, rx) = crate::sync::channel::<std::io::Error>(1);
let rx = rx.into_recv_async();
2023-08-05 17:41:06 +00:00
let span = tracing::info_span!(parent: None, "Background process task", %command);
span.follows_from(Span::current());
let handle = crate::sync::spawn(
"await-process",
async move {
let child_fut = async {
(f)(stdin).await?;
child.wait().await
};
2023-09-05 02:58:57 +00:00
let error = match child_fut.with_timeout(timeout).await {
Ok(Ok(status)) if status.success() => {
guard.disarm();
return;
}
Ok(Ok(status)) => {
std::io::Error::new(std::io::ErrorKind::Other, StatusError(status))
}
Ok(Err(e)) => e,
Err(_) => std::io::ErrorKind::TimedOut.into(),
};
let _ = tx.send(error);
let _ = child.kill().await;
}
.instrument(span),
);
let sleep = tokio::time::sleep(timeout);
2023-08-05 17:41:06 +00:00
2022-04-07 02:40:49 +00:00
ProcessRead {
inner: stdout,
err_recv: rx,
err_closed: false,
2021-10-20 23:58:32 +00:00
handle: DropHandle { inner: handle },
2023-07-14 00:21:28 +00:00
eof: false,
2023-08-05 17:41:06 +00:00
sleep: Box::pin(sleep),
2022-04-07 02:40:49 +00:00
}
}
}
impl<I> AsyncRead for ProcessRead<I>
where
2023-08-05 17:41:06 +00:00
I: AsyncRead + Unpin,
{
fn poll_read(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut ReadBuf<'_>,
) -> Poll<std::io::Result<()>> {
2023-08-05 17:41:06 +00:00
if !self.err_closed {
if let Poll::Ready(res) = Pin::new(&mut self.err_recv).poll(cx) {
self.err_closed = true;
2023-07-14 00:21:28 +00:00
2022-04-07 02:40:49 +00:00
if let Ok(err) = res {
return Poll::Ready(Err(err));
}
2023-07-14 00:21:28 +00:00
2023-08-05 17:41:06 +00:00
if self.eof {
2023-07-14 00:21:28 +00:00
return Poll::Ready(Ok(()));
}
}
2023-08-05 17:41:06 +00:00
if let Poll::Ready(()) = self.sleep.as_mut().poll(cx) {
self.err_closed = true;
return Poll::Ready(Err(std::io::ErrorKind::TimedOut.into()));
}
2022-04-07 02:40:49 +00:00
}
2023-08-05 17:41:06 +00:00
if !self.eof {
2023-07-14 00:21:28 +00:00
let before_size = buf.filled().len();
2023-08-05 17:41:06 +00:00
return match Pin::new(&mut self.inner).poll_read(cx, buf) {
2023-07-14 00:21:28 +00:00
Poll::Ready(Ok(())) => {
if buf.filled().len() == before_size {
2023-08-05 17:41:06 +00:00
self.eof = true;
2023-07-14 00:21:28 +00:00
2023-08-05 17:41:06 +00:00
if !self.err_closed {
2023-07-14 00:21:28 +00:00
// reached end of stream & haven't received process signal
return Poll::Pending;
}
}
Poll::Ready(Ok(()))
}
Poll::Ready(Err(e)) => {
2023-08-05 17:41:06 +00:00
self.eof = true;
2023-07-14 00:21:28 +00:00
Poll::Ready(Err(e))
}
Poll::Pending => Poll::Pending,
};
}
2023-08-05 17:41:06 +00:00
if self.err_closed && self.eof {
2023-07-14 00:21:28 +00:00
return Poll::Ready(Ok(()));
}
Poll::Pending
}
}
2021-10-20 23:58:32 +00:00
impl Drop for DropHandle {
fn drop(&mut self) {
2021-10-20 23:58:32 +00:00
self.inner.abort();
}
}
impl std::fmt::Display for StatusError {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
2023-07-14 00:21:28 +00:00
write!(f, "Command failed with bad status: {}", self.0)
}
}
impl std::error::Error for StatusError {}