Clean up lint errors

This commit is contained in:
Yehuda Katz 2019-07-23 21:10:48 -07:00
parent 5a8e041a48
commit 73deeb69db
24 changed files with 253 additions and 124 deletions

24
Cargo.lock generated
View File

@ -1705,6 +1705,7 @@ dependencies = [
"syntect 3.2.0 (registry+https://github.com/rust-lang/crates.io-index)", "syntect 3.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
"sys-info 0.5.7 (registry+https://github.com/rust-lang/crates.io-index)", "sys-info 0.5.7 (registry+https://github.com/rust-lang/crates.io-index)",
"sysinfo 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)", "sysinfo 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)",
"tempdir 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)",
"tempfile 3.1.0 (registry+https://github.com/rust-lang/crates.io-index)", "tempfile 3.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
"term 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)", "term 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)",
"toml 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)", "toml 0.5.1 (registry+https://github.com/rust-lang/crates.io-index)",
@ -2146,6 +2147,18 @@ dependencies = [
"proc-macro2 0.4.30 (registry+https://github.com/rust-lang/crates.io-index)", "proc-macro2 0.4.30 (registry+https://github.com/rust-lang/crates.io-index)",
] ]
[[package]]
name = "rand"
version = "0.4.6"
source = "registry+https://github.com/rust-lang/crates.io-index"
dependencies = [
"fuchsia-cprng 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
"libc 0.2.58 (registry+https://github.com/rust-lang/crates.io-index)",
"rand_core 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)",
"rdrand 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
"winapi 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)",
]
[[package]] [[package]]
name = "rand" name = "rand"
version = "0.6.5" version = "0.6.5"
@ -2829,6 +2842,15 @@ dependencies = [
"winapi 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)", "winapi 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)",
] ]
[[package]]
name = "tempdir"
version = "0.3.7"
source = "registry+https://github.com/rust-lang/crates.io-index"
dependencies = [
"rand 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
"remove_dir_all 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)",
]
[[package]] [[package]]
name = "tempfile" name = "tempfile"
version = "3.1.0" version = "3.1.0"
@ -3627,6 +3649,7 @@ dependencies = [
"checksum publicsuffix 1.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "5afecba86dcf1e4fd610246f89899d1924fe12e1e89f555eb7c7f710f3c5ad1d" "checksum publicsuffix 1.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "5afecba86dcf1e4fd610246f89899d1924fe12e1e89f555eb7c7f710f3c5ad1d"
"checksum quick-error 1.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "9274b940887ce9addde99c4eee6b5c44cc494b182b97e73dc8ffdcb3397fd3f0" "checksum quick-error 1.2.2 (registry+https://github.com/rust-lang/crates.io-index)" = "9274b940887ce9addde99c4eee6b5c44cc494b182b97e73dc8ffdcb3397fd3f0"
"checksum quote 0.6.12 (registry+https://github.com/rust-lang/crates.io-index)" = "faf4799c5d274f3868a4aae320a0a182cbd2baee377b378f080e16a23e9d80db" "checksum quote 0.6.12 (registry+https://github.com/rust-lang/crates.io-index)" = "faf4799c5d274f3868a4aae320a0a182cbd2baee377b378f080e16a23e9d80db"
"checksum rand 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)" = "552840b97013b1a26992c11eac34bdd778e464601a4c2054b5f0bff7c6761293"
"checksum rand 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)" = "6d71dacdc3c88c1fde3885a3be3fbab9f35724e6ce99467f7d9c5026132184ca" "checksum rand 0.6.5 (registry+https://github.com/rust-lang/crates.io-index)" = "6d71dacdc3c88c1fde3885a3be3fbab9f35724e6ce99467f7d9c5026132184ca"
"checksum rand 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)" = "d47eab0e83d9693d40f825f86948aa16eff6750ead4bdffc4ab95b8b3a7f052c" "checksum rand 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)" = "d47eab0e83d9693d40f825f86948aa16eff6750ead4bdffc4ab95b8b3a7f052c"
"checksum rand_chacha 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "556d3a1ca6600bfcbab7c7c91ccb085ac7fbbcd70e008a98742e7847f4f7bcef" "checksum rand_chacha 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)" = "556d3a1ca6600bfcbab7c7c91ccb085ac7fbbcd70e008a98742e7847f4f7bcef"
@ -3705,6 +3728,7 @@ dependencies = [
"checksum syntect 3.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "e80b8831c5a543192ffc3727f01cf0e57579c6ac15558e3048bfb5708892167b" "checksum syntect 3.2.0 (registry+https://github.com/rust-lang/crates.io-index)" = "e80b8831c5a543192ffc3727f01cf0e57579c6ac15558e3048bfb5708892167b"
"checksum sys-info 0.5.7 (registry+https://github.com/rust-lang/crates.io-index)" = "76d6cf7b349b6a6daaf7a3797227e2f4108c8dd398e0aca7e29b9fb239948541" "checksum sys-info 0.5.7 (registry+https://github.com/rust-lang/crates.io-index)" = "76d6cf7b349b6a6daaf7a3797227e2f4108c8dd398e0aca7e29b9fb239948541"
"checksum sysinfo 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)" = "c3e2cab189e59f72710e3dd5e1e0d5be0f6c5c999c326f2fdcdf3bf4483ec9fd" "checksum sysinfo 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)" = "c3e2cab189e59f72710e3dd5e1e0d5be0f6c5c999c326f2fdcdf3bf4483ec9fd"
"checksum tempdir 0.3.7 (registry+https://github.com/rust-lang/crates.io-index)" = "15f2b5fb00ccdf689e0149d1b1b3c03fead81c2b37735d812fa8bddbbf41b6d8"
"checksum tempfile 3.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "7a6e24d9338a0a5be79593e2fa15a648add6138caa803e2d5bc782c371732ca9" "checksum tempfile 3.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "7a6e24d9338a0a5be79593e2fa15a648add6138caa803e2d5bc782c371732ca9"
"checksum term 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "edd106a334b7657c10b7c540a0106114feadeb4dc314513e97df481d5d966f42" "checksum term 0.5.2 (registry+https://github.com/rust-lang/crates.io-index)" = "edd106a334b7657c10b7c540a0106114feadeb4dc314513e97df481d5d966f42"
"checksum termcolor 1.0.5 (registry+https://github.com/rust-lang/crates.io-index)" = "96d6098003bde162e4277c70665bd87c326f5a0c3f3fbfb285787fa482d54e6e" "checksum termcolor 1.0.5 (registry+https://github.com/rust-lang/crates.io-index)" = "96d6098003bde162e4277c70665bd87c326f5a0c3f3fbfb285787fa482d54e6e"

View File

@ -81,6 +81,7 @@ syntect = "3.2.0"
[dev-dependencies] [dev-dependencies]
pretty_assertions = "0.6.1" pretty_assertions = "0.6.1"
tempdir = "0.3.7"
[lib] [lib]
name = "nu" name = "nu"

View File

@ -42,7 +42,7 @@ crate mod trim;
crate mod vtable; crate mod vtable;
crate mod where_; crate mod where_;
crate use command::{command, filter, EvaluatedFilterCommandArgs, EvaluatedStaticCommandArgs}; crate use command::{command, EvaluatedStaticCommandArgs};
crate use config::Config; crate use config::Config;
crate use open::Open; crate use open::Open;
crate use rm::Remove; crate use rm::Remove;

View File

@ -27,18 +27,6 @@ impl ToDebug for UnevaluatedCallInfo {
} }
impl UnevaluatedCallInfo { impl UnevaluatedCallInfo {
fn name(&self) -> Result<&str, ShellError> {
let head = &self.args.head();
match head.item() {
hir::RawExpression::Literal(hir::Literal::Bare) => Ok(head.span.slice(&self.source)),
hir::RawExpression::Literal(hir::Literal::String(span)) => Ok(span.slice(&self.source)),
other => Err(ShellError::type_error(
"Command name",
head.type_name().spanned(head.span),
)),
}
}
fn evaluate( fn evaluate(
self, self,
registry: &registry::CommandRegistry, registry: &registry::CommandRegistry,
@ -94,20 +82,6 @@ impl CommandArgs {
} }
} }
pub enum EvaluatedInput {
Static(InputStream),
Filter(Spanned<Value>),
}
impl EvaluatedInput {
pub fn stream(self) -> InputStream {
match self {
EvaluatedInput::Static(stream) => stream,
EvaluatedInput::Filter(value) => vec![value].into(),
}
}
}
pub struct EvaluatedStaticCommandArgs { pub struct EvaluatedStaticCommandArgs {
pub args: EvaluatedCommandArgs, pub args: EvaluatedCommandArgs,
pub input: InputStream, pub input: InputStream,
@ -152,6 +126,7 @@ impl EvaluatedStaticCommandArgs {
#[get = "pub"] #[get = "pub"]
pub struct EvaluatedFilterCommandArgs { pub struct EvaluatedFilterCommandArgs {
args: EvaluatedCommandArgs, args: EvaluatedCommandArgs,
#[allow(unused)]
input: Spanned<Value>, input: Spanned<Value>,
} }
@ -189,8 +164,6 @@ pub struct EvaluatedCommandArgs {
} }
impl EvaluatedCommandArgs { impl EvaluatedCommandArgs {
pub fn parts(self) -> () {}
pub fn call_args(&self) -> &registry::EvaluatedArgs { pub fn call_args(&self) -> &registry::EvaluatedArgs {
&self.call_info.args &self.call_info.args
} }
@ -199,10 +172,6 @@ impl EvaluatedCommandArgs {
self.call_info.args.nth(pos) self.call_info.args.nth(pos)
} }
pub fn positional_iter(&self) -> impl Iterator<Item = &Spanned<Value>> {
self.call_info.args.positional_iter()
}
pub fn expect_nth(&self, pos: usize) -> Result<&Spanned<Value>, ShellError> { pub fn expect_nth(&self, pos: usize) -> Result<&Spanned<Value>, ShellError> {
self.call_info.args.expect_nth(pos) self.call_info.args.expect_nth(pos)
} }
@ -308,6 +277,7 @@ pub trait Sink {
} }
} }
#[allow(unused)]
pub struct FnFilterCommand { pub struct FnFilterCommand {
name: String, name: String,
func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>, func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>,
@ -397,6 +367,7 @@ pub fn command(
}) })
} }
#[allow(unused)]
pub fn filter( pub fn filter(
name: &str, name: &str,
func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>, func: fn(EvaluatedFilterCommandArgs) -> Result<OutputStream, ShellError>,

View File

@ -2,6 +2,6 @@ use crate::commands::command::CommandAction;
use crate::errors::ShellError; use crate::errors::ShellError;
use crate::prelude::*; use crate::prelude::*;
pub fn exit(_args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> { pub fn exit(_args: CommandArgs, _registry: &CommandRegistry) -> Result<OutputStream, ShellError> {
Ok(vec![Ok(ReturnSuccess::Action(CommandAction::Exit))].into()) Ok(vec![Ok(ReturnSuccess::Action(CommandAction::Exit))].into())
} }

View File

@ -50,7 +50,7 @@ pub fn from_yaml_string_to_value(
pub fn from_yaml( pub fn from_yaml(
args: CommandArgs, args: CommandArgs,
registry: &CommandRegistry, _registry: &CommandRegistry,
) -> Result<OutputStream, ShellError> { ) -> Result<OutputStream, ShellError> {
let span = args.name_span(); let span = args.name_span();
let out = args.input; let out = args.input;

View File

@ -1,11 +1,10 @@
use crate::context::SpanSource; use crate::context::SpanSource;
use crate::errors::ShellError; use crate::errors::ShellError;
use crate::object::{Primitive, Switch, Value}; use crate::object::{Primitive, Value};
use crate::parser::hir::SyntaxType; use crate::parser::hir::SyntaxType;
use crate::parser::parse::span::Span; use crate::parser::parse::span::Span;
use crate::parser::registry::{self, CommandConfig, NamedType, PositionalType}; use crate::parser::registry::{self, CommandConfig, NamedType};
use crate::prelude::*; use crate::prelude::*;
use indexmap::IndexMap;
use mime::Mime; use mime::Mime;
use std::path::{Path, PathBuf}; use std::path::{Path, PathBuf};
use std::str::FromStr; use std::str::FromStr;
@ -77,17 +76,10 @@ impl Command for Open {
} }
fn config(&self) -> CommandConfig { fn config(&self) -> CommandConfig {
let mut named = IndexMap::default(); CommandConfig::new(self.name())
named.insert("raw".to_string(), NamedType::Switch); .required("path", SyntaxType::Block)
.named("raw", NamedType::Switch)
CommandConfig { .sink()
name: self.name().to_string(),
positional: vec![PositionalType::mandatory("path", SyntaxType::Block)],
rest_positional: false,
named,
is_sink: true,
is_filter: false,
}
} }
} }

View File

@ -3,7 +3,7 @@ use crate::object::process::process_dict;
use crate::prelude::*; use crate::prelude::*;
use sysinfo::{RefreshKind, SystemExt}; use sysinfo::{RefreshKind, SystemExt};
pub fn ps(args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> { pub fn ps(args: CommandArgs, _registry: &CommandRegistry) -> Result<OutputStream, ShellError> {
let mut system = sysinfo::System::new_with_specifics(RefreshKind::new().with_processes()); let mut system = sysinfo::System::new_with_specifics(RefreshKind::new().with_processes());
system.refresh_processes(); system.refresh_processes();
let list = system.get_process_list(); let list = system.get_process_list();

View File

@ -2,7 +2,7 @@ use crate::errors::ShellError;
use crate::object::{SpannedDictBuilder, Value}; use crate::object::{SpannedDictBuilder, Value};
use crate::prelude::*; use crate::prelude::*;
pub fn size(args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> { pub fn size(args: CommandArgs, _registry: &CommandRegistry) -> Result<OutputStream, ShellError> {
let input = args.input; let input = args.input;
Ok(input Ok(input
.values .values

View File

@ -6,7 +6,7 @@ use crate::prelude::*;
use sys_info::*; use sys_info::*;
use sysinfo::{ComponentExt, DiskExt, NetworkExt, RefreshKind, SystemExt}; use sysinfo::{ComponentExt, DiskExt, NetworkExt, RefreshKind, SystemExt};
pub fn sysinfo(args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> { pub fn sysinfo(args: CommandArgs, _registry: &CommandRegistry) -> Result<OutputStream, ShellError> {
let name_span = args.name_span(); let name_span = args.name_span();
let mut idx = SpannedDictBuilder::new(name_span); let mut idx = SpannedDictBuilder::new(name_span);

View File

@ -1,7 +1,10 @@
use crate::object::Value; use crate::object::Value;
use crate::prelude::*; use crate::prelude::*;
pub fn to_array(args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> { pub fn to_array(
args: CommandArgs,
_registry: &CommandRegistry,
) -> Result<OutputStream, ShellError> {
let out = args.input.values.collect(); let out = args.input.values.collect();
Ok(out Ok(out

View File

@ -2,7 +2,7 @@ use crate::errors::ShellError;
use crate::object::Value; use crate::object::Value;
use crate::prelude::*; use crate::prelude::*;
pub fn trim(args: CommandArgs, registry: &CommandRegistry) -> Result<OutputStream, ShellError> { pub fn trim(args: CommandArgs, _registry: &CommandRegistry) -> Result<OutputStream, ShellError> {
let input = args.input; let input = args.input;
Ok(input Ok(input

View File

@ -1,6 +1,5 @@
use crate::errors::ShellError; use crate::errors::ShellError;
use crate::object::base as value; use crate::object::base as value;
use crate::object::{types, Value};
use crate::parser::hir::SyntaxType; use crate::parser::hir::SyntaxType;
use crate::parser::registry::{self, CommandConfig, PositionalType}; use crate::parser::registry::{self, CommandConfig, PositionalType};
use crate::prelude::*; use crate::prelude::*;

View File

@ -1,9 +1,5 @@
use crate::commands::command::{CallInfo, Sink, SinkCommandArgs, UnevaluatedCallInfo}; use crate::commands::command::{CallInfo, Sink, SinkCommandArgs, UnevaluatedCallInfo};
use crate::parser::{ use crate::parser::{hir, registry, Span};
hir,
registry::{self, CommandConfig},
Span,
};
use crate::prelude::*; use crate::prelude::*;
use derive_new::new; use derive_new::new;
@ -51,12 +47,6 @@ impl CommandRegistry {
} }
} }
fn get_config(&self, name: &str) -> Option<CommandConfig> {
let registry = self.registry.lock().unwrap();
registry.get(name).map(|c| c.config())
}
fn get_command(&self, name: &str) -> Option<Arc<dyn Command>> { fn get_command(&self, name: &str) -> Option<Arc<dyn Command>> {
let registry = self.registry.lock().unwrap(); let registry = self.registry.lock().unwrap();
@ -75,7 +65,7 @@ impl CommandRegistry {
} }
crate fn names(&self) -> Vec<String> { crate fn names(&self) -> Vec<String> {
let mut registry = self.registry.lock().unwrap(); let registry = self.registry.lock().unwrap();
registry.keys().cloned().collect() registry.keys().cloned().collect()
} }
} }

View File

@ -24,6 +24,7 @@ mod plugin;
mod shell; mod shell;
mod stream; mod stream;
mod traits; mod traits;
mod utils;
pub use crate::commands::command::{CallInfo, ReturnSuccess, ReturnValue}; pub use crate::commands::command::{CallInfo, ReturnSuccess, ReturnValue};
pub use crate::context::SpanSource; pub use crate::context::SpanSource;
@ -31,6 +32,7 @@ pub use crate::env::host::BasicHost;
pub use crate::parser::parse::span::SpannedItem; pub use crate::parser::parse::span::SpannedItem;
pub use crate::parser::Spanned; pub use crate::parser::Spanned;
pub use crate::plugin::{serve_plugin, Plugin}; pub use crate::plugin::{serve_plugin, Plugin};
pub use crate::utils::{AbsolutePath, RelativePath};
pub use cli::cli; pub use cli::cli;
pub use errors::ShellError; pub use errors::ShellError;
pub use object::base::{Primitive, Value}; pub use object::base::{Primitive, Value};

View File

@ -6,6 +6,6 @@ crate mod into;
crate mod process; crate mod process;
crate mod types; crate mod types;
crate use base::{Block, Primitive, Switch, Value}; crate use base::{Primitive, Value};
crate use dict::{Dictionary, SpannedDictBuilder}; crate use dict::{Dictionary, SpannedDictBuilder};
crate use files::dir_entry_dict; crate use files::dir_entry_dict;

View File

@ -266,6 +266,7 @@ pub enum Switch {
} }
impl Switch { impl Switch {
#[allow(unused)]
pub fn is_present(&self) -> bool { pub fn is_present(&self) -> bool {
match self { match self {
Switch::Present => true, Switch::Present => true,

View File

@ -77,7 +77,7 @@ fn parse_command_tail(
trace_remaining("nodes", tail.clone(), source); trace_remaining("nodes", tail.clone(), source);
for (name, kind) in config.named() { for (name, kind) in &config.named {
trace!(target: "nu::parse", "looking for {} : {:?}", name, kind); trace!(target: "nu::parse", "looking for {} : {:?}", name, kind);
match kind { match kind {
@ -115,7 +115,7 @@ fn parse_command_tail(
if tail.at_end() { if tail.at_end() {
return Err(ShellError::argument_error( return Err(ShellError::argument_error(
config.name().clone(), config.name.clone(),
ArgumentError::MissingValueForName(name.to_string()), ArgumentError::MissingValueForName(name.to_string()),
flag.span, flag.span,
)); ));
@ -139,14 +139,14 @@ fn parse_command_tail(
let mut positional = vec![]; let mut positional = vec![];
for arg in config.positional() { for arg in &config.positional {
trace!("Processing positional {:?}", arg); trace!("Processing positional {:?}", arg);
match arg { match arg {
PositionalType::Mandatory(..) => { PositionalType::Mandatory(..) => {
if tail.len() == 0 { if tail.len() == 0 {
return Err(ShellError::argument_error( return Err(ShellError::argument_error(
config.name().clone(), config.name.clone(),
ArgumentError::MissingMandatoryPositional(arg.name().to_string()), ArgumentError::MissingMandatoryPositional(arg.name().to_string()),
command_span, command_span,
)); ));
@ -207,7 +207,7 @@ fn extract_mandatory(
match flag { match flag {
None => Err(ShellError::argument_error( None => Err(ShellError::argument_error(
config.name().clone(), config.name.clone(),
ArgumentError::MissingMandatoryFlag(name.to_string()), ArgumentError::MissingMandatoryFlag(name.to_string()),
span, span,
)), )),

View File

@ -4,7 +4,6 @@ use crate::evaluate::{evaluate_baseline_expr, Scope};
use crate::parser::{hir, hir::SyntaxType, parse_command, CallNode, Spanned}; use crate::parser::{hir, hir::SyntaxType, parse_command, CallNode, Spanned};
use crate::prelude::*; use crate::prelude::*;
use derive_new::new; use derive_new::new;
use getset::Getters;
use indexmap::IndexMap; use indexmap::IndexMap;
use log::trace; use log::trace;
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
@ -70,8 +69,7 @@ impl PositionalType {
} }
} }
#[derive(Debug, Getters, Serialize, Deserialize, Clone)] #[derive(Debug, Serialize, Deserialize, Clone)]
#[get = "crate"]
pub struct CommandConfig { pub struct CommandConfig {
pub name: String, pub name: String,
pub positional: Vec<PositionalType>, pub positional: Vec<PositionalType>,
@ -81,6 +79,44 @@ pub struct CommandConfig {
pub is_sink: bool, pub is_sink: bool,
} }
impl CommandConfig {
pub fn new(name: impl Into<String>) -> CommandConfig {
CommandConfig {
name: name.into(),
positional: vec![],
rest_positional: false,
named: IndexMap::default(),
is_filter: false,
is_sink: false,
}
}
pub fn required(mut self, name: impl Into<String>, ty: impl Into<SyntaxType>) -> CommandConfig {
self.positional
.push(PositionalType::Mandatory(name.into(), ty.into()));
self
}
pub fn optional(mut self, name: impl Into<String>, ty: impl Into<SyntaxType>) -> CommandConfig {
self.positional
.push(PositionalType::Optional(name.into(), ty.into()));
self
}
pub fn named(mut self, name: impl Into<String>, ty: impl Into<NamedType>) -> CommandConfig {
self.named.insert(name.into(), ty.into());
self
}
pub fn sink(mut self) -> CommandConfig {
self.is_sink = true;
self
}
}
#[derive(Debug, Default, new, Serialize, Deserialize)] #[derive(Debug, Default, new, Serialize, Deserialize)]
pub struct EvaluatedArgs { pub struct EvaluatedArgs {
pub positional: Option<Vec<Spanned<Value>>>, pub positional: Option<Vec<Spanned<Value>>>,

View File

@ -34,8 +34,7 @@ macro_rules! trace_stream {
crate use crate::cli::MaybeOwned; crate use crate::cli::MaybeOwned;
crate use crate::commands::command::{ crate use crate::commands::command::{
Command, CommandAction, CommandArgs, EvaluatedCommandArgs, ReturnSuccess, ReturnValue, Sink, Command, CommandAction, CommandArgs, ReturnSuccess, ReturnValue, Sink, SinkCommandArgs,
SinkCommandArgs,
}; };
crate use crate::context::{CommandRegistry, Context}; crate use crate::context::{CommandRegistry, Context};
crate use crate::env::host::handle_unexpected; crate use crate::env::host::handle_unexpected;

View File

@ -1,5 +1,4 @@
use crate::context::CommandRegistry; use crate::context::CommandRegistry;
use crate::prelude::*;
use derive_new::new; use derive_new::new;
use rustyline::completion::Completer; use rustyline::completion::Completer;

62
src/utils.rs Normal file
View File

@ -0,0 +1,62 @@
use std::ops::Div;
use std::path::{Path, PathBuf};
pub struct AbsolutePath {
inner: PathBuf,
}
impl AbsolutePath {
pub fn new(path: impl AsRef<Path>) -> AbsolutePath {
let path = path.as_ref();
if path.is_absolute() {
AbsolutePath {
inner: path.to_path_buf(),
}
} else {
panic!("AbsolutePath::new must take an absolute path")
}
}
}
impl Div<&str> for AbsolutePath {
type Output = AbsolutePath;
fn div(self, rhs: &str) -> Self::Output {
AbsolutePath {
inner: self.inner.join(rhs),
}
}
}
impl AsRef<Path> for AbsolutePath {
fn as_ref(&self) -> &Path {
self.inner.as_path()
}
}
pub struct RelativePath {
inner: PathBuf,
}
impl RelativePath {
pub fn new(path: impl Into<PathBuf>) -> RelativePath {
let path = path.into();
if path.is_relative() {
RelativePath { inner: path }
} else {
panic!("RelativePath::new must take a relative path")
}
}
}
impl<T: AsRef<str>> Div<T> for RelativePath {
type Output = RelativePath;
fn div(self, rhs: T) -> Self::Output {
RelativePath {
inner: self.inner.join(rhs.as_ref()),
}
}
}

View File

@ -3,6 +3,8 @@ mod helpers;
use h::in_directory as cwd; use h::in_directory as cwd;
use helpers as h; use helpers as h;
use nu::AbsolutePath;
#[test] #[test]
fn lines() { fn lines() {
nu!(output, nu!(output,
@ -80,73 +82,102 @@ fn open_error_if_file_not_found() {
} }
#[test] #[test]
fn save_can_write_out_csv() { fn save_can_write_out_csv() -> Result<(), std::io::Error> {
let (playground_path, tests_dir) = h::setup_playground_for("save_test"); let (playground, tmp, dir) = h::setup_playground_for("save_test")?;
let full_path = format!("{}/{}", playground_path, tests_dir ); let tmp = AbsolutePath::new(tmp);
let expected_file = format!("{}/{}", full_path , "cargo_sample.csv");
nu!( let expected_file = tmp / "cargo_sample.csv";
_output,
cwd(&playground_path), let root = AbsolutePath::new(std::env::current_dir()?);
"open ../formats/cargo_sample.toml | inc package.version --minor | get package | save save_test/cargo_sample.csv"
let path = root / "tests" / "fixtures" / "formats" / "cargo_sample.toml";
let command = format!(
"open {} | inc package.version --minor | get package | save {}/cargo_sample.csv",
path.as_ref().display(),
dir
); );
nu!(_output, playground.path().display(), command);
let actual = h::file_contents(&expected_file); let actual = h::file_contents(&expected_file);
assert!(actual.contains("[list list],A shell for the GitHub era,2018,ISC,nu,0.2.0")); assert!(actual.contains("[list list],A shell for the GitHub era,2018,ISC,nu,0.2.0"));
Ok(())
} }
#[test] #[test]
fn rm_can_remove_a_file() { fn rm_can_remove_a_file() -> Result<(), std::io::Error> {
let directory = "tests/fixtures/nuplayground"; let _ = pretty_env_logger::try_init();
let file = format!("{}/rm_test.txt", directory);
h::create_file_at(&file); let (_playground, tmp, _) = h::setup_playground_for("remove_file")?;
nu!(_output, cwd(directory), "rm rm_test.txt"); let file = AbsolutePath::new(&tmp) / "rm_test.txt";
// let file = tmp.path().join("rm_test.txt");
assert!(!h::file_exists_at(&file)); h::create_file_at(&file)?;
nu!(_output, tmp.path().display(), "rm rm_test.txt");
assert!(!file.as_ref().exists());
Ok(())
} }
#[test] #[test]
fn rm_can_remove_directory_contents_with_recursive_flag() { fn rm_can_remove_directory_contents_with_recursive_flag() -> Result<(), std::io::Error> {
let (playground_path, tests_dir) = h::setup_playground_for("rm_test"); let _ = pretty_env_logger::try_init();
let (playground, tmp, dir) = h::setup_playground_for("rm_test")?;
for f in ["yehuda.txt", "jonathan.txt", "andres.txt"].iter() { for f in ["yehuda.txt", "jonathan.txt", "andres.txt"].iter() {
h::create_file_at(&format!("{}/{}/{}", playground_path, tests_dir, f)); h::create_file_at(&tmp.path().join(f))?;
} }
nu!( nu!(
_output, _output,
cwd("tests/fixtures/nuplayground"), playground.path().display(),
"rm rm_test --recursive" format!("rm {} --recursive", dir)
); );
assert!(!h::file_exists_at(&format!("{}/{}", playground_path, tests_dir))); assert!(!tmp.path().exists());
Ok(())
} }
#[test] #[test]
fn rm_error_if_attempting_to_delete_a_directory_without_recursive_flag() { fn rm_error_if_attempting_to_delete_a_directory_without_recursive_flag(
let (playground_path, tests_dir) = h::setup_playground_for("rm_test_2"); ) -> Result<(), std::io::Error> {
let full_path = format!("{}/{}", playground_path, tests_dir); let (playground, tmp, dir) = h::setup_playground_for("rm_test_2")?;
nu_error!(output, cwd("tests/fixtures/nuplayground"), "rm rm_test_2"); nu_error!(output, playground.path().display(), format!("rm {}", dir));
assert!(h::file_exists_at(&full_path)); assert!(tmp.path().exists());
assert!(output.contains("is a directory")); assert!(output.contains("is a directory"));
h::delete_directory_at(&full_path); h::delete_directory_at(tmp.path());
Ok(())
} }
#[test] #[test]
fn rm_error_if_attempting_to_delete_single_dot_as_argument() { fn rm_error_if_attempting_to_delete_single_dot_as_argument() -> Result<(), std::io::Error> {
nu_error!(output, cwd("tests/fixtures/nuplayground"), "rm ."); let (_playground, tmp, _) = h::setup_playground_for("rm_test_2")?;
nu_error!(output, tmp.path().display(), "rm .");
assert!(output.contains("may not be removed")); assert!(output.contains("may not be removed"));
Ok(())
} }
#[test] #[test]
fn rm_error_if_attempting_to_delete_two_dot_as_argument() { fn rm_error_if_attempting_to_delete_two_dot_as_argument() -> Result<(), std::io::Error> {
nu_error!(output, cwd("tests/fixtures/nuplayground"), "rm .."); let (_playground, tmp, _) = h::setup_playground_for("rm_test_2")?;
nu_error!(output, tmp.path().display(), "rm ..");
assert!(output.contains("may not be removed")); assert!(output.contains("may not be removed"));
Ok(())
} }

View File

@ -1,8 +1,10 @@
#![allow(dead_code)] #![allow(dead_code)]
pub use std::path::PathBuf; pub use std::path::{Path, PathBuf};
use log::trace;
use std::io::Read; use std::io::Read;
use tempdir::TempDir;
#[macro_export] #[macro_export]
macro_rules! nu { macro_rules! nu {
@ -79,20 +81,30 @@ macro_rules! nu_error {
}; };
} }
pub fn setup_playground_for(topic: &str) -> (String, String) { pub fn setup_playground_for(topic: &str) -> Result<(TempDir, TempDir, String), std::io::Error> {
let home = "tests/fixtures/nuplayground"; let home = TempDir::new("nuplayground")?;
let full_path = format!("{}/{}", home, topic); let child = TempDir::new_in(home.path(), topic)?;
let relative = child
.path()
.file_name()
.unwrap()
.to_str()
.expect(&format!(
"file name {} was not valid",
child.path().display()
))
.to_string();
if file_exists_at(&full_path) { trace!(
delete_directory_at(&full_path); "created {:?} dir={}",
child.path().display(),
child.path().is_dir()
);
Ok((home, child, relative))
} }
create_directory_at(&full_path); pub fn file_contents(full_path: impl AsRef<Path>) -> String {
(home.to_string(), topic.to_string())
}
pub fn file_contents(full_path: &str) -> String {
let mut file = std::fs::File::open(full_path).expect("can not open file"); let mut file = std::fs::File::open(full_path).expect("can not open file");
let mut contents = String::new(); let mut contents = String::new();
file.read_to_string(&mut contents) file.read_to_string(&mut contents)
@ -100,19 +112,26 @@ pub fn file_contents(full_path: &str) -> String {
contents contents
} }
pub fn create_file_at(full_path: &str) { pub fn create_file_at(full_path: impl AsRef<Path>) -> Result<(), std::io::Error> {
std::fs::write(PathBuf::from(full_path), "fake data".as_bytes()).expect("can not create file"); let full_path = full_path.as_ref();
assert!(
full_path.parent().unwrap().is_dir(),
"{:?} exists",
full_path.parent().unwrap().display(),
);
std::fs::write(full_path, "fake data".as_bytes())
} }
pub fn file_exists_at(full_path: &str) -> bool { pub fn file_exists_at(full_path: &str) -> bool {
PathBuf::from(full_path).exists() PathBuf::from(full_path).exists()
} }
pub fn delete_directory_at(full_path: &str) { pub fn delete_directory_at(full_path: &Path) {
std::fs::remove_dir_all(PathBuf::from(full_path)).expect("can not remove directory"); std::fs::remove_dir_all(PathBuf::from(full_path)).expect("can not remove directory");
} }
pub fn create_directory_at(full_path: &str) { pub fn create_directory_at(full_path: &Path) {
let path = PathBuf::from(full_path); let path = PathBuf::from(full_path);
println!("{:?} - is_dir: {:?}", path, path.is_dir()); println!("{:?} - is_dir: {:?}", path, path.is_dir());