use std::path::PathBuf;
use std::time::Duration;
use clap::Parser;
use grite_daemon::supervisor::Supervisor;
use tracing::{error, info};
use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, EnvFilter};
#[derive(Parser)]
#[command(name = "grite-daemon", about = "Grite daemon", version)]
struct Cli {
#[arg(long)]
endpoint: Option<String>,
#[arg(long, short)]
daemon: bool,
#[arg(long)]
pid_file: Option<PathBuf>,
#[arg(long, default_value = "info")]
log_level: String,
#[arg(long, default_value = "0")]
idle_timeout: u64,
}
#[tokio::main]
async fn main() {
let cli = Cli::parse();
let filter = EnvFilter::try_from_default_env()
.or_else(|_| EnvFilter::try_new(&cli.log_level))
.unwrap_or_else(|_| EnvFilter::new("info"));
tracing_subscriber::registry()
.with(filter)
.with(tracing_subscriber::fmt::layer())
.init();
info!("grite-daemon starting");
if cli.daemon {
info!("Daemonizing...");
}
if let Some(ref pid_file) = cli.pid_file {
let pid = std::process::id();
if let Err(e) = std::fs::write(pid_file, pid.to_string()) {
error!("Failed to write PID file: {}", e);
}
}
let shutdown = grite_daemon::signals::shutdown_signal();
let idle_timeout = if cli.idle_timeout > 0 {
Some(Duration::from_secs(cli.idle_timeout))
} else {
None
};
let endpoint = cli
.endpoint
.unwrap_or_else(libgrite_ipc::default_socket_path);
let supervisor = Supervisor::new(endpoint, idle_timeout);
if let Err(e) = supervisor.run(shutdown).await {
error!("Supervisor error: {}", e);
}
if let Some(ref pid_file) = cli.pid_file {
let _ = std::fs::remove_file(pid_file);
}
info!("grite-daemon stopped");
}