mirror of
https://github.com/kjuulh/dagger-rs.git
synced 2024-11-26 00:42:42 +01:00
110 lines
3.2 KiB
Rust
110 lines
3.2 KiB
Rust
use std::{fs::canonicalize, path::PathBuf, process::Stdio, sync::Arc};
|
|
|
|
use tokio::io::AsyncBufReadExt;
|
|
|
|
use crate::{config::Config, connect_params::ConnectParams};
|
|
|
|
#[derive(Clone, Debug)]
|
|
pub struct CliSession {
|
|
inner: Arc<InnerCliSession>,
|
|
}
|
|
|
|
impl CliSession {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
inner: Arc::new(InnerCliSession {}),
|
|
}
|
|
}
|
|
|
|
pub async fn connect(
|
|
&self,
|
|
config: &Config,
|
|
cli_path: &PathBuf,
|
|
) -> eyre::Result<(ConnectParams, tokio::process::Child)> {
|
|
self.inner.connect(config, cli_path).await
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct InnerCliSession {}
|
|
|
|
impl InnerCliSession {
|
|
pub async fn connect(
|
|
&self,
|
|
config: &Config,
|
|
cli_path: &PathBuf,
|
|
) -> eyre::Result<(ConnectParams, tokio::process::Child)> {
|
|
let proc = self.start(config, cli_path)?;
|
|
let params = self.get_conn(proc).await?;
|
|
Ok(params)
|
|
}
|
|
|
|
fn start(&self, config: &Config, cli_path: &PathBuf) -> eyre::Result<tokio::process::Child> {
|
|
let mut args: Vec<String> = vec!["session".into()];
|
|
if let Some(workspace) = &config.workdir_path {
|
|
let abs_path = canonicalize(workspace)?;
|
|
args.extend(["--workdir".into(), abs_path.to_string_lossy().to_string()])
|
|
}
|
|
if let Some(config_path) = &config.config_path {
|
|
let abs_path = canonicalize(config_path)?;
|
|
args.extend(["--project".into(), abs_path.to_string_lossy().to_string()])
|
|
}
|
|
|
|
let proc = tokio::process::Command::new(
|
|
cli_path
|
|
.to_str()
|
|
.ok_or(eyre::anyhow!("could not get string from path"))?,
|
|
)
|
|
.args(args.as_slice())
|
|
.stdin(Stdio::piped())
|
|
.stdout(Stdio::piped())
|
|
.stderr(Stdio::piped())
|
|
.spawn()?;
|
|
|
|
//TODO: Add retry mechanism
|
|
|
|
return Ok(proc);
|
|
}
|
|
|
|
async fn get_conn(
|
|
&self,
|
|
mut proc: tokio::process::Child,
|
|
) -> eyre::Result<(ConnectParams, tokio::process::Child)> {
|
|
let stdout = proc
|
|
.stdout
|
|
.take()
|
|
.ok_or(eyre::anyhow!("could not acquire stdout from child process"))?;
|
|
|
|
let stderr = proc
|
|
.stderr
|
|
.take()
|
|
.ok_or(eyre::anyhow!("could not acquire stderr from child process"))?;
|
|
|
|
let (sender, mut receiver) = tokio::sync::mpsc::channel(1);
|
|
|
|
tokio::spawn(async move {
|
|
let mut stdout_bufr = tokio::io::BufReader::new(stdout).lines();
|
|
while let Ok(Some(line)) = stdout_bufr.next_line().await {
|
|
if let Ok(conn) = serde_json::from_str::<ConnectParams>(&line) {
|
|
sender.send(conn).await.unwrap();
|
|
}
|
|
|
|
println!("dagger: {}", line);
|
|
}
|
|
});
|
|
|
|
tokio::spawn(async move {
|
|
let mut stdout_bufr = tokio::io::BufReader::new(stderr).lines();
|
|
while let Ok(Some(line)) = stdout_bufr.next_line().await {
|
|
println!("dagger: {}", line);
|
|
}
|
|
});
|
|
|
|
let conn = receiver.recv().await.ok_or(eyre::anyhow!(
|
|
"could not receive ok signal from dagger-engine"
|
|
))?;
|
|
|
|
Ok((conn, proc))
|
|
}
|
|
}
|