#![crate_name = "touch"] #![allow(unstable)] /* * This file is part of the uutils coreutils package. * * (c) Nick Platt * * For the full copyright and license information, please view the LICENSE file * that was distributed with this source code. */ extern crate getopts; extern crate time; use std::old_io::File; use std::old_io::fs::PathExtensions; #[path = "../common/util.rs"] #[macro_use] mod util; static NAME: &'static str = "touch"; static VERSION: &'static str = "1.0.0"; pub fn uumain(args: Vec) -> isize { let opts = [ getopts::optflag("a", "", "change only the access time"), getopts::optflag("c", "no-create", "do not create any files"), getopts::optopt( "d", "date", "parse argument and use it instead of current time", "STRING"), getopts::optflag("h", "no-dereference", "affect each symbolic link instead of any referenced file \ (only for systems that can change the timestamps of a symlink)"), getopts::optflag("m", "", "change only the modification time"), getopts::optopt( "r", "reference", "use this file's times instead of the current time", "FILE"), getopts::optopt( "t", "", "use [[CC]YY]MMDDhhmm[.ss] instead of the current time", "STAMP"), getopts::optopt( "", "time", "change only the specified time: \"access\", \"atime\", or \ \"use\" are equivalent to -a; \"modify\" or \"mtime\" are \ equivalent to -m", "WORD"), getopts::optflag("h", "help", "display this help and exit"), getopts::optflag("V", "version", "output version information and exit"), ]; let matches = match getopts::getopts(args.tail(), &opts) { Ok(m) => m, Err(e) => panic!("Invalid options\n{}", e) }; if matches.opt_present("version") { println!("{} {}", NAME, VERSION); return 0; } if matches.opt_present("help") || matches.free.is_empty() { println!("{} {}", NAME, VERSION); println!(""); println!("Usage: {} [OPTION]... FILE...", NAME); println!(""); println!("{}", getopts::usage("Update the access and modification times of \ each FILE to the current time.", &opts)); if matches.free.is_empty() { return 1; } return 0; } if matches.opt_present("date") && matches.opts_present(&["reference".to_string(), "t".to_string()]) || matches.opt_present("reference") && matches.opts_present(&["date".to_string(), "t".to_string()]) || matches.opt_present("t") && matches.opts_present(&["date".to_string(), "reference".to_string()]) { panic!("Invalid options: cannot specify reference time from more than one source"); } let (mut atime, mut mtime) = if matches.opt_present("reference") { let path = Path::new(matches.opt_str("reference").unwrap().to_string()); let stat = stat(&path, !matches.opt_present("no-dereference")); (stat.accessed, stat.modified) } else if matches.opts_present(&["date".to_string(), "t".to_string()]) { let timestamp = if matches.opt_present("date") { parse_date(matches.opt_str("date").unwrap().as_slice()) } else { parse_timestamp(matches.opt_str("t").unwrap().as_slice()) }; (timestamp, timestamp) } else { // FIXME: Should use Timespec. https://github.com/mozilla/rust/issues/10301 let now = (time::get_time().sec * 1000) as u64; (now, now) }; for filename in matches.free.iter() { let path = Path::new(filename.to_string()); if !path.exists() { // no-dereference included here for compatibility if matches.opts_present(&["no-create".to_string(), "no-dereference".to_string()]) { continue; } match File::create(&path) { Ok(fd) => fd, Err(e) => panic!("Unable to create file: {}\n{}", filename, e.desc) }; // Minor optimization: if no reference time was specified, we're done. if !matches.opts_present(&["date".to_string(), "reference".to_string(), "t".to_string()]) { continue; } } // If changing "only" atime or mtime, grab the existing value of the other. // Note that "-a" and "-m" may be passed together; this is not an xor. if matches.opts_present(&["a".to_string(), "m".to_string(), "time".to_string()]) { let stat = stat(&path, !matches.opt_present("no-dereference")); let time = matches.opt_strs("time"); if !(matches.opt_present("a") || time.contains(&"access".to_string()) || time.contains(&"atime".to_string()) || time.contains(&"use".to_string())) { atime = stat.accessed; } if !(matches.opt_present("m") || time.contains(&"modify".to_string()) || time.contains(&"mtime".to_string())) { mtime = stat.modified; } } match std::old_io::fs::change_file_times(&path, atime, mtime) { Ok(t) => t, Err(e) => panic!("Unable to modify times\n{}", e.desc) } } 0 } fn stat(path: &Path, follow: bool) -> std::old_io::FileStat { if follow { match std::old_io::fs::stat(path) { Ok(stat) => stat, Err(e) => panic!("Unable to open file\n{}", e.desc) } } else { match std::old_io::fs::lstat(path) { Ok(stat) => stat, Err(e) => panic!("Unable to open file\n{}", e.desc) } } } fn parse_date(str: &str) -> u64 { // This isn't actually compatible with GNU touch, but there doesn't seem to // be any simple specification for what format this parameter allows and I'm // not about to implement GNU parse_datetime. // http://git.savannah.gnu.org/gitweb/?p=gnulib.git;a=blob_plain;f=lib/parse-datetime.y match time::strptime(str, "%c") { Ok(tm) => (tm.to_timespec().sec * 1000) as u64, Err(e) => panic!("Unable to parse date\n{}", e) } } fn parse_timestamp(str: &str) -> u64 { let format = match str.chars().count() { 15 => "%Y%m%d%H%M.%S", 12 => "%Y%m%d%H%M", 13 => "%y%m%d%H%M.%S", 10 => "%y%m%d%H%M", 11 => "%m%d%H%M.%S", 8 => "%m%d%H%M", _ => panic!("Unknown timestamp format") }; match time::strptime(str, format) { Ok(tm) => (tm.to_timespec().sec * 1000) as u64, Err(e) => panic!("Unable to parse timestamp\n{}", e) } }