Make nth more stream-able (#3330)

This commit is contained in:
JT 2021-04-19 19:45:12 +12:00 committed by GitHub
parent 033df9457b
commit 9d24afcfe3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -1,7 +1,7 @@
use crate::prelude::*; use crate::prelude::*;
use nu_engine::WholeStreamCommand; use nu_engine::WholeStreamCommand;
use nu_errors::ShellError; use nu_errors::ShellError;
use nu_protocol::{ReturnSuccess, Signature, SyntaxShape, Value}; use nu_protocol::{Signature, SyntaxShape, Value};
use nu_source::Tagged; use nu_source::Tagged;
#[derive(Deserialize)] #[derive(Deserialize)]
@ -33,7 +33,7 @@ impl WholeStreamCommand for Nth {
"Return or skip only the selected rows." "Return or skip only the selected rows."
} }
fn run_with_actions(&self, args: CommandArgs) -> Result<ActionStream, ShellError> { fn run(&self, args: CommandArgs) -> Result<OutputStream, ShellError> {
nth(args) nth(args)
} }
@ -58,7 +58,7 @@ impl WholeStreamCommand for Nth {
} }
} }
fn nth(args: CommandArgs) -> Result<ActionStream, ShellError> { fn nth(args: CommandArgs) -> Result<OutputStream, ShellError> {
let ( let (
NthArgs { NthArgs {
row_number, row_number,
@ -68,22 +68,60 @@ fn nth(args: CommandArgs) -> Result<ActionStream, ShellError> {
input, input,
) = args.process()?; ) = args.process()?;
let row_numbers = vec![vec![row_number], and_rows] let mut rows: Vec<_> = and_rows.into_iter().map(|x| x.item as usize).collect();
.into_iter() rows.push(row_number.item as usize);
.flatten() rows.sort_unstable();
.map(|x| x.item)
.collect::<Vec<u64>>();
Ok(input Ok(NthIterator {
.enumerate() input,
.filter_map(move |(idx, item)| { rows,
if row_numbers.contains(&(idx as u64)) ^ skip { skip,
Some(ReturnSuccess::value(item)) current: 0,
}
.to_output_stream())
}
struct NthIterator {
input: InputStream,
rows: Vec<usize>,
skip: bool,
current: usize,
}
impl Iterator for NthIterator {
type Item = Value;
fn next(&mut self) -> Option<Self::Item> {
loop {
if !self.skip {
if let Some(row) = self.rows.get(0) {
if self.current == *row {
self.rows.remove(0);
self.current += 1;
return self.input.next();
} else {
self.current += 1;
let _ = self.input.next();
continue;
}
} else {
return None;
}
} else if let Some(row) = self.rows.get(0) {
if self.current == *row {
self.rows.remove(0);
self.current += 1;
let _ = self.input.next();
continue;
} else {
self.current += 1;
return self.input.next();
}
} else { } else {
None return self.input.next();
} }
}) }
.to_action_stream()) }
} }
#[cfg(test)] #[cfg(test)]