2019-07-23 22:22:11 +00:00
|
|
|
// TODO: Temporary redirect
|
|
|
|
crate use crate::context::CommandRegistry;
|
2019-06-22 01:36:57 +00:00
|
|
|
use crate::evaluate::{evaluate_baseline_expr, Scope};
|
2019-07-15 21:16:27 +00:00
|
|
|
use crate::parser::{hir, hir::SyntaxType, parse_command, CallNode, Spanned};
|
2019-05-28 06:45:18 +00:00
|
|
|
use crate::prelude::*;
|
2019-06-22 01:36:57 +00:00
|
|
|
use derive_new::new;
|
2019-05-26 06:54:41 +00:00
|
|
|
use indexmap::IndexMap;
|
2019-06-22 01:36:57 +00:00
|
|
|
use log::trace;
|
2019-07-02 07:56:20 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2019-06-22 01:36:57 +00:00
|
|
|
use std::fmt;
|
2019-05-26 06:54:41 +00:00
|
|
|
|
|
|
|
#[allow(unused)]
|
2019-07-16 07:08:35 +00:00
|
|
|
#[derive(Debug, Serialize, Deserialize, Clone)]
|
2019-05-28 06:45:18 +00:00
|
|
|
pub enum NamedType {
|
2019-06-01 05:50:16 +00:00
|
|
|
Switch,
|
2019-07-15 21:16:27 +00:00
|
|
|
Mandatory(SyntaxType),
|
|
|
|
Optional(SyntaxType),
|
2019-05-26 06:54:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unused)]
|
2019-07-02 07:56:20 +00:00
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
2019-05-28 06:45:18 +00:00
|
|
|
pub enum PositionalType {
|
2019-07-15 21:16:27 +00:00
|
|
|
Mandatory(String, SyntaxType),
|
|
|
|
Optional(String, SyntaxType),
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl PositionalType {
|
2019-07-15 21:16:27 +00:00
|
|
|
pub fn mandatory(name: &str, ty: SyntaxType) -> PositionalType {
|
|
|
|
PositionalType::Mandatory(name.to_string(), ty)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn mandatory_any(name: &str) -> PositionalType {
|
|
|
|
PositionalType::Mandatory(name.to_string(), SyntaxType::Any)
|
2019-07-13 02:07:06 +00:00
|
|
|
}
|
|
|
|
|
2019-07-16 19:10:25 +00:00
|
|
|
pub fn mandatory_block(name: &str) -> PositionalType {
|
|
|
|
PositionalType::Mandatory(name.to_string(), SyntaxType::Block)
|
|
|
|
}
|
|
|
|
|
2019-07-16 07:25:48 +00:00
|
|
|
pub fn optional(name: &str, ty: SyntaxType) -> PositionalType {
|
|
|
|
PositionalType::Optional(name.to_string(), ty)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn optional_any(name: &str) -> PositionalType {
|
|
|
|
PositionalType::Optional(name.to_string(), SyntaxType::Any)
|
|
|
|
}
|
|
|
|
|
2019-07-16 07:08:35 +00:00
|
|
|
#[allow(unused)]
|
2019-07-15 21:16:27 +00:00
|
|
|
crate fn to_coerce_hint(&self) -> Option<SyntaxType> {
|
2019-05-28 06:45:18 +00:00
|
|
|
match self {
|
2019-07-15 21:16:27 +00:00
|
|
|
PositionalType::Mandatory(_, SyntaxType::Block)
|
|
|
|
| PositionalType::Optional(_, SyntaxType::Block) => Some(SyntaxType::Block),
|
2019-07-03 20:31:15 +00:00
|
|
|
_ => None,
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
}
|
2019-06-30 06:14:40 +00:00
|
|
|
|
|
|
|
crate fn name(&self) -> &str {
|
|
|
|
match self {
|
2019-07-03 20:31:15 +00:00
|
|
|
PositionalType::Mandatory(s, _) => s,
|
|
|
|
PositionalType::Optional(s, _) => s,
|
2019-06-30 06:14:40 +00:00
|
|
|
}
|
|
|
|
}
|
2019-07-15 21:16:27 +00:00
|
|
|
|
|
|
|
crate fn syntax_type(&self) -> SyntaxType {
|
|
|
|
match *self {
|
|
|
|
PositionalType::Mandatory(_, t) => t,
|
|
|
|
PositionalType::Optional(_, t) => t,
|
|
|
|
}
|
|
|
|
}
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
|
2019-07-24 04:10:48 +00:00
|
|
|
#[derive(Debug, Serialize, Deserialize, Clone)]
|
2019-05-26 06:54:41 +00:00
|
|
|
pub struct CommandConfig {
|
2019-07-02 07:56:20 +00:00
|
|
|
pub name: String,
|
2019-07-13 02:07:06 +00:00
|
|
|
pub positional: Vec<PositionalType>,
|
2019-07-02 07:56:20 +00:00
|
|
|
pub rest_positional: bool,
|
|
|
|
pub named: IndexMap<String, NamedType>,
|
|
|
|
pub is_filter: bool,
|
|
|
|
pub is_sink: bool,
|
2019-05-28 06:45:18 +00:00
|
|
|
}
|
|
|
|
|
2019-07-24 04:10:48 +00:00
|
|
|
impl CommandConfig {
|
|
|
|
pub fn new(name: impl Into<String>) -> CommandConfig {
|
|
|
|
CommandConfig {
|
|
|
|
name: name.into(),
|
|
|
|
positional: vec![],
|
|
|
|
rest_positional: false,
|
|
|
|
named: IndexMap::default(),
|
|
|
|
is_filter: false,
|
|
|
|
is_sink: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn required(mut self, name: impl Into<String>, ty: impl Into<SyntaxType>) -> CommandConfig {
|
|
|
|
self.positional
|
|
|
|
.push(PositionalType::Mandatory(name.into(), ty.into()));
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn optional(mut self, name: impl Into<String>, ty: impl Into<SyntaxType>) -> CommandConfig {
|
|
|
|
self.positional
|
|
|
|
.push(PositionalType::Optional(name.into(), ty.into()));
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn named(mut self, name: impl Into<String>, ty: impl Into<NamedType>) -> CommandConfig {
|
|
|
|
self.named.insert(name.into(), ty.into());
|
|
|
|
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn sink(mut self) -> CommandConfig {
|
|
|
|
self.is_sink = true;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-02 07:56:20 +00:00
|
|
|
#[derive(Debug, Default, new, Serialize, Deserialize)]
|
2019-07-23 22:22:11 +00:00
|
|
|
pub struct EvaluatedArgs {
|
2019-06-22 01:36:57 +00:00
|
|
|
pub positional: Option<Vec<Spanned<Value>>>,
|
|
|
|
pub named: Option<IndexMap<String, Spanned<Value>>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(new)]
|
2019-07-23 22:22:11 +00:00
|
|
|
pub struct DebugEvaluatedPositional<'a> {
|
2019-06-22 01:36:57 +00:00
|
|
|
positional: &'a Option<Vec<Spanned<Value>>>,
|
|
|
|
}
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
impl fmt::Debug for DebugEvaluatedPositional<'a> {
|
2019-06-22 01:36:57 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match &self.positional {
|
|
|
|
None => write!(f, "None"),
|
|
|
|
Some(positional) => f
|
|
|
|
.debug_list()
|
2019-07-09 04:31:26 +00:00
|
|
|
.entries(positional.iter().map(|p| p.debug()))
|
2019-06-22 01:36:57 +00:00
|
|
|
.finish(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(new)]
|
2019-07-23 22:22:11 +00:00
|
|
|
pub struct DebugEvaluatedNamed<'a> {
|
2019-06-22 01:36:57 +00:00
|
|
|
named: &'a Option<IndexMap<String, Spanned<Value>>>,
|
|
|
|
}
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
impl fmt::Debug for DebugEvaluatedNamed<'a> {
|
2019-06-22 01:36:57 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match &self.named {
|
|
|
|
None => write!(f, "None"),
|
|
|
|
Some(named) => f
|
|
|
|
.debug_map()
|
2019-07-09 04:31:26 +00:00
|
|
|
.entries(named.iter().map(|(k, v)| (k, v.debug())))
|
2019-06-22 01:36:57 +00:00
|
|
|
.finish(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
pub struct DebugEvaluatedArgs<'a> {
|
|
|
|
args: &'a EvaluatedArgs,
|
2019-06-22 01:36:57 +00:00
|
|
|
}
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
impl fmt::Debug for DebugEvaluatedArgs<'a> {
|
2019-06-22 01:36:57 +00:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
let mut s = f.debug_struct("Args");
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
s.field(
|
|
|
|
"positional",
|
|
|
|
&DebugEvaluatedPositional::new(&self.args.positional),
|
|
|
|
);
|
|
|
|
s.field("named", &DebugEvaluatedNamed::new(&self.args.named));
|
2019-06-22 01:36:57 +00:00
|
|
|
|
|
|
|
s.finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
impl EvaluatedArgs {
|
|
|
|
pub fn debug(&'a self) -> DebugEvaluatedArgs<'a> {
|
|
|
|
DebugEvaluatedArgs { args: self }
|
2019-06-22 01:36:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn nth(&self, pos: usize) -> Option<&Spanned<Value>> {
|
|
|
|
match &self.positional {
|
|
|
|
None => None,
|
|
|
|
Some(array) => array.iter().nth(pos),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn expect_nth(&self, pos: usize) -> Result<&Spanned<Value>, ShellError> {
|
|
|
|
match &self.positional {
|
|
|
|
None => Err(ShellError::unimplemented("Better error: expect_nth")),
|
|
|
|
Some(array) => match array.iter().nth(pos) {
|
|
|
|
None => Err(ShellError::unimplemented("Better error: expect_nth")),
|
|
|
|
Some(item) => Ok(item),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn len(&self) -> usize {
|
|
|
|
match &self.positional {
|
|
|
|
None => 0,
|
|
|
|
Some(array) => array.len(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn has(&self, name: &str) -> bool {
|
|
|
|
match &self.named {
|
|
|
|
None => false,
|
|
|
|
Some(named) => named.contains_key(name),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get(&self, name: &str) -> Option<&Spanned<Value>> {
|
|
|
|
match &self.named {
|
|
|
|
None => None,
|
|
|
|
Some(named) => named.get(name),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn positional_iter(&'a self) -> PositionalIter<'a> {
|
|
|
|
match &self.positional {
|
|
|
|
None => PositionalIter::Empty,
|
|
|
|
Some(v) => {
|
|
|
|
let iter = v.iter();
|
|
|
|
PositionalIter::Array(iter)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum PositionalIter<'a> {
|
|
|
|
Empty,
|
|
|
|
Array(std::slice::Iter<'a, Spanned<Value>>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Iterator for PositionalIter<'a> {
|
|
|
|
type Item = &'a Spanned<Value>;
|
|
|
|
|
|
|
|
fn next(&mut self) -> Option<Self::Item> {
|
|
|
|
match self {
|
|
|
|
PositionalIter::Empty => None,
|
|
|
|
PositionalIter::Array(iter) => iter.next(),
|
|
|
|
}
|
|
|
|
}
|
2019-06-01 05:50:16 +00:00
|
|
|
}
|
|
|
|
|
2019-05-28 06:45:18 +00:00
|
|
|
impl CommandConfig {
|
2019-07-23 22:22:11 +00:00
|
|
|
crate fn parse_args(
|
2019-05-28 06:45:18 +00:00
|
|
|
&self,
|
2019-06-22 01:36:57 +00:00
|
|
|
call: &Spanned<CallNode>,
|
2019-07-23 22:22:11 +00:00
|
|
|
registry: &CommandRegistry,
|
2019-06-22 20:46:16 +00:00
|
|
|
source: &Text,
|
2019-07-23 22:22:11 +00:00
|
|
|
) -> Result<hir::Call, ShellError> {
|
2019-06-22 01:36:57 +00:00
|
|
|
let args = parse_command(self, registry, call, source)?;
|
2019-06-01 05:50:16 +00:00
|
|
|
|
2019-06-22 01:36:57 +00:00
|
|
|
trace!("parsed args: {:?}", args);
|
2019-06-01 05:50:16 +00:00
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
Ok(args)
|
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-07-23 22:22:11 +00:00
|
|
|
crate fn evaluate_args(
|
|
|
|
call: &hir::Call,
|
|
|
|
registry: &CommandRegistry,
|
2019-06-22 01:36:57 +00:00
|
|
|
scope: &Scope,
|
2019-06-22 20:46:16 +00:00
|
|
|
source: &Text,
|
2019-07-23 22:22:11 +00:00
|
|
|
) -> Result<EvaluatedArgs, ShellError> {
|
|
|
|
let positional: Result<Option<Vec<_>>, _> = call
|
2019-06-22 01:36:57 +00:00
|
|
|
.positional()
|
|
|
|
.as_ref()
|
|
|
|
.map(|p| {
|
|
|
|
p.iter()
|
2019-07-23 22:22:11 +00:00
|
|
|
.map(|e| evaluate_baseline_expr(e, &CommandRegistry::empty(), scope, source))
|
2019-06-22 01:36:57 +00:00
|
|
|
.collect()
|
|
|
|
})
|
|
|
|
.transpose();
|
|
|
|
|
|
|
|
let positional = positional?;
|
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
let named: Result<Option<IndexMap<String, Spanned<Value>>>, ShellError> = call
|
2019-06-22 01:36:57 +00:00
|
|
|
.named()
|
|
|
|
.as_ref()
|
|
|
|
.map(|n| {
|
|
|
|
let mut results = IndexMap::new();
|
|
|
|
|
|
|
|
for (name, value) in n.named.iter() {
|
|
|
|
match value {
|
|
|
|
hir::named::NamedValue::PresentSwitch(span) => {
|
|
|
|
results.insert(
|
|
|
|
name.clone(),
|
|
|
|
Spanned::from_item(Value::boolean(true), *span),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
hir::named::NamedValue::Value(expr) => {
|
|
|
|
results.insert(
|
|
|
|
name.clone(),
|
|
|
|
evaluate_baseline_expr(expr, registry, scope, source)?,
|
|
|
|
);
|
|
|
|
}
|
2019-06-01 05:50:16 +00:00
|
|
|
|
2019-06-22 01:36:57 +00:00
|
|
|
_ => {}
|
|
|
|
};
|
|
|
|
}
|
2019-06-01 05:50:16 +00:00
|
|
|
|
2019-06-22 01:36:57 +00:00
|
|
|
Ok(results)
|
|
|
|
})
|
|
|
|
.transpose();
|
2019-06-01 05:50:16 +00:00
|
|
|
|
2019-06-22 01:36:57 +00:00
|
|
|
let named = named?;
|
2019-06-01 05:50:16 +00:00
|
|
|
|
2019-07-23 22:22:11 +00:00
|
|
|
Ok(EvaluatedArgs::new(positional, named))
|
2019-05-26 06:54:41 +00:00
|
|
|
}
|