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

162 lines
4.3 KiB
Rust
Raw Normal View History

use crate::command_registry::CommandRegistry;
use crate::commands::WholeStreamCommand;
use crate::evaluate::evaluate_baseline_expr;
2019-05-24 21:35:22 +02:00
use crate::prelude::*;
2019-12-09 02:57:53 +01:00
use nu_errors::ShellError;
use nu_protocol::{ReturnSuccess, Scope, Signature, SyntaxShape, UntaggedValue};
use nu_source::Tagged;
2019-12-09 02:57:53 +01:00
use std::borrow::Borrow;
pub struct Format;
#[derive(Deserialize)]
pub struct FormatArgs {
pattern: Tagged<String>,
}
2020-05-29 10:22:52 +02:00
#[async_trait]
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."
}
2020-05-29 10:22:52 +02:00
async fn run(
2019-12-09 02:57:53 +01:00
&self,
args: CommandArgs,
registry: &CommandRegistry,
2019-12-09 02:57:53 +01:00
) -> Result<OutputStream, ShellError> {
format_command(args, registry).await
}
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
}]
}
}
async fn format_command(
args: CommandArgs,
registry: &CommandRegistry,
) -> Result<OutputStream, ShellError> {
let registry = Arc::new(registry.clone());
let scope = args.call_info.scope.clone();
let (FormatArgs { pattern }, input) = args.process(&registry).await?;
let format_pattern = format(&pattern);
let commands = Arc::new(format_pattern);
Ok(input
.then(move |value| {
let mut output = String::new();
let commands = commands.clone();
let registry = registry.clone();
let scope = scope.clone();
async move {
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()),
&*registry,
);
let result = evaluate_baseline_expr(
&full_column_path.0,
&registry,
Scope::append_it(scope.clone(), value.clone()),
)
.await;
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
}
}
}
}
ReturnSuccess::value(UntaggedValue::string(output).into_untagged_value())
}
})
.to_output_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;
#[test]
fn examples_work_as_expected() {
use crate::examples::test as test_examples;
test_examples(Format {})
}
}