diff --git a/src/parsing/utils.rs b/src/parsing/utils.rs index f3e6cf6..820f92a 100644 --- a/src/parsing/utils.rs +++ b/src/parsing/utils.rs @@ -12,8 +12,9 @@ use nom::sequence::{delimited, terminated}; /// - `inner`: The parser to trim /// - Returns: A parser that trims leading and trailing whitespace from the input and then runs the value from the inner parser pub fn trim<'a, Parser, R>(inner: Parser) -> impl FnMut(&'a str) -> IResult<&'a str, R> - where - Parser: Fn(&'a str) -> IResult<&'a str, R> { +where + Parser: Fn(&'a str) -> IResult<&'a str, R>, +{ delimited( multispace0, inner, @@ -27,8 +28,9 @@ pub fn trim<'a, Parser, R>(inner: Parser) -> impl FnMut(&'a str) -> IResult<&'a /// - `inner`: The parser to run inside the parentheses /// - Returns: A parser that parses a parenthesized expression pub fn parenthesized<'a, Parser, R>(inner: Parser) -> impl FnMut(&'a str) -> IResult<&'a str, R> - where - Parser: Fn(&'a str) -> IResult<&'a str, R> { +where + Parser: Fn(&'a str) -> IResult<&'a str, R>, +{ delimited( char('('), trim(inner), @@ -42,15 +44,19 @@ pub fn parenthesized<'a, Parser, R>(inner: Parser) -> impl FnMut(&'a str) -> IRe /// - `predicate`: The predicate to call to validate the input /// - Returns: A parser that takes `n` characters from the input pub fn take_where>(n: usize, predicate: F) -> impl Fn(Input) -> IResult - where Input: InputTake + InputIter + InputLength + Slice>, F: Fn(::Item) -> bool, { +where + Input: InputTake + InputIter + InputLength + Slice>, + F: Fn(::Item) -> bool, +{ move |input: Input| { take_while_m_n(n, n, |it| predicate(it))(input) } } pub fn exhausted<'a, Parser, R>(inner: Parser) -> impl FnMut(&'a str) -> IResult<&'a str, R> - where - Parser: Fn(&'a str) -> IResult<&'a str, R> { +where + Parser: Fn(&'a str) -> IResult<&'a str, R>, +{ terminated(inner, eof) } diff --git a/src/routing/routes/index.rs b/src/routing/routes/index.rs index 259f143..7e292bf 100644 --- a/src/routing/routes/index.rs +++ b/src/routing/routes/index.rs @@ -1,16 +1,15 @@ use axum::body::Body; use axum::http::StatusCode; use axum::response::{Html, IntoResponse, Response}; -use axum::Router; -use axum::routing::get; use tokio::fs::File; use tokio_util::io::ReaderStream; -pub fn router() -> Router { - Router::new() - .route("/", get(index)) - .route("/openapi", get(open_api)) -} +use crate::router; + +router!( + get "/" => index, + get "/openapi" => open_api +); async fn index() -> &'static str { "Welcome to the Simplify Truths API!\n" diff --git a/src/routing/routes/simplify.rs b/src/routing/routes/simplify.rs index 2dcb837..b54d9e3 100644 --- a/src/routing/routes/simplify.rs +++ b/src/routing/routes/simplify.rs @@ -1,22 +1,18 @@ -use axum::{Router, routing::get}; use axum::extract::{Path, Query}; use axum::http::StatusCode; use axum::response::{IntoResponse, Response}; use crate::expressions::expression::Expression; use crate::expressions::truth_table::TruthTable; +use crate::{router, routes}; use crate::routing::error::{Error, ErrorKind}; use crate::routing::options::{SimplifyAndTableOptions, SimplifyOptions}; use crate::routing::response::SimplifyResponse; -pub fn router() -> Router<()> { - Router::new() - .nest("/simplify", - Router::new() - .route("/:exp", get(simplify)) - .route("/table/:exp", get(simplify_and_table)), - ) -} +router!("/simplify", routes!( + get "/:exp" => simplify, + get "/table/:exp" => simplify_and_table +)); async fn simplify(Path(path): Path, Query(query): Query) -> Response { match Expression::try_from(path.as_str()) { diff --git a/src/routing/routes/table.rs b/src/routing/routes/table.rs index 477f141..87dbc62 100644 --- a/src/routing/routes/table.rs +++ b/src/routing/routes/table.rs @@ -1,21 +1,17 @@ use axum::extract::{Path, Query}; use axum::http::StatusCode; use axum::response::{IntoResponse, Response}; -use axum::Router; -use axum::routing::get; +use crate::{router, routes}; use crate::expressions::expression::Expression; use crate::expressions::truth_table::TruthTable; use crate::routing::error::{Error, ErrorKind}; use crate::routing::options::TruthTableOptions; use crate::routing::response::TruthTableResponse; -pub fn router() -> Router<()> { - Router::new() - .nest("/table", Router::new() - .route("/:exp", get(table)), - ) -} +router!("/table", routes!( + get "/:exp" => table +)); // TODO Expression as input in body async fn table(Path(value): Path, Query(query): Query) -> Response { diff --git a/src/routing/routes/util.rs b/src/routing/routes/util.rs index 086b946..08f3e69 100644 --- a/src/routing/routes/util.rs +++ b/src/routing/routes/util.rs @@ -1,16 +1,14 @@ use axum::extract::Path; use axum::response::{IntoResponse, Response}; -use axum::Router; -use axum::routing::get; use crate::expressions::expression::Expression; +use crate::router; use crate::routing::error::{Error, ErrorKind}; use crate::routing::response::IsLegalResponse; -pub fn router() -> Router<()> { - Router::new() - .route("/is-legal/:exp", get(is_legal)) -} +router!( + get "/is-legal/:exp" => is_legal +); async fn is_legal(Path(path): Path) -> Response { match Expression::try_from(path.as_str()) { diff --git a/src/utils/axum.rs b/src/utils/axum.rs new file mode 100644 index 0000000..6efaa15 --- /dev/null +++ b/src/utils/axum.rs @@ -0,0 +1,42 @@ +/// Create an axum router function with the given body or routes. +/// # Examples +/// ``` +/// router!( +/// get "/" => index, +/// get "/openapi" => open_api +/// ); +/// router!("/simplify", routes!( +/// get "/:exp" => simplify, +/// get "/table/:exp" => simplify_and_table +/// )); +/// ``` +#[macro_export] +macro_rules! router { + ($body:expr) => { + pub(crate) fn router() -> axum::Router<()> { + $body + } + }; + ($route:expr, $router:expr) => { + router!(axum::Router::new().nest($route, $router)); + }; + ($($method:ident $route:expr => $func:expr),* $(,)?) => { + router!($crate::routes!($($method $route => $func),*)); + }; +} + +/// Create a router with the given routes. +/// # Examples +/// ``` +/// routes!( +/// get "/" => index, +/// post "/" => create +/// ); +/// ``` +#[macro_export] +macro_rules! routes { + ($($method:ident $route:expr => $func:expr),* $(,)?) => { + axum::Router::new() + $(.route($route, axum::routing::$method($func)))* + }; +} diff --git a/src/utils/mod.rs b/src/utils/mod.rs index a81c6df..d23a11a 100644 --- a/src/utils/mod.rs +++ b/src/utils/mod.rs @@ -1,2 +1,3 @@ pub mod array; -pub mod serialize; \ No newline at end of file +pub mod serialize; +mod axum; \ No newline at end of file