2019-07-03 20:31:15 +00:00
|
|
|
#[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<OutputStream, ShellError> {
|
|
|
|
fn command($args: CommandArgs, ( $($param_name),*, ): ( $($param_type),*, )) -> Result<OutputStream, ShellError> {
|
|
|
|
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,
|
2019-07-13 02:07:06 +00:00
|
|
|
is_filter: false,
|
|
|
|
is_sink: false,
|
|
|
|
|
2019-07-03 20:31:15 +00:00
|
|
|
named: {
|
2019-07-08 16:44:53 +00:00
|
|
|
use $crate::parser::registry::NamedType;
|
2019-07-03 20:31:15 +00:00
|
|
|
|
|
|
|
#[allow(unused_mut)]
|
|
|
|
let mut named: indexmap::IndexMap<String, NamedType> = 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()?
|
|
|
|
})
|
|
|
|
}
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2019-07-12 19:22:08 +00:00
|
|
|
// mandatory positional block
|
|
|
|
(
|
|
|
|
Named { $export:ident $args:ident $body:block }
|
|
|
|
Positional { $($positional_count:tt)* }
|
|
|
|
Rest { , $param_name:ident : Block $($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_block(
|
|
|
|
stringify!($param_name)
|
|
|
|
), ],
|
|
|
|
optional_positional: vec![ $($optional_positional)* ],
|
|
|
|
rest_positional: $rest_positional,
|
|
|
|
named: {
|
|
|
|
$($config_named)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Function {
|
|
|
|
$($function)* ($param_name : Block)
|
|
|
|
}
|
|
|
|
|
|
|
|
Extract {
|
|
|
|
$($extract:tt)* {
|
|
|
|
use $crate::object::types::ExtractType;
|
|
|
|
let value = $args.expect_nth($($positional_count)*)?;
|
|
|
|
Block::extract(value)?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2019-07-03 20:31:15 +00:00
|
|
|
// 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(
|
2019-07-12 19:22:08 +00:00
|
|
|
stringify!($param_name)
|
2019-07-03 20:31:15 +00:00
|
|
|
), ],
|
|
|
|
optional_positional: vec![ $($optional_positional)* ],
|
|
|
|
rest_positional: $rest_positional,
|
|
|
|
named: {
|
|
|
|
$($config_named)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Function {
|
|
|
|
$($function)* ($param_name : $param_kind)
|
|
|
|
}
|
|
|
|
|
|
|
|
Extract {
|
|
|
|
$($extract:tt)* {
|
2019-07-08 16:44:53 +00:00
|
|
|
use $crate::object::types::ExtractType;
|
2019-07-05 22:08:58 +00:00
|
|
|
let value = $args.expect_nth($($positional_count)*)?;
|
2019-07-08 16:44:53 +00:00
|
|
|
// let value = $param_kind.check(value)?;
|
|
|
|
$param_kind::extract(value)?
|
2019-07-03 20:31:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
($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<OutputStream, ShellError> {
|
|
|
|
// fn command($args: CommandArgs, $param: $kind) -> Result<OutputStream, ShellError> {
|
|
|
|
// $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<String, NamedType> = 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,
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// };
|
|
|
|
}
|