use std::fmt::Display; use serde::{Deserialize, Serialize}; use crate::{DeclId, Type}; /// The syntactic shapes that values must match to be passed into a command. You can think of this as the type-checking that occurs when you call a function. #[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] pub enum SyntaxShape { /// Any syntactic form is allowed Any, /// A binary literal Binary, /// A block is allowed, eg `{start this thing}` Block, /// A boolean value, eg `true` or `false` Boolean, /// A dotted path to navigate the table CellPath, /// A closure is allowed, eg `{|| start this thing}` Closure(Option>), /// A custom shape with custom completion logic Custom(Box, DeclId), /// A datetime value, eg `2022-02-02` or `2019-10-12T07:20:50.52+00:00` DateTime, /// A directory is allowed Directory, /// A duration value is allowed, eg `19day` Duration, /// An error value Error, /// A general expression, eg `1 + 2` or `foo --bar` Expression, /// A filepath is allowed Filepath, /// A filesize value is allowed, eg `10kb` Filesize, /// A floating point value, eg `1.0` Float, /// A dotted path to navigate the table (including variable) FullCellPath, /// A glob pattern is allowed, eg `foo*` GlobPattern, /// Only an integer value is allowed Int, /// A module path pattern used for imports ImportPattern, /// A specific match to a word or symbol Keyword(Vec, Box), /// A list is allowed, eg `[first second]` List(Box), /// A general math expression, eg `1 + 2` MathExpression, /// A block of matches, used by `match` MatchBlock, /// A match pattern, eg `{a: $foo}` MatchPattern, /// Nothing Nothing, /// Only a numeric (integer or float) value is allowed Number, /// One of a list of possible items, checked in order OneOf(Vec), /// An operator, eg `+` Operator, /// A range is allowed (eg, `1..3`) Range, /// A record value, eg `{x: 1, y: 2}` Record(Vec<(String, SyntaxShape)>), /// A math expression which expands shorthand forms on the lefthand side, eg `foo > 1` /// The shorthand allows us to more easily reach columns inside of the row being passed in RowCondition, /// A signature for a definition, `[x:int, --foo]` Signature, /// Strings and string-like bare words are allowed String, /// A table is allowed, eg `[[first, second]; [1, 2]]` Table(Vec<(String, SyntaxShape)>), /// A variable with optional type, `x` or `x: int` VarWithOptType, } impl SyntaxShape { pub fn to_type(&self) -> Type { let mk_ty = |tys: &[(String, SyntaxShape)]| { tys.iter() .map(|(key, val)| (key.clone(), val.to_type())) .collect() }; match self { SyntaxShape::Any => Type::Any, SyntaxShape::Block => Type::Block, SyntaxShape::Closure(_) => Type::Closure, SyntaxShape::Binary => Type::Binary, SyntaxShape::CellPath => Type::Any, SyntaxShape::Custom(custom, _) => custom.to_type(), SyntaxShape::DateTime => Type::Date, SyntaxShape::Duration => Type::Duration, SyntaxShape::Expression => Type::Any, SyntaxShape::Filepath => Type::String, SyntaxShape::Directory => Type::String, SyntaxShape::Float => Type::Float, SyntaxShape::Filesize => Type::Filesize, SyntaxShape::FullCellPath => Type::Any, SyntaxShape::GlobPattern => Type::String, SyntaxShape::Error => Type::Error, SyntaxShape::ImportPattern => Type::Any, SyntaxShape::Int => Type::Int, SyntaxShape::List(x) => { let contents = x.to_type(); Type::List(Box::new(contents)) } SyntaxShape::Keyword(_, expr) => expr.to_type(), SyntaxShape::MatchBlock => Type::Any, SyntaxShape::MatchPattern => Type::Any, SyntaxShape::MathExpression => Type::Any, SyntaxShape::Nothing => Type::Nothing, SyntaxShape::Number => Type::Number, SyntaxShape::OneOf(_) => Type::Any, SyntaxShape::Operator => Type::Any, SyntaxShape::Range => Type::Range, SyntaxShape::Record(entries) => Type::Record(mk_ty(entries)), SyntaxShape::RowCondition => Type::Bool, SyntaxShape::Boolean => Type::Bool, SyntaxShape::Signature => Type::Signature, SyntaxShape::String => Type::String, SyntaxShape::Table(columns) => Type::Table(mk_ty(columns)), SyntaxShape::VarWithOptType => Type::Any, } } } impl Display for SyntaxShape { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { let mk_fmt = |tys: &[(String, SyntaxShape)]| -> String { tys.iter() .map(|(x, y)| format!("{x}: {y}")) .collect::>() .join(", ") }; match self { SyntaxShape::Keyword(kw, shape) => { write!(f, "\"{}\" {}", String::from_utf8_lossy(kw), shape) } SyntaxShape::Any => write!(f, "any"), SyntaxShape::String => write!(f, "string"), SyntaxShape::CellPath => write!(f, "cell-path"), SyntaxShape::FullCellPath => write!(f, "cell-path"), SyntaxShape::Number => write!(f, "number"), SyntaxShape::Range => write!(f, "range"), SyntaxShape::Int => write!(f, "int"), SyntaxShape::Float => write!(f, "float"), SyntaxShape::Filepath => write!(f, "path"), SyntaxShape::Directory => write!(f, "directory"), SyntaxShape::GlobPattern => write!(f, "glob"), SyntaxShape::ImportPattern => write!(f, "import"), SyntaxShape::Block => write!(f, "block"), SyntaxShape::Closure(args) => { if let Some(args) = args { let arg_vec: Vec<_> = args.iter().map(|x| x.to_string()).collect(); let arg_string = arg_vec.join(", "); write!(f, "closure({arg_string})") } else { write!(f, "closure()") } } SyntaxShape::Binary => write!(f, "binary"), SyntaxShape::List(x) => write!(f, "list<{x}>"), SyntaxShape::Table(columns) => { if columns.is_empty() { write!(f, "table") } else { write!(f, "table<{}>", mk_fmt(columns)) } } SyntaxShape::Record(entries) => { if entries.is_empty() { write!(f, "record") } else { write!(f, "record<{}>", mk_fmt(entries)) } } SyntaxShape::Filesize => write!(f, "filesize"), SyntaxShape::Duration => write!(f, "duration"), SyntaxShape::DateTime => write!(f, "datetime"), SyntaxShape::Operator => write!(f, "operator"), SyntaxShape::RowCondition => write!(f, "condition"), SyntaxShape::MathExpression => write!(f, "variable"), SyntaxShape::VarWithOptType => write!(f, "vardecl"), SyntaxShape::Signature => write!(f, "signature"), SyntaxShape::MatchPattern => write!(f, "match-pattern"), SyntaxShape::MatchBlock => write!(f, "match-block"), SyntaxShape::Expression => write!(f, "expression"), SyntaxShape::Boolean => write!(f, "bool"), SyntaxShape::Error => write!(f, "error"), SyntaxShape::Custom(x, _) => write!(f, "custom<{x}>"), SyntaxShape::OneOf(list) => { let arg_vec: Vec<_> = list.iter().map(|x| x.to_string()).collect(); let arg_string = arg_vec.join(", "); write!(f, "one_of({arg_string})") } SyntaxShape::Nothing => write!(f, "nothing"), } } }