1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
// 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;
use quote::quote;
use std::env;

#[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 = if env::var("CARGO_PKG_NAME").unwrap() == "substrate-test-utils" {
		syn::Ident::new("substrate_test_utils", Span::call_site().into())
	} else {
		let crate_name = crate_name("substrate-test-utils")
			.map_err(|e| syn::Error::new_spanned(&sig, e))?;

		syn::Ident::new(&crate_name, Span::call_site().into())
	};

	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())
}