mirror of
https://github.com/tokio-rs/axum.git
synced 2024-12-04 14:04:41 +01:00
refactor: add some informative header (X-Forwarded-For
, X-Forwarded-Host
, X-Forwarded-Proto
, Forwarded
)
This commit is contained in:
parent
e01086b179
commit
40971a46c5
1 changed files with 50 additions and 13 deletions
|
@ -1,5 +1,5 @@
|
|||
//! Reverse proxy listening in "localhost:4000" will proxy all `GET` requests to "localhost:3000" except for path /https is example.com
|
||||
//! endpoint.
|
||||
//! Reverse proxy listening in "localhost:4000" will proxy all `GET` requests to "localhost:3000"
|
||||
//! except for path /https is example.com endpoint.
|
||||
//!
|
||||
//! On unix like OS: make sure `ca-certificates` is installed.
|
||||
//!
|
||||
|
@ -9,6 +9,8 @@
|
|||
//! cargo run -p example-reverse-proxy
|
||||
//! ```
|
||||
|
||||
use axum::extract::ConnectInfo;
|
||||
use axum::http::header::FORWARDED;
|
||||
use axum::http::{header::HOST, StatusCode};
|
||||
use axum::{
|
||||
body::Body,
|
||||
|
@ -19,6 +21,7 @@ use axum::{
|
|||
};
|
||||
use hyper_tls::HttpsConnector;
|
||||
use hyper_util::{client::legacy::connect::HttpConnector, rt::TokioExecutor};
|
||||
use std::net::SocketAddr;
|
||||
|
||||
type Client = hyper_util::client::legacy::Client<HttpsConnector<HttpConnector>, Body>;
|
||||
|
||||
|
@ -38,16 +41,31 @@ async fn main() {
|
|||
.await
|
||||
.unwrap();
|
||||
println!("listening on {}", listener.local_addr().unwrap());
|
||||
axum::serve(listener, app).await.unwrap();
|
||||
axum::serve(
|
||||
listener,
|
||||
app.into_make_service_with_connect_info::<SocketAddr>(),
|
||||
)
|
||||
.await
|
||||
.unwrap();
|
||||
}
|
||||
|
||||
async fn handler(State(client): State<Client>, mut req: Request) -> Result<Response, StatusCode> {
|
||||
let path = req.uri().path();
|
||||
let path_query = req
|
||||
.uri()
|
||||
.path_and_query()
|
||||
.map(|v| v.as_str())
|
||||
.unwrap_or(path);
|
||||
async fn handler(
|
||||
State(client): State<Client>,
|
||||
ConnectInfo(addr): ConnectInfo<SocketAddr>,
|
||||
mut req: Request,
|
||||
) -> Result<Response, StatusCode> {
|
||||
let uri = req.uri();
|
||||
|
||||
let ip = addr.ip().to_string();
|
||||
let host = uri
|
||||
.authority()
|
||||
.map(|a| a.as_str())
|
||||
.unwrap_or("127.0.0.1:4000")
|
||||
.to_string();
|
||||
let proto = uri.scheme_str().unwrap_or("http").to_string();
|
||||
|
||||
let path = uri.path();
|
||||
let path_query = uri.path_and_query().map(|v| v.as_str()).unwrap_or(path);
|
||||
|
||||
let mut uri = format!("http://127.0.0.1:3000{}", path_query);
|
||||
if path == "/https" {
|
||||
|
@ -56,13 +74,32 @@ async fn handler(State(client): State<Client>, mut req: Request) -> Result<Respo
|
|||
|
||||
*req.uri_mut() = Uri::try_from(uri).unwrap();
|
||||
|
||||
//? Remove incorrect header host, hyper will add automatically for you.
|
||||
req.headers_mut().remove(HOST).unwrap();
|
||||
// Remove incorrect header host, hyper will add automatically for you.
|
||||
req.headers_mut().remove(HOST);
|
||||
|
||||
// Add some informative header (de-facto)
|
||||
req.headers_mut()
|
||||
.insert("X-Forwarded-For", ip.parse().unwrap());
|
||||
req.headers_mut()
|
||||
.insert("X-Forwarded-Host", host.parse().unwrap());
|
||||
req.headers_mut()
|
||||
.insert("X-Forwarded-Proto", proto.parse().unwrap());
|
||||
|
||||
// a standardized
|
||||
req.headers_mut().insert(
|
||||
FORWARDED,
|
||||
format!("for={ip};host={host};proto={proto};")
|
||||
.parse()
|
||||
.unwrap(),
|
||||
);
|
||||
|
||||
Ok(client
|
||||
.request(req)
|
||||
.await
|
||||
.map_err(|_| StatusCode::BAD_REQUEST)?
|
||||
.map_err(|err| {
|
||||
eprintln!("{:?}", err);
|
||||
StatusCode::BAD_REQUEST
|
||||
})?
|
||||
.into_response())
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in a new issue