mirror of
https://github.com/RGBCube/uutils-coreutils
synced 2025-07-31 21:17:46 +00:00
update cat, cat tests to new API
This commit updates the `cat` utility as well as its testbench to work with the latest APIs.
This commit is contained in:
parent
5b34d19a83
commit
cc48e83e98
2 changed files with 78 additions and 55 deletions
|
@ -1,5 +1,5 @@
|
||||||
#![crate_name = "cat"]
|
#![crate_name = "cat"]
|
||||||
#![feature(collections, core, old_io, old_path, rustc_private)]
|
#![feature(collections, rustc_private)]
|
||||||
|
|
||||||
#![feature(box_syntax, unsafe_destructor)]
|
#![feature(box_syntax, unsafe_destructor)]
|
||||||
|
|
||||||
|
@ -15,11 +15,15 @@
|
||||||
/* last synced with: cat (GNU coreutils) 8.13 */
|
/* last synced with: cat (GNU coreutils) 8.13 */
|
||||||
|
|
||||||
extern crate getopts;
|
extern crate getopts;
|
||||||
|
extern crate libc;
|
||||||
|
|
||||||
use std::old_io::{print, File};
|
use std::fs::File;
|
||||||
use std::old_io::stdio::{stdout_raw, stdin_raw, stderr};
|
use std::io::{stdout, stdin, stderr, Write, Read};
|
||||||
use std::old_io::{IoResult};
|
use std::io::Result;
|
||||||
use std::ptr::{copy_nonoverlapping_memory};
|
use std::intrinsics::{copy_nonoverlapping};
|
||||||
|
use libc::consts::os::posix88::STDIN_FILENO;
|
||||||
|
use libc::funcs::posix88::unistd::isatty;
|
||||||
|
use libc::types::os::arch::c95::c_int;
|
||||||
|
|
||||||
pub fn uumain(args: Vec<String>) -> i32 {
|
pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
let program = &args[0];
|
let program = &args[0];
|
||||||
|
@ -48,7 +52,7 @@ pub fn uumain(args: Vec<String>) -> i32 {
|
||||||
println!("Usage:");
|
println!("Usage:");
|
||||||
println!(" {0} [OPTION]... [FILE]...", program);
|
println!(" {0} [OPTION]... [FILE]...", program);
|
||||||
println!("");
|
println!("");
|
||||||
print(&getopts::usage("Concatenate FILE(s), or standard input, to \
|
print!("{}", &getopts::usage("Concatenate FILE(s), or standard input, to \
|
||||||
standard output.", &opts)[..]);
|
standard output.", &opts)[..]);
|
||||||
println!("");
|
println!("");
|
||||||
println!("With no FILE, or when FILE is -, read standard input.");
|
println!("With no FILE, or when FILE is -, read standard input.");
|
||||||
|
@ -99,13 +103,13 @@ fn write_lines(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
|
|
||||||
let mut in_buf = [0; 1024 * 31];
|
let mut in_buf = [0; 1024 * 31];
|
||||||
let mut out_buf = [0; 1024 * 64];
|
let mut out_buf = [0; 1024 * 64];
|
||||||
let mut writer = UnsafeWriter::new(out_buf.as_mut_slice(), stdout_raw());
|
let mut writer = UnsafeWriter::new(&mut out_buf[..], stdout());
|
||||||
let mut at_line_start = true;
|
let mut at_line_start = true;
|
||||||
while let Ok(n) = reader.read(&mut in_buf) {
|
while let Ok(n) = reader.read(&mut in_buf) {
|
||||||
if n == 0 { break }
|
if n == 0 { break }
|
||||||
|
|
||||||
let in_buf = &in_buf[..n];
|
let in_buf = &in_buf[..n];
|
||||||
let mut buf_pos = range(0, n);
|
let mut buf_pos = 0..n;
|
||||||
loop {
|
loop {
|
||||||
writer.possibly_flush();
|
writer.possibly_flush();
|
||||||
let pos = match buf_pos.next() {
|
let pos = match buf_pos.next() {
|
||||||
|
@ -119,9 +123,9 @@ fn write_lines(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
line_counter += 1;
|
line_counter += 1;
|
||||||
}
|
}
|
||||||
if show_ends {
|
if show_ends {
|
||||||
writer.write_u8('$' as u8).unwrap();
|
writer.write_all(&['$' as u8]).unwrap();
|
||||||
}
|
}
|
||||||
writer.write_u8('\n' as u8).unwrap();
|
writer.write_all(&['\n' as u8]).unwrap();
|
||||||
if interactive {
|
if interactive {
|
||||||
writer.flush().unwrap();
|
writer.flush().unwrap();
|
||||||
}
|
}
|
||||||
|
@ -137,13 +141,13 @@ fn write_lines(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
Some(p) => {
|
Some(p) => {
|
||||||
writer.write_all(&in_buf[pos..pos + p]).unwrap();
|
writer.write_all(&in_buf[pos..pos + p]).unwrap();
|
||||||
if show_ends {
|
if show_ends {
|
||||||
writer.write_u8('$' as u8).unwrap();
|
writer.write_all(&['$' as u8]).unwrap();
|
||||||
}
|
}
|
||||||
writer.write_u8('\n' as u8).unwrap();
|
writer.write_all(&['\n' as u8]).unwrap();
|
||||||
if interactive {
|
if interactive {
|
||||||
writer.flush().unwrap();
|
writer.flush().unwrap();
|
||||||
}
|
}
|
||||||
buf_pos = range(pos + p + 1, n);
|
buf_pos = pos + p + 1..n;
|
||||||
at_line_start = true;
|
at_line_start = true;
|
||||||
},
|
},
|
||||||
None => {
|
None => {
|
||||||
|
@ -165,11 +169,11 @@ fn write_bytes(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
for (mut reader, interactive) in files.iter().filter_map(|p| open(&p[..])) {
|
for (mut reader, interactive) in files.iter().filter_map(|p| open(&p[..])) {
|
||||||
|
|
||||||
// Flush all 1024 iterations.
|
// Flush all 1024 iterations.
|
||||||
let mut flush_counter = range(0usize, 1024);
|
let mut flush_counter = 0usize..1024;
|
||||||
|
|
||||||
let mut in_buf = [0; 1024 * 32];
|
let mut in_buf = [0; 1024 * 32];
|
||||||
let mut out_buf = [0; 1024 * 64];
|
let mut out_buf = [0; 1024 * 64];
|
||||||
let mut writer = UnsafeWriter::new(out_buf.as_mut_slice(), stdout_raw());
|
let mut writer = UnsafeWriter::new(&mut out_buf[..], stdout());
|
||||||
let mut at_line_start = true;
|
let mut at_line_start = true;
|
||||||
while let Ok(n) = reader.read(&mut in_buf) {
|
while let Ok(n) = reader.read(&mut in_buf) {
|
||||||
if n == 0 { break }
|
if n == 0 { break }
|
||||||
|
@ -177,7 +181,7 @@ fn write_bytes(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
for &byte in in_buf[..n].iter() {
|
for &byte in in_buf[..n].iter() {
|
||||||
if flush_counter.next().is_none() {
|
if flush_counter.next().is_none() {
|
||||||
writer.possibly_flush();
|
writer.possibly_flush();
|
||||||
flush_counter = range(0usize, 1024);
|
flush_counter = 0usize..1024;
|
||||||
}
|
}
|
||||||
if byte == '\n' as u8 {
|
if byte == '\n' as u8 {
|
||||||
if !at_line_start || !squeeze_blank {
|
if !at_line_start || !squeeze_blank {
|
||||||
|
@ -186,9 +190,9 @@ fn write_bytes(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
line_counter += 1;
|
line_counter += 1;
|
||||||
}
|
}
|
||||||
if show_ends {
|
if show_ends {
|
||||||
writer.write_u8('$' as u8).unwrap();
|
writer.write_all(&['$' as u8]).unwrap();
|
||||||
}
|
}
|
||||||
writer.write_u8('\n' as u8).unwrap();
|
writer.write_all(&['\n' as u8]).unwrap();
|
||||||
if interactive {
|
if interactive {
|
||||||
writer.flush().unwrap();
|
writer.flush().unwrap();
|
||||||
}
|
}
|
||||||
|
@ -205,14 +209,14 @@ fn write_bytes(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
// this by having the whole loop inside show_nonprint.
|
// this by having the whole loop inside show_nonprint.
|
||||||
if byte == '\t' as u8 {
|
if byte == '\t' as u8 {
|
||||||
if show_tabs {
|
if show_tabs {
|
||||||
writer.write_str("^I")
|
writer.write_all("^I".as_bytes())
|
||||||
} else {
|
} else {
|
||||||
writer.write_u8(byte)
|
writer.write_all(&[byte])
|
||||||
}
|
}
|
||||||
} else if show_nonprint {
|
} else if show_nonprint {
|
||||||
let byte = match byte {
|
let byte = match byte {
|
||||||
128 ... 255 => {
|
128 ... 255 => {
|
||||||
writer.write_str("M-").unwrap();
|
writer.write_all("M-".as_bytes()).unwrap();
|
||||||
byte - 128
|
byte - 128
|
||||||
},
|
},
|
||||||
_ => byte,
|
_ => byte,
|
||||||
|
@ -220,10 +224,10 @@ fn write_bytes(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
match byte {
|
match byte {
|
||||||
0 ... 31 => writer.write_all(&['^' as u8, byte + 64]),
|
0 ... 31 => writer.write_all(&['^' as u8, byte + 64]),
|
||||||
127 => writer.write_all(&['^' as u8, byte - 64]),
|
127 => writer.write_all(&['^' as u8, byte - 64]),
|
||||||
_ => writer.write_u8(byte),
|
_ => writer.write_all(&[byte]),
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
writer.write_u8(byte)
|
writer.write_all(&[byte])
|
||||||
}.unwrap();
|
}.unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -231,7 +235,7 @@ fn write_bytes(files: Vec<String>, number: NumberingMode, squeeze_blank: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_fast(files: Vec<String>) {
|
fn write_fast(files: Vec<String>) {
|
||||||
let mut writer = stdout_raw();
|
let mut writer = stdout();
|
||||||
let mut in_buf = [0; 1024 * 64];
|
let mut in_buf = [0; 1024 * 64];
|
||||||
|
|
||||||
for (mut reader, _) in files.iter().filter_map(|p| open(&p[..])) {
|
for (mut reader, _) in files.iter().filter_map(|p| open(&p[..])) {
|
||||||
|
@ -255,15 +259,15 @@ fn exec(files: Vec<String>, number: NumberingMode, show_nonprint: bool,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn open(path: &str) -> Option<(Box<Reader>, bool)> {
|
fn open(path: &str) -> Option<(Box<Read>, bool)> {
|
||||||
if path == "-" {
|
if path == "-" {
|
||||||
let stdin = stdin_raw();
|
let stdin = stdin();
|
||||||
let interactive = stdin.isatty();
|
let interactive = unsafe { isatty(STDIN_FILENO) } != 0 as c_int;
|
||||||
return Some((box stdin as Box<Reader>, interactive));
|
return Some((box stdin as Box<Read>, interactive));
|
||||||
}
|
}
|
||||||
|
|
||||||
match File::open(&std::old_path::Path::new(path)) {
|
match File::open(path) {
|
||||||
Ok(f) => Some((box f as Box<Reader>, false)),
|
Ok(f) => Some((box f as Box<Read>, false)),
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
(writeln!(&mut stderr(), "cat: {0}: {1}", path, e.to_string())).unwrap();
|
(writeln!(&mut stderr(), "cat: {0}: {1}", path, e.to_string())).unwrap();
|
||||||
None
|
None
|
||||||
|
@ -271,14 +275,14 @@ fn open(path: &str) -> Option<(Box<Reader>, bool)> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct UnsafeWriter<'a, W> {
|
struct UnsafeWriter<'a, W: Write> {
|
||||||
inner: W,
|
inner: W,
|
||||||
buf: &'a mut [u8],
|
buf: &'a mut [u8],
|
||||||
pos: usize,
|
pos: usize,
|
||||||
threshold: usize,
|
threshold: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, W: Writer> UnsafeWriter<'a, W> {
|
impl<'a, W: Write> UnsafeWriter<'a, W> {
|
||||||
fn new(buf: &'a mut [u8], inner: W) -> UnsafeWriter<'a, W> {
|
fn new(buf: &'a mut [u8], inner: W) -> UnsafeWriter<'a, W> {
|
||||||
let threshold = buf.len()/2;
|
let threshold = buf.len()/2;
|
||||||
UnsafeWriter {
|
UnsafeWriter {
|
||||||
|
@ -289,11 +293,14 @@ impl<'a, W: Writer> UnsafeWriter<'a, W> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn flush_buf(&mut self) -> IoResult<()> {
|
fn flush_buf(&mut self) -> Result<()> {
|
||||||
if self.pos != 0 {
|
if self.pos != 0 {
|
||||||
let ret = self.inner.write_all(&self.buf[..self.pos]);
|
let ret = self.inner.write(&self.buf[..self.pos]);
|
||||||
self.pos = 0;
|
self.pos = 0;
|
||||||
ret
|
match ret {
|
||||||
|
Ok(_) => Ok(()),
|
||||||
|
Err(e) => Err(e),
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -312,26 +319,27 @@ fn fail() -> ! {
|
||||||
panic!("assertion failed");
|
panic!("assertion failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, W: Writer> Writer for UnsafeWriter<'a, W> {
|
impl<'a, W: Write> Write for UnsafeWriter<'a, W> {
|
||||||
fn write_all(&mut self, buf: &[u8]) -> IoResult<()> {
|
fn write(&mut self, buf: &[u8]) -> Result<usize> {
|
||||||
let dst = &mut self.buf[self.pos..];
|
let dst = &mut self.buf[self.pos..];
|
||||||
if buf.len() > dst.len() {
|
let len = buf.len();
|
||||||
|
if len > dst.len() {
|
||||||
fail();
|
fail();
|
||||||
}
|
}
|
||||||
unsafe {
|
unsafe {
|
||||||
copy_nonoverlapping_memory(dst.as_mut_ptr(), buf.as_ptr(), buf.len())
|
copy_nonoverlapping(buf.as_ptr(), dst.as_mut_ptr(), len)
|
||||||
}
|
}
|
||||||
self.pos += buf.len();
|
self.pos += len;
|
||||||
Ok(())
|
Ok(len)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn flush(&mut self) -> IoResult<()> {
|
fn flush(&mut self) -> Result<()> {
|
||||||
self.flush_buf().and_then(|()| self.inner.flush())
|
self.flush_buf().and_then(|()| self.inner.flush())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[unsafe_destructor]
|
#[unsafe_destructor]
|
||||||
impl<'a, W: Writer> Drop for UnsafeWriter<'a, W> {
|
impl<'a, W: Write> Drop for UnsafeWriter<'a, W> {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
let _ = self.flush_buf();
|
let _ = self.flush_buf();
|
||||||
}
|
}
|
||||||
|
|
39
test/cat.rs
39
test/cat.rs
|
@ -1,6 +1,6 @@
|
||||||
#![allow(unstable)]
|
use std::io::Write;
|
||||||
|
use std::process::Command;
|
||||||
use std::old_io::process::Command;
|
use std::process::Stdio;
|
||||||
use std::str;
|
use std::str;
|
||||||
|
|
||||||
static PROGNAME: &'static str = "./cat";
|
static PROGNAME: &'static str = "./cat";
|
||||||
|
@ -17,29 +17,44 @@ fn test_output_multi_files_print_all_chars() {
|
||||||
Err(err) => panic!("{}", err),
|
Err(err) => panic!("{}", err),
|
||||||
};
|
};
|
||||||
|
|
||||||
let out = str::from_utf8(po.output.as_slice()).unwrap();
|
let out = str::from_utf8(&po.stdout[..]).unwrap();
|
||||||
assert_eq!(out,
|
assert_eq!(out,
|
||||||
" 1\tabcde$\n 2\tfghij$\n 3\tklmno$\n 4\tpqrst$\n 5\tuvwxyz$\n 6\t^@^A^B^C^D^E^F^G^H^I$\n 7\t^K^L^M^N^O^P^Q^R^S^T^U^V^W^X^Y^Z^[^\\^]^^^_ !\"#$%&\'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~^?M-^@M-^AM-^BM-^CM-^DM-^EM-^FM-^GM-^HM-^IM-^JM-^KM-^LM-^MM-^NM-^OM-^PM-^QM-^RM-^SM-^TM-^UM-^VM-^WM-^XM-^YM-^ZM-^[M-^\\M-^]M-^^M-^_M- M-!M-\"M-#M-$M-%M-&M-\'M-(M-)M-*M-+M-,M--M-.M-/M-0M-1M-2M-3M-4M-5M-6M-7M-8M-9M-:M-;M-<M-=M->M-?M-@M-AM-BM-CM-DM-EM-FM-GM-HM-IM-JM-KM-LM-MM-NM-OM-PM-QM-RM-SM-TM-UM-VM-WM-XM-YM-ZM-[M-\\M-]M-^M-_M-`M-aM-bM-cM-dM-eM-fM-gM-hM-iM-jM-kM-lM-mM-nM-oM-pM-qM-rM-sM-tM-uM-vM-wM-xM-yM-zM-{M-|M-}M-~M-^?");
|
" 1\tabcde$\n 2\tfghij$\n 3\tklmno$\n 4\tpqrst$\n 5\tuvwxyz$\n 6\t^@^A^B^C^D^E^F^G^H^I$\n 7\t^K^L^M^N^O^P^Q^R^S^T^U^V^W^X^Y^Z^[^\\^]^^^_ !\"#$%&\'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~^?M-^@M-^AM-^BM-^CM-^DM-^EM-^FM-^GM-^HM-^IM-^JM-^KM-^LM-^MM-^NM-^OM-^PM-^QM-^RM-^SM-^TM-^UM-^VM-^WM-^XM-^YM-^ZM-^[M-^\\M-^]M-^^M-^_M- M-!M-\"M-#M-$M-%M-&M-\'M-(M-)M-*M-+M-,M--M-.M-/M-0M-1M-2M-3M-4M-5M-6M-7M-8M-9M-:M-;M-<M-=M->M-?M-@M-AM-BM-CM-DM-EM-FM-GM-HM-IM-JM-KM-LM-MM-NM-OM-PM-QM-RM-SM-TM-UM-VM-WM-XM-YM-ZM-[M-\\M-]M-^M-_M-`M-aM-bM-cM-dM-eM-fM-gM-hM-iM-jM-kM-lM-mM-nM-oM-pM-qM-rM-sM-tM-uM-vM-wM-xM-yM-zM-{M-|M-}M-~M-^?");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_stdin_squeeze() {
|
fn test_stdin_squeeze() {
|
||||||
let mut process= Command::new(PROGNAME).arg("-A").spawn().unwrap();
|
let mut process = Command::new(PROGNAME)
|
||||||
|
.arg("-A")
|
||||||
|
.stdin(Stdio::piped())
|
||||||
|
.stdout(Stdio::piped())
|
||||||
|
.spawn()
|
||||||
|
.unwrap_or_else(|e| panic!("{}", e));
|
||||||
|
|
||||||
process.stdin.take().unwrap().write_all(b"\x00\x01\x02").unwrap();
|
process.stdin.take().unwrap_or_else(|| panic!("Could not grab child process stdin"))
|
||||||
let po = process.wait_with_output().unwrap();
|
.write_all("\x00\x01\x02".as_bytes()).unwrap_or_else(|e| panic!("{}", e));
|
||||||
let out = str::from_utf8(po.output.as_slice()).unwrap();
|
|
||||||
|
let po = process.wait_with_output().unwrap_or_else(|e| panic!("{}", e));
|
||||||
|
let out = str::from_utf8(&po.stdout[..]).unwrap_or_else(|e| panic!("{}", e));
|
||||||
|
|
||||||
assert_eq!(out, "^@^A^B");
|
assert_eq!(out, "^@^A^B");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_stdin_number_non_blank() {
|
fn test_stdin_number_non_blank() {
|
||||||
let mut process = Command::new(PROGNAME).arg("-b").arg("-").spawn().unwrap();
|
let mut process = Command::new(PROGNAME)
|
||||||
|
.arg("-b")
|
||||||
|
.arg("-")
|
||||||
|
.stdin(Stdio::piped())
|
||||||
|
.stdout(Stdio::piped())
|
||||||
|
.spawn()
|
||||||
|
.unwrap_or_else(|e| panic!("{}", e));
|
||||||
|
|
||||||
process.stdin.take().unwrap().write_all(b"\na\nb\n\n\nc").unwrap();
|
process.stdin.take().unwrap_or_else(|| panic!("Could not grab child process stdin"))
|
||||||
let po = process.wait_with_output().unwrap();
|
.write_all("\na\nb\n\n\nc".as_bytes()).unwrap_or_else(|e| panic!("{}", e));
|
||||||
let out = str::from_utf8(po.output.as_slice()).unwrap();
|
|
||||||
|
let po = process.wait_with_output().unwrap_or_else(|e| panic!("{}", e));
|
||||||
|
let out = str::from_utf8(&po.stdout[..]).unwrap_or_else(|e| panic!("{}", e));
|
||||||
|
|
||||||
assert_eq!(out, "\n 1\ta\n 2\tb\n\n\n 3\tc");
|
assert_eq!(out, "\n 1\ta\n 2\tb\n\n\n 3\tc");
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue