2021-02-02 03:07:58 +01:00
|
|
|
//! Macros for Actix system and runtime.
|
|
|
|
//!
|
|
|
|
//! The [`actix-rt`](https://docs.rs/actix-rt) crate must be available for macro output to compile.
|
|
|
|
//!
|
|
|
|
//! # Entry-point
|
|
|
|
//! See docs for the [`#[main]`](macro@main) macro.
|
|
|
|
//!
|
|
|
|
//! # Tests
|
|
|
|
//! See docs for the [`#[test]`](macro@test) macro.
|
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;
|
|
|
|
|
2021-02-02 03:07:58 +01:00
|
|
|
/// Marks async entry-point function to be executed by Actix system.
|
2019-11-25 16:49:11 +01:00
|
|
|
///
|
2021-02-02 03:07:58 +01:00
|
|
|
/// # Examples
|
2021-01-09 15:13:16 +01:00
|
|
|
/// ```
|
2019-11-25 16:49:11 +01:00
|
|
|
/// #[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
|
2021-06-08 18:33:05 +02:00
|
|
|
pub fn main(args: TokenStream, item: TokenStream) -> TokenStream {
|
2021-10-14 11:54:39 +02:00
|
|
|
let mut input = match syn::parse::<syn::ItemFn>(item.clone()) {
|
|
|
|
Ok(input) => input,
|
|
|
|
// on parse err, make IDEs happy; see fn docs
|
|
|
|
Err(err) => return input_and_compile_error(item, err),
|
|
|
|
};
|
|
|
|
|
2021-06-08 18:33:05 +02:00
|
|
|
let args = syn::parse_macro_input!(args as syn::AttributeArgs);
|
|
|
|
|
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;
|
2019-11-25 16:49:11 +01:00
|
|
|
|
2019-12-10 03:47:35 +01:00
|
|
|
if sig.asyncness.is_none() {
|
2021-02-02 03:07:58 +01:00
|
|
|
return syn::Error::new_spanned(
|
|
|
|
sig.fn_token,
|
|
|
|
"the async keyword is missing from the function declaration",
|
|
|
|
)
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
2019-11-25 16:49:11 +01:00
|
|
|
}
|
|
|
|
|
2021-06-08 18:33:05 +02:00
|
|
|
let mut system = syn::parse_str::<syn::Path>("::actix_rt::System").unwrap();
|
|
|
|
|
|
|
|
for arg in &args {
|
|
|
|
match arg {
|
|
|
|
syn::NestedMeta::Meta(syn::Meta::NameValue(syn::MetaNameValue {
|
|
|
|
lit: syn::Lit::Str(lit),
|
|
|
|
path,
|
|
|
|
..
|
|
|
|
})) => match path
|
|
|
|
.get_ident()
|
|
|
|
.map(|i| i.to_string().to_lowercase())
|
|
|
|
.as_deref()
|
|
|
|
{
|
|
|
|
Some("system") => match lit.parse() {
|
|
|
|
Ok(path) => system = path,
|
|
|
|
Err(_) => {
|
|
|
|
return syn::Error::new_spanned(lit, "Expected path")
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
return syn::Error::new_spanned(arg, "Unknown attribute specified")
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
return syn::Error::new_spanned(arg, "Unknown attribute specified")
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-10 03:47:35 +01:00
|
|
|
sig.asyncness = None;
|
|
|
|
|
2021-01-09 15:58:15 +01:00
|
|
|
(quote! {
|
|
|
|
#(#attrs)*
|
|
|
|
#vis #sig {
|
2021-06-08 18:33:05 +02:00
|
|
|
<#system>::new().block_on(async move { #body })
|
2021-01-09 15:58:15 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.into()
|
2019-11-25 16:49:11 +01:00
|
|
|
}
|
|
|
|
|
2021-02-02 03:07:58 +01:00
|
|
|
/// Marks async test function to be executed in an Actix system.
|
2019-11-25 16:49:11 +01:00
|
|
|
///
|
2021-02-02 03:07:58 +01:00
|
|
|
/// # Examples
|
|
|
|
/// ```
|
2019-11-25 16:49:11 +01:00
|
|
|
/// #[actix_rt::test]
|
|
|
|
/// async fn my_test() {
|
|
|
|
/// assert!(true);
|
|
|
|
/// }
|
|
|
|
/// ```
|
|
|
|
#[proc_macro_attribute]
|
2021-10-14 11:54:39 +02:00
|
|
|
pub fn test(args: TokenStream, item: TokenStream) -> TokenStream {
|
|
|
|
let mut input = match syn::parse::<syn::ItemFn>(item.clone()) {
|
|
|
|
Ok(input) => input,
|
|
|
|
// on parse err, make IDEs happy; see fn docs
|
|
|
|
Err(err) => return input_and_compile_error(item, err),
|
|
|
|
};
|
|
|
|
|
|
|
|
let args = syn::parse_macro_input!(args as syn::AttributeArgs);
|
|
|
|
|
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,
|
2021-02-02 03:07:58 +01:00
|
|
|
"the async keyword is missing from the function declaration",
|
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;
|
|
|
|
|
2021-02-02 03:07:58 +01:00
|
|
|
let missing_test_attr = if has_test_attr {
|
|
|
|
quote!()
|
2019-11-25 16:49:11 +01:00
|
|
|
} else {
|
2021-02-02 03:07:58 +01:00
|
|
|
quote!(#[test])
|
2019-11-25 16:49:11 +01:00
|
|
|
};
|
|
|
|
|
2021-10-14 11:54:39 +02:00
|
|
|
let mut system = syn::parse_str::<syn::Path>("::actix_rt::System").unwrap();
|
|
|
|
|
|
|
|
for arg in &args {
|
|
|
|
match arg {
|
|
|
|
syn::NestedMeta::Meta(syn::Meta::NameValue(syn::MetaNameValue {
|
|
|
|
lit: syn::Lit::Str(lit),
|
|
|
|
path,
|
|
|
|
..
|
|
|
|
})) => match path
|
|
|
|
.get_ident()
|
|
|
|
.map(|i| i.to_string().to_lowercase())
|
|
|
|
.as_deref()
|
|
|
|
{
|
|
|
|
Some("system") => match lit.parse() {
|
|
|
|
Ok(path) => system = path,
|
|
|
|
Err(_) => {
|
|
|
|
return syn::Error::new_spanned(lit, "Expected path")
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
return syn::Error::new_spanned(arg, "Unknown attribute specified")
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_ => {
|
|
|
|
return syn::Error::new_spanned(arg, "Unknown attribute specified")
|
|
|
|
.to_compile_error()
|
|
|
|
.into();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-02 03:07:58 +01:00
|
|
|
(quote! {
|
|
|
|
#missing_test_attr
|
|
|
|
#(#attrs)*
|
|
|
|
#vis #sig {
|
2021-10-14 11:54:39 +02:00
|
|
|
<#system>::new().block_on(async { #body })
|
2021-02-02 03:07:58 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.into()
|
2019-11-25 16:49:11 +01:00
|
|
|
}
|
2021-10-14 11:54:39 +02:00
|
|
|
|
|
|
|
/// Converts the error to a token stream and appends it to the original input.
|
|
|
|
///
|
|
|
|
/// Returning the original input in addition to the error is good for IDEs which can gracefully
|
|
|
|
/// recover and show more precise errors within the macro body.
|
|
|
|
///
|
|
|
|
/// See <https://github.com/rust-analyzer/rust-analyzer/issues/10468> for more info.
|
|
|
|
fn input_and_compile_error(mut item: TokenStream, err: syn::Error) -> TokenStream {
|
|
|
|
let compile_err = TokenStream::from(err.to_compile_error());
|
|
|
|
item.extend(compile_err);
|
|
|
|
return item;
|
|
|
|
}
|