2019-11-25 16:49:11 +01:00
|
|
|
//! Macros for use with Tokio
|
2020-12-13 00:24:00 +01:00
|
|
|
|
|
|
|
#![deny(rust_2018_idioms, nonstandard_style)]
|
|
|
|
#![doc(html_logo_url = "https://actix.rs/img/logo.png")]
|
|
|
|
#![doc(html_favicon_url = "https://actix.rs/favicon.ico")]
|
2019-11-25 16:49:11 +01:00
|
|
|
|
|
|
|
use proc_macro::TokenStream;
|
|
|
|
use quote::quote;
|
|
|
|
|
|
|
|
/// Marks async function to be executed by actix system.
|
|
|
|
///
|
|
|
|
/// ## Usage
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// #[actix_rt::main]
|
|
|
|
/// async fn main() {
|
|
|
|
/// println!("Hello world");
|
|
|
|
/// }
|
|
|
|
/// ```
|
2020-01-28 12:27:33 +01:00
|
|
|
#[allow(clippy::needless_doctest_main)]
|
2019-11-25 16:49:11 +01:00
|
|
|
#[proc_macro_attribute]
|
|
|
|
#[cfg(not(test))] // Work around for rust-lang/rust#62127
|
|
|
|
pub fn main(_: TokenStream, item: TokenStream) -> TokenStream {
|
2019-12-10 03:47:35 +01:00
|
|
|
let mut input = syn::parse_macro_input!(item as syn::ItemFn);
|
2019-11-25 16:49:11 +01:00
|
|
|
let attrs = &input.attrs;
|
2019-12-10 03:47:35 +01:00
|
|
|
let vis = &input.vis;
|
|
|
|
let sig = &mut input.sig;
|
|
|
|
let body = &input.block;
|
|
|
|
let name = &sig.ident;
|
2019-11-25 16:49:11 +01:00
|
|
|
|
2019-12-10 03:47:35 +01:00
|
|
|
if sig.asyncness.is_none() {
|
|
|
|
return syn::Error::new_spanned(sig.fn_token, "only async fn is supported")
|
2019-11-25 16:49:11 +01:00
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
|
2019-12-10 03:47:35 +01:00
|
|
|
sig.asyncness = None;
|
|
|
|
|
2020-12-03 00:59:13 +01:00
|
|
|
if cfg!(feature = "actix-reexport") {
|
|
|
|
(quote! {
|
|
|
|
#(#attrs)*
|
|
|
|
#vis #sig {
|
|
|
|
actix::System::new(stringify!(#name))
|
|
|
|
.block_on(async move { #body })
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.into()
|
|
|
|
} else {
|
|
|
|
(quote! {
|
|
|
|
#(#attrs)*
|
|
|
|
#vis #sig {
|
|
|
|
actix_rt::System::new(stringify!(#name))
|
|
|
|
.block_on(async move { #body })
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.into()
|
|
|
|
}
|
2019-11-25 16:49:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Marks async test function to be executed by actix runtime.
|
|
|
|
///
|
|
|
|
/// ## Usage
|
|
|
|
///
|
|
|
|
/// ```no_run
|
|
|
|
/// #[actix_rt::test]
|
|
|
|
/// async fn my_test() {
|
|
|
|
/// assert!(true);
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
#[proc_macro_attribute]
|
|
|
|
pub fn test(_: TokenStream, item: TokenStream) -> TokenStream {
|
2020-04-21 19:11:48 +02:00
|
|
|
let mut input = syn::parse_macro_input!(item as syn::ItemFn);
|
2019-11-25 16:49:11 +01:00
|
|
|
let attrs = &input.attrs;
|
2020-04-21 19:11:48 +02:00
|
|
|
let vis = &input.vis;
|
|
|
|
let sig = &mut input.sig;
|
|
|
|
let body = &input.block;
|
2019-11-25 16:49:11 +01:00
|
|
|
let mut has_test_attr = false;
|
|
|
|
|
|
|
|
for attr in attrs {
|
|
|
|
if attr.path.is_ident("test") {
|
|
|
|
has_test_attr = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-21 19:11:48 +02:00
|
|
|
if sig.asyncness.is_none() {
|
2019-11-26 05:01:46 +01:00
|
|
|
return syn::Error::new_spanned(
|
|
|
|
input.sig.fn_token,
|
2019-12-14 18:01:55 +01:00
|
|
|
format!("only async fn is supported, {}", input.sig.ident),
|
2019-11-26 05:01:46 +01:00
|
|
|
)
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
2019-11-25 16:49:11 +01:00
|
|
|
}
|
|
|
|
|
2020-04-21 19:11:48 +02:00
|
|
|
sig.asyncness = None;
|
|
|
|
|
2019-11-25 16:49:11 +01:00
|
|
|
let result = if has_test_attr {
|
|
|
|
quote! {
|
|
|
|
#(#attrs)*
|
2020-04-21 19:11:48 +02:00
|
|
|
#vis #sig {
|
2019-11-25 16:49:11 +01:00
|
|
|
actix_rt::System::new("test")
|
|
|
|
.block_on(async { #body })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
quote! {
|
|
|
|
#[test]
|
|
|
|
#(#attrs)*
|
2020-04-21 19:11:48 +02:00
|
|
|
#vis #sig {
|
2019-11-25 16:49:11 +01:00
|
|
|
actix_rt::System::new("test")
|
|
|
|
.block_on(async { #body })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
result.into()
|
|
|
|
}
|