mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-27 05:47:58 +00:00
104 lines
3.1 KiB
Rust
104 lines
3.1 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2020-2021 Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: GPL-3.0-or-later WITH Classpath-exception-2.0
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
use proc_macro::{Span, TokenStream};
|
|
use proc_macro_crate::{crate_name, FoundCrate};
|
|
use quote::quote;
|
|
|
|
#[proc_macro_attribute]
|
|
pub fn test(args: TokenStream, item: TokenStream) -> TokenStream {
|
|
impl_test(args, item)
|
|
}
|
|
|
|
fn impl_test(args: TokenStream, item: TokenStream) -> TokenStream {
|
|
let input = syn::parse_macro_input!(item as syn::ItemFn);
|
|
let args = syn::parse_macro_input!(args as syn::AttributeArgs);
|
|
|
|
parse_knobs(input, args).unwrap_or_else(|e| e.to_compile_error().into())
|
|
}
|
|
|
|
fn parse_knobs(
|
|
mut input: syn::ItemFn,
|
|
args: syn::AttributeArgs,
|
|
) -> Result<TokenStream, syn::Error> {
|
|
let sig = &mut input.sig;
|
|
let body = &input.block;
|
|
let attrs = &input.attrs;
|
|
let vis = input.vis;
|
|
|
|
if sig.inputs.len() != 1 {
|
|
let msg = "the test function accepts only one argument of type sc_service::TaskExecutor";
|
|
return Err(syn::Error::new_spanned(&sig, msg));
|
|
}
|
|
let (task_executor_name, task_executor_type) = match sig.inputs.pop().map(|x| x.into_value()) {
|
|
Some(syn::FnArg::Typed(x)) => (x.pat, x.ty),
|
|
_ => {
|
|
let msg =
|
|
"the test function accepts only one argument of type sc_service::TaskExecutor";
|
|
return Err(syn::Error::new_spanned(&sig, msg));
|
|
}
|
|
};
|
|
|
|
let crate_name = match crate_name("substrate-test-utils") {
|
|
Ok(FoundCrate::Itself) => syn::Ident::new("substrate_test_utils", Span::call_site().into()),
|
|
Ok(FoundCrate::Name(crate_name)) => syn::Ident::new(&crate_name, Span::call_site().into()),
|
|
Err(e) => return Err(syn::Error::new_spanned(&sig, e)),
|
|
};
|
|
|
|
let header = {
|
|
quote! {
|
|
#[#crate_name::tokio::test(#(#args)*)]
|
|
}
|
|
};
|
|
|
|
let result = quote! {
|
|
#header
|
|
#(#attrs)*
|
|
#vis #sig {
|
|
use #crate_name::futures::future::FutureExt;
|
|
|
|
let #task_executor_name: #task_executor_type = (|fut, _| {
|
|
#crate_name::tokio::spawn(fut).map(drop)
|
|
})
|
|
.into();
|
|
let timeout_task = #crate_name::tokio::time::delay_for(
|
|
std::time::Duration::from_secs(
|
|
std::env::var("SUBSTRATE_TEST_TIMEOUT")
|
|
.ok()
|
|
.and_then(|x| x.parse().ok())
|
|
.unwrap_or(600))
|
|
).fuse();
|
|
let actual_test_task = async move {
|
|
#body
|
|
}
|
|
.fuse();
|
|
|
|
#crate_name::futures::pin_mut!(timeout_task, actual_test_task);
|
|
|
|
#crate_name::futures::select! {
|
|
_ = timeout_task => {
|
|
panic!("The test took too long!");
|
|
},
|
|
_ = actual_test_task => {},
|
|
}
|
|
}
|
|
};
|
|
|
|
Ok(result.into())
|
|
}
|