use nu_errors::ShellError; use nu_protocol::hir::ClassifiedBlock; use nu_protocol::{ ReturnSuccess, Scope, ShellTypeName, Signature, SyntaxShape, UntaggedValue, Value, }; use nu_source::{AnchorLocation, TaggedItem}; use crate::prelude::*; use indexmap::indexmap; use num_bigint::BigInt; use indexmap::IndexMap; use crate::command_registry::CommandRegistry; use crate::commands::classified::block::run_block; use crate::commands::command::CommandArgs; use crate::commands::{ whole_stream_command, BuildString, Command, Each, Echo, Get, Keep, StrCollect, WholeStreamCommand, }; use crate::evaluation_context::EvaluationContext; use crate::stream::{InputStream, OutputStream}; use async_trait::async_trait; use futures::executor::block_on; use serde::Deserialize; pub fn test_examples(cmd: Command) -> Result<(), ShellError> { let examples = cmd.examples(); let mut base_context = EvaluationContext::basic()?; base_context.add_commands(vec![ // Mocks whole_stream_command(MockLs {}), // Minimal restricted commands to aid in testing whole_stream_command(Echo {}), whole_stream_command(BuildString {}), whole_stream_command(Get {}), whole_stream_command(Keep {}), whole_stream_command(Each {}), whole_stream_command(StrCollect), cmd, ]); for sample_pipeline in examples { let mut ctx = base_context.clone(); let block = parse_line(sample_pipeline.example, &mut ctx)?; if let Some(expected) = &sample_pipeline.result { let result = block_on(evaluate_block(block, &mut ctx))?; ctx.with_errors(|reasons| { reasons .iter() .cloned() .take(1) .next() .and_then(|err| Some(err)) }) .map_or(Ok(()), |reason| Err(reason))?; if expected.len() != result.len() { let rows_returned = format!("expected: {}\nactual: {}", expected.len(), result.len()); let failed_call = format!("command: {}\n", sample_pipeline.example); panic!( "example command produced unexpected number of results.\n {} {}", failed_call, rows_returned ); } for (e, a) in expected.iter().zip(result.iter()) { if !values_equal(e, a) { let row_errored = format!("expected: {:#?}\nactual: {:#?}", e, a); let failed_call = format!("command: {}\n", sample_pipeline.example); panic!( "example command produced unexpected result.\n {} {}", failed_call, row_errored ); } } } } Ok(()) } pub fn test(cmd: impl WholeStreamCommand + 'static) -> Result<(), ShellError> { let examples = cmd.examples(); let mut base_context = EvaluationContext::basic()?; base_context.add_commands(vec![ whole_stream_command(Echo {}), whole_stream_command(BuildString {}), whole_stream_command(Get {}), whole_stream_command(Keep {}), whole_stream_command(Each {}), whole_stream_command(cmd), whole_stream_command(StrCollect), ]); for sample_pipeline in examples { let mut ctx = base_context.clone(); let block = parse_line(sample_pipeline.example, &mut ctx)?; if let Some(expected) = &sample_pipeline.result { let result = block_on(evaluate_block(block, &mut ctx))?; ctx.with_errors(|reasons| { reasons .iter() .cloned() .take(1) .next() .and_then(|err| Some(err)) }) .map_or(Ok(()), |reason| Err(reason))?; if expected.len() != result.len() { let rows_returned = format!("expected: {}\nactual: {}", expected.len(), result.len()); let failed_call = format!("command: {}\n", sample_pipeline.example); panic!( "example command produced unexpected number of results.\n {} {}", failed_call, rows_returned ); } for (e, a) in expected.iter().zip(result.iter()) { if !values_equal(e, a) { let row_errored = format!("expected: {:#?}\nactual: {:#?}", e, a); let failed_call = format!("command: {}\n", sample_pipeline.example); panic!( "example command produced unexpected result.\n {} {}", failed_call, row_errored ); } } } } Ok(()) } pub fn test_anchors(cmd: Command) -> Result<(), ShellError> { let examples = cmd.examples(); let mut base_context = EvaluationContext::basic()?; base_context.add_commands(vec![ // Minimal restricted commands to aid in testing whole_stream_command(MockCommand {}), whole_stream_command(MockEcho {}), whole_stream_command(MockLs {}), whole_stream_command(BuildString {}), whole_stream_command(Get {}), whole_stream_command(Keep {}), whole_stream_command(Each {}), whole_stream_command(StrCollect), cmd, ]); for sample_pipeline in examples { let pipeline_with_anchor = format!("mock --open --path | {}", sample_pipeline.example); let mut ctx = base_context.clone(); let block = parse_line(&pipeline_with_anchor, &mut ctx)?; let result = block_on(evaluate_block(block, &mut ctx))?; ctx.with_errors(|reasons| { reasons .iter() .cloned() .take(1) .next() .and_then(|err| Some(err)) }) .map_or(Ok(()), |reason| Err(reason))?; for actual in result.iter() { if !is_anchor_carried(actual, mock_path()) { let failed_call = format!("command: {}\n", pipeline_with_anchor); panic!( "example command didn't carry anchor tag correctly.\n {} {:#?} {:#?}", failed_call, actual, mock_path() ); } } } Ok(()) } /// Parse and run a nushell pipeline fn parse_line(line: &str, ctx: &mut EvaluationContext) -> Result { let line = if line.ends_with('\n') { &line[..line.len() - 1] } else { line }; let lite_result = nu_parser::lite_parse(&line, 0)?; // TODO ensure the command whose examples we're testing is actually in the pipeline let mut classified_block = nu_parser::classify_block(&lite_result, ctx.registry()); classified_block.block.expand_it_usage(); Ok(classified_block) } async fn evaluate_block( block: ClassifiedBlock, ctx: &mut EvaluationContext, ) -> Result, ShellError> { let input_stream = InputStream::empty(); let env = ctx.get_env(); let scope = Scope::from_env(env); Ok(run_block(&block.block, ctx, input_stream, scope) .await? .drain_vec() .await) } // TODO probably something already available to do this // TODO perhaps better panic messages when things don't compare // Deep value comparisons that ignore tags fn values_equal(expected: &Value, actual: &Value) -> bool { use nu_protocol::UntaggedValue::*; match (&expected.value, &actual.value) { (Primitive(e), Primitive(a)) => e == a, (Row(e), Row(a)) => { if e.entries.len() != a.entries.len() { return false; } e.entries .iter() .zip(a.entries.iter()) .all(|((ek, ev), (ak, av))| ek == ak && values_equal(ev, av)) } (Table(e), Table(a)) => e.iter().zip(a.iter()).all(|(e, a)| values_equal(e, a)), (e, a) => unimplemented!("{} {}", e.type_name(), a.type_name()), } } fn is_anchor_carried(actual: &Value, anchor: AnchorLocation) -> bool { actual.tag.anchor() == Some(anchor) } #[derive(Deserialize)] struct Arguments { path: Option, open: bool, } struct MockCommand; #[async_trait] impl WholeStreamCommand for MockCommand { fn name(&self) -> &str { "mock" } fn signature(&self) -> Signature { Signature::build("mock") .switch("open", "fake opening sources", Some('o')) .switch("path", "file open", Some('p')) } fn usage(&self) -> &str { "Generates tables and metadata that mimics behavior of real commands in controlled ways." } async fn run( &self, args: CommandArgs, registry: &CommandRegistry, ) -> Result { let name_tag = args.call_info.name_tag.clone(); let ( Arguments { path: mocked_path, open: open_mock, }, _input, ) = args.process(®istry).await?; let out = UntaggedValue::string("Yehuda Katz in Ecuador"); if open_mock { if let Some(true) = mocked_path { let mocked_path = Some(mock_path()); let value = out.tagged(name_tag.span).map_anchored(&mocked_path); return Ok(OutputStream::one(Ok(ReturnSuccess::Value( value.item.into_value(value.tag), )))); } } Ok(OutputStream::one(Ok(ReturnSuccess::Value( out.into_value(name_tag), )))) } } struct MockEcho; #[derive(Deserialize)] pub struct MockEchoArgs { pub rest: Vec, } #[async_trait] impl WholeStreamCommand for MockEcho { fn name(&self) -> &str { "echo" } fn signature(&self) -> Signature { Signature::build("echo").rest(SyntaxShape::Any, "the values to echo") } fn usage(&self) -> &str { "Mock echo." } async fn run( &self, args: CommandArgs, registry: &CommandRegistry, ) -> Result { let name_tag = args.call_info.name_tag.clone(); let (MockEchoArgs { rest }, input) = args.process(®istry).await?; let mut base_value = UntaggedValue::string("Yehuda Katz in Ecuador").into_value(name_tag); let input: Vec = input.collect().await; if let Some(first) = input.get(0) { base_value = first.clone() } let stream = rest.into_iter().map(move |i| { let base_value = base_value.clone(); match i.as_string() { Ok(s) => OutputStream::one(Ok(ReturnSuccess::Value( UntaggedValue::string(s).into_value(base_value.tag.clone()), ))), _ => match i { Value { value: UntaggedValue::Table(table), .. } => futures::stream::iter( table .into_iter() .map(move |mut v| { v.tag = base_value.tag(); v }) .map(ReturnSuccess::value), ) .to_output_stream(), _ => OutputStream::one(Ok(ReturnSuccess::Value(Value { value: i.value.clone(), tag: base_value.tag.clone(), }))), }, } }); Ok(futures::stream::iter(stream).flatten().to_output_stream()) } } struct MockLs; #[async_trait] impl WholeStreamCommand for MockLs { fn name(&self) -> &str { "ls" } fn signature(&self) -> Signature { Signature::build("ls") } fn usage(&self) -> &str { "Mock ls." } async fn run( &self, args: CommandArgs, _: &CommandRegistry, ) -> Result { let name_tag = args.call_info.name_tag.clone(); let mut base_value = UntaggedValue::string("Andrés N. Robalino in Portland").into_value(name_tag); let input: Vec = args.input.collect().await; if let Some(first) = input.get(0) { base_value = first.clone() } Ok(futures::stream::iter( file_listing() .iter() .map(|row| Value { value: row.value.clone(), tag: base_value.tag.clone(), }) .collect::>() .into_iter() .map(ReturnSuccess::value), ) .to_output_stream()) } } fn int(s: impl Into) -> Value { UntaggedValue::int(s).into_untagged_value() } fn string(input: impl Into) -> Value { UntaggedValue::string(input.into()).into_untagged_value() } fn row(entries: IndexMap) -> Value { UntaggedValue::row(entries).into_untagged_value() } fn date(input: impl Into) -> Value { let key = input.into().tagged_unknown(); crate::value::Date::naive_from_str(key.borrow_tagged()) .expect("date from string failed") .into_untagged_value() } fn file_listing() -> Vec { vec![ row(indexmap! { "modified".to_string() => date("2019-07-23"), "name".to_string() => string("Andrés.txt"), "type".to_string() => string("File"), "chickens".to_string() => int(10), }), row(indexmap! { "modified".to_string() => date("2019-07-23"), "name".to_string() => string("Jonathan"), "type".to_string() => string("Dir"), "chickens".to_string() => int(5), }), row(indexmap! { "modified".to_string() => date("2019-09-24"), "name".to_string() => string("Andrés.txt"), "type".to_string() => string("File"), "chickens".to_string() => int(20), }), row(indexmap! { "modified".to_string() => date("2019-09-24"), "name".to_string() => string("Yehuda"), "type".to_string() => string("Dir"), "chickens".to_string() => int(4), }), ] } fn mock_path() -> AnchorLocation { let path = String::from("path/to/las_best_arepas_in_the_world.txt"); AnchorLocation::File(path) }