mirror of
https://github.com/nushell/nushell
synced 2025-01-03 16:58:58 +00:00
commit
a2eba38e81
6 changed files with 141 additions and 1 deletions
|
@ -35,12 +35,14 @@ pub fn create_default_context() -> Rc<RefCell<EngineState>> {
|
||||||
working_set.add_decl(Box::new(LetEnv));
|
working_set.add_decl(Box::new(LetEnv));
|
||||||
working_set.add_decl(Box::new(Lines));
|
working_set.add_decl(Box::new(Lines));
|
||||||
working_set.add_decl(Box::new(Ls));
|
working_set.add_decl(Box::new(Ls));
|
||||||
|
working_set.add_decl(Box::new(Mkdir));
|
||||||
working_set.add_decl(Box::new(Module));
|
working_set.add_decl(Box::new(Module));
|
||||||
working_set.add_decl(Box::new(Mv));
|
working_set.add_decl(Box::new(Mv));
|
||||||
working_set.add_decl(Box::new(Ps));
|
working_set.add_decl(Box::new(Ps));
|
||||||
working_set.add_decl(Box::new(Select));
|
working_set.add_decl(Box::new(Select));
|
||||||
working_set.add_decl(Box::new(Sys));
|
working_set.add_decl(Box::new(Sys));
|
||||||
working_set.add_decl(Box::new(Table));
|
working_set.add_decl(Box::new(Table));
|
||||||
|
working_set.add_decl(Box::new(Touch));
|
||||||
working_set.add_decl(Box::new(Use));
|
working_set.add_decl(Box::new(Use));
|
||||||
working_set.add_decl(Box::new(Where));
|
working_set.add_decl(Box::new(Where));
|
||||||
working_set.add_decl(Box::new(Wrap));
|
working_set.add_decl(Box::new(Wrap));
|
||||||
|
|
|
@ -60,7 +60,7 @@ impl Command for Cp {
|
||||||
}
|
}
|
||||||
|
|
||||||
let any_source_is_dir = sources.iter().any(|f| matches!(f, Ok(f) if f.is_dir()));
|
let any_source_is_dir = sources.iter().any(|f| matches!(f, Ok(f) if f.is_dir()));
|
||||||
let recursive = call.named.iter().any(|p| &p.0 == "recursive");
|
let recursive: bool = call.has_flag("recursive");
|
||||||
if any_source_is_dir && !recursive {
|
if any_source_is_dir && !recursive {
|
||||||
return Err(ShellError::MoveNotPossibleSingle(
|
return Err(ShellError::MoveNotPossibleSingle(
|
||||||
"Directories must be copied using \"--recursive\"".to_string(),
|
"Directories must be copied using \"--recursive\"".to_string(),
|
||||||
|
|
74
crates/nu-command/src/filesystem/mkdir.rs
Normal file
74
crates/nu-command/src/filesystem/mkdir.rs
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
use std::collections::VecDeque;
|
||||||
|
use std::env::current_dir;
|
||||||
|
|
||||||
|
use nu_engine::CallExt;
|
||||||
|
use nu_protocol::ast::Call;
|
||||||
|
use nu_protocol::engine::{Command, EvaluationContext};
|
||||||
|
use nu_protocol::{ShellError, Signature, SyntaxShape, Value, ValueStream};
|
||||||
|
|
||||||
|
pub struct Mkdir;
|
||||||
|
|
||||||
|
impl Command for Mkdir {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"mkdir"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build("mkdir")
|
||||||
|
.rest(
|
||||||
|
"rest",
|
||||||
|
SyntaxShape::Filepath,
|
||||||
|
"the name(s) of the path(s) to create",
|
||||||
|
)
|
||||||
|
.switch("show-created-paths", "show the path(s) created.", Some('s'))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Make directories, creates intermediary directories as required."
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
context: &EvaluationContext,
|
||||||
|
call: &Call,
|
||||||
|
_input: Value,
|
||||||
|
) -> Result<Value, ShellError> {
|
||||||
|
let path = current_dir()?;
|
||||||
|
let mut directories = call
|
||||||
|
.rest::<String>(context, 0)?
|
||||||
|
.into_iter()
|
||||||
|
.map(|dir| path.join(dir))
|
||||||
|
.peekable();
|
||||||
|
|
||||||
|
let show_created_paths = call.has_flag("show-created-paths");
|
||||||
|
let mut stream: VecDeque<Value> = VecDeque::new();
|
||||||
|
|
||||||
|
if directories.peek().is_none() {
|
||||||
|
return Err(ShellError::MissingParameter(
|
||||||
|
"requires directory paths".to_string(),
|
||||||
|
call.head,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i, dir) in directories.enumerate() {
|
||||||
|
let span = call.positional[i].span;
|
||||||
|
let dir_res = std::fs::create_dir_all(&dir);
|
||||||
|
|
||||||
|
if let Err(reason) = dir_res {
|
||||||
|
return Err(ShellError::CreateNotPossible(
|
||||||
|
format!("failed to create directory: {}", reason),
|
||||||
|
call.positional[i].span,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
if show_created_paths {
|
||||||
|
let val = format!("{:}", dir.to_string_lossy());
|
||||||
|
stream.push_back(Value::String { val, span });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let stream = ValueStream::from_stream(stream.into_iter());
|
||||||
|
let span = call.head;
|
||||||
|
Ok(Value::Stream { stream, span })
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,10 +1,14 @@
|
||||||
mod cd;
|
mod cd;
|
||||||
mod cp;
|
mod cp;
|
||||||
mod ls;
|
mod ls;
|
||||||
|
mod mkdir;
|
||||||
mod mv;
|
mod mv;
|
||||||
|
mod touch;
|
||||||
mod util;
|
mod util;
|
||||||
|
|
||||||
pub use cd::Cd;
|
pub use cd::Cd;
|
||||||
pub use cp::Cp;
|
pub use cp::Cp;
|
||||||
pub use ls::Ls;
|
pub use ls::Ls;
|
||||||
|
pub use mkdir::Mkdir;
|
||||||
pub use mv::Mv;
|
pub use mv::Mv;
|
||||||
|
pub use touch::Touch;
|
||||||
|
|
52
crates/nu-command/src/filesystem/touch.rs
Normal file
52
crates/nu-command/src/filesystem/touch.rs
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
use std::fs::OpenOptions;
|
||||||
|
|
||||||
|
use nu_engine::CallExt;
|
||||||
|
use nu_protocol::ast::Call;
|
||||||
|
use nu_protocol::engine::{Command, EvaluationContext};
|
||||||
|
use nu_protocol::{ShellError, Signature, SyntaxShape, Value};
|
||||||
|
|
||||||
|
pub struct Touch;
|
||||||
|
|
||||||
|
impl Command for Touch {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"touch"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build("touch")
|
||||||
|
.required(
|
||||||
|
"filename",
|
||||||
|
SyntaxShape::Filepath,
|
||||||
|
"the path of the file you want to create",
|
||||||
|
)
|
||||||
|
.rest("rest", SyntaxShape::Filepath, "additional files to create")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Creates one or more files."
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run(
|
||||||
|
&self,
|
||||||
|
context: &EvaluationContext,
|
||||||
|
call: &Call,
|
||||||
|
_input: Value,
|
||||||
|
) -> Result<Value, ShellError> {
|
||||||
|
let target: String = call.req(context, 0)?;
|
||||||
|
let rest: Vec<String> = call.rest(context, 1)?;
|
||||||
|
|
||||||
|
for (index, item) in vec![target].into_iter().chain(rest).enumerate() {
|
||||||
|
match OpenOptions::new().write(true).create(true).open(&item) {
|
||||||
|
Ok(_) => continue,
|
||||||
|
Err(err) => {
|
||||||
|
return Err(ShellError::CreateNotPossible(
|
||||||
|
format!("Failed to create file: {}", err),
|
||||||
|
call.positional[index].span,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Value::Nothing { span: call.head })
|
||||||
|
}
|
||||||
|
}
|
|
@ -27,6 +27,10 @@ pub enum ShellError {
|
||||||
#[diagnostic(code(nu::shell::unknown_operator), url(docsrs))]
|
#[diagnostic(code(nu::shell::unknown_operator), url(docsrs))]
|
||||||
UnknownOperator(String, #[label = "unsupported operator"] Span),
|
UnknownOperator(String, #[label = "unsupported operator"] Span),
|
||||||
|
|
||||||
|
#[error("Missing parameter: {0}.")]
|
||||||
|
#[diagnostic(code(nu::shell::missing_parameter), url(docsrs))]
|
||||||
|
MissingParameter(String, #[label = "missing parameter: {0}"] Span),
|
||||||
|
|
||||||
#[error("External commands not yet supported")]
|
#[error("External commands not yet supported")]
|
||||||
#[diagnostic(code(nu::shell::external_commands), url(docsrs))]
|
#[diagnostic(code(nu::shell::external_commands), url(docsrs))]
|
||||||
ExternalNotSupported(#[label = "external not supported"] Span),
|
ExternalNotSupported(#[label = "external not supported"] Span),
|
||||||
|
@ -109,6 +113,10 @@ pub enum ShellError {
|
||||||
#[error("Move not possible")]
|
#[error("Move not possible")]
|
||||||
#[diagnostic(code(nu::shell::move_not_possible_single), url(docsrs))]
|
#[diagnostic(code(nu::shell::move_not_possible_single), url(docsrs))]
|
||||||
MoveNotPossibleSingle(String, #[label("{0}")] Span),
|
MoveNotPossibleSingle(String, #[label("{0}")] Span),
|
||||||
|
|
||||||
|
#[error("Create not possible")]
|
||||||
|
#[diagnostic(code(nu::shell::create_not_possible), url(docsrs))]
|
||||||
|
CreateNotPossible(String, #[label("{0}")] Span),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<std::io::Error> for ShellError {
|
impl From<std::io::Error> for ShellError {
|
||||||
|
|
Loading…
Reference in a new issue