use crate::commands::WholeStreamCommand; use crate::context::CommandRegistry; use crate::prelude::*; use futures::stream::StreamExt; use nu_errors::ShellError; use nu_protocol::{ReturnSuccess, Signature, SyntaxShape, UntaggedValue, Value}; use nu_source::Tagged; pub struct Compact; #[derive(Deserialize)] pub struct CompactArgs { rest: Vec>, } #[async_trait] impl WholeStreamCommand for Compact { fn name(&self) -> &str { "compact" } fn signature(&self) -> Signature { Signature::build("compact").rest(SyntaxShape::Any, "the columns to compact from the table") } fn usage(&self) -> &str { "Creates a table with non-empty rows" } async fn run( &self, args: CommandArgs, registry: &CommandRegistry, ) -> Result { compact(args, registry) } fn examples(&self) -> Vec { vec![ Example { description: "Filter out all null entries in a list", example: "echo [1 2 $null 3 $null $null] | compact target", result: Some(vec![ UntaggedValue::int(1).into(), UntaggedValue::int(2).into(), UntaggedValue::int(3).into(), ]), }, Example { description: "Filter out all directory entries having no 'target'", example: "ls -af | compact target", result: None, }, ] } } pub fn compact(args: CommandArgs, registry: &CommandRegistry) -> Result { let registry = registry.clone(); let stream = async_stream! { let (CompactArgs { rest: columns }, mut input) = args.process(®istry).await?; while let Some(item) = input.next().await { if columns.is_empty() { if !item.is_empty() { yield ReturnSuccess::value(item); } } else { match item { Value { value: UntaggedValue::Row(ref r), .. } => if columns .iter() .all(|field| r.get_data(field).borrow().is_some()) { yield ReturnSuccess::value(item); } _ => {}, } }; } }; Ok(stream.to_output_stream()) } #[cfg(test)] mod tests { use super::Compact; #[test] fn examples_work_as_expected() { use crate::examples::test as test_examples; test_examples(Compact {}) } }