use nu_protocol::ast::{Block, Expr, Expression, PathMember, Pipeline, Statement}; use nu_protocol::{engine::StateWorkingSet, Span}; #[derive(Debug)] pub enum FlatShape { Garbage, Bool, Int, Float, Range, InternalCall, External, ExternalArg, Literal, Operator, Signature, String, Variable, Custom(String), } pub fn flatten_block(working_set: &StateWorkingSet, block: &Block) -> Vec<(Span, FlatShape)> { let mut output = vec![]; for stmt in &block.stmts { output.extend(flatten_statement(working_set, stmt)); } output } pub fn flatten_statement( working_set: &StateWorkingSet, stmt: &Statement, ) -> Vec<(Span, FlatShape)> { match stmt { Statement::Pipeline(pipeline) => flatten_pipeline(working_set, pipeline), _ => vec![], } } pub fn flatten_expression( working_set: &StateWorkingSet, expr: &Expression, ) -> Vec<(Span, FlatShape)> { if let Some(custom_completion) = &expr.custom_completion { return vec![(expr.span, FlatShape::Custom(custom_completion.clone()))]; } match &expr.expr { Expr::BinaryOp(lhs, op, rhs) => { let mut output = vec![]; output.extend(flatten_expression(working_set, lhs)); output.extend(flatten_expression(working_set, op)); output.extend(flatten_expression(working_set, rhs)); output } Expr::Block(block_id) => flatten_block(working_set, working_set.get_block(*block_id)), Expr::Call(call) => { let mut output = vec![(call.head, FlatShape::InternalCall)]; for positional in &call.positional { output.extend(flatten_expression(working_set, positional)); } output } Expr::ExternalCall(name, args) => { let mut output = vec![(*name, FlatShape::External)]; for arg in args { output.push((*arg, FlatShape::ExternalArg)); } output } Expr::Garbage => { vec![(expr.span, FlatShape::Garbage)] } Expr::Int(_) => { vec![(expr.span, FlatShape::Int)] } Expr::Float(_) => { vec![(expr.span, FlatShape::Float)] } Expr::FullCellPath(column_path) => { let mut output = vec![]; output.extend(flatten_expression(working_set, &column_path.head)); for path_element in &column_path.tail { match path_element { PathMember::String { span, .. } => output.push((*span, FlatShape::String)), PathMember::Int { span, .. } => output.push((*span, FlatShape::Int)), } } output } Expr::Range(from, next, to, op) => { let mut output = vec![]; if let Some(f) = from { output.extend(flatten_expression(working_set, f)); } if let Some(s) = next { output.extend(vec![(op.next_op_span, FlatShape::Operator)]); output.extend(flatten_expression(working_set, s)); } output.extend(vec![(op.span, FlatShape::Operator)]); if let Some(t) = to { output.extend(flatten_expression(working_set, t)); } output } Expr::Bool(_) => { vec![(expr.span, FlatShape::Bool)] } Expr::List(list) => { let mut output = vec![]; for l in list { output.extend(flatten_expression(working_set, l)); } output } Expr::Keyword(_, span, expr) => { let mut output = vec![(*span, FlatShape::Operator)]; output.extend(flatten_expression(working_set, expr)); output } Expr::Operator(_) => { vec![(expr.span, FlatShape::Operator)] } Expr::Signature(_) => { vec![(expr.span, FlatShape::Signature)] } Expr::String(_) => { vec![(expr.span, FlatShape::String)] } Expr::RowCondition(_, expr) => flatten_expression(working_set, expr), Expr::Subexpression(block_id) => { flatten_block(working_set, working_set.get_block(*block_id)) } Expr::Table(headers, cells) => { let mut output = vec![]; for e in headers { output.extend(flatten_expression(working_set, e)); } for row in cells { for expr in row { output.extend(flatten_expression(working_set, expr)); } } output } Expr::Var(_) => { vec![(expr.span, FlatShape::Variable)] } } } pub fn flatten_pipeline( working_set: &StateWorkingSet, pipeline: &Pipeline, ) -> Vec<(Span, FlatShape)> { let mut output = vec![]; for expr in &pipeline.expressions { output.extend(flatten_expression(working_set, expr)) } output }