nu-cmd-extra crate infrastructure in place with the Bits command as the model for adding other commands (#9327)

I wanted to get the infrastructure in place for starters for our
*nu-cmd-extra* crate...

The plan is to put inside here the following commands...

* bits
* bytes
* math

I thought it would be easier to do one at a time as well as get the
nu-cmd-extra crate out there on crates.io
for this upcoming release...

Once this lands the infrastructure will be in place to move over the
other noted commands for now...
And then add other stuff we do NOT want to be in 1.0.
This commit is contained in:
Michael Angerman 2023-06-01 10:46:16 -07:00 committed by GitHub
parent e48b94965b
commit 356e05177c
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
22 changed files with 190 additions and 20 deletions

13
Cargo.lock generated
View file

@ -2704,6 +2704,18 @@ dependencies = [
"sqlparser 0.33.0", "sqlparser 0.33.0",
] ]
[[package]]
name = "nu-cmd-extra"
version = "0.80.1"
dependencies = [
"nu-cmd-lang",
"nu-engine",
"nu-parser",
"nu-protocol",
"nu-test-support",
"num-traits",
]
[[package]] [[package]]
name = "nu-cmd-lang" name = "nu-cmd-lang"
version = "0.80.1" version = "0.80.1"
@ -2774,6 +2786,7 @@ dependencies = [
"notify", "notify",
"nu-ansi-term", "nu-ansi-term",
"nu-cmd-dataframe", "nu-cmd-dataframe",
"nu-cmd-extra",
"nu-cmd-lang", "nu-cmd-lang",
"nu-color-config", "nu-color-config",
"nu-engine", "nu-engine",

View file

@ -28,6 +28,7 @@ members = [
"crates/nu-engine", "crates/nu-engine",
"crates/nu-parser", "crates/nu-parser",
"crates/nu-system", "crates/nu-system",
"crates/nu-cmd-extra",
"crates/nu-cmd-lang", "crates/nu-cmd-lang",
"crates/nu-cmd-dataframe", "crates/nu-cmd-dataframe",
"crates/nu-command", "crates/nu-command",
@ -108,8 +109,6 @@ plugin = [
"nu-protocol/plugin", "nu-protocol/plugin",
"nu-engine/plugin", "nu-engine/plugin",
] ]
# extra used to be more useful but now it's the same as default. Leaving it in for backcompat with existing build scripts
extra = ["default", "nu-cmd-lang/extra"]
default = ["plugin", "which-support", "trash-support", "sqlite"] default = ["plugin", "which-support", "trash-support", "sqlite"]
stable = ["default"] stable = ["default"]
wasi = ["nu-cmd-lang/wasi"] wasi = ["nu-cmd-lang/wasi"]
@ -121,7 +120,8 @@ static-link-openssl = ["dep:openssl", "nu-cmd-lang/static-link-openssl"]
which-support = ["nu-command/which-support", "nu-cmd-lang/which-support"] which-support = ["nu-command/which-support", "nu-cmd-lang/which-support"]
trash-support = ["nu-command/trash-support", "nu-cmd-lang/trash-support"] trash-support = ["nu-command/trash-support", "nu-cmd-lang/trash-support"]
# Extra # Extra feature for nushell
extra = ["nu-command/extra"]
# Dataframe feature for nushell # Dataframe feature for nushell
dataframe = ["nu-command/dataframe", "nu-cmd-lang/dataframe"] dataframe = ["nu-command/dataframe", "nu-cmd-lang/dataframe"]

View file

@ -0,0 +1,29 @@
[package]
authors = ["The Nushell Project Developers"]
description = "Nushell's extra commands that are not part of the 1.0 api standard."
edition = "2021"
license = "MIT"
name = "nu-cmd-extra"
repository = "https://github.com/nushell/nushell/tree/main/crates/nu-cmd-extra"
version = "0.80.1"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
[lib]
bench = false
[dependencies]
nu-cmd-lang = { path = "../nu-cmd-lang", version = "0.80.1" }
nu-engine = { path = "../nu-engine", version = "0.80.1" }
nu-parser = { path = "../nu-parser", version = "0.80.1" }
nu-protocol = { path = "../nu-protocol", version = "0.80.1" }
# Potential dependencies for extras
num-traits = "0.2"
[features]
extra = ["default"]
default = []
[dev-dependencies]
nu-test-support = { path = "../nu-test-support", version = "0.80.1" }

View file

@ -0,0 +1,21 @@
MIT License
Copyright (c) 2019 - 2023 The Nushell Project Developers
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.

View file

@ -0,0 +1,74 @@
#[cfg(test)]
use nu_protocol::engine::Command;
#[cfg(test)]
pub fn test_examples(cmd: impl Command + 'static) {
test_examples::test_examples(cmd);
}
#[cfg(test)]
mod test_examples {
use nu_cmd_lang::example_support::{
check_all_signature_input_output_types_entries_have_examples,
check_example_evaluates_to_expected_output,
check_example_input_and_output_types_match_command_signature,
};
use nu_protocol::{
engine::{Command, EngineState, StateWorkingSet},
Type,
};
use std::collections::HashSet;
pub fn test_examples(cmd: impl Command + 'static) {
let examples = cmd.examples();
let signature = cmd.signature();
let mut engine_state = make_engine_state(cmd.clone_box());
let cwd = std::env::current_dir().expect("Could not get current working directory.");
let mut witnessed_type_transformations = HashSet::<(Type, Type)>::new();
for example in examples {
if example.result.is_none() {
continue;
}
witnessed_type_transformations.extend(
check_example_input_and_output_types_match_command_signature(
&example,
&cwd,
&mut make_engine_state(cmd.clone_box()),
&signature.input_output_types,
signature.operates_on_cell_paths(),
signature.vectorizes_over_list,
),
);
check_example_evaluates_to_expected_output(&example, cwd.as_path(), &mut engine_state);
}
check_all_signature_input_output_types_entries_have_examples(
signature,
witnessed_type_transformations,
);
}
fn make_engine_state(cmd: Box<dyn Command>) -> Box<EngineState> {
let mut engine_state = Box::new(EngineState::new());
let delta = {
// Base functions that are needed for testing
// Try to keep this working set small to keep tests running as fast as possible
let mut working_set = StateWorkingSet::new(&engine_state);
// Adding the command that is being tested to the working set
working_set.add_decl(cmd);
working_set.render()
};
engine_state
.merge_delta(delta)
.expect("Error merging delta");
engine_state
}
}

View file

@ -8,6 +8,7 @@ mod shift_left;
mod shift_right; mod shift_right;
mod xor; mod xor;
use nu_protocol::engine::StateWorkingSet;
use nu_protocol::Spanned; use nu_protocol::Spanned;
pub use and::SubCommand as BitsAnd; pub use and::SubCommand as BitsAnd;
@ -20,6 +21,30 @@ pub use shift_left::SubCommand as BitsShiftLeft;
pub use shift_right::SubCommand as BitsShiftRight; pub use shift_right::SubCommand as BitsShiftRight;
pub use xor::SubCommand as BitsXor; pub use xor::SubCommand as BitsXor;
pub fn add_bits_decls(working_set: &mut StateWorkingSet) {
macro_rules! bind_command {
( $command:expr ) => {
working_set.add_decl(Box::new($command));
};
( $( $command:expr ),* ) => {
$( working_set.add_decl(Box::new($command)); )*
};
}
// Dataframe commands
bind_command!(
Bits,
BitsAnd,
BitsNot,
BitsOr,
BitsXor,
BitsRotateLeft,
BitsRotateRight,
BitsShiftLeft,
BitsShiftRight
);
}
#[derive(Clone, Copy)] #[derive(Clone, Copy)]
enum NumberBytes { enum NumberBytes {
One, One,

View file

@ -0,0 +1,9 @@
mod bits;
pub use bits::add_bits_decls;
use nu_protocol::engine::StateWorkingSet;
pub fn add_extra_decls(working_set: &mut StateWorkingSet) {
add_bits_decls(working_set);
}

View file

@ -0,0 +1,6 @@
mod example_test;
pub mod extra;
pub use extra::*;
#[cfg(test)]
pub use example_test::test_examples;

View file

@ -32,6 +32,4 @@ trash-support = []
sqlite = [] sqlite = []
dataframe = [] dataframe = []
static-link-openssl = [] static-link-openssl = []
extra = []
wasi = [] wasi = []

View file

@ -15,6 +15,7 @@ bench = false
[dependencies] [dependencies]
nu-cmd-lang = { path = "../nu-cmd-lang", version = "0.80.1" } nu-cmd-lang = { path = "../nu-cmd-lang", version = "0.80.1" }
nu-cmd-dataframe = { path = "../nu-cmd-dataframe", version = "0.80.1", optional = true } nu-cmd-dataframe = { path = "../nu-cmd-dataframe", version = "0.80.1", optional = true }
nu-cmd-extra = { path = "../nu-cmd-extra", version = "0.80.1", optional = true }
nu-color-config = { path = "../nu-color-config", version = "0.80.1" } nu-color-config = { path = "../nu-color-config", version = "0.80.1" }
nu-engine = { path = "../nu-engine", version = "0.80.1" } nu-engine = { path = "../nu-engine", version = "0.80.1" }
nu-explore = { path = "../nu-explore", version = "0.80.1" } nu-explore = { path = "../nu-explore", version = "0.80.1" }
@ -117,6 +118,7 @@ version = "0.48"
[features] [features]
dataframe = ["dep:nu-cmd-dataframe"] dataframe = ["dep:nu-cmd-dataframe"]
extra = ["dep:nu-cmd-extra"]
plugin = ["nu-parser/plugin"] plugin = ["nu-parser/plugin"]
sqlite = [ sqlite = [
"rusqlite", "rusqlite",

View file

@ -4,6 +4,9 @@ use crate::*;
#[cfg(feature = "dataframe")] #[cfg(feature = "dataframe")]
use nu_cmd_dataframe::*; use nu_cmd_dataframe::*;
#[cfg(feature = "extra")]
use nu_cmd_extra::*;
pub fn create_default_context() -> EngineState { pub fn create_default_context() -> EngineState {
let mut engine_state = nu_cmd_lang::create_default_context(); let mut engine_state = nu_cmd_lang::create_default_context();
@ -20,6 +23,10 @@ pub fn create_default_context() -> EngineState {
// they have to be registered before the main declarations. This helps to make // they have to be registered before the main declarations. This helps to make
// them only accessible if the correct input value category is used with the // them only accessible if the correct input value category is used with the
// declaration // declaration
#[cfg(feature = "extra")]
add_extra_decls(&mut working_set);
#[cfg(feature = "dataframe")] #[cfg(feature = "dataframe")]
add_dataframe_decls(&mut working_set); add_dataframe_decls(&mut working_set);
@ -205,19 +212,6 @@ pub fn create_default_context() -> EngineState {
StrUpcase StrUpcase
}; };
// Bits
bind_command! {
Bits,
BitsAnd,
BitsNot,
BitsOr,
BitsXor,
BitsRotateLeft,
BitsRotateRight,
BitsShiftLeft,
BitsShiftRight,
}
// Bytes // Bytes
bind_command! { bind_command! {
Bytes, Bytes,

View file

@ -1,4 +1,3 @@
mod bits;
mod bytes; mod bytes;
mod charting; mod charting;
mod conversions; mod conversions;
@ -30,7 +29,6 @@ mod system;
pub mod util; pub mod util;
mod viewers; mod viewers;
pub use bits::*;
pub use bytes::*; pub use bytes::*;
pub use charting::*; pub use charting::*;
pub use conversions::*; pub use conversions::*;

View file

@ -1,3 +1,4 @@
#[cfg(feature = "extra")]
mod test_bits; mod test_bits;
mod test_cell_path; mod test_cell_path;
mod test_commandline; mod test_commandline;