mirror of
https://gitlab.com/fabinfra/fabaccess/bffh.git
synced 2024-11-25 08:07:57 +01:00
228 lines
7.5 KiB
Rust
228 lines
7.5 KiB
Rust
use super::Initiator;
|
|
use super::InitiatorCallbacks;
|
|
use crate::resources::state::State;
|
|
use crate::utils::linebuffer::LineBuffer;
|
|
use async_process::{Child, ChildStderr, ChildStdout, Command, Stdio};
|
|
use futures_lite::{ready, AsyncRead};
|
|
use miette::{miette, IntoDiagnostic};
|
|
use serde::{Deserialize, Serialize};
|
|
use std::collections::HashMap;
|
|
use std::future::Future;
|
|
use std::io;
|
|
use std::pin::Pin;
|
|
use std::task::{Context, Poll};
|
|
use crate::resources::modules::fabaccess::Status;
|
|
|
|
#[derive(Debug, Serialize, Deserialize)]
|
|
pub enum InputMessage {
|
|
#[serde(rename = "state")]
|
|
SetState(Status),
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
pub struct OutputLine {}
|
|
|
|
pub struct Process {
|
|
pub cmd: String,
|
|
pub args: Vec<String>,
|
|
state: Option<ProcessState>,
|
|
buffer: LineBuffer,
|
|
err_buffer: LineBuffer,
|
|
callbacks: InitiatorCallbacks,
|
|
}
|
|
|
|
impl Process {
|
|
fn spawn(&mut self) -> io::Result<()> {
|
|
let mut child = Command::new(&self.cmd)
|
|
.args(&self.args)
|
|
.stdin(Stdio::null())
|
|
.stdout(Stdio::piped())
|
|
.stderr(Stdio::piped())
|
|
.spawn()?;
|
|
self.state = Some(ProcessState::new(
|
|
child
|
|
.stdout
|
|
.take()
|
|
.expect("Child just spawned with piped stdout has no stdout"),
|
|
child
|
|
.stderr
|
|
.take()
|
|
.expect("Child just spawned with piped stderr has no stderr"),
|
|
child,
|
|
));
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
struct ProcessState {
|
|
pub stdout: ChildStdout,
|
|
pub stderr: ChildStderr,
|
|
pub stderr_closed: bool,
|
|
pub child: Child,
|
|
}
|
|
|
|
impl ProcessState {
|
|
pub fn new(stdout: ChildStdout, stderr: ChildStderr, child: Child) -> Self {
|
|
Self { stdout, stderr, stderr_closed: false, child }
|
|
}
|
|
|
|
fn try_process(&mut self, buffer: &[u8], callbacks: &mut InitiatorCallbacks) -> usize {
|
|
tracing::trace!("trying to process current buffer");
|
|
|
|
let mut end = 0;
|
|
|
|
while let Some(idx) = buffer[end..].iter().position(|b| *b == b'\n') {
|
|
if idx == 0 {
|
|
end += 1;
|
|
continue;
|
|
}
|
|
let line = &buffer[end..(end + idx)];
|
|
self.process_line(line, callbacks);
|
|
end = idx;
|
|
}
|
|
|
|
end
|
|
}
|
|
|
|
fn process_line(&mut self, line: &[u8], callbacks: &mut InitiatorCallbacks) {
|
|
if !line.is_empty() {
|
|
let res = std::str::from_utf8(line);
|
|
if let Err(error) = &res {
|
|
tracing::warn!(%error, "Initiator sent line with invalid UTF-8");
|
|
return;
|
|
}
|
|
let string = res.unwrap().trim();
|
|
// Ignore whitespace-only lines
|
|
if !string.is_empty() {
|
|
match serde_json::from_str::<InputMessage>(res.unwrap()) {
|
|
Ok(state) => {
|
|
tracing::trace!(?state, "got new state for process initiator");
|
|
let InputMessage::SetState(status) = state;
|
|
callbacks.set_status(status);
|
|
}
|
|
Err(error) => tracing::warn!(%error, "process initiator did not send a valid line"),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Future for Process {
|
|
type Output = ();
|
|
|
|
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
|
if let Process {
|
|
state: Some(state),
|
|
buffer,
|
|
err_buffer,
|
|
callbacks,
|
|
..
|
|
} = self.get_mut()
|
|
{
|
|
match state.child.try_status() {
|
|
Err(error) => {
|
|
tracing::error!(%error, "checking child exit code returned an error");
|
|
return Poll::Ready(());
|
|
}
|
|
Ok(Some(exitcode)) => {
|
|
tracing::warn!(%exitcode, "child process exited");
|
|
return Poll::Ready(());
|
|
}
|
|
Ok(None) => {
|
|
tracing::trace!("process initiator checking on process");
|
|
|
|
let stdout = &mut state.stdout;
|
|
|
|
loop {
|
|
let buf = buffer.get_mut_write(512);
|
|
match AsyncRead::poll_read(Pin::new(stdout), cx, buf) {
|
|
Poll::Pending => break,
|
|
Poll::Ready(Ok(read)) => {
|
|
buffer.advance_valid(read);
|
|
continue;
|
|
}
|
|
Poll::Ready(Err(error)) => {
|
|
tracing::warn!(%error, "reading from child stdout errored");
|
|
return Poll::Ready(());
|
|
}
|
|
}
|
|
}
|
|
|
|
let processed = state.try_process(buffer, callbacks);
|
|
buffer.consume(processed);
|
|
|
|
if !state.stderr_closed {
|
|
let stderr = &mut state.stderr;
|
|
loop {
|
|
let buf = err_buffer.get_mut_write(512);
|
|
match AsyncRead::poll_read(Pin::new(stderr), cx, buf) {
|
|
Poll::Pending => break,
|
|
Poll::Ready(Ok(read)) => {
|
|
err_buffer.advance_valid(read);
|
|
continue;
|
|
}
|
|
Poll::Ready(Err(error)) => {
|
|
tracing::warn!(%error, "reading from child stderr errored");
|
|
state.stderr_closed = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
{
|
|
let mut consumed = 0;
|
|
|
|
while let Some(idx) = buffer[consumed..].iter().position(|b| *b == b'\n') {
|
|
if idx == 0 {
|
|
consumed += 1;
|
|
continue;
|
|
}
|
|
let line = &buffer[consumed..(consumed + idx)];
|
|
match std::str::from_utf8(line) {
|
|
Ok(line) => tracing::debug!(line, "initiator STDERR"),
|
|
Err(error) => tracing::debug!(%error,
|
|
"invalid UTF-8 on initiator STDERR"),
|
|
}
|
|
consumed = idx;
|
|
}
|
|
err_buffer.consume(consumed);
|
|
}
|
|
|
|
return Poll::Pending;
|
|
}
|
|
}
|
|
} else {
|
|
tracing::warn!("process initiator has no process attached!");
|
|
}
|
|
|
|
Poll::Ready(())
|
|
}
|
|
}
|
|
|
|
impl Initiator for Process {
|
|
fn new(params: &HashMap<String, String>, callbacks: InitiatorCallbacks) -> miette::Result<Self>
|
|
where
|
|
Self: Sized,
|
|
{
|
|
let cmd = params
|
|
.get("cmd")
|
|
.ok_or(miette!("Process initiator requires a `cmd` parameter."))?
|
|
.clone();
|
|
let args = params
|
|
.get("args")
|
|
.map(|argv| argv.split_whitespace().map(|s| s.to_string()).collect())
|
|
.unwrap_or_else(Vec::new);
|
|
let mut this = Self {
|
|
cmd,
|
|
args,
|
|
state: None,
|
|
buffer: LineBuffer::new(),
|
|
err_buffer: LineBuffer::new(),
|
|
callbacks,
|
|
};
|
|
this.spawn().into_diagnostic()?;
|
|
Ok(this)
|
|
}
|
|
}
|