nushell/src/commands/mv.rs

344 lines
13 KiB
Rust
Raw Normal View History

use crate::commands::command::RunnablePerItemContext;
2019-08-14 20:08:10 +00:00
use crate::errors::ShellError;
use crate::parser::hir::SyntaxType;
use crate::parser::registry::{CommandRegistry, Signature};
use crate::prelude::*;
use std::path::PathBuf;
pub struct Move;
#[derive(Deserialize)]
pub struct MoveArgs {
2019-08-21 12:08:23 +00:00
src: Tagged<PathBuf>,
dst: Tagged<PathBuf>,
}
2019-08-15 05:02:02 +00:00
impl PerItemCommand for Move {
2019-08-14 20:08:10 +00:00
fn name(&self) -> &str {
"mv"
}
fn signature(&self) -> Signature {
Signature::build("mv")
.required("source", SyntaxType::Path)
.required("destination", SyntaxType::Path)
.named("file", SyntaxType::Any)
2019-08-14 20:08:10 +00:00
}
fn run(
&self,
call_info: &CallInfo,
_registry: &CommandRegistry,
shell_manager: &ShellManager,
_input: Tagged<Value>,
) -> Result<VecDeque<ReturnValue>, ShellError> {
call_info.process(shell_manager, mv)?.run()
}
2019-08-14 20:08:10 +00:00
}
2019-08-21 12:08:23 +00:00
fn mv(
MoveArgs { src, dst }: MoveArgs,
RunnablePerItemContext {
name,
shell_manager,
}: &RunnablePerItemContext,
2019-08-15 05:02:02 +00:00
) -> Result<VecDeque<ReturnValue>, ShellError> {
2019-08-21 12:08:23 +00:00
let source = src.item.clone();
let mut destination = dst.item.clone();
let name_span = name;
2019-08-14 20:08:10 +00:00
let sources: Vec<_> = match glob::glob(&source.to_string_lossy()) {
Ok(files) => files.collect(),
Err(_) => {
return Err(ShellError::labeled_error(
"Invalid pattern.",
"Invalid pattern.",
2019-08-21 12:08:23 +00:00
src.tag,
))
}
};
2019-08-14 20:08:10 +00:00
2019-08-21 12:08:23 +00:00
if "." == destination.to_string_lossy() {
destination = PathBuf::from(shell_manager.path());
}
2019-08-14 20:08:10 +00:00
2019-08-21 12:08:23 +00:00
let destination_file_name = {
match destination.file_name() {
Some(name) => PathBuf::from(name),
None => {
return Err(ShellError::labeled_error(
"Rename aborted. Not a valid destination",
"Rename aborted. Not a valid destination",
2019-08-21 12:08:23 +00:00
dst.span(),
))
}
}
};
2019-08-14 20:08:10 +00:00
if sources.len() == 1 {
if let Ok(entry) = &sources[0] {
let entry_file_name = match entry.file_name() {
Some(name) => name,
None => {
return Err(ShellError::labeled_error(
"Rename aborted. Not a valid entry name",
"Rename aborted. Not a valid entry name",
name_span,
))
}
};
2019-08-14 20:08:10 +00:00
if destination.exists() && destination.is_dir() {
destination = match dunce::canonicalize(&destination) {
Ok(path) => path,
Err(e) => {
return Err(ShellError::labeled_error(
format!("Rename aborted. {:}", e.to_string()),
format!("Rename aborted. {:}", e.to_string()),
name_span,
))
}
};
destination.push(entry_file_name);
2019-08-14 20:08:10 +00:00
}
if entry.is_file() {
match std::fs::rename(&entry, &destination) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
name_span,
2019-08-14 20:08:10 +00:00
));
}
Ok(o) => o,
};
}
if entry.is_dir() {
match std::fs::create_dir_all(&destination) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
name_span,
2019-08-14 20:08:10 +00:00
));
}
Ok(o) => o,
};
#[cfg(not(windows))]
{
match std::fs::rename(&entry, &destination) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
name_span,
2019-08-14 20:08:10 +00:00
));
}
Ok(o) => o,
};
}
#[cfg(windows)]
{
2019-08-21 12:08:23 +00:00
use crate::utils::FileStructure;
2019-08-14 20:08:10 +00:00
let mut sources: FileStructure = FileStructure::new();
sources.walk_decorate(&entry)?;
2019-08-14 20:08:10 +00:00
let strategy = |(source_file, depth_level)| {
let mut new_dst = destination.clone();
let path = dunce::canonicalize(&source_file)?;
2019-08-14 20:08:10 +00:00
let mut comps: Vec<_> = path
.components()
.map(|fragment| fragment.as_os_str())
.rev()
.take(1 + depth_level)
.collect();
comps.reverse();
for fragment in comps.iter() {
new_dst.push(fragment);
}
Ok((PathBuf::from(&source_file), PathBuf::from(new_dst)))
2019-08-14 20:08:10 +00:00
};
let sources = sources.paths_applying_with(strategy)?;
for (ref src, ref dst) in sources {
2019-08-14 20:08:10 +00:00
if src.is_dir() {
if !dst.exists() {
match std::fs::create_dir_all(dst) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
name_span,
2019-08-14 20:08:10 +00:00
));
}
Ok(o) => o,
}
2019-08-14 20:08:10 +00:00
}
}
if src.is_file() {
match std::fs::rename(src, dst) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
name_span,
2019-08-14 20:08:10 +00:00
));
}
Ok(o) => o,
}
2019-08-14 20:08:10 +00:00
}
}
match std::fs::remove_dir_all(entry) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
e.to_string(),
),
name_span,
));
}
Ok(o) => o,
};
2019-08-14 20:08:10 +00:00
}
}
}
} else {
if destination.exists() {
if !sources.iter().all(|x| {
if let Ok(entry) = x.as_ref() {
entry.is_file()
} else {
false
}
}) {
2019-08-14 20:08:10 +00:00
return Err(ShellError::labeled_error(
"Rename aborted (directories found). Renaming in patterns not supported yet (try moving the directory directly)",
"Rename aborted (directories found). Renaming in patterns not supported yet (try moving the directory directly)",
2019-08-21 12:08:23 +00:00
src.tag,
2019-08-14 20:08:10 +00:00
));
}
for entry in sources {
if let Ok(entry) = entry {
let entry_file_name = match entry.file_name() {
Some(name) => name,
None => {
return Err(ShellError::labeled_error(
"Rename aborted. Not a valid entry name",
"Rename aborted. Not a valid entry name",
name_span,
))
}
};
2019-08-14 20:08:10 +00:00
let mut to = PathBuf::from(&destination);
to.push(entry_file_name);
2019-08-14 20:08:10 +00:00
if entry.is_file() {
match std::fs::rename(&entry, &to) {
Err(e) => {
return Err(ShellError::labeled_error(
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
format!(
"Rename {:?} to {:?} aborted. {:}",
entry_file_name,
destination_file_name,
2019-08-14 20:08:10 +00:00
e.to_string(),
),
name_span,
2019-08-14 20:08:10 +00:00
));
}
Ok(o) => o,
};
}
}
}
} else {
return Err(ShellError::labeled_error(
format!("Rename aborted. (Does {:?} exist?)", destination_file_name),
format!("Rename aborted. (Does {:?} exist?)", destination_file_name),
2019-08-21 12:08:23 +00:00
dst.span(),
2019-08-14 20:08:10 +00:00
));
}
}
2019-08-15 05:02:02 +00:00
Ok(VecDeque::new())
2019-08-14 20:08:10 +00:00
}