Add example tests (nu-plugin-test-support) for plugins in repo (#12281)

# Description

Uses the new `nu-plugin-test-support` crate to test the examples of
commands provided by plugins in the repo.

Also fixed some of the examples to pass.

# User-Facing Changes

- Examples that are more guaranteed to work

# Tests + Formatting
- 🟢 `toolkit fmt`
- 🟢 `toolkit clippy`
- 🟢 `toolkit test`
- 🟢 `toolkit test stdlib`
This commit is contained in:
Devyn Cairns 2024-03-25 19:20:35 -07:00 committed by GitHub
parent efe1c99a3b
commit 2ae4408ced
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
19 changed files with 139 additions and 6 deletions

4
Cargo.lock generated
View File

@ -3214,6 +3214,7 @@ name = "nu_plugin_custom_values"
version = "0.1.0" version = "0.1.0"
dependencies = [ dependencies = [
"nu-plugin", "nu-plugin",
"nu-plugin-test-support",
"nu-protocol", "nu-protocol",
"serde", "serde",
"typetag", "typetag",
@ -3223,7 +3224,9 @@ dependencies = [
name = "nu_plugin_example" name = "nu_plugin_example"
version = "0.91.1" version = "0.91.1"
dependencies = [ dependencies = [
"nu-cmd-lang",
"nu-plugin", "nu-plugin",
"nu-plugin-test-support",
"nu-protocol", "nu-protocol",
] ]
@ -3235,6 +3238,7 @@ dependencies = [
"ical", "ical",
"indexmap", "indexmap",
"nu-plugin", "nu-plugin",
"nu-plugin-test-support",
"nu-protocol", "nu-protocol",
"rust-ini", "rust-ini",
] ]

View File

@ -61,6 +61,19 @@ impl PluginTest {
&mut self.engine_state &mut self.engine_state
} }
/// Make additional command declarations available for use by tests.
///
/// This can be used to pull in commands from `nu-cmd-lang` for example, as required.
pub fn add_decl(
&mut self,
decl: Box<dyn nu_protocol::engine::Command>,
) -> Result<&mut Self, ShellError> {
let mut working_set = StateWorkingSet::new(&self.engine_state);
working_set.add_decl(decl);
self.engine_state.merge_delta(working_set.render())?;
Ok(self)
}
/// Evaluate some Nushell source code with the plugin commands in scope with the given input to /// Evaluate some Nushell source code with the plugin commands in scope with the given input to
/// the pipeline. /// the pipeline.
/// ///

View File

@ -14,3 +14,6 @@ nu-plugin = { path = "../nu-plugin", version = "0.91.1" }
nu-protocol = { path = "../nu-protocol", version = "0.91.1", features = ["plugin"] } nu-protocol = { path = "../nu-protocol", version = "0.91.1", features = ["plugin"] }
serde = { workspace = true, default-features = false } serde = { workspace = true, default-features = false }
typetag = "0.2" typetag = "0.2"
[dev-dependencies]
nu-plugin-test-support = { path = "../nu-plugin-test-support", version = "0.91.1" }

View File

@ -29,3 +29,11 @@ impl SimplePluginCommand for Generate {
Ok(CoolCustomValue::new("abc").into_value(call.head)) Ok(CoolCustomValue::new("abc").into_value(call.head))
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("custom_values", crate::CustomValuePlugin.into())?
.test_command_examples(&Generate)
}

View File

@ -54,3 +54,11 @@ impl SimplePluginCommand for Generate2 {
} }
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("custom_values", crate::CustomValuePlugin.into())?
.test_command_examples(&Generate2)
}

View File

@ -1,7 +1,9 @@
use std::cmp::Ordering;
use nu_protocol::{CustomValue, ShellError, Span, Value}; use nu_protocol::{CustomValue, ShellError, Span, Value};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
#[derive(Clone, Debug, Serialize, Deserialize)] #[derive(Clone, Debug, Serialize, Deserialize, PartialEq, Eq, PartialOrd, Ord)]
pub struct SecondCustomValue { pub struct SecondCustomValue {
pub(crate) something: String, pub(crate) something: String,
} }
@ -59,6 +61,16 @@ impl CustomValue for SecondCustomValue {
)) ))
} }
fn partial_cmp(&self, other: &Value) -> Option<Ordering> {
if let Value::CustomValue { val, .. } = other {
val.as_any()
.downcast_ref()
.and_then(|other: &SecondCustomValue| PartialOrd::partial_cmp(self, other))
} else {
None
}
}
fn as_any(&self) -> &dyn std::any::Any { fn as_any(&self) -> &dyn std::any::Any {
self self
} }

View File

@ -23,9 +23,9 @@ impl SimplePluginCommand for Update {
result: Some(CoolCustomValue::new("abcxyz").into_value(Span::test_data())), result: Some(CoolCustomValue::new("abcxyz").into_value(Span::test_data())),
}, },
PluginExample { PluginExample {
example: "custom-value generate | custom-value update".into(), example: "custom-value generate2 | custom-value update".into(),
description: "Update a SecondCustomValue".into(), description: "Update a SecondCustomValue".into(),
result: Some(CoolCustomValue::new("xyzabc").into_value(Span::test_data())), result: Some(SecondCustomValue::new("xyzabc").into_value(Span::test_data())),
}, },
]) ])
} }
@ -56,3 +56,11 @@ impl SimplePluginCommand for Update {
.into()) .into())
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("custom_values", crate::CustomValuePlugin.into())?
.test_command_examples(&Update)
}

View File

