mirror of
https://github.com/uutils/coreutils
synced 2024-11-16 09:48:03 +00:00
commit
3a3fbc2840
4 changed files with 140 additions and 46 deletions
|
@ -161,7 +161,6 @@ To do
|
||||||
- install
|
- install
|
||||||
- join
|
- join
|
||||||
- ls
|
- ls
|
||||||
- mktemp (almost done, some options are not working)
|
|
||||||
- mv (almost done, one more option)
|
- mv (almost done, one more option)
|
||||||
- numfmt
|
- numfmt
|
||||||
- od (in progress, needs lots of work)
|
- od (in progress, needs lots of work)
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
#![crate_name = "uu_mktemp"]
|
#![crate_name = "uu_mktemp"]
|
||||||
|
|
||||||
/*
|
// This file is part of the uutils coreutils package.
|
||||||
* This file is part of the uutils coreutils package.
|
//
|
||||||
*
|
// (c) Sunrin SHIMURA
|
||||||
* (c) Sunrin SHIMURA
|
// Collaborator: Jian Zeng
|
||||||
*
|
//
|
||||||
* For the full copyright and license information, please view the LICENSE
|
// For the full copyright and license information, please view the LICENSE
|
||||||
* file that was distributed with this source code.
|
// file that was distributed with this source code.
|
||||||
*/
|
//
|
||||||
|
|
||||||
extern crate getopts;
|
extern crate getopts;
|
||||||
extern crate libc;
|
extern crate libc;
|
||||||
|
@ -26,6 +26,8 @@ use std::iter;
|
||||||
use rand::Rng;
|
use rand::Rng;
|
||||||
use tempfile::NamedTempFileOptions;
|
use tempfile::NamedTempFileOptions;
|
||||||
|
|
||||||
|
mod tempdir;
|
||||||
|
|
||||||
static NAME: &'static str = "mktemp";
|
static NAME: &'static str = "mktemp";
|
||||||
static VERSION: &'static str = env!("CARGO_PKG_VERSION");
|
static VERSION: &'static str = env!("CARGO_PKG_VERSION");
|
||||||
static DEFAULT_TEMPLATE: &'static str = "tmp.XXXXXXXXXX";
|
static DEFAULT_TEMPLATE: &'static str = "tmp.XXXXXXXXXX";
|
||||||
|
@ -34,7 +36,9 @@ static DEFAULT_TEMPLATE: &'static str = "tmp.XXXXXXXXXX";
|
||||||
pub fn uumain(args: Vec<String>) -> i32 {
|
pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
let mut opts = getopts::Options::new();
|
let mut opts = getopts::Options::new();
|
||||||
opts.optflag("d", "directory", "Make a directory instead of a file");
|
opts.optflag("d", "directory", "Make a directory instead of a file");
|
||||||
opts.optflag("u", "dry-run", "do not create anything; merely print a name (unsafe)");
|
opts.optflag("u",
|
||||||
|
"dry-run",
|
||||||
|
"do not create anything; merely print a name (unsafe)");
|
||||||
opts.optflag("q", "quiet", "Fail silently if an error occurs.");
|
opts.optflag("q", "quiet", "Fail silently if an error occurs.");
|
||||||
opts.optopt("", "suffix", "append SUFF to TEMPLATE; SUFF must not contain a path separator. This option is implied if TEMPLATE does not end with X.", "SUFF");
|
opts.optopt("", "suffix", "append SUFF to TEMPLATE; SUFF must not contain a path separator. This option is implied if TEMPLATE does not end with X.", "SUFF");
|
||||||
opts.optopt("p", "tmpdir", "interpret TEMPLATE relative to DIR; if DIR is not specified, use $TMPDIR if set, else /tmp. With this option, TEMPLATE must not be an absolute name; unlike with -t, TEMPLATE may contain slashes, but mktemp creates only the final component", "DIR");
|
opts.optopt("p", "tmpdir", "interpret TEMPLATE relative to DIR; if DIR is not specified, use $TMPDIR if set, else /tmp. With this option, TEMPLATE must not be an absolute name; unlike with -t, TEMPLATE may contain slashes, but mktemp creates only the final component", "DIR");
|
||||||
|
@ -47,12 +51,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
// >> early return options
|
// >> early return options
|
||||||
let matches = match opts.parse(&args[1..]) {
|
let matches = match opts.parse(&args[1..]) {
|
||||||
Ok(m) => m,
|
Ok(m) => m,
|
||||||
Err(f) => crash!(1, "Invalid options\n{}", f)
|
Err(f) => crash!(1, "Invalid options\n{}", f),
|
||||||
};
|
|
||||||
|
|
||||||
if matches.opt_present("quiet") {
|
|
||||||
// TODO: close stderror. `crash!` macro always write output to stderror
|
|
||||||
crash!(1, "quiet option is not supported yet.");
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if matches.opt_present("help") {
|
if matches.opt_present("help") {
|
||||||
|
@ -72,6 +71,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
let make_dir = matches.opt_present("directory");
|
let make_dir = matches.opt_present("directory");
|
||||||
let dry_run = matches.opt_present("dry-run");
|
let dry_run = matches.opt_present("dry-run");
|
||||||
let suffix_opt = matches.opt_str("suffix");
|
let suffix_opt = matches.opt_str("suffix");
|
||||||
|
let suppress_file_err = matches.opt_present("quiet");
|
||||||
|
|
||||||
|
|
||||||
let template = if matches.free.is_empty() {
|
let template = if matches.free.is_empty() {
|
||||||
|
@ -81,15 +81,20 @@ pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
};
|
};
|
||||||
|
|
||||||
let (prefix, rand, suffix) = match parse_template(template) {
|
let (prefix, rand, suffix) = match parse_template(template) {
|
||||||
Some((p, r, s)) => match suffix_opt {
|
Some((p, r, s)) => {
|
||||||
Some(suf) => if s == "" {
|
match suffix_opt {
|
||||||
|
Some(suf) => {
|
||||||
|
if s == "" {
|
||||||
(p, r, suf)
|
(p, r, suf)
|
||||||
} else {
|
} else {
|
||||||
crash!(1, "Template should end with 'X' when you specify suffix option.")
|
crash!(1,
|
||||||
},
|
"Template should end with 'X' when you specify suffix option.")
|
||||||
None => (p, r, s.to_owned())
|
}
|
||||||
},
|
}
|
||||||
None => ("",0, "".to_owned())
|
None => (p, r, s.to_owned()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None => ("", 0, "".to_owned()),
|
||||||
};
|
};
|
||||||
|
|
||||||
if rand < 3 {
|
if rand < 3 {
|
||||||
|
@ -104,18 +109,18 @@ pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
let tmpdir = match matches.opt_str("tmpdir") {
|
let tmpdir = match matches.opt_str("tmpdir") {
|
||||||
Some(s) => {
|
Some(s) => {
|
||||||
if PathBuf::from(prefix).is_absolute() {
|
if PathBuf::from(prefix).is_absolute() {
|
||||||
crash!(1, "template must not be an absolute path when tempdir is specified.");
|
show_info!("invalid template, ‘{}’; with --tmpdir, it may not be absolute", template);
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
PathBuf::from(s)
|
PathBuf::from(s)
|
||||||
|
}
|
||||||
},
|
None => env::temp_dir(),
|
||||||
None => env::temp_dir()
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if dry_run {
|
if dry_run {
|
||||||
dry_exec(tmpdir, prefix, rand, &suffix)
|
dry_exec(tmpdir, prefix, rand, &suffix)
|
||||||
} else {
|
} else {
|
||||||
exec(tmpdir, prefix, rand , &suffix, make_dir)
|
exec(tmpdir, prefix, rand, &suffix, make_dir, suppress_file_err)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -123,7 +128,8 @@ pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
fn print_help(opts: &getopts::Options) {
|
fn print_help(opts: &getopts::Options) {
|
||||||
let usage = format!(" Create a temporary file or directory, safely, and print its name.
|
let usage = format!(" Create a temporary file or directory, safely, and print its name.
|
||||||
TEMPLATE must contain at least 3 consecutive 'X's in last component.
|
TEMPLATE must contain at least 3 consecutive 'X's in last component.
|
||||||
If TEMPLATE is not specified, use {}, and --tmpdir is implied", DEFAULT_TEMPLATE);
|
If TEMPLATE is not specified, use {}, and --tmpdir is implied",
|
||||||
|
DEFAULT_TEMPLATE);
|
||||||
|
|
||||||
println!("{} {}", NAME, VERSION);
|
println!("{} {}", NAME, VERSION);
|
||||||
println!("SYNOPSIS");
|
println!("SYNOPSIS");
|
||||||
|
@ -132,10 +138,10 @@ If TEMPLATE is not specified, use {}, and --tmpdir is implied", DEFAULT_TEMPLATE
|
||||||
print!("{}", opts.usage(&usage[..]));
|
print!("{}", opts.usage(&usage[..]));
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_template(temp :&str) -> Option<(&str, usize, &str)> {
|
fn parse_template(temp: &str) -> Option<(&str, usize, &str)> {
|
||||||
let right = match temp.rfind('X') {
|
let right = match temp.rfind('X') {
|
||||||
Some(r) => r+1,
|
Some(r) => r + 1,
|
||||||
None => return None
|
None => return None,
|
||||||
};
|
};
|
||||||
let left = temp[..right].rfind(|c| c != 'X').map_or(0, |i| i + 1);
|
let left = temp[..right].rfind(|c| c != 'X').map_or(0, |i| i + 1);
|
||||||
let prefix = &temp[..left];
|
let prefix = &temp[..left];
|
||||||
|
@ -170,10 +176,20 @@ pub fn dry_exec(mut tmpdir: PathBuf, prefix: &str, rand: usize, suffix: &str) ->
|
||||||
0
|
0
|
||||||
}
|
}
|
||||||
|
|
||||||
fn exec(tmpdir: PathBuf, prefix: &str, rand: usize, suffix: &str, make_dir: bool) -> i32 {
|
fn exec(tmpdir: PathBuf, prefix: &str, rand: usize, suffix: &str, make_dir: bool, quiet: bool) -> i32 {
|
||||||
// TODO: respect make_dir option
|
|
||||||
if make_dir {
|
if make_dir {
|
||||||
crash!(1, "Directory option is not supported yet. Sorry.");
|
match tempdir::new_in(&tmpdir, prefix, rand, suffix) {
|
||||||
|
Ok(ref f) => {
|
||||||
|
println!("{}", f);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
if !quiet {
|
||||||
|
show_info!("{}", e);
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let tmpfile = NamedTempFileOptions::new()
|
let tmpfile = NamedTempFileOptions::new()
|
||||||
|
@ -184,11 +200,15 @@ fn exec(tmpdir: PathBuf, prefix: &str, rand: usize, suffix: &str, make_dir: bool
|
||||||
|
|
||||||
let tmpfile = match tmpfile {
|
let tmpfile = match tmpfile {
|
||||||
Ok(f) => f,
|
Ok(f) => f,
|
||||||
Err(_) => crash!(1, "failed to create tempfile")
|
Err(e) => {
|
||||||
|
if !quiet {
|
||||||
|
show_info!("failed to create tempfile: {}", e);
|
||||||
|
}
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let tmpname = tmpfile
|
let tmpname = tmpfile.path()
|
||||||
.path()
|
|
||||||
.to_string_lossy()
|
.to_string_lossy()
|
||||||
.to_string();
|
.to_string();
|
||||||
|
|
||||||
|
|
43
src/mktemp/tempdir.rs
Normal file
43
src/mktemp/tempdir.rs
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
// Mainly taken from crate `tempdir`
|
||||||
|
|
||||||
|
extern crate rand;
|
||||||
|
use rand::{Rng, thread_rng};
|
||||||
|
|
||||||
|
use std::io::Result as IOResult;
|
||||||
|
use std::io::{Error, ErrorKind};
|
||||||
|
use std::path::Path;
|
||||||
|
|
||||||
|
// How many times should we (re)try finding an unused random name? It should be
|
||||||
|
// enough that an attacker will run out of luck before we run out of patience.
|
||||||
|
const NUM_RETRIES: u32 = 1 << 31;
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
fn create_dir<P: AsRef<Path>>(path: P) -> IOResult<()> {
|
||||||
|
use std::fs::DirBuilder;
|
||||||
|
use std::os::unix::fs::DirBuilderExt;
|
||||||
|
|
||||||
|
DirBuilder::new().mode(0o700).create(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(windows)]
|
||||||
|
fn create_dir<P: AsRef<Path>>(path: P) -> IOResult<()> {
|
||||||
|
::std::fs::create_dir(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new_in<P: AsRef<Path>>(tmpdir: P, prefix: &str, rand: usize, suffix: &str) -> IOResult<String> {
|
||||||
|
|
||||||
|
let mut rng = thread_rng();
|
||||||
|
for _ in 0..NUM_RETRIES {
|
||||||
|
let rand_chars: String = rng.gen_ascii_chars().take(rand).collect();
|
||||||
|
let leaf = format!("{}{}{}", prefix, rand_chars, suffix);
|
||||||
|
let path = tmpdir.as_ref().join(&leaf);
|
||||||
|
match create_dir(&path) {
|
||||||
|
Ok(_) => return Ok(path.to_string_lossy().into_owned()),
|
||||||
|
Err(ref e) if e.kind() == ErrorKind::AlreadyExists => {}
|
||||||
|
Err(e) => return Err(e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Err(Error::new(ErrorKind::AlreadyExists,
|
||||||
|
"too many temporary directories already exist"))
|
||||||
|
}
|
|
@ -43,7 +43,30 @@ fn test_mktemp_mktemp() {
|
||||||
assert!(!exit_success8);
|
assert!(!exit_success8);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: test directory option when implemented
|
#[test]
|
||||||
|
fn test_mktemp_make_temp_dir() {
|
||||||
|
let ts = TestSet::new(UTIL_NAME);
|
||||||
|
|
||||||
|
let pathname = ts.fixtures.as_string();
|
||||||
|
|
||||||
|
let exit_success1 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE1).run().success;
|
||||||
|
let exit_success2 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE2).run().success;
|
||||||
|
let exit_success3 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE3).run().success;
|
||||||
|
let exit_success4 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE4).run().success;
|
||||||
|
let exit_success5 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE5).run().success;
|
||||||
|
let exit_success6 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE6).run().success;
|
||||||
|
let exit_success7 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE7).run().success;
|
||||||
|
let exit_success8 = ts.util_cmd().env(TMPDIR, &pathname).arg("-d").arg(TEST_TEMPLATE8).run().success;
|
||||||
|
|
||||||
|
assert!(exit_success1);
|
||||||
|
assert!(!exit_success2);
|
||||||
|
assert!(!exit_success3);
|
||||||
|
assert!(!exit_success4);
|
||||||
|
assert!(exit_success5);
|
||||||
|
assert!(exit_success6);
|
||||||
|
assert!(exit_success7);
|
||||||
|
assert!(!exit_success8);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_mktemp_dry_run() {
|
fn test_mktemp_dry_run() {
|
||||||
|
@ -71,7 +94,16 @@ fn test_mktemp_dry_run() {
|
||||||
assert!(!exit_success8);
|
assert!(!exit_success8);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TOOD: test quiet option when correctry implemented
|
#[test]
|
||||||
|
fn test_mktemp_quiet() {
|
||||||
|
let ts = TestSet::new(UTIL_NAME);
|
||||||
|
|
||||||
|
let result1 = ts.util_cmd().arg("-p").arg("/definitely/not/exist/I/promise").arg("-q").run();
|
||||||
|
let result2 = ts.util_cmd().arg("-d").arg("-p").arg("/definitely/not/exist/I/promise").arg("-q").run();
|
||||||
|
|
||||||
|
assert!(result1.stderr.is_empty() && result1.stdout.is_empty() && !result1.success);
|
||||||
|
assert!(result2.stderr.is_empty() && result2.stdout.is_empty() && !result2.success);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_mktemp_suffix() {
|
fn test_mktemp_suffix() {
|
||||||
|
|
Loading…
Reference in a new issue