2019-05-28 06:45:18 +00:00
|
|
|
use crate::evaluate::{evaluate_expr, Scope};
|
|
|
|
use crate::prelude::*;
|
2019-05-26 06:54:41 +00:00
|
|
|
use indexmap::IndexMap;
|
|
|
|
|
|
|
|
#[allow(unused)]
|
2019-05-28 06:45:18 +00:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum NamedType {
|
2019-06-01 05:50:16 +00:00
|
|
|
Switch,
|
|
|
|
Mandatory(NamedValue),
|
|
|
|
Optional(NamedValue),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum NamedValue {
|
|
|
|
Single,
|
|
|
|
Tuple,
|
|
|
|
|
|
|
|
#[allow(unused)]
|
|
|
|
Block,
|
|
|
|
|
|
|
|
#[allow(unused)]
|
|
|
|
Array,
|
2019-05-26 06:54:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
2019-05-28 06:45:18 +00:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub enum PositionalType {
|
|
|
|
Value(String),
|
|
|
|
Block(String),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PositionalType {
|
|
|
|
crate fn name(&self) -> String {
|
|
|
|
match self {
|
|
|
|
PositionalType::Value(s) => s.clone(),
|
|
|
|
PositionalType::Block(s) => s.clone(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
crate fn evaluate(&self, arg: ast::Expression, scope: &Scope) -> Result<Value, ShellError> {
|
|
|
|
match self {
|
|
|
|
PositionalType::Value(_) => evaluate_expr(&arg, scope),
|
|
|
|
PositionalType::Block(_) => match arg {
|
|
|
|
ast::Expression::Block(b) => Ok(Value::block(b.expr)),
|
|
|
|
ast::Expression::Binary(b) => {
|
|
|
|
if let Some(s) = b.left.as_string() {
|
|
|
|
Ok(Value::block(ast::Expression::Binary(Box::new(
|
|
|
|
ast::Binary::new(
|
|
|
|
ast::Expression::Path(Box::new(ast::Path::new(
|
|
|
|
ast::Expression::VariableReference(ast::Variable::It),
|
|
|
|
vec![s],
|
|
|
|
))),
|
|
|
|
b.operator,
|
|
|
|
b.right,
|
|
|
|
),
|
|
|
|
))))
|
|
|
|
} else {
|
|
|
|
Ok(Value::block(ast::Expression::Binary(b)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
other => Ok(Value::block(other)), // other =>
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
2019-05-26 06:54:41 +00:00
|
|
|
pub struct CommandConfig {
|
|
|
|
crate name: String,
|
2019-05-28 06:45:18 +00:00
|
|
|
crate mandatory_positional: Vec<PositionalType>,
|
|
|
|
crate optional_positional: Vec<PositionalType>,
|
2019-05-26 06:54:41 +00:00
|
|
|
crate rest_positional: bool,
|
2019-05-28 06:45:18 +00:00
|
|
|
crate named: IndexMap<String, NamedType>,
|
|
|
|
}
|
|
|
|
|
2019-06-04 21:42:31 +00:00
|
|
|
#[derive(Debug, Default)]
|
2019-06-01 05:50:16 +00:00
|
|
|
pub struct Args {
|
|
|
|
pub positional: Vec<Value>,
|
|
|
|
pub named: IndexMap<String, Value>,
|
|
|
|
}
|
|
|
|
|
2019-05-28 06:45:18 +00:00
|
|
|
impl CommandConfig {
|
|
|
|
crate fn evaluate_args(
|
|
|
|
&self,
|
2019-06-01 05:50:16 +00:00
|
|
|
args: impl Iterator<Item = &'expr ast::Expression>,
|
2019-05-28 06:45:18 +00:00
|
|
|
scope: &Scope,
|
2019-06-01 05:50:16 +00:00
|
|
|
) -> Result<Args, ShellError> {
|
|
|
|
let mut positional: Vec<Value> = vec![];
|
|
|
|
let mut named: IndexMap<String, Value> = IndexMap::default();
|
|
|
|
|
|
|
|
let mut args: Vec<ast::Expression> = args.cloned().collect();
|
|
|
|
|
|
|
|
for (key, ty) in self.named.iter() {
|
|
|
|
let index = args.iter().position(|a| a.is_flag(&key));
|
|
|
|
|
|
|
|
match (index, ty) {
|
|
|
|
(Some(i), NamedType::Switch) => {
|
|
|
|
args.remove(i);
|
|
|
|
named.insert(key.clone(), Value::boolean(true));
|
|
|
|
}
|
|
|
|
|
|
|
|
(None, NamedType::Switch) => {}
|
|
|
|
|
|
|
|
(Some(i), NamedType::Optional(v)) => {
|
|
|
|
args.remove(i);
|
|
|
|
named.insert(key.clone(), extract_named(&mut args, i, v)?);
|
|
|
|
}
|
|
|
|
|
|
|
|
(None, NamedType::Optional(_)) => {}
|
|
|
|
|
|
|
|
(Some(i), NamedType::Mandatory(v)) => {
|
|
|
|
args.remove(i);
|
|
|
|
named.insert(key.clone(), extract_named(&mut args, i, v)?);
|
|
|
|
}
|
|
|
|
|
|
|
|
(None, NamedType::Mandatory(_)) => {
|
|
|
|
return Err(ShellError::string(&format!(
|
|
|
|
"Expected mandatory argument {}, but it was missing",
|
|
|
|
key
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut args = args.into_iter();
|
2019-05-28 06:45:18 +00:00
|
|
|
|
|
|
|
for param in &self.mandatory_positional {
|
|
|
|
let arg = args.next();
|
|
|
|
|
|
|
|
let value = match arg {
|
|
|
|
None => {
|
|
|
|
return Err(ShellError::string(format!(
|
|
|
|
"expected mandatory positional argument {}",
|
|
|
|
param.name()
|
|
|
|
)))
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(arg) => param.evaluate(arg.clone(), scope)?,
|
|
|
|
};
|
|
|
|
|
2019-06-01 05:50:16 +00:00
|
|
|
positional.push(value);
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if self.rest_positional {
|
|
|
|
let rest: Result<Vec<Value>, _> =
|
2019-06-01 05:50:16 +00:00
|
|
|
args.map(|i| evaluate_expr(&i, &Scope::empty())).collect();
|
|
|
|
positional.extend(rest?);
|
2019-05-28 06:45:18 +00:00
|
|
|
} else {
|
2019-06-01 05:50:16 +00:00
|
|
|
let rest: Vec<ast::Expression> = args.collect();
|
|
|
|
|
|
|
|
if rest.len() > 0 {
|
|
|
|
return Err(ShellError::string(&format!(
|
|
|
|
"Too many arguments, extras: {:?}",
|
|
|
|
rest
|
|
|
|
)));
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-01 05:50:16 +00:00
|
|
|
Ok(Args { positional, named })
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
|
|
|
crate fn signature(&self) -> String {
|
|
|
|
format!("TODO")
|
|
|
|
}
|
2019-05-26 06:54:41 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 05:50:16 +00:00
|
|
|
fn extract_named(
|
|
|
|
v: &mut Vec<ast::Expression>,
|
|
|
|
position: usize,
|
|
|
|
ty: &NamedValue,
|
|
|
|
) -> Result<Value, ShellError> {
|
|
|
|
match ty {
|
|
|
|
NamedValue::Single => {
|
|
|
|
let expr = v.remove(position);
|
|
|
|
expect_simple_expr(expr)
|
|
|
|
}
|
|
|
|
|
|
|
|
NamedValue::Tuple => {
|
|
|
|
let expr = v.remove(position);
|
|
|
|
let next = v.remove(position);
|
|
|
|
|
|
|
|
let list = vec![expect_simple_expr(expr)?, expect_simple_expr(next)?];
|
|
|
|
Ok(Value::List(list))
|
|
|
|
}
|
|
|
|
|
|
|
|
other => Err(ShellError::string(&format!(
|
|
|
|
"Unimplemented named argument {:?}",
|
|
|
|
other
|
|
|
|
))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn expect_simple_expr(expr: ast::Expression) -> Result<Value, ShellError> {
|
|
|
|
match expr {
|
|
|
|
ast::Expression::Leaf(l) => Ok(match l {
|
|
|
|
ast::Leaf::Bare(s) => Value::string(s.to_string()),
|
|
|
|
ast::Leaf::String(s) => Value::string(s),
|
|
|
|
ast::Leaf::Boolean(b) => Value::boolean(b),
|
|
|
|
ast::Leaf::Int(i) => Value::int(i),
|
2019-06-02 16:28:40 +00:00
|
|
|
ast::Leaf::Unit(i, unit) => unit.compute(i),
|
2019-06-01 05:50:16 +00:00
|
|
|
}),
|
|
|
|
|
|
|
|
// TODO: Diagnostic
|
|
|
|
other => Err(ShellError::string(&format!(
|
|
|
|
"Expected a value, found {}",
|
|
|
|
other.print()
|
|
|
|
))),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-26 06:54:41 +00:00
|
|
|
pub trait CommandRegistry {
|
|
|
|
fn get(&self, name: &str) -> CommandConfig;
|
|
|
|
}
|