nushell/crates/nu-command/src/commands/format/command.rs

144 lines
3.8 KiB
Rust
Raw Normal View History

2019-05-24 21:35:22 +02:00
use crate::prelude::*;
use nu_engine::evaluate_baseline_expr;
use nu_engine::WholeStreamCommand;
2019-12-09 02:57:53 +01:00
use nu_errors::ShellError;
use nu_protocol::{Signature, SyntaxShape, UntaggedValue};
use nu_source::Tagged;
2019-12-09 02:57:53 +01:00
use std::borrow::Borrow;
pub struct Format;
impl WholeStreamCommand for Format {
2019-12-09 02:57:53 +01:00
fn name(&self) -> &str {
"format"
}
fn signature(&self) -> Signature {
Signature::build("format").required(
"pattern",
SyntaxShape::String,
2019-12-09 02:57:53 +01:00
"the pattern to output. Eg) \"{foo}: {bar}\"",
)
}
fn usage(&self) -> &str {
"Format columns into a string using a simple pattern."
}
fn run(&self, args: CommandArgs) -> Result<OutputStream, ShellError> {
format_command(args)
}
2020-05-12 07:17:17 +02:00
fn examples(&self) -> Vec<Example> {
vec![Example {
2020-05-12 07:17:17 +02:00
description: "Print filenames with their sizes",
example: "ls | format '{name}: {size}'",
result: None,
2020-05-12 07:17:17 +02:00
}]
}
}
fn format_command(args: CommandArgs) -> Result<OutputStream, ShellError> {
let ctx = Arc::new(EvaluationContext::from_args(&args));
let args = args.evaluate_once()?;
let pattern: Tagged<String> = args.req(0)?;
let format_pattern = format(&pattern);
let commands = Arc::new(format_pattern);
Ok(args
.input
.map(move |value| {
let mut output = String::new();
let commands = commands.clone();
let ctx = ctx.clone();
for command in &*commands {
match command {
FormatCommand::Text(s) => {
output.push_str(&s);
}
FormatCommand::Column(c) => {
// FIXME: use the correct spans
let full_column_path = nu_parser::parse_full_column_path(
&(c.to_string()).spanned(Span::unknown()),
&ctx.scope,
);
ctx.scope.enter_scope();
ctx.scope.add_var("$it", value.clone());
let result = evaluate_baseline_expr(&full_column_path.0, &*ctx);
ctx.scope.exit_scope();
if let Ok(c) = result {
output.push_str(&value::format_leaf(c.borrow()).plain_string(100_000))
} else {
// That column doesn't match, so don't emit anything
}
}
}
}
Ok(UntaggedValue::string(output).into_untagged_value())
})
.to_input_stream())
2019-12-09 02:57:53 +01:00
}
#[derive(Debug)]
enum FormatCommand {
Text(String),
Column(String),
}
2020-04-07 09:51:17 +02:00
fn format(input: &str) -> Vec<FormatCommand> {
2019-12-09 02:57:53 +01:00
let mut output = vec![];
2020-04-07 09:51:17 +02:00
let mut loop_input = input.chars();
2019-12-09 02:57:53 +01:00
loop {
2020-04-07 09:51:17 +02:00
let mut before = String::new();
while let Some(c) = loop_input.next() {
if c == '{' {
break;
}
before.push(c);
}
2019-12-09 02:57:53 +01:00
if !before.is_empty() {
output.push(FormatCommand::Text(before.to_string()));
}
2020-04-07 09:51:17 +02:00
// Look for column as we're now at one
let mut column = String::new();
2019-12-09 02:57:53 +01:00
2020-04-07 09:51:17 +02:00
while let Some(c) = loop_input.next() {
if c == '}' {
break;
}
column.push(c);
}
if !column.is_empty() {
2019-12-09 02:57:53 +01:00
output.push(FormatCommand::Column(column.to_string()));
}
2020-04-07 09:51:17 +02:00
if before.is_empty() && column.is_empty() {
2019-12-09 02:57:53 +01:00
break;
2019-05-24 21:35:22 +02:00
}
}
2019-12-09 02:57:53 +01:00
2020-04-07 09:51:17 +02:00
output
2019-05-24 21:35:22 +02:00
}
#[cfg(test)]
mod tests {
use super::Format;
use super::ShellError;
#[test]
fn examples_work_as_expected() -> Result<(), ShellError> {
use crate::examples::test as test_examples;
2021-02-12 11:13:14 +01:00
test_examples(Format {})
}
}