mirror of
https://github.com/uutils/coreutils
synced 2024-12-15 15:52:42 +00:00
30bba07f9c
squashed: a2c6b27 - build: automatically generate main() files c942f0f - remove MULTICALL=1 build from travis cb7b35b - make: remove unnecessary shell command 69bbb31 - update README 03a3168 - all: move main() into separate file that links against util crate 8276384 - make: always build multicall binary aa4edeb - make: avoid 'rustc --crate-file-name'
119 lines
4.1 KiB
Rust
119 lines
4.1 KiB
Rust
#![crate_id(name = "paste", vers = "1.0.0", author = "Arcterus")]
|
|
|
|
/*
|
|
* 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.
|
|
*/
|
|
|
|
#![feature(macro_rules)]
|
|
|
|
extern crate getopts;
|
|
extern crate libc;
|
|
|
|
use std::io;
|
|
|
|
#[path = "../common/util.rs"]
|
|
mod util;
|
|
|
|
static NAME: &'static str = "paste";
|
|
static VERSION: &'static str = "1.0.0";
|
|
|
|
pub fn uumain(args: Vec<String>) -> int {
|
|
let program = args.get(0).clone();
|
|
|
|
let opts = [
|
|
getopts::optflag("s", "serial", "paste one file at a time instead of in parallel"),
|
|
getopts::optopt("d", "delimiters", "reuse characters from LIST instead of TABs", "LIST"),
|
|
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(f) => crash!(1, "{}", f)
|
|
};
|
|
if matches.opt_present("help") {
|
|
println!("{} {}", NAME, VERSION);
|
|
println!("");
|
|
println!("Usage:");
|
|
println!(" {0:s} [OPTION]... [FILE]...", program);
|
|
println!("");
|
|
print!("{}", getopts::usage("Write lines consisting of the sequentially corresponding lines from each FILE, separated by TABs, to standard output.", opts));
|
|
} else if matches.opt_present("version") {
|
|
println!("{} {}", NAME, VERSION);
|
|
} else {
|
|
let serial = matches.opt_present("serial");
|
|
let delimiters = match matches.opt_str("delimiters") {
|
|
Some(m) => m,
|
|
None => "\t".to_string()
|
|
};
|
|
paste(matches.free, serial, delimiters.as_slice());
|
|
}
|
|
|
|
0
|
|
}
|
|
|
|
fn paste(filenames: Vec<String>, serial: bool, delimiters: &str) {
|
|
let mut files: Vec<io::BufferedReader<Box<Reader>>> = filenames.move_iter().map(|name|
|
|
io::BufferedReader::new(
|
|
if name.as_slice() == "-" {
|
|
box io::stdio::stdin_raw() as Box<Reader>
|
|
} else {
|
|
box crash_if_err!(1, io::File::open(&Path::new(name))) as Box<Reader>
|
|
}
|
|
)
|
|
).collect();
|
|
let delimiters: Vec<String> = delimiters.chars().map(|x| x.to_str()).collect();
|
|
let mut delim_count = 0;
|
|
if serial {
|
|
for file in files.mut_iter() {
|
|
let mut output = String::new();
|
|
loop {
|
|
match file.read_line() {
|
|
Ok(line) => {
|
|
output.push_str(line.as_slice().trim_right());
|
|
output.push_str(delimiters.get(delim_count % delimiters.len()).as_slice());
|
|
}
|
|
Err(f) => if f.kind == io::EndOfFile {
|
|
break
|
|
} else {
|
|
crash!(1, "{}", f.to_str())
|
|
}
|
|
}
|
|
delim_count += 1;
|
|
}
|
|
println!("{}", output.as_slice().slice_to(output.len() - 1));
|
|
}
|
|
} else {
|
|
let mut eof = Vec::from_elem(files.len(), false);
|
|
loop {
|
|
let mut output = "".to_string();
|
|
let mut eof_count = 0;
|
|
for (i, file) in files.mut_iter().enumerate() {
|
|
if *eof.get(i) {
|
|
eof_count += 1;
|
|
} else {
|
|
match file.read_line() {
|
|
Ok(line) => output.push_str(line.as_slice().slice_to(line.len() - 1)),
|
|
Err(f) => if f.kind == io::EndOfFile {
|
|
*eof.get_mut(i) = true;
|
|
eof_count += 1;
|
|
} else {
|
|
crash!(1, "{}", f.to_str());
|
|
}
|
|
}
|
|
}
|
|
output.push_str(delimiters.get(delim_count % delimiters.len()).as_slice());
|
|
delim_count += 1;
|
|
}
|
|
if files.len() == eof_count {
|
|
break;
|
|
}
|
|
println!("{}", output.as_slice().slice_to(output.len() - 1));
|
|
delim_count = 0;
|
|
}
|
|
}
|
|
}
|