forked from extern/nushell
Add umkdir
command (#10785)
A `mkdir` command, which uses `uu_mkdir` as backend. close #10515.
This commit is contained in:
parent
3f61ca19f0
commit
72f7b9b7cc
11
Cargo.lock
generated
11
Cargo.lock
generated
@ -2884,6 +2884,7 @@ dependencies = [
|
|||||||
"ureq",
|
"ureq",
|
||||||
"url",
|
"url",
|
||||||
"uu_cp",
|
"uu_cp",
|
||||||
|
"uu_mkdir",
|
||||||
"uu_whoami",
|
"uu_whoami",
|
||||||
"uuid",
|
"uuid",
|
||||||
"wax",
|
"wax",
|
||||||
@ -5621,6 +5622,16 @@ dependencies = [
|
|||||||
"xattr",
|
"xattr",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "uu_mkdir"
|
||||||
|
version = "0.0.22"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "4776960a036a4ec375f0701004a41013d66d2e3e46a19e9216fd18d4d92f88f3"
|
||||||
|
dependencies = [
|
||||||
|
"clap",
|
||||||
|
"uucore",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "uu_whoami"
|
name = "uu_whoami"
|
||||||
version = "0.0.22"
|
version = "0.0.22"
|
||||||
|
@ -89,6 +89,7 @@ ureq = { version = "2.8", default-features = false, features = ["charset", "gzip
|
|||||||
url = "2.2"
|
url = "2.2"
|
||||||
uu_cp = "0.0.22"
|
uu_cp = "0.0.22"
|
||||||
uu_whoami = "0.0.22"
|
uu_whoami = "0.0.22"
|
||||||
|
uu_mkdir = "0.0.22"
|
||||||
uuid = { version = "1.5", features = ["v4"] }
|
uuid = { version = "1.5", features = ["v4"] }
|
||||||
wax = { version = "0.6" }
|
wax = { version = "0.6" }
|
||||||
which = { version = "5.0", optional = true }
|
which = { version = "5.0", optional = true }
|
||||||
|
@ -204,6 +204,7 @@ pub fn add_shell_command_context(mut engine_state: EngineState) -> EngineState {
|
|||||||
Cd,
|
Cd,
|
||||||
Ls,
|
Ls,
|
||||||
Mkdir,
|
Mkdir,
|
||||||
|
UMkdir,
|
||||||
Mv,
|
Mv,
|
||||||
Cp,
|
Cp,
|
||||||
UCp,
|
UCp,
|
||||||
|
@ -10,6 +10,7 @@ mod save;
|
|||||||
mod start;
|
mod start;
|
||||||
mod touch;
|
mod touch;
|
||||||
mod ucp;
|
mod ucp;
|
||||||
|
mod umkdir;
|
||||||
mod util;
|
mod util;
|
||||||
mod watch;
|
mod watch;
|
||||||
|
|
||||||
@ -25,4 +26,5 @@ pub use save::Save;
|
|||||||
pub use start::Start;
|
pub use start::Start;
|
||||||
pub use touch::Touch;
|
pub use touch::Touch;
|
||||||
pub use ucp::UCp;
|
pub use ucp::UCp;
|
||||||
|
pub use umkdir::UMkdir;
|
||||||
pub use watch::Watch;
|
pub use watch::Watch;
|
||||||
|
98
crates/nu-command/src/filesystem/umkdir.rs
Normal file
98
crates/nu-command/src/filesystem/umkdir.rs
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
use nu_engine::env::current_dir;
|
||||||
|
use nu_engine::CallExt;
|
||||||
|
use nu_protocol::ast::Call;
|
||||||
|
use nu_protocol::engine::{Command, EngineState, Stack};
|
||||||
|
use nu_protocol::{Category, Example, PipelineData, ShellError, Signature, SyntaxShape, Type};
|
||||||
|
|
||||||
|
use uu_mkdir::mkdir;
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct UMkdir;
|
||||||
|
|
||||||
|
const IS_RECURSIVE: bool = true;
|
||||||
|
// This is the same default as Rust's std uses:
|
||||||
|
// https://doc.rust-lang.org/nightly/std/os/unix/fs/trait.DirBuilderExt.html#tymethod.mode
|
||||||
|
const DEFAULT_MODE: u32 = 0o777;
|
||||||
|
|
||||||
|
impl Command for UMkdir {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"umkdir"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Create directories, with intermediary directories if required using uutils/coreutils mkdir."
|
||||||
|
}
|
||||||
|
|
||||||
|
fn search_terms(&self) -> Vec<&str> {
|
||||||
|
vec!["directory", "folder", "create", "make_dirs", "coreutils"]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build("umkdir")
|
||||||
|
.input_output_types(vec![(Type::Nothing, Type::Nothing)])
|
||||||
|
.rest(
|
||||||
|
"rest",
|
||||||
|
SyntaxShape::Directory,
|
||||||
|
"the name(s) of the path(s) to create",
|
||||||
|
)
|
||||||
|
.switch(
|
||||||
|
"verbose",
|
||||||
|
"print a message for each created directory.",
|
||||||
|
Some('v'),
|
||||||
|
)
|
||||||
|
.category(Category::FileSystem)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
engine_state: &EngineState,
|
||||||
|
stack: &mut Stack,
|
||||||
|
call: &Call,
|
||||||
|
_input: PipelineData,
|
||||||
|
) -> Result<PipelineData, ShellError> {
|
||||||
|
let path = current_dir(engine_state, stack)?;
|
||||||
|
let mut directories = call
|
||||||
|
.rest::<String>(engine_state, stack, 0)?
|
||||||
|
.into_iter()
|
||||||
|
.map(|dir| path.join(dir))
|
||||||
|
.peekable();
|
||||||
|
|
||||||
|
let is_verbose = call.has_flag("verbose");
|
||||||
|
|
||||||
|
if directories.peek().is_none() {
|
||||||
|
return Err(ShellError::MissingParameter {
|
||||||
|
param_name: "requires directory paths".to_string(),
|
||||||
|
span: call.head,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
for dir in directories {
|
||||||
|
if let Err(error) = mkdir(&dir, IS_RECURSIVE, DEFAULT_MODE, is_verbose) {
|
||||||
|
return Err(ShellError::GenericError(
|
||||||
|
format!("{}", error),
|
||||||
|
format!("{}", error),
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
Vec::new(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(PipelineData::empty())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn examples(&self) -> Vec<Example> {
|
||||||
|
vec![
|
||||||
|
Example {
|
||||||
|
description: "Make a directory named foo",
|
||||||
|
example: "umkdir foo",
|
||||||
|
result: None,
|
||||||
|
},
|
||||||
|
Example {
|
||||||
|
description: "Make multiple directories and show the paths created",
|
||||||
|
example: "umkdir -v foo/bar foo2",
|
||||||
|
result: None,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
@ -105,6 +105,7 @@ mod touch;
|
|||||||
mod transpose;
|
mod transpose;
|
||||||
mod try_;
|
mod try_;
|
||||||
mod ucp;
|
mod ucp;
|
||||||
|
mod umkdir;
|
||||||
mod uniq;
|
mod uniq;
|
||||||
mod uniq_by;
|
mod uniq_by;
|
||||||
mod update;
|
mod update;
|
||||||
|
124
crates/nu-command/tests/commands/umkdir.rs
Normal file
124
crates/nu-command/tests/commands/umkdir.rs
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
use nu_test_support::fs::files_exist_at;
|
||||||
|
use nu_test_support::playground::Playground;
|
||||||
|
use nu_test_support::{nu, pipeline};
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn creates_directory() {
|
||||||
|
Playground::setup("umkdir_test_1", |dirs, _| {
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
"umkdir my_new_directory"
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = dirs.test().join("my_new_directory");
|
||||||
|
|
||||||
|
assert!(expected.exists());
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn accepts_and_creates_directories() {
|
||||||
|
Playground::setup("umkdir_test_2", |dirs, _| {
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
"umkdir dir_1 dir_2 dir_3"
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(files_exist_at(
|
||||||
|
vec![Path::new("dir_1"), Path::new("dir_2"), Path::new("dir_3")],
|
||||||
|
dirs.test()
|
||||||
|
));
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn creates_intermediary_directories() {
|
||||||
|
Playground::setup("umkdir_test_3", |dirs, _| {
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
"umkdir some_folder/another/deeper_one"
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = dirs.test().join("some_folder/another/deeper_one");
|
||||||
|
|
||||||
|
assert!(expected.exists());
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn create_directory_two_parents_up_using_multiple_dots() {
|
||||||
|
Playground::setup("umkdir_test_4", |dirs, sandbox| {
|
||||||
|
sandbox.within("foo").mkdir("bar");
|
||||||
|
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test().join("foo/bar"),
|
||||||
|
"umkdir .../boo"
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = dirs.test().join("boo");
|
||||||
|
|
||||||
|
assert!(expected.exists());
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn print_created_paths() {
|
||||||
|
Playground::setup("umkdir_test_2", |dirs, _| {
|
||||||
|
let actual = nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
pipeline("umkdir -v dir_1 dir_2 dir_3")
|
||||||
|
);
|
||||||
|
|
||||||
|
assert!(files_exist_at(
|
||||||
|
vec![Path::new("dir_1"), Path::new("dir_2"), Path::new("dir_3")],
|
||||||
|
dirs.test()
|
||||||
|
));
|
||||||
|
|
||||||
|
assert!(actual.out.contains("dir_1"));
|
||||||
|
assert!(actual.out.contains("dir_2"));
|
||||||
|
assert!(actual.out.contains("dir_3"));
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn creates_directory_three_dots() {
|
||||||
|
Playground::setup("umkdir_test_1", |dirs, _| {
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
"umkdir test..."
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = dirs.test().join("test...");
|
||||||
|
|
||||||
|
assert!(expected.exists());
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn creates_directory_four_dots() {
|
||||||
|
Playground::setup("umkdir_test_1", |dirs, _| {
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
"umkdir test...."
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = dirs.test().join("test....");
|
||||||
|
|
||||||
|
assert!(expected.exists());
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn creates_directory_three_dots_quotation_marks() {
|
||||||
|
Playground::setup("umkdir_test_1", |dirs, _| {
|
||||||
|
nu!(
|
||||||
|
cwd: dirs.test(),
|
||||||
|
"umkdir 'test...'"
|
||||||
|
);
|
||||||
|
|
||||||
|
let expected = dirs.test().join("test...");
|
||||||
|
|
||||||
|
assert!(expected.exists());
|
||||||
|
})
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user