|
mod app; |
|
mod chat; |
|
mod common; |
|
|
|
use app::{ |
|
config::handle_config_update, |
|
constant::{ |
|
PKG_VERSION, ROUTE_ABOUT_PATH, ROUTE_API_PATH, ROUTE_BASIC_CALIBRATION_PATH, |
|
ROUTE_BUILD_KEY_PATH, ROUTE_CONFIG_PATH, ROUTE_ENV_EXAMPLE_PATH, ROUTE_GET_CHECKSUM, |
|
ROUTE_GET_HASH, ROUTE_GET_TIMESTAMP_HEADER, ROUTE_HEALTH_PATH, ROUTE_LOGS_PATH, |
|
ROUTE_README_PATH, ROUTE_ROOT_PATH, ROUTE_STATIC_PATH, ROUTE_TOKENS_ADD_PATH, |
|
ROUTE_TOKENS_DELETE_PATH, ROUTE_TOKENS_GET_PATH, ROUTE_TOKENS_PATH, |
|
ROUTE_TOKENS_RELOAD_PATH, ROUTE_TOKENS_UPDATE_PATH, ROUTE_USER_INFO_PATH, |
|
}, |
|
lazy::{AUTH_TOKEN, ROUTE_CHAT_PATH, ROUTE_MODELS_PATH}, |
|
model::*, |
|
}; |
|
use axum::{ |
|
routing::{get, post}, |
|
Router, |
|
}; |
|
use chat::{ |
|
route::{ |
|
handle_about, handle_add_tokens, handle_api_page, handle_basic_calibration, |
|
handle_build_key, handle_build_key_page, handle_config_page, handle_delete_tokens, |
|
handle_env_example, handle_get_checksum, handle_get_hash, handle_get_timestamp_header, |
|
handle_get_tokens, handle_health, handle_logs, handle_logs_post, handle_readme, |
|
handle_reload_tokens, handle_root, handle_static, handle_tokens_page, handle_update_tokens, |
|
handle_user_info, |
|
}, |
|
service::{handle_chat, handle_models}, |
|
}; |
|
use common::utils::{load_tokens, parse_string_from_env, parse_usize_from_env}; |
|
use std::sync::Arc; |
|
use tokio::signal; |
|
use tokio::sync::Mutex; |
|
use tower_http::{cors::CorsLayer, limit::RequestBodyLimitLayer}; |
|
|
|
#[tokio::main] |
|
async fn main() { |
|
|
|
std::panic::set_hook(Box::new(|info| { |
|
|
|
if let Some(msg) = info.payload().downcast_ref::<String>() { |
|
eprintln!("{}", msg); |
|
} else if let Some(msg) = info.payload().downcast_ref::<&str>() { |
|
eprintln!("{}", msg); |
|
} |
|
})); |
|
|
|
|
|
dotenvy::dotenv().ok(); |
|
|
|
if AUTH_TOKEN.is_empty() { |
|
panic!("AUTH_TOKEN must be set") |
|
}; |
|
|
|
|
|
AppConfig::init(); |
|
|
|
|
|
let token_infos = load_tokens(); |
|
|
|
|
|
let state = Arc::new(Mutex::new(AppState::new(token_infos))); |
|
|
|
|
|
if let Err(e) = AppConfig::load_saved_config() { |
|
eprintln!("加载保存的配置失败: {}", e); |
|
} |
|
|
|
|
|
let state_for_reload = state.clone(); |
|
|
|
|
|
tokio::spawn(async move { |
|
loop { |
|
|
|
let now = std::time::SystemTime::now() |
|
.duration_since(std::time::UNIX_EPOCH) |
|
.unwrap() |
|
.as_secs(); |
|
|
|
|
|
let next_reload = (now / 1000 + 1) * 1000; |
|
let wait_duration = next_reload - now; |
|
|
|
|
|
tokio::time::sleep(std::time::Duration::from_secs(wait_duration)).await; |
|
|
|
let mut app_state = state_for_reload.lock().await; |
|
app_state.update_checksum(); |
|
|
|
} |
|
}); |
|
|
|
|
|
let state_for_shutdown = state.clone(); |
|
|
|
|
|
let shutdown_signal = async move { |
|
let ctrl_c = async { |
|
signal::ctrl_c() |
|
.await |
|
.expect("failed to install Ctrl+C handler"); |
|
}; |
|
|
|
#[cfg(unix)] |
|
let terminate = async { |
|
signal::unix::signal(signal::unix::SignalKind::terminate()) |
|
.expect("failed to install signal handler") |
|
.recv() |
|
.await; |
|
}; |
|
|
|
#[cfg(not(unix))] |
|
let terminate = std::future::pending::<()>(); |
|
|
|
tokio::select! { |
|
_ = ctrl_c => {}, |
|
_ = terminate => {}, |
|
} |
|
|
|
println!("正在关闭服务器..."); |
|
|
|
|
|
if let Err(e) = AppConfig::save_config() { |
|
eprintln!("保存配置失败: {}", e); |
|
} else { |
|
println!("配置已保存"); |
|
} |
|
|
|
|
|
let state = state_for_shutdown.lock().await; |
|
if let Err(e) = state.save_logs().await { |
|
eprintln!("保存日志失败: {}", e); |
|
} else { |
|
println!("日志已保存"); |
|
} |
|
}; |
|
|
|
|
|
let app = Router::new() |
|
.route(ROUTE_ROOT_PATH, get(handle_root)) |
|
.route(ROUTE_HEALTH_PATH, get(handle_health)) |
|
.route(ROUTE_TOKENS_PATH, get(handle_tokens_page)) |
|
.route(ROUTE_MODELS_PATH.as_str(), get(handle_models)) |
|
.route(ROUTE_TOKENS_GET_PATH, post(handle_get_tokens)) |
|
.route(ROUTE_TOKENS_RELOAD_PATH, post(handle_reload_tokens)) |
|
.route(ROUTE_TOKENS_UPDATE_PATH, post(handle_update_tokens)) |
|
.route(ROUTE_TOKENS_ADD_PATH, post(handle_add_tokens)) |
|
.route(ROUTE_TOKENS_DELETE_PATH, post(handle_delete_tokens)) |
|
.route(ROUTE_CHAT_PATH.as_str(), post(handle_chat)) |
|
.route(ROUTE_LOGS_PATH, get(handle_logs)) |
|
.route(ROUTE_LOGS_PATH, post(handle_logs_post)) |
|
.route(ROUTE_ENV_EXAMPLE_PATH, get(handle_env_example)) |
|
.route(ROUTE_CONFIG_PATH, get(handle_config_page)) |
|
.route(ROUTE_CONFIG_PATH, post(handle_config_update)) |
|
.route(ROUTE_STATIC_PATH, get(handle_static)) |
|
.route(ROUTE_ABOUT_PATH, get(handle_about)) |
|
.route(ROUTE_README_PATH, get(handle_readme)) |
|
.route(ROUTE_API_PATH, get(handle_api_page)) |
|
.route(ROUTE_GET_HASH, get(handle_get_hash)) |
|
.route(ROUTE_GET_CHECKSUM, get(handle_get_checksum)) |
|
.route(ROUTE_GET_TIMESTAMP_HEADER, get(handle_get_timestamp_header)) |
|
.route(ROUTE_BASIC_CALIBRATION_PATH, post(handle_basic_calibration)) |
|
.route(ROUTE_USER_INFO_PATH, post(handle_user_info)) |
|
.route(ROUTE_BUILD_KEY_PATH, get(handle_build_key_page)) |
|
.route(ROUTE_BUILD_KEY_PATH, post(handle_build_key)) |
|
.layer(RequestBodyLimitLayer::new( |
|
1024 * 1024 * parse_usize_from_env("REQUEST_BODY_LIMIT_MB", 2), |
|
)) |
|
.layer(CorsLayer::permissive()) |
|
.with_state(state); |
|
|
|
|
|
let port = parse_string_from_env("PORT", "3000"); |
|
let addr = format!("0.0.0.0:{}", port); |
|
println!("服务器运行在端口 {}", port); |
|
println!("当前版本: v{}", PKG_VERSION); |
|
|
|
|
|
|
|
|
|
let listener = tokio::net::TcpListener::bind(addr).await.unwrap(); |
|
let server = axum::serve(listener, app); |
|
tokio::select! { |
|
result = server => { |
|
if let Err(e) = result { |
|
eprintln!("服务器错误: {}", e); |
|
} |
|
} |
|
_ = shutdown_signal => { |
|
println!("服务器已关闭"); |
|
} |
|
} |
|
} |
|
|