coreutils/src/cp/cp.rs

155 lines
4 KiB
Rust
Raw Normal View History

2014-07-06 08:13:36 +00:00
#![crate_name = "cp"]
#![feature(rustc_private, path_ext)]
/*
* This file is part of the uutils coreutils package.
*
* (c) Jordy Dickinson <jordy.dickinson@gmail.com>
*
* For the full copyright and license information, please view the LICENSE file
* that was distributed with this source code.
*/
extern crate getopts;
2015-01-08 12:54:22 +00:00
#[macro_use] extern crate log;
use getopts::{getopts, optflag, usage};
use std::fs;
use std::fs::{PathExt};
use std::io::{ErrorKind, Result};
use std::path::Path;
#[derive(Clone, Eq, PartialEq)]
pub enum Mode {
Copy,
Help,
Version,
}
static NAME: &'static str = "cp";
static VERSION: &'static str = "1.0.0";
2014-12-11 06:36:58 +00:00
pub fn uumain(args: Vec<String>) -> i32 {
2014-05-30 08:35:54 +00:00
let opts = [
optflag("h", "help", "display this help and exit"),
optflag("", "version", "output version information and exit"),
];
let matches = match getopts(&args[1..], &opts) {
Ok(m) => m,
Err(e) => {
2014-06-15 10:50:40 +00:00
error!("error: {}", e);
2014-10-30 09:06:47 +00:00
panic!()
},
};
2014-07-20 01:13:55 +00:00
let progname = &args[0];
2014-11-19 20:55:25 +00:00
let usage = usage("Copy SOURCE to DEST, or multiple SOURCE(s) to DIRECTORY.", &opts);
let mode = if matches.opt_present("version") {
2014-11-19 20:50:37 +00:00
Mode::Version
} else if matches.opt_present("help") {
2014-11-19 20:50:37 +00:00
Mode::Help
} else {
2014-11-19 20:50:37 +00:00
Mode::Copy
};
match mode {
2014-11-19 20:50:37 +00:00
Mode::Copy => copy(matches),
Mode::Help => help(&progname, &usage),
2014-11-19 20:50:37 +00:00
Mode::Version => version(),
}
0
}
fn version() {
println!("{} {}", NAME, VERSION);
}
fn help(progname: &String, usage: &String) {
let msg = format!("Usage: {0} SOURCE DEST\n \
or: {0} SOURCE... DIRECTORY\n \
or: {0} -t DIRECTORY SOURCE\n\
\n\
{1}", progname, usage);
println!("{}", msg);
}
fn copy(matches: getopts::Matches) {
let sources: Vec<String> = if matches.free.is_empty() {
error!("error: Missing SOURCE argument. Try --help.");
2014-10-30 09:06:47 +00:00
panic!()
} else {
// All but the last argument:
matches.free[..matches.free.len() - 1].iter().map(|arg| arg.clone()).collect()
};
let dest = if matches.free.len() < 2 {
error!("error: Missing DEST argument. Try --help.");
2014-10-30 09:06:47 +00:00
panic!()
} else {
// Only the last argument:
Path::new(&matches.free[matches.free.len() - 1])
};
assert!(sources.len() >= 1);
if sources.len() == 1 {
let source = Path::new(&sources[0]);
let same_file = paths_refer_to_same_file(source, dest).unwrap_or_else(|err| {
match err.kind() {
ErrorKind::NotFound => false,
_ => {
error!("error: {}", err);
panic!()
}
}
});
if same_file {
2014-11-21 09:09:43 +00:00
error!("error: \"{}\" and \"{}\" are the same file",
source.display(),
dest.display());
2014-10-30 09:06:47 +00:00
panic!();
}
if let Err(err) = fs::copy(source, dest) {
error!("error: {}", err);
2014-10-30 09:06:47 +00:00
panic!();
}
} else {
if !fs::metadata(dest).unwrap().is_dir() {
error!("error: TARGET must be a directory");
2014-10-30 09:06:47 +00:00
panic!();
}
for src in sources.iter() {
let source = Path::new(&src);
if !fs::metadata(source).unwrap().is_file() {
error!("error: \"{}\" is not a file", source.display());
continue;
}
let mut full_dest = dest.to_path_buf();
full_dest.push(source.to_str().unwrap());
println!("{}", full_dest.display());
let io_result = fs::copy(source, full_dest);
2015-01-24 02:56:37 +00:00
if let Err(err) = io_result {
error!("error: {}", err);
2014-10-30 09:06:47 +00:00
panic!()
}
}
}
}
pub fn paths_refer_to_same_file(p1: &Path, p2: &Path) -> Result<bool> {
// We have to take symlinks and relative paths into account.
let pathbuf1 = try!(p1.canonicalize());
let pathbuf2 = try!(p2.canonicalize());
Ok(pathbuf1 == pathbuf2)
}