@ -17,3 +17,7 @@ bench = false
[dependencies] [dependencies]
nu-plugin = { path = "../nu-plugin", version = "0.91.1" } nu-plugin = { path = "../nu-plugin", version = "0.91.1" }
nu-protocol = { path = "../nu-protocol", version = "0.91.1", features = ["plugin"] } nu-protocol = { path = "../nu-protocol", version = "0.91.1", features = ["plugin"] }
[dev-dependencies]
nu-plugin-test-support = { path = "../nu-plugin-test-support", version = "0.91.1" }
nu-cmd-lang = { path = "../nu-cmd-lang", version = "0.91.1" }

View File

@ -51,3 +51,9 @@ impl PluginCommand for CollectExternal {
}) })
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("example", Example.into())?.test_command_examples(&CollectExternal)
}

View File

@ -45,3 +45,9 @@ impl PluginCommand for ForEach {
Ok(PipelineData::Empty) Ok(PipelineData::Empty)
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("example", Example.into())?.test_command_examples(&ForEach)
}

View File

@ -76,3 +76,12 @@ impl PluginCommand for Generate {
.into_pipeline_data(None)) .into_pipeline_data(None))
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_cmd_lang::If;
use nu_plugin_test_support::PluginTest;
PluginTest::new("example", Example.into())?
.add_decl(Box::new(If))?
.test_command_examples(&Generate)
}

View File

@ -41,3 +41,9 @@ impl SimplePluginCommand for One {
Ok(Value::nothing(call.head)) Ok(Value::nothing(call.head))
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("example", Example.into())?.test_command_examples(&One)
}

View File

@ -46,3 +46,9 @@ impl PluginCommand for Seq {
Ok(PipelineData::ListStream(list_stream, None)) Ok(PipelineData::ListStream(list_stream, None))
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("example", Example.into())?.test_command_examples(&Seq)
}

View File

@ -20,7 +20,7 @@ impl PluginCommand for Sum {
(Type::List(Type::Float.into()), Type::Float), (Type::List(Type::Float.into()), Type::Float),
]) ])
.plugin_examples(vec![PluginExample { .plugin_examples(vec![PluginExample {
example: "seq 1 5 | example sum".into(), example: "example seq 1 5 | example sum".into(),
description: "sum values from 1 to 5".into(), description: "sum values from 1 to 5".into(),
result: Some(Value::test_int(15)), result: Some(Value::test_int(15)),
}]) }])
@ -88,3 +88,9 @@ impl IntOrFloat {
} }
} }
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("example", Example.into())?.test_command_examples(&Sum)
}

View File

@ -16,3 +16,6 @@ indexmap = { workspace = true }
eml-parser = "0.1" eml-parser = "0.1"
ical = "0.10" ical = "0.10"
rust-ini = "0.20.0" rust-ini = "0.20.0"
[dev-dependencies]
nu-plugin-test-support = { path = "../nu-plugin-test-support", version = "0.91.1" }

View File

@ -53,6 +53,7 @@ pub fn examples() -> Vec<PluginExample> {
example: "'From: test@email.com example: "'From: test@email.com
Subject: Welcome Subject: Welcome
To: someone@somewhere.com To: someone@somewhere.com
Test' | from eml" Test' | from eml"
.into(), .into(),
result: Some(Value::test_record(record! { result: Some(Value::test_record(record! {
@ -73,6 +74,7 @@ Test' | from eml"
example: "'From: test@email.com example: "'From: test@email.com
Subject: Welcome Subject: Welcome
To: someone@somewhere.com To: someone@somewhere.com
Test' | from eml -b 1" Test' | from eml -b 1"
.into(), .into(),
result: Some(Value::test_record(record! { result: Some(Value::test_record(record! {
@ -164,3 +166,10 @@ fn from_eml(input: &Value, body_preview: usize, head: Span) -> Result<Value, Lab
Ok(Value::record(collected.into_iter().collect(), head)) Ok(Value::record(collected.into_iter().collect(), head))
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("formats", crate::FromCmds.into())?.test_command_examples(&FromEml)
}

View File

@ -75,7 +75,8 @@ impl SimplePluginCommand for FromIcs {
pub fn examples() -> Vec<PluginExample> { pub fn examples() -> Vec<PluginExample> {
vec![PluginExample { vec![PluginExample {
example: "'BEGIN:VCALENDAR example: "
'BEGIN:VCALENDAR
END:VCALENDAR' | from ics" END:VCALENDAR' | from ics"
.into(), .into(),
description: "Converts ics formatted string to table".into(), description: "Converts ics formatted string to table".into(),
@ -263,3 +264,10 @@ fn params_to_value(params: Vec<(String, Vec<String>)>, span: Span) -> Value {
Value::record(row.into_iter().collect(), span) Value::record(row.into_iter().collect(), span)
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("formats", crate::FromCmds.into())?.test_command_examples(&FromIcs)
}

View File

@ -89,3 +89,10 @@ b=2' | from ini"
})), })),
}] }]
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("formats", crate::FromCmds.into())?.test_command_examples(&FromIni)
}

View File

@ -153,3 +153,10 @@ fn params_to_value(params: Vec<(String, Vec<String>)>, span: Span) -> Value {
Value::record(row.into_iter().collect(), span) Value::record(row.into_iter().collect(), span)
} }
#[test]
fn test_examples() -> Result<(), nu_protocol::ShellError> {
use nu_plugin_test_support::PluginTest;
PluginTest::new("formats", crate::FromCmds.into())?.test_command_examples(&FromVcf)
}