From c3e45a04fc50636c5b0ba2f7d4377f2f87eb8ff3 Mon Sep 17 00:00:00 2001
From: Bernhard Schuster <bernhard@ahoi.io>
Date: Mon, 26 Sep 2022 22:11:36 +0200
Subject: [PATCH] [orchestra] fix: require the initialization with `F: FnOnce`
 to be `Send` (#6051)

* add regression test for missing Send requirement

* fix: require the initialization with `F: FnOnce` to be `Send` as well

If creating intermediate variables of the builder type within
a future, rustc will complain about the future not being send,
while the thing itself isn't even using the closure based field
initialization. Adding an additional bound, resolves this and
pushes the error message "closer" to the user, and out of the
generated code.

* import fixins
---
 polkadot/node/orchestra/examples/solo.rs      | 37 +++++++++++--------
 .../orchestra/proc-macro/src/impl_builder.rs  |  6 +--
 2 files changed, 24 insertions(+), 19 deletions(-)

diff --git a/polkadot/node/orchestra/examples/solo.rs b/polkadot/node/orchestra/examples/solo.rs
index 67ebc292e46..cc82656759c 100644
--- a/polkadot/node/orchestra/examples/solo.rs
+++ b/polkadot/node/orchestra/examples/solo.rs
@@ -46,24 +46,29 @@ impl<Context> Fortified {
 	}
 }
 
-fn main() {
-	use futures::{executor, pin_mut};
+async fn setup() {
+	let builder = Solo::builder();
+
+	let builder = builder.goblin_tower(Fortified::default());
+
+	let builder = builder.spawner(DummySpawner);
+	let (orchestra, _handle): (Solo<_>, _) = builder.build().unwrap();
 
-	executor::block_on(async move {
-		let (orchestra, _handle): (Solo<_>, _) = Solo::builder()
-			.goblin_tower(Fortified::default())
-			.spawner(DummySpawner)
-			.build()
-			.unwrap();
+	let orchestra_fut = orchestra
+		.running_subsystems
+		.into_future()
+		.timeout(std::time::Duration::from_millis(300))
+		.fuse();
 
-		let orchestra_fut = orchestra
-			.running_subsystems
-			.into_future()
-			.timeout(std::time::Duration::from_millis(300))
-			.fuse();
+	futures::pin_mut!(orchestra_fut);
 
-		pin_mut!(orchestra_fut);
+	orchestra_fut.await;
+}
+
+fn assert_t_impl_trait_send<T: Send>(_: &T) {}
 
-		orchestra_fut.await
-	});
+fn main() {
+	let x = setup();
+	assert_t_impl_trait_send(&x);
+	futures::executor::block_on(x);
 }
diff --git a/polkadot/node/orchestra/proc-macro/src/impl_builder.rs b/polkadot/node/orchestra/proc-macro/src/impl_builder.rs
index 1be25d45b5e..1a88dc92612 100644
--- a/polkadot/node/orchestra/proc-macro/src/impl_builder.rs
+++ b/polkadot/node/orchestra/proc-macro/src/impl_builder.rs
@@ -172,7 +172,7 @@ pub(crate) fn impl_builder(info: &OrchestraInfo) -> proc_macro2::TokenStream {
 					pub fn #field_name_with<'a, F>(self, subsystem_init_fn: F ) ->
 						#builder <InitStateSpawner, #( #post_setter_state_generics, )* #( #baggage_passthrough_state_generics, )*>
 					where
-						F: 'static + FnOnce(#handle) ->
+						F: 'static + Send + FnOnce(#handle) ->
 							::std::result::Result<#field_type, #error_ty>,
 					{
 						let boxed_func = Init::<#field_type>::Fn(
@@ -206,7 +206,7 @@ pub(crate) fn impl_builder(info: &OrchestraInfo) -> proc_macro2::TokenStream {
 						-> #builder <InitStateSpawner, #( #post_replace_state_generics, )* #( #baggage_passthrough_state_generics, )*>
 					where
 						#field_type: 'static,
-						F: 'static + FnOnce(#field_type) -> NEW,
+						F: 'static + Send + FnOnce(#field_type) -> NEW,
 						NEW: #support_crate ::Subsystem<#subsystem_ctx_name< #subsystem_consumes >, #error_ty>,
 					{
 						let replacement: Init<NEW> = match self.#field_name {
@@ -333,7 +333,7 @@ pub(crate) fn impl_builder(info: &OrchestraInfo) -> proc_macro2::TokenStream {
 
 	let mut ts = quote! {
 		/// Convenience alias.
-		type SubsystemInitFn<T> = Box<dyn FnOnce(#handle) -> ::std::result::Result<T, #error_ty> >;
+		type SubsystemInitFn<T> = Box<dyn FnOnce(#handle) -> ::std::result::Result<T, #error_ty> + Send + 'static>;
 
 		/// Type for the initialized field of the orchestra builder
 		pub enum Init<T> {
-- 
GitLab