mirror of
https://github.com/tokio-rs/axum.git
synced 2025-01-19 15:33:17 +01:00
139 lines
4.2 KiB
Rust
139 lines
4.2 KiB
Rust
//! Run with
|
|
//!
|
|
//! ```not_rust
|
|
//! cargo run -p example-unix-domain-socket
|
|
//! ```
|
|
|
|
#[cfg(unix)]
|
|
#[tokio::main]
|
|
async fn main() {
|
|
unix::server().await;
|
|
}
|
|
|
|
#[cfg(not(unix))]
|
|
fn main() {
|
|
println!("This example requires unix")
|
|
}
|
|
|
|
#[cfg(unix)]
|
|
mod unix {
|
|
use axum::{
|
|
body::Body,
|
|
extract::connect_info::{self, ConnectInfo},
|
|
http::{Method, Request, StatusCode},
|
|
routing::get,
|
|
Router,
|
|
};
|
|
use http_body_util::BodyExt;
|
|
use hyper::body::Incoming;
|
|
use hyper_util::{
|
|
rt::{TokioExecutor, TokioIo},
|
|
server,
|
|
};
|
|
use std::{convert::Infallible, path::PathBuf, sync::Arc};
|
|
use tokio::net::{unix::UCred, UnixListener, UnixStream};
|
|
use tower::Service;
|
|
use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt};
|
|
|
|
pub async fn server() {
|
|
tracing_subscriber::registry()
|
|
.with(
|
|
tracing_subscriber::EnvFilter::try_from_default_env()
|
|
.unwrap_or_else(|_| "debug".into()),
|
|
)
|
|
.with(tracing_subscriber::fmt::layer())
|
|
.init();
|
|
|
|
let path = PathBuf::from("/tmp/axum/helloworld");
|
|
|
|
let _ = tokio::fs::remove_file(&path).await;
|
|
tokio::fs::create_dir_all(path.parent().unwrap())
|
|
.await
|
|
.unwrap();
|
|
|
|
let uds = UnixListener::bind(path.clone()).unwrap();
|
|
tokio::spawn(async move {
|
|
let app = Router::new().route("/", get(handler));
|
|
|
|
let mut make_service = app.into_make_service_with_connect_info::<UdsConnectInfo>();
|
|
|
|
// See https://github.com/tokio-rs/axum/blob/main/examples/serve-with-hyper/src/main.rs for
|
|
// more details about this setup
|
|
loop {
|
|
let (socket, _remote_addr) = uds.accept().await.unwrap();
|
|
|
|
let tower_service = unwrap_infallible(make_service.call(&socket).await);
|
|
|
|
tokio::spawn(async move {
|
|
let socket = TokioIo::new(socket);
|
|
|
|
let hyper_service =
|
|
hyper::service::service_fn(move |request: Request<Incoming>| {
|
|
tower_service.clone().call(request)
|
|
});
|
|
|
|
if let Err(err) = server::conn::auto::Builder::new(TokioExecutor::new())
|
|
.serve_connection_with_upgrades(socket, hyper_service)
|
|
.await
|
|
{
|
|
eprintln!("failed to serve connection: {err:#}");
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
let stream = TokioIo::new(UnixStream::connect(path).await.unwrap());
|
|
let (mut sender, conn) = hyper::client::conn::http1::handshake(stream).await.unwrap();
|
|
tokio::task::spawn(async move {
|
|
if let Err(err) = conn.await {
|
|
println!("Connection failed: {:?}", err);
|
|
}
|
|
});
|
|
|
|
let request = Request::builder()
|
|
.method(Method::GET)
|
|
.uri("http://uri-doesnt-matter.com")
|
|
.body(Body::empty())
|
|
.unwrap();
|
|
|
|
let response = sender.send_request(request).await.unwrap();
|
|
|
|
assert_eq!(response.status(), StatusCode::OK);
|
|
|
|
let body = response.collect().await.unwrap().to_bytes();
|
|
let body = String::from_utf8(body.to_vec()).unwrap();
|
|
assert_eq!(body, "Hello, World!");
|
|
}
|
|
|
|
async fn handler(ConnectInfo(info): ConnectInfo<UdsConnectInfo>) -> &'static str {
|
|
println!("new connection from `{:?}`", info);
|
|
|
|
"Hello, World!"
|
|
}
|
|
|
|
#[derive(Clone, Debug)]
|
|
#[allow(dead_code)]
|
|
struct UdsConnectInfo {
|
|
peer_addr: Arc<tokio::net::unix::SocketAddr>,
|
|
peer_cred: UCred,
|
|
}
|
|
|
|
impl connect_info::Connected<&UnixStream> for UdsConnectInfo {
|
|
fn connect_info(target: &UnixStream) -> Self {
|
|
let peer_addr = target.peer_addr().unwrap();
|
|
let peer_cred = target.peer_cred().unwrap();
|
|
|
|
Self {
|
|
peer_addr: Arc::new(peer_addr),
|
|
peer_cred,
|
|
}
|
|
}
|
|
}
|
|
|
|
fn unwrap_infallible<T>(result: Result<T, Infallible>) -> T {
|
|
match result {
|
|
Ok(value) => value,
|
|
Err(err) => match err {},
|
|
}
|
|
}
|
|
}
|