tetcore_test_utils_derive/
lib.rs

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 Tetcore.

// 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 tc_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 tc_service::TaskExecutor";
			return Err(syn::Error::new_spanned(&sig, msg));
		}
	};

	let crate_name = if env::var("CARGO_PKG_NAME").unwrap() == "tetcore-test-utils" {
		syn::Ident::new("tetcore_test_utils", Span::call_site().into())
	} else {
		let crate_name = crate_name("tetcore-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("TETCORE_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())
}