use crate::commands::WholeStreamCommand; use crate::data::{Primitive, Value}; use crate::errors::ShellError; use crate::prelude::*; use log::trace; pub struct Lines; impl WholeStreamCommand for Lines { fn name(&self) -> &str { "lines" } fn signature(&self) -> Signature { Signature::build("lines") } fn usage(&self) -> &str { "Split single string into rows, one per line." } fn run( &self, args: CommandArgs, registry: &CommandRegistry, ) -> Result { lines(args, registry) } } // TODO: "Amount remaining" wrapper fn lines(args: CommandArgs, registry: &CommandRegistry) -> Result { let args = args.evaluate_once(registry)?; let tag = args.name_tag(); let input = args.input; let stream = input .values .map(move |v| match v.item { Value::Primitive(Primitive::String(s)) => { let split_result: Vec<_> = s.lines().filter(|s| s.trim() != "").collect(); trace!("split result = {:?}", split_result); let mut result = VecDeque::new(); for s in split_result { result.push_back(ReturnSuccess::value( Value::Primitive(Primitive::String(s.into())).tagged_unknown(), )); } result } _ => { let mut result = VecDeque::new(); result.push_back(Err(ShellError::labeled_error_with_secondary( "Expected a string from pipeline", "requires string input", &tag, "value originates from here", v.tag(), ))); result } }) .flatten(); Ok(stream.to_output_stream()) }