redsunlib/src/server.rs

216 lines
5.9 KiB
Rust
Raw Normal View History

2021-03-17 15:30:33 -07:00
use cookie::Cookie;
use futures_lite::{future::Boxed, Future, FutureExt};
use hyper::{
header::HeaderValue,
2021-03-17 15:30:33 -07:00
service::{make_service_fn, service_fn},
HeaderMap,
};
use hyper::{Body, Method, Request, Response, Server as HyperServer};
use route_recognizer::{Params, Router};
use std::{pin::Pin, result::Result};
use time::Duration;
type BoxResponse = Pin<Box<dyn Future<Output = Result<Response<Body>, String>> + Send>>;
pub struct Route<'a> {
router: &'a mut Router<fn(Request<Body>) -> BoxResponse>,
path: String,
}
pub struct Server {
pub default_headers: HeaderMap,
router: Router<fn(Request<Body>) -> BoxResponse>,
}
#[macro_export]
macro_rules! headers(
{ $($key:expr => $value:expr),+ } => {
{
let mut m = hyper::HeaderMap::new();
$(
2021-03-20 22:10:31 -07:00
if let Ok(val) = hyper::header::HeaderValue::from_str($value) {
m.insert($key, val);
}
2021-03-17 15:30:33 -07:00
)+
m
}
};
);
pub trait RequestExt {
fn params(&self) -> Params;
fn param(&self, name: &str) -> Option<String>;
fn set_params(&mut self, params: Params) -> Option<Params>;
fn cookies(&self) -> Vec<Cookie>;
fn cookie(&self, name: &str) -> Option<Cookie>;
}
pub trait ResponseExt {
fn cookies(&self) -> Vec<Cookie>;
fn insert_cookie(&mut self, cookie: Cookie);
fn remove_cookie(&mut self, name: String);
}
impl RequestExt for Request<Body> {
fn params(&self) -> Params {
self.extensions().get::<Params>().unwrap_or(&Params::new()).clone()
2021-03-17 15:30:33 -07:00
// self.extensions()
// .get::<RequestMeta>()
// .and_then(|meta| meta.route_params())
// .expect("Routerify: No RouteParams added while processing request")
}
fn param(&self, name: &str) -> Option<String> {
2021-03-26 20:00:47 -07:00
self.params().find(name).map(std::borrow::ToOwned::to_owned)
2021-03-17 15:30:33 -07:00
}
fn set_params(&mut self, params: Params) -> Option<Params> {
self.extensions_mut().insert(params)
}
fn cookies(&self) -> Vec<Cookie> {
2021-05-20 12:24:06 -07:00
self.headers().get("Cookie").map_or(Vec::new(), |header| {
header
.to_str()
.unwrap_or_default()
.split("; ")
.map(|cookie| Cookie::parse(cookie).unwrap_or_else(|_| Cookie::named("")))
.collect()
})
2021-03-17 15:30:33 -07:00
}
fn cookie(&self, name: &str) -> Option<Cookie> {
2021-05-20 12:24:06 -07:00
self.cookies().into_iter().find(|c| c.name() == name)
2021-03-17 15:30:33 -07:00
}
}
impl ResponseExt for Response<Body> {
fn cookies(&self) -> Vec<Cookie> {
2021-05-20 12:24:06 -07:00
self.headers().get("Cookie").map_or(Vec::new(), |header| {
header
.to_str()
.unwrap_or_default()
.split("; ")
.map(|cookie| Cookie::parse(cookie).unwrap_or_else(|_| Cookie::named("")))
.collect()
})
2021-03-17 15:30:33 -07:00
}
fn insert_cookie(&mut self, cookie: Cookie) {
2021-03-20 22:10:31 -07:00
if let Ok(val) = HeaderValue::from_str(&cookie.to_string()) {
self.headers_mut().append("Set-Cookie", val);
}
2021-03-17 15:30:33 -07:00
}
fn remove_cookie(&mut self, name: String) {
let mut cookie = Cookie::named(name);
cookie.set_path("/");
cookie.set_max_age(Duration::second());
2021-03-20 22:10:31 -07:00
if let Ok(val) = HeaderValue::from_str(&cookie.to_string()) {
self.headers_mut().append("Set-Cookie", val);
}
2021-03-17 15:30:33 -07:00
}
}
impl Route<'_> {
fn method(&mut self, method: Method, dest: fn(Request<Body>) -> BoxResponse) -> &mut Self {
self.router.add(&format!("/{}{}", method.as_str(), self.path), dest);
self
}
/// Add an endpoint for `GET` requests
pub fn get(&mut self, dest: fn(Request<Body>) -> BoxResponse) -> &mut Self {
self.method(Method::GET, dest)
}
/// Add an endpoint for `POST` requests
pub fn post(&mut self, dest: fn(Request<Body>) -> BoxResponse) -> &mut Self {
self.method(Method::POST, dest)
}
}
impl Server {
pub fn new() -> Self {
Server {
default_headers: HeaderMap::new(),
router: Router::new(),
}
}
pub fn at(&mut self, path: &str) -> Route {
Route {
path: path.to_owned(),
router: &mut self.router,
}
}
pub fn listen(self, addr: String) -> Boxed<Result<(), hyper::Error>> {
let make_svc = make_service_fn(move |_conn| {
2021-05-20 12:24:06 -07:00
// For correct borrowing, these values need to be borrowed
2021-03-17 15:30:33 -07:00
let router = self.router.clone();
let default_headers = self.default_headers.clone();
// This is the `Service` that will handle the connection.
// `service_fn` is a helper to convert a function that
// returns a Response into a `Service`.
// let shared_router = router.clone();
async move {
Ok::<_, String>(service_fn(move |req: Request<Body>| {
let headers = default_headers.clone();
// Remove double slashes
let mut path = req.uri().path().replace("//", "/");
// Remove trailing slashes
2021-05-20 12:24:06 -07:00
if path != "/" && path.ends_with('/') {
2021-03-17 15:30:33 -07:00
path.pop();
}
// Match the visited path with an added route
match router.recognize(&format!("/{}{}", req.method().as_str(), path)) {
// If a route was configured for this path
Ok(found) => {
let mut parammed = req;
parammed.set_params(found.params().clone());
2021-03-17 15:30:33 -07:00
// Run the route's function
2021-03-31 13:03:44 -07:00
let func = (found.handler().to_owned().to_owned())(parammed);
2021-03-17 15:30:33 -07:00
async move {
2021-03-31 13:03:44 -07:00
let res: Result<Response<Body>, String> = func.await;
2021-03-17 15:30:33 -07:00
// Add default headers to response
res.map(|mut response| {
response.headers_mut().extend(headers);
response
})
}
.boxed()
}
// If there was a routing error
Err(e) => async move {
// Return a 404 error
let res: Result<Response<Body>, String> = Ok(Response::builder().status(404).body(e.into()).unwrap_or_default());
// Add default headers to response
res.map(|mut response| {
response.headers_mut().extend(headers);
response
})
}
.boxed(),
}
}))
}
});
2021-05-20 12:24:06 -07:00
// Build SocketAddr from provided address
2021-03-17 15:30:33 -07:00
let address = &addr.parse().unwrap_or_else(|_| panic!("Cannot parse {} as address (example format: 0.0.0.0:8080)", addr));
2021-05-20 12:24:06 -07:00
// Bind server to address specified above. Gracefully shut down if CTRL+C is pressed
let server = HyperServer::bind(address).serve(make_svc).with_graceful_shutdown(async {
// Wait for the CTRL+C signal
2021-09-09 17:28:55 -07:00
tokio::signal::ctrl_c().await.expect("Failed to install CTRL+C signal handler");
2021-05-20 12:24:06 -07:00
});
2021-03-17 15:30:33 -07:00
2021-05-20 12:24:06 -07:00
server.boxed()
2021-03-17 15:30:33 -07:00
}
}