2014-07-06 08:13:36 +00:00
|
|
|
#![crate_name = "truncate"]
|
2014-02-01 05:18:57 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is part of the uutils coreutils package.
|
|
|
|
*
|
|
|
|
* (c) Arcterus <arcterus@mail.com>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
2014-03-31 16:40:21 +00:00
|
|
|
#![feature(macro_rules)]
|
2014-02-05 04:17:36 +00:00
|
|
|
|
2014-02-16 21:29:31 +00:00
|
|
|
extern crate getopts;
|
2014-04-07 22:43:34 +00:00
|
|
|
extern crate libc;
|
2014-02-01 05:18:57 +00:00
|
|
|
|
2014-02-08 03:05:53 +00:00
|
|
|
use std::io::{File, Open, ReadWrite, fs};
|
2014-02-01 05:18:57 +00:00
|
|
|
use std::u64;
|
2014-02-07 06:39:07 +00:00
|
|
|
|
2014-02-23 22:17:48 +00:00
|
|
|
#[path = "../common/util.rs"]
|
2014-02-07 06:39:07 +00:00
|
|
|
mod util;
|
2014-02-01 05:18:57 +00:00
|
|
|
|
2014-06-01 00:41:32 +00:00
|
|
|
#[deriving(Eq, PartialEq)]
|
2014-02-01 05:18:57 +00:00
|
|
|
enum TruncateMode {
|
|
|
|
Reference,
|
|
|
|
Extend,
|
|
|
|
Reduce,
|
|
|
|
AtMost,
|
|
|
|
AtLeast,
|
|
|
|
RoundDown,
|
|
|
|
RoundUp
|
|
|
|
}
|
|
|
|
|
2014-02-07 06:39:07 +00:00
|
|
|
static NAME: &'static str = "truncate";
|
|
|
|
|
2014-06-08 07:56:37 +00:00
|
|
|
pub fn uumain(args: Vec<String>) -> int {
|
2014-05-16 08:32:58 +00:00
|
|
|
let program = args.get(0).clone();
|
2014-02-01 05:18:57 +00:00
|
|
|
|
2014-05-30 08:35:54 +00:00
|
|
|
let opts = [
|
2014-02-07 06:39:07 +00:00
|
|
|
getopts::optflag("c", "no-create", "do not create files that do not exist"),
|
|
|
|
getopts::optflag("o", "io-blocks", "treat SIZE as the number of I/O blocks of the file rather than bytes (NOT IMPLEMENTED)"),
|
|
|
|
getopts::optopt("r", "reference", "base the size of each file on the size of RFILE", "RFILE"),
|
|
|
|
getopts::optopt("s", "size", "set or adjust the size of each file according to SIZE, which is in bytes unless --io-blocks is specified", "SIZE"),
|
|
|
|
getopts::optflag("h", "help", "display this help and exit"),
|
|
|
|
getopts::optflag("V", "version", "output version information and exit")
|
2014-02-01 05:18:57 +00:00
|
|
|
];
|
2014-02-07 06:39:07 +00:00
|
|
|
let matches = match getopts::getopts(args.tail(), opts) {
|
2014-02-01 05:18:57 +00:00
|
|
|
Ok(m) => m,
|
|
|
|
Err(f) => {
|
2014-06-15 10:50:40 +00:00
|
|
|
crash!(1, "{}", f)
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if matches.opt_present("help") {
|
|
|
|
println!("truncate 1.0.0");
|
|
|
|
println!("");
|
|
|
|
println!("Usage:");
|
|
|
|
println!(" {0:s} [OPTION]... FILE...", program);
|
|
|
|
println!("");
|
2014-02-07 06:39:07 +00:00
|
|
|
print!("{}", getopts::usage("Shrink or extend the size of each file to the specified size.", opts));
|
2014-02-01 18:51:20 +00:00
|
|
|
print!("
|
|
|
|
SIZE is an integer with an optional prefix and optional unit.
|
|
|
|
The available units (K, M, G, T, P, E, Z, and Y) use the following format:
|
|
|
|
'KB' => 1000 (kilobytes)
|
|
|
|
'K' => 1024 (kibibytes)
|
|
|
|
'MB' => 1000*1000 (megabytes)
|
|
|
|
'M' => 1024*1024 (mebibytes)
|
|
|
|
'GB' => 1000*1000*1000 (gigabytes)
|
|
|
|
'G' => 1024*1024*1024 (gibibytes)
|
|
|
|
SIZE may also be prefixed by one of the following to adjust the size of each
|
|
|
|
file based on its current size:
|
|
|
|
'+' => extend by
|
|
|
|
'-' => reduce by
|
|
|
|
'<' => at most
|
|
|
|
'>' => at least
|
|
|
|
'/' => round down to multiple of
|
|
|
|
'%' => round up to multiple of
|
|
|
|
");
|
2014-02-01 05:18:57 +00:00
|
|
|
} else if matches.opt_present("version") {
|
|
|
|
println!("truncate 1.0.0");
|
|
|
|
} else if matches.free.is_empty() {
|
2014-06-09 03:26:51 +00:00
|
|
|
show_error!("missing an argument");
|
2014-06-09 01:49:06 +00:00
|
|
|
return 1;
|
2014-02-01 05:18:57 +00:00
|
|
|
} else {
|
|
|
|
let no_create = matches.opt_present("no-create");
|
|
|
|
let io_blocks = matches.opt_present("io-blocks");
|
|
|
|
let reference = matches.opt_str("reference");
|
|
|
|
let size = matches.opt_str("size");
|
|
|
|
if reference.is_none() && size.is_none() {
|
2014-02-07 06:39:07 +00:00
|
|
|
crash!(1, "you must specify either --reference or --size");
|
2014-02-01 05:18:57 +00:00
|
|
|
} else {
|
2014-06-09 01:49:06 +00:00
|
|
|
match truncate(no_create, io_blocks, reference, size, matches.free) {
|
|
|
|
Ok(()) => ( /* pass */ ),
|
|
|
|
Err(e) => return e
|
|
|
|
}
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
}
|
2014-06-08 07:56:37 +00:00
|
|
|
|
2014-06-12 04:41:53 +00:00
|
|
|
0
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
|
2014-06-09 01:49:06 +00:00
|
|
|
fn truncate(no_create: bool, _: bool, reference: Option<String>, size: Option<String>, filenames: Vec<String>) -> Result<(), int> {
|
2014-02-01 05:18:57 +00:00
|
|
|
let (refsize, mode) = match reference {
|
|
|
|
Some(rfilename) => {
|
2014-02-08 03:05:53 +00:00
|
|
|
let rfile = match File::open(&Path::new(rfilename.clone())) {
|
2014-02-05 04:17:36 +00:00
|
|
|
Ok(m) => m,
|
|
|
|
Err(f) => {
|
2014-02-07 06:39:07 +00:00
|
|
|
crash!(1, "{}", f.to_str())
|
2014-02-05 04:17:36 +00:00
|
|
|
}
|
|
|
|
};
|
2014-06-09 01:49:06 +00:00
|
|
|
match fs::stat(rfile.path()) {
|
|
|
|
Ok(stat) => (stat.size, Reference),
|
|
|
|
Err(f) => {
|
2014-06-09 03:26:51 +00:00
|
|
|
show_error!("{}", f.to_str());
|
2014-06-09 01:49:06 +00:00
|
|
|
return Err(1);
|
|
|
|
}
|
|
|
|
}
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
2014-05-17 10:32:14 +00:00
|
|
|
None => parse_size(size.unwrap().as_slice())
|
2014-02-01 05:18:57 +00:00
|
|
|
};
|
|
|
|
for filename in filenames.iter() {
|
2014-05-17 10:32:14 +00:00
|
|
|
let filename = filename.as_slice();
|
2014-02-01 05:18:57 +00:00
|
|
|
let path = Path::new(filename);
|
2014-02-05 04:17:36 +00:00
|
|
|
if path.exists() || !no_create {
|
|
|
|
match File::open_mode(&path, Open, ReadWrite) {
|
|
|
|
Ok(mut file) => {
|
2014-06-09 01:49:06 +00:00
|
|
|
let fsize = match fs::stat(file.path()) {
|
|
|
|
Ok(stat) => stat.size,
|
|
|
|
Err(f) => {
|
|
|
|
show_warning!("{}", f.to_str());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
};
|
2014-02-05 04:17:36 +00:00
|
|
|
let tsize = match mode {
|
|
|
|
Reference => refsize,
|
|
|
|
Extend => fsize + refsize,
|
|
|
|
Reduce => fsize - refsize,
|
|
|
|
AtMost => if fsize > refsize { refsize } else { fsize },
|
|
|
|
AtLeast => if fsize < refsize { refsize } else { fsize },
|
|
|
|
RoundDown => fsize - fsize % refsize,
|
|
|
|
RoundUp => fsize + fsize % refsize
|
|
|
|
};
|
|
|
|
match file.truncate(tsize as i64) {
|
|
|
|
Ok(_) => {}
|
|
|
|
Err(f) => {
|
2014-06-09 03:26:51 +00:00
|
|
|
show_error!("{}", f.to_str());
|
2014-06-09 01:49:06 +00:00
|
|
|
return Err(1);
|
2014-02-05 04:17:36 +00:00
|
|
|
}
|
2014-02-01 05:31:10 +00:00
|
|
|
}
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
2014-02-05 04:17:36 +00:00
|
|
|
Err(f) => {
|
2014-06-09 03:26:51 +00:00
|
|
|
show_error!("{}", f.to_str());
|
2014-06-09 01:49:06 +00:00
|
|
|
return Err(1);
|
2014-02-05 04:17:36 +00:00
|
|
|
}
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
2014-02-05 04:17:36 +00:00
|
|
|
}
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
2014-06-12 04:41:53 +00:00
|
|
|
Ok(())
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
|
2014-05-17 10:32:14 +00:00
|
|
|
fn parse_size(size: &str) -> (u64, TruncateMode) {
|
2014-02-01 05:18:57 +00:00
|
|
|
let mode = match size.char_at(0) {
|
|
|
|
'+' => Extend,
|
|
|
|
'-' => Reduce,
|
|
|
|
'<' => AtMost,
|
|
|
|
'>' => AtLeast,
|
|
|
|
'/' => RoundDown,
|
|
|
|
'*' => RoundUp,
|
|
|
|
_ => Reference /* assume that the size is just a number */
|
|
|
|
};
|
|
|
|
let bytes = {
|
|
|
|
let mut slice =
|
|
|
|
if mode == Reference {
|
|
|
|
let size: &str = size;
|
|
|
|
size
|
|
|
|
} else {
|
|
|
|
size.slice_from(1)
|
|
|
|
};
|
|
|
|
if slice.char_at(slice.len() - 1).is_alphabetic() {
|
|
|
|
slice = slice.slice_to(slice.len() - 1);
|
|
|
|
if slice.len() > 0 && slice.char_at(slice.len() - 1).is_alphabetic() {
|
|
|
|
slice = slice.slice_to(slice.len() - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
slice
|
2014-05-28 06:33:39 +00:00
|
|
|
}.to_string().into_bytes();
|
2014-05-23 12:28:40 +00:00
|
|
|
let mut number = match u64::parse_bytes(bytes.as_slice(), 10) {
|
2014-02-01 05:18:57 +00:00
|
|
|
Some(num) => num,
|
|
|
|
None => {
|
2014-02-07 06:39:07 +00:00
|
|
|
crash!(1, "'{}' is not a valid number.", size)
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
};
|
2014-02-07 06:39:07 +00:00
|
|
|
if size.char_at(size.len() - 1).is_alphabetic() {
|
2014-05-25 09:35:21 +00:00
|
|
|
number *= match size.char_at(size.len() - 1).to_ascii().to_uppercase().to_char() {
|
|
|
|
'B' => match size.char_at(size.len() - 2).to_ascii().to_uppercase().to_char() {
|
2014-02-01 05:18:57 +00:00
|
|
|
'K' => 1000,
|
|
|
|
'M' => 1000 * 1000,
|
|
|
|
'G' => 1000 * 1000 * 1000,
|
|
|
|
'T' => 1000 * 1000 * 1000 * 1000,
|
|
|
|
'P' => 1000 * 1000 * 1000 * 1000 * 1000,
|
|
|
|
'E' => 1000 * 1000 * 1000 * 1000 * 1000 * 1000,
|
|
|
|
'Z' => 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000,
|
|
|
|
'Y' => 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000 * 1000,
|
|
|
|
letter => {
|
2014-02-07 06:39:07 +00:00
|
|
|
crash!(1, "'{}B' is not a valid suffix.", letter)
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
'K' => 1024,
|
|
|
|
'M' => 1024 * 1024,
|
|
|
|
'G' => 1024 * 1024 * 1024,
|
|
|
|
'T' => 1024 * 1024 * 1024 * 1024,
|
|
|
|
'P' => 1024 * 1024 * 1024 * 1024 * 1024,
|
|
|
|
'E' => 1024 * 1024 * 1024 * 1024 * 1024 * 1024,
|
|
|
|
'Z' => 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024,
|
|
|
|
'Y' => 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024 * 1024,
|
|
|
|
letter => {
|
2014-02-07 06:39:07 +00:00
|
|
|
crash!(1, "'{}' is not a valid suffix.", letter)
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2014-02-07 06:39:07 +00:00
|
|
|
(number, mode)
|
2014-02-01 05:18:57 +00:00
|
|
|
}
|
|
|
|
|