160 lines
4.9 KiB
Rust
Raw Normal View History

#![forbid(unsafe_code)]
use std::path::PathBuf;
use std::{env, ffi, fs, io, process};
extern crate weezl;
use weezl::{decode as delzw, encode as enlzw, BitOrder};
fn main() {
let args = env::args_os().skip(1);
let flags = Flags::from_args(args).unwrap_or_else(|ParamError| explain());
let out = io::stdout();
let out = out.lock();
let mut files = flags.files;
let input = files.pop().unwrap_or_else(explain);
if !files.is_empty() {
return explain();
}
let operation = flags.operation.unwrap_or_else(explain);
let min_code = if flags.min_code < 2 || flags.min_code > 12 {
return explain();
} else {
flags.min_code
};
let bit_order = flags.bit_order;
let result = match (input, operation) {
(Input::File(file), Operation::Encode) => (|| {
let data = fs::File::open(file)?;
let file = io::BufReader::with_capacity(1 << 26, data);
let mut encoder = enlzw::Encoder::new(bit_order, min_code);
encoder.into_stream(out).encode_all(file).status
})(),
(Input::Stdin, Operation::Encode) => {
let input = io::BufReader::with_capacity(1 << 26, io::stdin());
let mut encoder = enlzw::Encoder::new(bit_order, min_code);
encoder.into_stream(out).encode_all(input).status
}
(Input::File(file), Operation::Decode) => (|| {
let data = fs::File::open(file)?;
let file = io::BufReader::with_capacity(1 << 26, data);
let mut decoder = delzw::Decoder::new(bit_order, min_code);
decoder.into_stream(out).decode_all(file).status
})(),
(Input::Stdin, Operation::Decode) => {
let input = io::BufReader::with_capacity(1 << 26, io::stdin());
let mut decoder = delzw::Decoder::new(bit_order, min_code);
decoder.into_stream(out).decode_all(input).status
}
};
result.expect("Operation Failed: ");
}
struct Flags {
files: Vec<Input>,
operation: Option<Operation>,
min_code: u8,
bit_order: BitOrder,
}
struct ParamError;
enum Input {
File(PathBuf),
Stdin,
}
enum Operation {
Encode,
Decode,
}
fn explain<T>() -> T {
println!(
"Usage: lzw [-e|-d] <file>\n\
Arguments:\n\
-e\t operation encode (default)\n\
-d\t operation decode\n\
<file>\tfilepath or '-' for stdin"
);
process::exit(1);
}
impl Default for Flags {
fn default() -> Flags {
Flags {
files: vec![],
operation: None,
min_code: 8,
bit_order: BitOrder::Msb,
}
}
}
impl Flags {
fn from_args(mut args: impl Iterator<Item = ffi::OsString>) -> Result<Self, ParamError> {
let mut flags = Flags::default();
let mut operation = None;
loop {
match args.next().as_ref().and_then(|s| s.to_str()) {
Some("-d") | Some("--decode") => {
if operation.is_some() {
return Err(ParamError);
}
operation = Some(Operation::Decode);
}
Some("-e") | Some("--encode") => {
if operation.is_some() {
return Err(ParamError);
}
operation = Some(Operation::Encode);
}
Some("-w") | Some("--word-bits") => match args.next() {
None => return Err(ParamError),
Some(bits) => {
let st = bits.to_str().ok_or(ParamError)?;
flags.min_code = st.parse().ok().ok_or(ParamError)?;
}
},
Some("-le") | Some("--little-endian") => {
flags.bit_order = BitOrder::Lsb;
}
Some("-be") | Some("--big-endian") | Some("-ne") | Some("--network-endian") => {
flags.bit_order = BitOrder::Msb;
}
Some("-") => {
flags.files.push(Input::Stdin);
}
Some(other) if other.starts_with('-') => {
// Reserved for future use.
// -a: self-describing archive format, similar to actual compress
// -b: maximum bits
// -v: verbosity
// some compress compatibility mode? Probably through arg(0) though.
return Err(ParamError);
}
Some(file) => {
flags.files.push(Input::File(file.into()));
}
None => break,
};
}
flags.files.extend(args.map(|file| {
if let Some("-") = file.to_str() {
Input::Stdin
} else {
Input::File(file.into())
}
}));
flags.operation = operation;
Ok(flags)
}
}