2020-05-18 12:56:01 +00:00
|
|
|
use futures::executor::block_on;
|
|
|
|
|
|
|
|
use nu_errors::ShellError;
|
|
|
|
use nu_protocol::hir::ClassifiedBlock;
|
2020-09-25 23:40:02 +00:00
|
|
|
use nu_protocol::{Scope, ShellTypeName, Value};
|
2020-05-18 12:56:01 +00:00
|
|
|
|
|
|
|
use crate::commands::classified::block::run_block;
|
2020-08-27 05:44:18 +00:00
|
|
|
use crate::commands::{whole_stream_command, BuildString, Each, Echo, StrCollect};
|
2020-09-19 21:29:51 +00:00
|
|
|
use crate::evaluation_context::EvaluationContext;
|
2020-05-18 12:56:01 +00:00
|
|
|
use crate::stream::InputStream;
|
|
|
|
use crate::WholeStreamCommand;
|
|
|
|
|
|
|
|
pub fn test(cmd: impl WholeStreamCommand + 'static) {
|
|
|
|
let examples = cmd.examples();
|
2020-09-19 21:29:51 +00:00
|
|
|
let mut base_context = EvaluationContext::basic().expect("could not create basic context");
|
2020-07-03 08:43:55 +00:00
|
|
|
|
2020-05-18 12:56:01 +00:00
|
|
|
base_context.add_commands(vec![
|
|
|
|
whole_stream_command(Echo {}),
|
2020-07-03 08:43:55 +00:00
|
|
|
whole_stream_command(BuildString {}),
|
2020-08-27 05:44:18 +00:00
|
|
|
whole_stream_command(Each {}),
|
2020-05-18 12:56:01 +00:00
|
|
|
whole_stream_command(cmd),
|
2020-07-20 23:33:39 +00:00
|
|
|
whole_stream_command(StrCollect),
|
2020-05-18 12:56:01 +00:00
|
|
|
]);
|
|
|
|
|
|
|
|
for example in examples {
|
|
|
|
let mut ctx = base_context.clone();
|
|
|
|
let block = parse_line(example.example, &mut ctx).expect("failed to parse example");
|
|
|
|
if let Some(expected) = example.result {
|
|
|
|
let result = block_on(evaluate_block(block, &mut ctx)).expect("failed to run example");
|
2020-07-03 08:43:55 +00:00
|
|
|
|
|
|
|
let errors = ctx.get_errors();
|
|
|
|
|
|
|
|
assert!(
|
|
|
|
errors.is_empty(),
|
|
|
|
"errors while running command.\ncommand: {}\nerrors: {:?}",
|
|
|
|
example.example,
|
|
|
|
errors
|
|
|
|
);
|
|
|
|
|
|
|
|
assert!(expected.len() == result.len(), "example command produced unexpected number of results.\ncommand: {}\nexpected number: {}\nactual: {}",
|
|
|
|
example.example,
|
|
|
|
expected.len(),
|
|
|
|
result.len(),);
|
|
|
|
|
2020-05-18 12:56:01 +00:00
|
|
|
assert!(
|
|
|
|
expected
|
|
|
|
.iter()
|
|
|
|
.zip(result.iter())
|
|
|
|
.all(|(e, a)| values_equal(e, a)),
|
2020-07-03 08:43:55 +00:00
|
|
|
"example command produced unexpected result.\ncommand: {}\nexpected: {:?}\nactual: {:?}",
|
2020-05-18 12:56:01 +00:00
|
|
|
example.example,
|
2020-06-13 21:49:57 +00:00
|
|
|
expected,
|
|
|
|
result,
|
2020-05-18 12:56:01 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Parse and run a nushell pipeline
|
2020-09-19 21:29:51 +00:00
|
|
|
fn parse_line(
|
|
|
|
line: &'static str,
|
|
|
|
ctx: &mut EvaluationContext,
|
|
|
|
) -> Result<ClassifiedBlock, ShellError> {
|
2020-05-18 12:56:01 +00:00
|
|
|
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,
|
2020-09-19 21:29:51 +00:00
|
|
|
ctx: &mut EvaluationContext,
|
2020-05-18 12:56:01 +00:00
|
|
|
) -> Result<Vec<Value>, ShellError> {
|
|
|
|
let input_stream = InputStream::empty();
|
|
|
|
let env = ctx.get_env();
|
|
|
|
|
2020-09-25 23:40:02 +00:00
|
|
|
let scope = Scope::from_env(env);
|
|
|
|
|
|
|
|
Ok(run_block(&block.block, ctx, input_stream, scope)
|
|
|
|
.await?
|
|
|
|
.into_vec()
|
|
|
|
.await)
|
2020-05-18 12:56:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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()),
|
|
|
|
}
|
|
|
|
}
|