1use nu_protocol::{
2 DeclId, GetSpan, Span, SyntaxShape, VarId,
3 ast::{
4 Argument, Block, Expr, Expression, ExternalArgument, ImportPatternMember, ListItem,
5 MatchPattern, PathMember, Pattern, Pipeline, PipelineElement, PipelineRedirection,
6 RecordItem,
7 },
8 engine::StateWorkingSet,
9};
10use std::fmt::{Display, Formatter, Result};
11
12#[derive(Debug, Eq, PartialEq, Ord, Clone, PartialOrd)]
13pub enum FlatShape {
14 Binary,
15 Block,
16 Bool,
17 Closure,
18 Custom(DeclId),
19 DateTime,
20 Directory,
21 External(Box<Span>),
25 ExternalArg,
26 ExternalResolved,
27 Filepath,
28 Flag,
29 Float,
30 Garbage,
31 GlobInterpolation,
32 GlobPattern,
33 Int,
34 InternalCall(DeclId),
35 Keyword,
36 List,
37 Literal,
38 MatchPattern,
39 Nothing,
40 Operator,
41 Pipe,
42 Range,
43 RawString,
44 Record,
45 Redirection,
46 Signature,
47 String,
48 StringInterpolation,
49 Table,
50 Variable(VarId),
51 VarDecl(VarId),
52}
53
54impl FlatShape {
55 pub fn as_str(&self) -> &str {
56 match self {
57 FlatShape::Binary => "shape_binary",
58 FlatShape::Block => "shape_block",
59 FlatShape::Bool => "shape_bool",
60 FlatShape::Closure => "shape_closure",
61 FlatShape::Custom(_) => "shape_custom",
62 FlatShape::DateTime => "shape_datetime",
63 FlatShape::Directory => "shape_directory",
64 FlatShape::External(_) => "shape_external",
65 FlatShape::ExternalArg => "shape_externalarg",
66 FlatShape::ExternalResolved => "shape_external_resolved",
67 FlatShape::Filepath => "shape_filepath",
68 FlatShape::Flag => "shape_flag",
69 FlatShape::Float => "shape_float",
70 FlatShape::Garbage => "shape_garbage",
71 FlatShape::GlobInterpolation => "shape_glob_interpolation",
72 FlatShape::GlobPattern => "shape_globpattern",
73 FlatShape::Int => "shape_int",
74 FlatShape::InternalCall(_) => "shape_internalcall",
75 FlatShape::Keyword => "shape_keyword",
76 FlatShape::List => "shape_list",
77 FlatShape::Literal => "shape_literal",
78 FlatShape::MatchPattern => "shape_match_pattern",
79 FlatShape::Nothing => "shape_nothing",
80 FlatShape::Operator => "shape_operator",
81 FlatShape::Pipe => "shape_pipe",
82 FlatShape::Range => "shape_range",
83 FlatShape::RawString => "shape_raw_string",
84 FlatShape::Record => "shape_record",
85 FlatShape::Redirection => "shape_redirection",
86 FlatShape::Signature => "shape_signature",
87 FlatShape::String => "shape_string",
88 FlatShape::StringInterpolation => "shape_string_interpolation",
89 FlatShape::Table => "shape_table",
90 FlatShape::Variable(_) => "shape_variable",
91 FlatShape::VarDecl(_) => "shape_vardecl",
92 }
93 }
94}
95
96impl Display for FlatShape {
97 fn fmt(&self, f: &mut Formatter) -> Result {
98 f.write_str(self.as_str())
99 }
100}
101
102fn flatten_block_into(
110 working_set: &StateWorkingSet,
111 block: &Block,
112 output: &mut Vec<(Span, FlatShape)>,
113) {
114 for pipeline in &block.pipelines {
115 flatten_pipeline_into(working_set, pipeline, output);
116 }
117}
118
119fn flatten_pipeline_into(
120 working_set: &StateWorkingSet,
121 pipeline: &Pipeline,
122 output: &mut Vec<(Span, FlatShape)>,
123) {
124 for expr in &pipeline.elements {
125 flatten_pipeline_element_into(working_set, expr, output)
126 }
127}
128
129fn flatten_pipeline_element_into(
130 working_set: &StateWorkingSet,
131 pipeline_element: &PipelineElement,
132 output: &mut Vec<(Span, FlatShape)>,
133) {
134 if let Some(span) = pipeline_element.pipe {
135 output.push((span, FlatShape::Pipe));
136 }
137
138 flatten_expression_into(working_set, &pipeline_element.expr, output);
139
140 if let Some(redirection) = pipeline_element.redirection.as_ref() {
141 match redirection {
142 PipelineRedirection::Single { target, .. } => {
143 output.push((target.span(), FlatShape::Redirection));
144 if let Some(expr) = target.expr() {
145 flatten_expression_into(working_set, expr, output);
146 }
147 }
148 PipelineRedirection::Separate { out, err } => {
149 let (out, err) = if out.span() <= err.span() {
150 (out, err)
151 } else {
152 (err, out)
153 };
154
155 output.push((out.span(), FlatShape::Redirection));
156 if let Some(expr) = out.expr() {
157 flatten_expression_into(working_set, expr, output);
158 }
159 output.push((err.span(), FlatShape::Redirection));
160 if let Some(expr) = err.expr() {
161 flatten_expression_into(working_set, expr, output);
162 }
163 }
164 }
165 }
166}
167
168fn flatten_positional_arg_into(
169 working_set: &StateWorkingSet,
170 positional: &Expression,
171 shape: &SyntaxShape,
172 output: &mut Vec<(Span, FlatShape)>,
173) {
174 if matches!(shape, SyntaxShape::ExternalArgument)
175 && matches!(positional.expr, Expr::String(..) | Expr::GlobPattern(..))
176 {
177 output.push((positional.span, FlatShape::ExternalArg));
179 } else {
180 flatten_expression_into(working_set, positional, output)
181 }
182}
183
184fn flatten_expression_into(
185 working_set: &StateWorkingSet,
186 expr: &Expression,
187 output: &mut Vec<(Span, FlatShape)>,
188) {
189 match &expr.expr {
190 Expr::AttributeBlock(ab) => {
191 for attr in &ab.attributes {
192 flatten_expression_into(working_set, &attr.expr, output);
193 }
194 flatten_expression_into(working_set, &ab.item, output);
195 }
196 Expr::BinaryOp(lhs, op, rhs) => {
197 flatten_expression_into(working_set, lhs, output);
198 flatten_expression_into(working_set, op, output);
199 flatten_expression_into(working_set, rhs, output);
200 }
201 Expr::UnaryNot(not) => {
202 output.push((
203 Span::new(expr.span.start, expr.span.start + 3),
204 FlatShape::Operator,
205 ));
206 flatten_expression_into(working_set, not, output);
207 }
208 Expr::Collect(_, expr) => {
209 flatten_expression_into(working_set, expr, output);
210 }
211 Expr::Closure(block_id) => {
212 let outer_span = expr.span;
213
214 let block = working_set.get_block(*block_id);
215 let flattened = flatten_block(working_set, block);
216
217 if let Some(first) = flattened.first() {
218 if first.0.start > outer_span.start {
219 output.push((
220 Span::new(outer_span.start, first.0.start),
221 FlatShape::Closure,
222 ));
223 }
224 }
225
226 let last = if let Some(last) = flattened.last() {
227 if last.0.end < outer_span.end {
228 Some((Span::new(last.0.end, outer_span.end), FlatShape::Closure))
229 } else {
230 None
231 }
232 } else {
233 Some((outer_span, FlatShape::Closure))
235 };
236
237 output.extend(flattened);
238 if let Some(last) = last {
239 output.push(last);
240 }
241 }
242 Expr::Block(block_id) | Expr::RowCondition(block_id) | Expr::Subexpression(block_id) => {
243 let outer_span = expr.span;
244
245 let flattened = flatten_block(working_set, working_set.get_block(*block_id));
246
247 if let Some(first) = flattened.first() {
248 if first.0.start > outer_span.start {
249 output.push((Span::new(outer_span.start, first.0.start), FlatShape::Block));
250 }
251 }
252
253 let last = if let Some(last) = flattened.last() {
254 if last.0.end < outer_span.end {
255 Some((Span::new(last.0.end, outer_span.end), FlatShape::Block))
256 } else {
257 None
258 }
259 } else {
260 None
261 };
262
263 output.extend(flattened);
264 if let Some(last) = last {
265 output.push(last);
266 }
267 }
268 Expr::Call(call) => {
269 let decl = working_set.get_decl(call.decl_id);
270
271 if call.head.end != 0 {
272 output.push((call.head, FlatShape::InternalCall(call.decl_id)));
274 }
275
276 let signature = decl.signature();
278 let mut positional_args = signature
279 .required_positional
280 .iter()
281 .chain(&signature.optional_positional);
282
283 let arg_start = output.len();
284 for arg in &call.arguments {
285 match arg {
286 Argument::Positional(positional) => {
287 let positional_arg = positional_args.next();
288 let shape = positional_arg
289 .or(signature.rest_positional.as_ref())
290 .map(|arg| &arg.shape)
291 .unwrap_or(&SyntaxShape::Any);
292
293 flatten_positional_arg_into(working_set, positional, shape, output)
294 }
295 Argument::Unknown(positional) => {
296 let shape = signature
297 .rest_positional
298 .as_ref()
299 .map(|arg| &arg.shape)
300 .unwrap_or(&SyntaxShape::Any);
301
302 flatten_positional_arg_into(working_set, positional, shape, output)
303 }
304 Argument::Named(named) => {
305 if named.0.span.end != 0 {
306 output.push((named.0.span, FlatShape::Flag));
308 }
309 if let Some(expr) = &named.2 {
310 flatten_expression_into(working_set, expr, output);
311 }
312 }
313 Argument::Spread(expr) => {
314 output.push((
315 Span::new(expr.span.start - 3, expr.span.start),
316 FlatShape::Operator,
317 ));
318 flatten_expression_into(working_set, expr, output);
319 }
320 }
321 }
322 output[arg_start..].sort();
324 }
325 Expr::ExternalCall(head, args) => {
326 if let Expr::String(..) | Expr::GlobPattern(..) = &head.expr {
327 output.push((
328 head.span,
329 FlatShape::External(Box::new(working_set.get_span(head.span_id))),
336 ));
337 } else {
338 flatten_expression_into(working_set, head, output);
339 }
340
341 for arg in args.as_ref() {
342 match arg {
343 ExternalArgument::Regular(expr) => {
344 if let Expr::String(..) | Expr::GlobPattern(..) = &expr.expr {
345 output.push((expr.span, FlatShape::ExternalArg));
346 } else {
347 flatten_expression_into(working_set, expr, output);
348 }
349 }
350 ExternalArgument::Spread(expr) => {
351 output.push((
352 Span::new(expr.span.start - 3, expr.span.start),
353 FlatShape::Operator,
354 ));
355 flatten_expression_into(working_set, expr, output);
356 }
357 }
358 }
359 }
360 Expr::Garbage => output.push((expr.span, FlatShape::Garbage)),
361 Expr::Nothing => output.push((expr.span, FlatShape::Nothing)),
362 Expr::DateTime(_) => output.push((expr.span, FlatShape::DateTime)),
363 Expr::Binary(_) => output.push((expr.span, FlatShape::Binary)),
364 Expr::Int(_) => output.push((expr.span, FlatShape::Int)),
365 Expr::Float(_) => output.push((expr.span, FlatShape::Float)),
366 Expr::MatchBlock(matches) => {
367 for (pattern, expr) in matches {
368 flatten_pattern_into(pattern, output);
369 flatten_expression_into(working_set, expr, output);
370 }
371 }
372 Expr::ValueWithUnit(value) => {
373 flatten_expression_into(working_set, &value.expr, output);
374 output.push((value.unit.span, FlatShape::String));
375 }
376 Expr::CellPath(cell_path) => {
377 output.extend(cell_path.members.iter().map(|member| match *member {
378 PathMember::String { span, .. } => (span, FlatShape::String),
379 PathMember::Int { span, .. } => (span, FlatShape::Int),
380 }));
381 }
382 Expr::FullCellPath(cell_path) => {
383 flatten_expression_into(working_set, &cell_path.head, output);
384 output.extend(cell_path.tail.iter().map(|member| match *member {
385 PathMember::String { span, .. } => (span, FlatShape::String),
386 PathMember::Int { span, .. } => (span, FlatShape::Int),
387 }));
388 }
389 Expr::ImportPattern(import_pattern) => {
390 output.push((import_pattern.head.span, FlatShape::String));
391
392 for member in &import_pattern.members {
393 match member {
394 ImportPatternMember::Glob { span } => output.push((*span, FlatShape::String)),
395 ImportPatternMember::Name { span, .. } => {
396 output.push((*span, FlatShape::String))
397 }
398 ImportPatternMember::List { names } => {
399 output.extend(names.iter().map(|&(_, span)| (span, FlatShape::String)))
400 }
401 }
402 }
403 }
404 Expr::Overlay(_) => output.push((expr.span, FlatShape::String)),
405 Expr::Range(range) => {
406 if let Some(f) = &range.from {
407 flatten_expression_into(working_set, f, output);
408 }
409 if let Some(s) = &range.next {
410 output.push((range.operator.next_op_span, FlatShape::Operator));
411 flatten_expression_into(working_set, s, output);
412 }
413 output.push((range.operator.span, FlatShape::Operator));
414 if let Some(t) = &range.to {
415 flatten_expression_into(working_set, t, output);
416 }
417 }
418 Expr::Bool(_) => output.push((expr.span, FlatShape::Bool)),
419 Expr::Filepath(_, _) => output.push((expr.span, FlatShape::Filepath)),
420 Expr::Directory(_, _) => output.push((expr.span, FlatShape::Directory)),
421 Expr::GlobPattern(_, _) => output.push((expr.span, FlatShape::GlobPattern)),
422 Expr::List(list) => {
423 let outer_span = expr.span;
424 let mut last_end = outer_span.start;
425
426 for item in list {
427 match item {
428 ListItem::Item(expr) => {
429 let flattened = flatten_expression(working_set, expr);
430
431 if let Some(first) = flattened.first() {
432 if first.0.start > last_end {
433 output.push((Span::new(last_end, first.0.start), FlatShape::List));
434 }
435 }
436
437 if let Some(last) = flattened.last() {
438 last_end = last.0.end;
439 }
440
441 output.extend(flattened);
442 }
443 ListItem::Spread(op_span, expr) => {
444 if op_span.start > last_end {
445 output.push((Span::new(last_end, op_span.start), FlatShape::List));
446 }
447 output.push((*op_span, FlatShape::Operator));
448 last_end = op_span.end;
449
450 let flattened_inner = flatten_expression(working_set, expr);
451 if let Some(first) = flattened_inner.first() {
452 if first.0.start > last_end {
453 output.push((Span::new(last_end, first.0.start), FlatShape::List));
454 }
455 }
456 if let Some(last) = flattened_inner.last() {
457 last_end = last.0.end;
458 }
459 output.extend(flattened_inner);
460 }
461 }
462 }
463
464 if last_end < outer_span.end {
465 output.push((Span::new(last_end, outer_span.end), FlatShape::List));
466 }
467 }
468 Expr::StringInterpolation(exprs) => {
469 let mut flattened = vec![];
470 for expr in exprs {
471 flatten_expression_into(working_set, expr, &mut flattened);
472 }
473
474 if let Some(first) = flattened.first() {
475 if first.0.start != expr.span.start {
476 output.push((
478 Span::new(expr.span.start, expr.span.start + 2),
479 FlatShape::StringInterpolation,
480 ));
481 flattened.push((
482 Span::new(expr.span.end - 1, expr.span.end),
483 FlatShape::StringInterpolation,
484 ));
485 }
486 }
487 output.extend(flattened);
488 }
489 Expr::GlobInterpolation(exprs, quoted) => {
490 let mut flattened = vec![];
491 for expr in exprs {
492 flatten_expression_into(working_set, expr, &mut flattened);
493 }
494
495 if *quoted {
496 output.push((
498 Span::new(expr.span.start, expr.span.start + 2),
499 FlatShape::GlobInterpolation,
500 ));
501 flattened.push((
502 Span::new(expr.span.end - 1, expr.span.end),
503 FlatShape::GlobInterpolation,
504 ));
505 }
506 output.extend(flattened);
507 }
508 Expr::Record(list) => {
509 let outer_span = expr.span;
510 let mut last_end = outer_span.start;
511
512 for l in list {
513 match l {
514 RecordItem::Pair(key, val) => {
515 let flattened_lhs = flatten_expression(working_set, key);
516 let flattened_rhs = flatten_expression(working_set, val);
517
518 if let Some(first) = flattened_lhs.first() {
519 if first.0.start > last_end {
520 output
521 .push((Span::new(last_end, first.0.start), FlatShape::Record));
522 }
523 }
524 if let Some(last) = flattened_lhs.last() {
525 last_end = last.0.end;
526 }
527 output.extend(flattened_lhs);
528
529 if let Some(first) = flattened_rhs.first() {
530 if first.0.start > last_end {
531 output
532 .push((Span::new(last_end, first.0.start), FlatShape::Record));
533 }
534 }
535 if let Some(last) = flattened_rhs.last() {
536 last_end = last.0.end;
537 }
538
539 output.extend(flattened_rhs);
540 }
541 RecordItem::Spread(op_span, record) => {
542 if op_span.start > last_end {
543 output.push((Span::new(last_end, op_span.start), FlatShape::Record));
544 }
545 output.push((*op_span, FlatShape::Operator));
546 last_end = op_span.end;
547
548 let flattened = flatten_expression(working_set, record);
549 if let Some(first) = flattened.first() {
550 if first.0.start > last_end {
551 output
552 .push((Span::new(last_end, first.0.start), FlatShape::Record));
553 }
554 }
555 if let Some(last) = flattened.last() {
556 last_end = last.0.end;
557 }
558 output.extend(flattened);
559 }
560 }
561 }
562 if last_end < outer_span.end {
563 output.push((Span::new(last_end, outer_span.end), FlatShape::Record));
564 }
565 }
566 Expr::Keyword(kw) => {
567 output.push((kw.span, FlatShape::Keyword));
568 flatten_expression_into(working_set, &kw.expr, output);
569 }
570 Expr::Operator(_) => output.push((expr.span, FlatShape::Operator)),
571 Expr::Signature(_) => output.push((expr.span, FlatShape::Signature)),
572 Expr::String(_) => output.push((expr.span, FlatShape::String)),
573 Expr::RawString(_) => output.push((expr.span, FlatShape::RawString)),
574 Expr::Table(table) => {
575 let outer_span = expr.span;
576 let mut last_end = outer_span.start;
577
578 for col in table.columns.as_ref() {
579 let flattened = flatten_expression(working_set, col);
580 if let Some(first) = flattened.first() {
581 if first.0.start > last_end {
582 output.push((Span::new(last_end, first.0.start), FlatShape::Table));
583 }
584 }
585
586 if let Some(last) = flattened.last() {
587 last_end = last.0.end;
588 }
589
590 output.extend(flattened);
591 }
592 for row in table.rows.as_ref() {
593 for expr in row.as_ref() {
594 let flattened = flatten_expression(working_set, expr);
595 if let Some(first) = flattened.first() {
596 if first.0.start > last_end {
597 output.push((Span::new(last_end, first.0.start), FlatShape::Table));
598 }
599 }
600
601 if let Some(last) = flattened.last() {
602 last_end = last.0.end;
603 }
604
605 output.extend(flattened);
606 }
607 }
608
609 if last_end < outer_span.end {
610 output.push((Span::new(last_end, outer_span.end), FlatShape::Table));
611 }
612 }
613 Expr::Var(var_id) => output.push((expr.span, FlatShape::Variable(*var_id))),
614 Expr::VarDecl(var_id) => output.push((expr.span, FlatShape::VarDecl(*var_id))),
615 }
616}
617
618fn flatten_pattern_into(match_pattern: &MatchPattern, output: &mut Vec<(Span, FlatShape)>) {
619 match &match_pattern.pattern {
620 Pattern::Garbage => output.push((match_pattern.span, FlatShape::Garbage)),
621 Pattern::IgnoreValue => output.push((match_pattern.span, FlatShape::Nothing)),
622 Pattern::IgnoreRest => output.push((match_pattern.span, FlatShape::Nothing)),
623 Pattern::List(items) => {
624 if let Some(first) = items.first() {
625 if let Some(last) = items.last() {
626 output.push((
627 Span::new(match_pattern.span.start, first.span.start),
628 FlatShape::MatchPattern,
629 ));
630 for item in items {
631 flatten_pattern_into(item, output);
632 }
633 output.push((
634 Span::new(last.span.end, match_pattern.span.end),
635 FlatShape::MatchPattern,
636 ))
637 }
638 } else {
639 output.push((match_pattern.span, FlatShape::MatchPattern));
640 }
641 }
642 Pattern::Record(items) => {
643 if let Some(first) = items.first() {
644 if let Some(last) = items.last() {
645 output.push((
646 Span::new(match_pattern.span.start, first.1.span.start),
647 FlatShape::MatchPattern,
648 ));
649 for (_, pattern) in items {
650 flatten_pattern_into(pattern, output);
651 }
652 output.push((
653 Span::new(last.1.span.end, match_pattern.span.end),
654 FlatShape::MatchPattern,
655 ))
656 }
657 } else {
658 output.push((match_pattern.span, FlatShape::MatchPattern));
659 }
660 }
661 Pattern::Expression(_) | Pattern::Value(_) => {
662 output.push((match_pattern.span, FlatShape::MatchPattern))
663 }
664 Pattern::Variable(var_id) => output.push((match_pattern.span, FlatShape::VarDecl(*var_id))),
665 Pattern::Rest(var_id) => output.push((match_pattern.span, FlatShape::VarDecl(*var_id))),
666 Pattern::Or(patterns) => {
667 for pattern in patterns {
668 flatten_pattern_into(pattern, output);
669 }
670 }
671 }
672}
673
674pub fn flatten_block(working_set: &StateWorkingSet, block: &Block) -> Vec<(Span, FlatShape)> {
675 let mut output = Vec::new();
676 flatten_block_into(working_set, block, &mut output);
677 output
678}
679
680pub fn flatten_pipeline(
681 working_set: &StateWorkingSet,
682 pipeline: &Pipeline,
683) -> Vec<(Span, FlatShape)> {
684 let mut output = Vec::new();
685 flatten_pipeline_into(working_set, pipeline, &mut output);
686 output
687}
688
689pub fn flatten_pipeline_element(
690 working_set: &StateWorkingSet,
691 pipeline_element: &PipelineElement,
692) -> Vec<(Span, FlatShape)> {
693 let mut output = Vec::new();
694 flatten_pipeline_element_into(working_set, pipeline_element, &mut output);
695 output
696}
697
698pub fn flatten_expression(
699 working_set: &StateWorkingSet,
700 expr: &Expression,
701) -> Vec<(Span, FlatShape)> {
702 let mut output = Vec::new();
703 flatten_expression_into(working_set, expr, &mut output);
704 output
705}