我正在使用 Rusts 的Axum网络框架。
我需要有多个后备方案。
第一个后备方案是用于传送静态文件,例如/script.js
位于/style.css
我的static
文件夹中的文件。
第二个回退应该是如果任何路由不匹配,那么我需要提供主页(/
匹配时也会提供主页)。例如,对于渲染客户端渲染页面的虚假路由。
代码:
dotenv::dotenv().expect("Failed to load environment variables from .env file!");
let state = AppState {
pool: PgPoolOptions::new().max_connections(70).connect(&std::env::var("DATABASE_URL").expect("DATABASE_URL must be set.")).await.unwrap(),
};
let api_path = "/api";
tracing_subscriber::fmt()
.with_max_level(tracing::Level::DEBUG)
.init();
let app = axum::Router::new()
.route("/", get(home_get))
.route_with_tsr(format!("{api_path}/submit").as_str(), post(submit_post))
.route_with_tsr(format!("{api_path}/search").as_str(), post(search_post))
.fallback_service(ServeDir::new(Path::new(&std::env::var("STATIC_FILES").expect("STATIC_FILES must be set."))))
.fallback(get(home_get))
.route_layer(axum::middleware::from_fn_with_state(state.clone(),info_middleware))
.layer(RequestBodyLimitLayer::new(4096))
.layer(TraceLayer::new_for_http())
.with_state(state);
let listener = tokio::net::TcpListener::bind(":::8080").await.unwrap();
axum::serve(listener, app.into_make_service_with_connect_info::<SocketAddr>()).await.unwrap();
这不起作用。转到主页时,静态文件不会被发送,但500 Internal Error
会被发送。
跟踪日志:
2025-01-19T04:32:37.437061Z DEBUG request{method=GET uri=/ version=HTTP/1.1}: tower_http::trace::on_request: started processing request
2025-01-19T04:32:37.437244Z DEBUG request{method=GET uri=/ version=HTTP/1.1}: tower_http::trace::on_response: finished processing request latency=0 ms status=200
2025-01-19T04:32:37.478539Z DEBUG request{method=GET uri=/style.css?v=1737261157 version=HTTP/1.1}: tower_http::trace::on_request: started processing request
2025-01-19T04:32:37.480165Z DEBUG request{method=GET uri=/style.css?v=1737261157 version=HTTP/1.1}: tower_http::trace::on_response: finished processing request latency=1 ms status=500
2025-01-19T04:32:37.480295Z ERROR request{method=GET uri=/style.css?v=1737261157 version=HTTP/1.1}: tower_http::trace::on_failure: response failed classification=Status code: 500 Internal Server Error latency=1 ms
2025-01-19T04:32:37.482660Z DEBUG request{method=GET uri=/script.js?v=1737261157 version=HTTP/1.1}: tower_http::trace::on_request: started processing request
2025-01-19T04:32:37.487678Z DEBUG request{method=GET uri=/script.js?v=1737261157 version=HTTP/1.1}: tower_http::trace::on_response: finished processing request latency=5 ms status=500
2025-01-19T04:32:37.487877Z ERROR request{method=GET uri=/script.js?v=1737261157 version=HTTP/1.1}: tower_http::trace::on_failure: response failed classification=Status code: 500 Internal Server Error latency=5 ms
如果我注释掉第二个 fallback .fallback(get(home_get))
,那么它就开始工作了。
如何实现两种后备方案?
路由器只能有一个后备路由/服务。调用后
.fallback()
,此处理程序将替换已注册的后备服务.fallback_service()
。因此,500 与两个后备的存在无关(因为这是不可能的);相反,home_get
处理程序可能会返回此代码。如果.fallback_service()
完全删除调用,您应该会看到相同的行为。这是一个您需要修复的单独错误。您可能想要做的是将处理程序
ServeDir::fallback
链接home_get
为该服务的嵌套回退。请注意,您需要创建一个Service
没有状态概念的普通塔,因此您需要再次为将MethodRouter
用作回退服务提供状态(假设此处理程序甚至需要访问状态;如果不需要,那么您可以跳过提供状态)。例如: