aboutsummaryrefslogtreecommitdiff
path: root/vendor/weezl/bin/lzw.rs
diff options
context:
space:
mode:
authorValentin Popov <valentin@popov.link>2024-01-08 00:21:28 +0300
committerValentin Popov <valentin@popov.link>2024-01-08 00:21:28 +0300
commit1b6a04ca5504955c571d1c97504fb45ea0befee4 (patch)
tree7579f518b23313e8a9748a88ab6173d5e030b227 /vendor/weezl/bin/lzw.rs
parent5ecd8cf2cba827454317368b68571df0d13d7842 (diff)
downloadfparkan-1b6a04ca5504955c571d1c97504fb45ea0befee4.tar.xz
fparkan-1b6a04ca5504955c571d1c97504fb45ea0befee4.zip
Initial vendor packages
Signed-off-by: Valentin Popov <valentin@popov.link>
Diffstat (limited to 'vendor/weezl/bin/lzw.rs')
-rw-r--r--vendor/weezl/bin/lzw.rs159
1 files changed, 159 insertions, 0 deletions
diff --git a/vendor/weezl/bin/lzw.rs b/vendor/weezl/bin/lzw.rs
new file mode 100644
index 0000000..cb55dce
--- /dev/null
+++ b/vendor/weezl/bin/lzw.rs
@@ -0,0 +1,159 @@
+#![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)
+ }
+}