#[macro_export] macro_rules! command { ( Named { $export:tt $args:ident $body:block } Positional { $($number:tt)* } Rest {} CommandConfig { name: $config_name:tt, mandatory_positional: vec![ $($mandatory_positional:tt)* ], optional_positional: vec![ $($optional_positional:tt)* ], rest_positional: $rest_positional:tt, named: { $( ($named_param:tt : $named_type:tt) )* } } Function { $( ( $param_name:tt : $param_type:tt ) )* } Extract { $($extract:tt)* } ) => { #[allow(non_camel_case_types)] pub struct $export; impl Command for $export { fn run(&self, $args: CommandArgs) -> Result { fn command($args: CommandArgs, ( $($param_name),*, ): ( $($param_type),*, )) -> Result { let output = $body; Ok(output.boxed().to_output_stream()) } let tuple = ( $($extract),*, ); command( $args, tuple ) } fn name(&self) -> &str { stringify!($config_name) } fn config(&self) -> $crate::parser::registry::CommandConfig { $crate::parser::registry::CommandConfig { name: self.name().to_string(), positional: vec![$($mandatory_positional)*], rest_positional: false, named: { use $crate::parser::registry::NamedType; #[allow(unused_mut)] let mut named: indexmap::IndexMap = indexmap::IndexMap::new(); $( named.insert(stringify!($named_param).to_string(), NamedType::$named_type); )* named } } } } }; // switch ( Named { $export:tt $args:ident $body:block } Positional { $($positional_count:tt)* } Rest { , -- $param_name:ident : Switch $($rest:tt)* } CommandConfig { name: $config_name:tt, mandatory_positional: vec![ $($mandatory_positional:tt)* ], optional_positional: vec![ $($optional_positional:tt)* ], rest_positional: $rest_positional:tt, named: { $($config_named:tt)* } } Function { $($function:tt)* } Extract { $($extract:tt)* } ) => { command!( Named { $export $args $body } Positional { $($positional_count)* + 1 } Rest { $($rest)* } CommandConfig { name: $config_name, mandatory_positional: vec![ $($mandatory_positional)* ], optional_positional: vec![ $($optional_positional)* ], rest_positional: $rest_positional, named: { $($config_named)* ($param_name : Switch) } } Function { $($function)* ($param_name : Switch) } Extract { ($($extract)* { use std::convert::TryInto; $args.get(stringify!($param_name)).clone().try_into()? }) } ); }; // mandatory named arguments ( Named { $export:tt $args:ident $body:block } Positional { $($positional_count:tt)* } Rest { , -- $param_name:ident : $param_kind:tt $($rest:tt)* } CommandConfig { name: $config_name:tt, mandatory_positional: vec![ $($mandatory_positional:tt)* ], optional_positional: vec![ $($optional_positional:tt)* ], rest_positional: $rest_positional:tt, named: { $($config_named:tt)* } } Function { $($function:tt)* } Extract { $($extract:tt)* } ) => { command!( Named { $export $args $body } Positional { $($positional_count)* + 1 } Rest { $($rest)* } CommandConfig { name: $config_name, mandatory_positional: vec![ $($mandatory_positional)* ], optional_positional: vec![ $($optional_positional)* ], rest_positional: $rest_positional, named: { $($config_named)* ($param_name : Mandatory(NamedValue::Single)) } } Function { $($function)* ($param_name : $param_kind) } Extract { ($($extract)* { use std::convert::TryInto; $args.get(stringify!($param_name)).clone().try_into()? }) } ); }; // optional named arguments ( Named { $export:tt $args:ident $body:block } Positional { $($positional_count:tt)* } Rest { , -- $param_name:ident ? : $param_kind:tt $($rest:tt)* } CommandConfig { name: $config_name:tt, mandatory_positional: vec![ $($mandatory_positional:tt)* ], optional_positional: vec![ $($optional_positional:tt)* ], rest_positional: $rest_positional:tt, named: { $($config_named:tt)* } } Function { $($function:tt)* } Extract { $($extract:tt)* } ) => { command!( Named { $export $args $body } Positional { $($positional_count)* + 1 } Rest { $($rest)* } CommandConfig { name: $config_name, mandatory_positional: vec![ $($mandatory_positional)* ], optional_positional: vec![ $($optional_positional)* ], rest_positional: $rest_positional, named: { $($config_named)* ($param_name : Optional(NamedValue::Single)) } } Function { $($function)* ($param_name : $param_kind) } Extract { ($($extract)* { use std::convert::TryInto; $args.get(stringify!($param_name)).clone().try_into()? }) } ); }; // mandatory positional argument ( Named { $export:ident $args:ident $body:block } Positional { $($positional_count:tt)* } Rest { , $param_name:ident : $param_kind:tt $($rest:tt)* } CommandConfig { name: $config_name:tt, mandatory_positional: vec![ $($mandatory_positional:tt)* ], optional_positional: vec![ $($optional_positional:tt)* ], rest_positional: $rest_positional:tt, named: { $($config_named:tt)* } } Function { $($function:tt)* } Extract { $($extract:tt)* } ) => { command!( Named { $export $args $body } Positional { $($positional_count)* + 1 } Rest { $($rest)* } CommandConfig { name: $config_name, mandatory_positional: vec![ $($mandatory_positional)* $crate::parser::registry::PositionalType::mandatory( stringify!($param_name), stringify!($param_kind) ), ], optional_positional: vec![ $($optional_positional)* ], rest_positional: $rest_positional, named: { $($config_named)* } } Function { $($function)* ($param_name : $param_kind) } Extract { $($extract:tt)* { use $crate::object::types::ExtractType; let value = $args.expect_nth($($positional_count)*)?; // let value = $param_kind.check(value)?; $param_kind::extract(value)? } } ); }; ($export:ident as $config_name:tt ( $args:ident $($command_rest:tt)* ) $body:block) => { command!( Named { $export $args $body } Positional { 0 } Rest { $($command_rest)* } CommandConfig { name: $config_name, mandatory_positional: vec![], optional_positional: vec![], rest_positional: false, named: {} } Function { } Extract { } ); }; // ($export:ident as $name:tt ( $args:ident, -- $param:ident : $kind:ident ) $body:block) => { // #[allow(non_camel_case_types)] // pub struct $export; // impl Command for $export { // fn run(&self, $args: CommandArgs) -> Result { // fn command($args: CommandArgs, $param: $kind) -> Result { // $body // } // use std::convert::TryInto; // let param = $args.get(stringify!($param)).try_into()?; // command($args, param) // } // fn name(&self) -> &str { // stringify!($name) // } // fn config(&self) -> CommandConfig { // let mut named: IndexMap = IndexMap::new(); // named.insert(stringify!($param).to_string(), NamedType::$kind); // CommandConfig { // name: self.name().to_string(), // mandatory_positional: vec![], // optional_positional: vec![], // rest_positional: false, // named, // } // } // } // }; }