use std::path::PathBuf; use std::sync::Arc; use codeg_lib::app_state::AppState; use codeg_lib::web::event_bridge::{EventEmitter, WebEventBroadcaster}; use codeg_lib::web::{ find_static_dir_standalone, generate_random_token, get_local_addresses, WebServerState, }; #[tokio::main] async fn main() { codeg_lib::process::ensure_user_npm_prefix_in_path(); let port: u16 = std::env::var("CODEG_PORT") .ok() .and_then(|v| v.parse().ok()) .unwrap_or(3080); let host = std::env::var("CODEG_HOST").unwrap_or_else(|_| "0.0.0.0".to_string()); let token = std::env::var("CODEG_TOKEN").unwrap_or_else(|_| generate_random_token()); let data_dir = std::env::var("CODEG_DATA_DIR") .map(PathBuf::from) .unwrap_or_else(|_| default_data_dir()); let static_dir_env = std::env::var("CODEG_STATIC_DIR").ok(); let static_dir = find_static_dir_standalone(static_dir_env.as_deref()); let app_version = env!("CARGO_PKG_VERSION"); eprintln!("[SERVER] codeg-server v{}", app_version); eprintln!("[SERVER] Data directory: {}", data_dir.display()); eprintln!("[SERVER] Static directory: {}", static_dir.display()); // Initialize database let db = codeg_lib::db::init_database(&data_dir, app_version) .await .expect("Failed to initialize database"); // Create shared broadcaster let broadcaster = Arc::new(WebEventBroadcaster::new()); let emitter = EventEmitter::WebOnly(broadcaster.clone()); // Build AppState let state = Arc::new(AppState { db, connection_manager: codeg_lib::app_state::default_connection_manager(), terminal_manager: codeg_lib::app_state::default_terminal_manager(), event_broadcaster: broadcaster, emitter, data_dir, web_server_state: WebServerState::new(), chat_channel_manager: codeg_lib::app_state::default_chat_channel_manager(), }); // Start chat channel background tasks (event subscriber, command dispatcher, scheduler, auto-connect) state .chat_channel_manager .start_background( state.event_broadcaster.clone(), state.db.conn.clone(), state.connection_manager.clone_ref(), state.emitter.clone(), ) .await; // Build router let router = codeg_lib::web::router::build_router(state, token.clone(), static_dir); // Bind let addr = format!("{}:{}", host, port); let listener = tokio::net::TcpListener::bind(&addr) .await .unwrap_or_else(|e| { eprintln!("[SERVER] Failed to bind {}: {}", addr, e); std::process::exit(1); }); let actual_port = listener.local_addr().map(|a| a.port()).unwrap_or(port); let addresses = get_local_addresses(actual_port); eprintln!("[SERVER] Token: {}", token); eprintln!("[SERVER] Listening on:"); for addr in &addresses { eprintln!(" {}", addr); } // Start serving if let Err(e) = axum::serve(listener, router).await { eprintln!("[SERVER] Server error: {}", e); std::process::exit(1); } } fn default_data_dir() -> PathBuf { dirs::data_dir() .map(|d| d.join("codeg")) .unwrap_or_else(|| PathBuf::from(".codeg-data")) }