mirror of
https://github.com/aljazceru/goose.git
synced 2026-01-28 10:44:30 +01:00
113 lines
4.0 KiB
Rust
113 lines
4.0 KiB
Rust
use anyhow::{Context, Result};
|
|
use etcetera::{choose_app_strategy, AppStrategy};
|
|
use std::fs;
|
|
use std::path::PathBuf;
|
|
use tracing_appender::rolling::Rotation;
|
|
use tracing_subscriber::{
|
|
filter::LevelFilter, fmt, layer::SubscriberExt, util::SubscriberInitExt, EnvFilter, Layer,
|
|
Registry,
|
|
};
|
|
|
|
use goose::config::APP_STRATEGY;
|
|
use goose::tracing::langfuse_layer;
|
|
|
|
/// Returns the directory where log files should be stored.
|
|
/// Creates the directory structure if it doesn't exist.
|
|
fn get_log_directory() -> Result<PathBuf> {
|
|
// choose_app_strategy().state_dir()
|
|
// - macOS/Linux: ~/.local/state/goose/logs/server
|
|
// - Windows: ~\AppData\Roaming\Block\goose\data\logs\server
|
|
// - Windows has no convention for state_dir, use data_dir instead
|
|
let home_dir =
|
|
choose_app_strategy(APP_STRATEGY.clone()).context("HOME environment variable not set")?;
|
|
|
|
let base_log_dir = home_dir
|
|
.in_state_dir("logs/server")
|
|
.unwrap_or_else(|| home_dir.in_data_dir("logs/server"));
|
|
|
|
// Create date-based subdirectory
|
|
let now = chrono::Local::now();
|
|
let date_dir = base_log_dir.join(now.format("%Y-%m-%d").to_string());
|
|
|
|
// Ensure log directory exists
|
|
fs::create_dir_all(&date_dir).context("Failed to create log directory")?;
|
|
|
|
Ok(date_dir)
|
|
}
|
|
|
|
/// Sets up the logging infrastructure for the application.
|
|
/// This includes:
|
|
/// - File-based logging with JSON formatting (DEBUG level)
|
|
/// - Console output for development (INFO level)
|
|
/// - Optional Langfuse integration (DEBUG level)
|
|
pub fn setup_logging(name: Option<&str>) -> Result<()> {
|
|
// Set up file appender for goose module logs
|
|
let log_dir = get_log_directory()?;
|
|
let timestamp = chrono::Local::now().format("%Y%m%d_%H%M%S").to_string();
|
|
|
|
// Create log file name by prefixing with timestamp
|
|
let log_filename = if name.is_some() {
|
|
format!("{}-{}.log", timestamp, name.unwrap())
|
|
} else {
|
|
format!("{}.log", timestamp)
|
|
};
|
|
|
|
// Create non-rolling file appender for detailed logs
|
|
let file_appender =
|
|
tracing_appender::rolling::RollingFileAppender::new(Rotation::NEVER, log_dir, log_filename);
|
|
|
|
// Create JSON file logging layer
|
|
let file_layer = fmt::layer()
|
|
.with_target(true)
|
|
.with_level(true)
|
|
.with_writer(file_appender)
|
|
.with_ansi(false)
|
|
.with_file(true);
|
|
|
|
// Create console logging layer for development - INFO and above only
|
|
let console_layer = fmt::layer()
|
|
.with_target(true)
|
|
.with_level(true)
|
|
.with_ansi(true)
|
|
.with_file(true)
|
|
.with_line_number(true)
|
|
.pretty();
|
|
|
|
// Base filter for all logging
|
|
let env_filter = EnvFilter::try_from_default_env().unwrap_or_else(|_| {
|
|
// Set default levels for different modules
|
|
EnvFilter::new("")
|
|
// Set mcp-server module to DEBUG
|
|
.add_directive("mcp_server=debug".parse().unwrap())
|
|
// Set mcp-client to DEBUG
|
|
.add_directive("mcp_client=debug".parse().unwrap())
|
|
// Set goose module to DEBUG
|
|
.add_directive("goose=debug".parse().unwrap())
|
|
// Set goose-server to INFO
|
|
.add_directive("goose_server=info".parse().unwrap())
|
|
// Set tower-http to INFO for request logging
|
|
.add_directive("tower_http=info".parse().unwrap())
|
|
// Set everything else to WARN
|
|
.add_directive(LevelFilter::WARN.into())
|
|
});
|
|
|
|
// Build the subscriber with required layers
|
|
let subscriber = Registry::default()
|
|
.with(file_layer.with_filter(env_filter))
|
|
.with(console_layer.with_filter(LevelFilter::INFO));
|
|
|
|
// Initialize with Langfuse if available
|
|
if let Some(langfuse) = langfuse_layer::create_langfuse_observer() {
|
|
subscriber
|
|
.with(langfuse.with_filter(LevelFilter::DEBUG))
|
|
.try_init()
|
|
.context("Failed to set global subscriber")?;
|
|
} else {
|
|
subscriber
|
|
.try_init()
|
|
.context("Failed to set global subscriber")?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|