pesde/src/scripts.rs

107 lines
3.4 KiB
Rust
Raw Normal View History

use crate::Project;
use std::{
ffi::OsStr,
2024-12-16 22:00:37 +00:00
fmt::{Debug, Display, Formatter},
2024-07-22 15:41:45 +01:00
path::Path,
process::Stdio,
};
use tokio::{
io::{AsyncBufReadExt, BufReader},
process::Command,
};
2024-12-16 22:00:37 +00:00
use tracing::instrument;
2024-08-03 21:18:38 +01:00
/// Script names used by pesde
2024-07-24 10:55:15 +01:00
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum ScriptName {
2024-08-03 21:18:38 +01:00
/// Generates a config for syncing tools for Roblox. For example, for Rojo it should create a `default.project.json` file
2024-07-24 10:55:15 +01:00
RobloxSyncConfigGenerator,
2024-08-03 21:18:38 +01:00
/// Prints a sourcemap for a Wally package, used for finding the library export file
2024-07-24 10:55:15 +01:00
#[cfg(feature = "wally-compat")]
SourcemapGenerator,
}
impl Display for ScriptName {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
match self {
ScriptName::RobloxSyncConfigGenerator => write!(f, "roblox_sync_config_generator"),
#[cfg(feature = "wally-compat")]
ScriptName::SourcemapGenerator => write!(f, "sourcemap_generator"),
}
}
}
2024-12-16 22:00:37 +00:00
#[instrument(skip(project), level = "debug")]
pub(crate) async fn execute_script<A: IntoIterator<Item = S> + Debug, S: AsRef<OsStr> + Debug>(
2024-08-11 16:27:51 +01:00
script_name: ScriptName,
2024-07-22 15:41:45 +01:00
script_path: &Path,
args: A,
project: &Project,
2024-07-22 15:41:45 +01:00
return_stdout: bool,
) -> Result<Option<String>, std::io::Error> {
match Command::new("lune")
.arg("run")
2024-07-22 15:41:45 +01:00
.arg(script_path.as_os_str())
2024-08-08 16:59:59 +01:00
.arg("--")
.args(args)
.current_dir(project.package_dir())
2024-08-11 15:16:25 +01:00
.stdin(Stdio::inherit())
.stdout(Stdio::piped())
.stderr(Stdio::piped())
.spawn()
{
Ok(mut child) => {
let mut stdout = BufReader::new(child.stdout.take().unwrap()).lines();
let mut stderr = BufReader::new(child.stderr.take().unwrap()).lines();
2024-08-11 16:27:51 +01:00
let script = script_name.to_string();
let script_2 = script.to_string();
tokio::spawn(async move {
while let Some(line) = stderr.next_line().await.transpose() {
match line {
Ok(line) => {
2024-12-16 22:00:37 +00:00
tracing::error!("[{script}]: {line}");
}
Err(e) => {
2024-12-16 22:00:37 +00:00
tracing::error!("ERROR IN READING STDERR OF {script}: {e}");
break;
}
}
}
});
2024-07-22 15:41:45 +01:00
let mut stdout_str = String::new();
while let Some(line) = stdout.next_line().await.transpose() {
match line {
Ok(line) => {
2024-07-22 15:41:45 +01:00
if return_stdout {
stdout_str.push_str(&line);
stdout_str.push('\n');
2024-08-08 16:59:59 +01:00
} else {
2024-12-16 22:00:37 +00:00
tracing::info!("[{script_2}]: {line}");
2024-07-22 15:41:45 +01:00
}
}
Err(e) => {
2024-12-16 22:00:37 +00:00
tracing::error!("ERROR IN READING STDOUT OF {script_2}: {e}");
break;
}
}
}
2024-07-22 15:41:45 +01:00
if return_stdout {
Ok(Some(stdout_str))
} else {
Ok(None)
}
}
Err(e) if e.kind() == std::io::ErrorKind::NotFound => {
2024-12-16 22:00:37 +00:00
tracing::warn!("Lune could not be found in PATH: {e}");
2024-07-22 15:41:45 +01:00
Ok(None)
}
Err(e) => Err(e),
}
}