use bytes::Bytes;
use http_body_util::Full;
use hyper::Response;
use hyper::service::Service;
use hyper_util::rt::TokioExecutor;
use hyper_util::rt::TokioIo;
use hyper_util::server::conn::auto::Builder;
use routerify_ng::{Router, RouterService};
use std::io;
use std::net::SocketAddr;
use std::sync::Arc;
use tokio::net::TcpListener;
mod users {
use http::Request;
use routerify_ng::ext::RequestExt;
use std::sync::Mutex;
use super::*;
#[derive(Clone)]
struct State {
count: Arc<Mutex<u8>>,
}
async fn list(req: Request<Full<Bytes>>) -> Result<Response<Full<Bytes>>, io::Error> {
let count = req.data::<State>().unwrap().count.lock().unwrap();
Ok(Response::new(Full::from(format!("Suppliers: {}", count))))
}
pub fn router() -> Router<io::Error> {
let state = State {
count: Arc::new(Mutex::new(20)),
};
Router::builder().data(state).get("/", list).build().unwrap()
}
}
mod offers {
use std::sync::Mutex;
use http::Request;
use routerify_ng::ext::RequestExt;
use super::*;
#[derive(Clone)]
struct State {
count: Arc<Mutex<u8>>,
}
async fn list(req: Request<Full<Bytes>>) -> Result<Response<Full<Bytes>>, io::Error> {
let count = req.data::<State>().unwrap().count.lock().unwrap();
println!("I can also access parent state: {:?}", req.data::<String>().unwrap());
Ok(Response::new(Full::from(format!("Suppliers: {}", count))))
}
pub fn router() -> Router<io::Error> {
let state = State {
count: Arc::new(Mutex::new(100)),
};
Router::builder().data(state).get("/abc", list).build().unwrap()
}
}
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
let scopes = Router::builder()
.data("Parent State data".to_owned())
.scope("/offers", offers::router())
.scope("/users", users::router())
.build()
.unwrap();
let router = Router::builder().scope("/v1", scopes).build().unwrap();
dbg!(&router);
let router_service = Arc::new(RouterService::new(router).unwrap());
let addr = SocketAddr::from(([127, 0, 0, 1], 3001));
let listener = TcpListener::bind(addr).await?;
println!("App is running on: {}", addr);
loop {
let (stream, _) = listener.accept().await?;
let router_service = router_service.clone();
tokio::spawn(async move {
let request_service = router_service.call(&stream).await.unwrap();
let io = TokioIo::new(stream);
let builder = Builder::new(TokioExecutor::new());
if let Err(err) = builder.serve_connection(io, request_service).await {
eprintln!("Error serving connection: {:?}", err);
}
});
}
}