Update argument parsing to clap v3

This commit is contained in:
Tangent Wantwight 2022-05-22 20:37:03 -04:00
parent 15520f26bd
commit bfe7d4b1d7
8 changed files with 205 additions and 207 deletions

102
Cargo.lock generated
View file

@ -11,15 +11,6 @@ dependencies = [
"memchr", "memchr",
] ]
[[package]]
name = "ansi_term"
version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d52a9bb7ec0cf484c551830a7ce27bd20d67eac647e1befb56b0be4ee39a55d2"
dependencies = [
"winapi",
]
[[package]] [[package]]
name = "atty" name = "atty"
version = "0.2.14" version = "0.2.14"
@ -97,17 +88,41 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
[[package]] [[package]]
name = "clap" name = "clap"
version = "2.34.0" version = "3.1.18"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a0610544180c38b88101fecf2dd634b174a62eef6946f84dfc6a7127512b381c" checksum = "d2dbdf4bdacb33466e854ce889eee8dfd5729abf7ccd7664d0a2d60cd384440b"
dependencies = [ dependencies = [
"ansi_term",
"atty", "atty",
"bitflags", "bitflags",
"clap_derive",
"clap_lex",
"indexmap",
"lazy_static",
"strsim", "strsim",
"termcolor",
"textwrap", "textwrap",
"unicode-width", ]
"vec_map",
[[package]]
name = "clap_derive"
version = "3.1.18"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "25320346e922cffe59c0bbc5410c8d8784509efb321488971081313cb1e1a33c"
dependencies = [
"heck",
"proc-macro-error",
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "clap_lex"
version = "0.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a37c35f1112dad5e6e0b1adaff798507497a18fceeb30cceb3bae7d1427b9213"
dependencies = [
"os_str_bytes",
] ]
[[package]] [[package]]
@ -352,6 +367,12 @@ dependencies = [
"http", "http",
] ]
[[package]]
name = "heck"
version = "0.4.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2540771e65fc8cb83cd6e8a237f70c319bd5c29f78ed1084ba5d50eeac86f7f9"
[[package]] [[package]]
name = "hermit-abi" name = "hermit-abi"
version = "0.1.19" version = "0.1.19"
@ -562,6 +583,12 @@ version = "0.3.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5"
[[package]]
name = "os_str_bytes"
version = "6.0.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "029d8d0b2f198229de29dca79676f2738ff952edf3fde542eb8bf94d8c21b435"
[[package]] [[package]]
name = "percent-encoding" name = "percent-encoding"
version = "2.1.0" version = "2.1.0"
@ -606,6 +633,30 @@ version = "0.2.16"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872" checksum = "eb9f9e6e233e5c4a35559a617bf40a4ec447db2e84c20b55a6f83167b7e57872"
[[package]]
name = "proc-macro-error"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c"
dependencies = [
"proc-macro-error-attr",
"proc-macro2",
"quote",
"syn",
"version_check",
]
[[package]]
name = "proc-macro-error-attr"
version = "1.0.4"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869"
dependencies = [
"proc-macro2",
"quote",
"version_check",
]
[[package]] [[package]]
name = "proc-macro2" name = "proc-macro2"
version = "1.0.39" version = "1.0.39"
@ -784,9 +835,9 @@ dependencies = [
[[package]] [[package]]
name = "strsim" name = "strsim"
version = "0.8.0" version = "0.10.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8ea5119cdb4c55b55d432abb513a0429384878c15dde60cc77b1c99de1a95a6a" checksum = "73473c0e59e6d5812c5dfe2a064a6444949f089e20eec9a2e5506596494e4623"
[[package]] [[package]]
name = "syn" name = "syn"
@ -824,12 +875,9 @@ dependencies = [
[[package]] [[package]]
name = "textwrap" name = "textwrap"
version = "0.11.0" version = "0.15.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d326610f408c7a4eb6f51c37c330e496b08506c9457c9d34287ecc38809fb060" checksum = "b1141d4d61095b28419e22cb0bbf02755f5e54e0526f97f1e3d1d160e60885fb"
dependencies = [
"unicode-width",
]
[[package]] [[package]]
name = "thiserror" name = "thiserror"
@ -1056,12 +1104,6 @@ dependencies = [
"tinyvec", "tinyvec",
] ]
[[package]]
name = "unicode-width"
version = "0.1.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3ed742d4ea2bd1176e236172c8429aaf54486e7ac098db29ffe6529e0ce50973"
[[package]] [[package]]
name = "url" name = "url"
version = "2.2.2" version = "2.2.2"
@ -1080,12 +1122,6 @@ version = "0.7.6"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9"
[[package]]
name = "vec_map"
version = "0.8.2"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f1bddf1187be692e79c5ffeab891132dfb0f236ed36a43c7ed39f1165ee20191"
[[package]] [[package]]
name = "version_check" name = "version_check"
version = "0.9.4" version = "0.9.4"

View file

@ -7,7 +7,7 @@ edition = "2018"
[dependencies] [dependencies]
byteorder = "1" byteorder = "1"
bytes = "1" bytes = "1"
clap = "^2.33" clap = { version="^3.1.18", features=["cargo", "derive"] }
custom_error = "^1.7" custom_error = "^1.7"
env_logger = "^0.9" env_logger = "^0.9"
futures = "^0.3" futures = "^0.3"

View file

@ -1,24 +1,18 @@
use clap::{App, AppSettings, ArgMatches, SubCommand}; use clap::Args;
use super::stdin_stream; use super::stdin_stream;
use webmetro::{ use webmetro::{
error::WebmetroError, error::WebmetroError,
stream_parser::StreamEbml, stream_parser::StreamEbml,
webm::{ webm::{SimpleBlock, WebmElement::*},
SimpleBlock,
WebmElement::*
}
}; };
pub fn options() -> App<'static, 'static> { /// Dumps WebM parsing events from parsing stdin
SubCommand::with_name("dump") #[derive(Args, Debug)]
.setting(AppSettings::Hidden) pub struct DumpArgs;
.about("Dumps WebM parsing events from parsing stdin")
}
#[tokio::main] #[tokio::main]
pub async fn run(_args: &ArgMatches) -> Result<(), WebmetroError> { pub async fn run(_args: DumpArgs) -> Result<(), WebmetroError> {
let mut events = stdin_stream().parse_ebml(); let mut events = stdin_stream().parse_ebml();
while let Some(element) = events.next().await? { while let Some(element) = events.next().await? {
@ -26,7 +20,7 @@ pub async fn run(_args: &ArgMatches) -> Result<(), WebmetroError> {
// suppress printing byte arrays // suppress printing byte arrays
Tracks(slice) => println!("Tracks[{}]", slice.len()), Tracks(slice) => println!("Tracks[{}]", slice.len()),
SimpleBlock(SimpleBlock { timecode, .. }) => println!("SimpleBlock@{}", timecode), SimpleBlock(SimpleBlock { timecode, .. }) => println!("SimpleBlock@{}", timecode),
other => println!("{:?}", other) other => println!("{:?}", other),
} }
} }
Ok(()) Ok(())

View file

@ -1,6 +1,6 @@
use std::{io, io::prelude::*, pin::Pin}; use std::{io, io::prelude::*, pin::Pin};
use clap::{App, Arg, ArgMatches, SubCommand}; use clap::Args;
use futures::prelude::*; use futures::prelude::*;
use super::stdin_stream; use super::stdin_stream;
@ -11,16 +11,16 @@ use webmetro::{
stream_parser::StreamEbml, stream_parser::StreamEbml,
}; };
pub fn options() -> App<'static, 'static> { /// Copies WebM from stdin to stdout, applying the same cleanup & stripping the relay server does.
SubCommand::with_name("filter") #[derive(Args, Debug)]
.about("Copies WebM from stdin to stdout, applying the same cleanup & stripping the relay server does.") pub struct FilterArgs {
.arg(Arg::with_name("throttle") /// Slow down output to "real time" speed as determined by the timestamps (useful for streaming static files)
.long("throttle") #[clap(long, short)]
.help("Slow down output to \"real time\" speed as determined by the timestamps (useful for streaming static files)")) throttle: bool,
} }
#[tokio::main] #[tokio::main]
pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> { pub async fn run(args: FilterArgs) -> Result<(), WebmetroError> {
let mut timecode_fixer = ChunkTimecodeFixer::new(); let mut timecode_fixer = ChunkTimecodeFixer::new();
let mut chunk_stream: Pin<Box<dyn Stream<Item = Result<Chunk, WebmetroError>> + Send>> = let mut chunk_stream: Pin<Box<dyn Stream<Item = Result<Chunk, WebmetroError>> + Send>> =
Box::pin( Box::pin(
@ -30,7 +30,7 @@ pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> {
.map_ok(move |chunk| timecode_fixer.process(chunk)), .map_ok(move |chunk| timecode_fixer.process(chunk)),
); );
if args.is_present("throttle") { if args.throttle {
chunk_stream = Box::pin(Throttle::new(chunk_stream)); chunk_stream = Box::pin(Throttle::new(chunk_stream));
} }

View file

@ -17,14 +17,11 @@ pub fn stdin_stream() -> impl Stream<Item = Result<Bytes, std::io::Error>> + Siz
FramedRead::new(tokio::io::stdin(), BytesCodec::new()).map_ok(|bytes| bytes.freeze()) FramedRead::new(tokio::io::stdin(), BytesCodec::new()).map_ok(|bytes| bytes.freeze())
} }
pub fn parse_time(arg: Option<&str>) -> Result<Option<Duration>, WebmetroError> { pub fn parse_time(arg: &str) -> Result<Duration, WebmetroError> {
match arg { match arg.parse() {
Some(string) => match string.parse() { Ok(secs) => Ok(Duration::from_secs(secs)),
Ok(secs) => Ok(Some(Duration::from_secs(secs))),
Err(err) => Err(WebmetroError::ApplicationError { Err(err) => Err(WebmetroError::ApplicationError {
message: err.to_string(), message: err.to_string(),
}), }),
},
None => Ok(None),
} }
} }

View file

@ -1,74 +1,53 @@
use std::net::ToSocketAddrs; use std::net::ToSocketAddrs;
use std::sync::{ use std::sync::{Arc, Mutex, Weak};
Arc,
Mutex,
Weak
};
use bytes::{Bytes, Buf}; use bytes::{Buf, Bytes};
use clap::{App, Arg, ArgMatches, SubCommand}; use clap::Args;
use futures::{ use futures::{prelude::*, stream::FuturesUnordered, Stream};
prelude::*,
Stream,
stream::FuturesUnordered,
};
use hyper::{ use hyper::{
Body, header::{CACHE_CONTROL, CONTENT_TYPE},
Response, Body, Response,
header::{
CACHE_CONTROL,
CONTENT_TYPE
}
}; };
use stream::iter; use stream::iter;
use warp::{ use warp::{self, path, Filter};
self, use weak_table::WeakValueHashMap;
Filter,
path
};
use weak_table::{
WeakValueHashMap
};
use webmetro::{ use webmetro::{
channel::{ channel::{Channel, Handle, Listener, Transmitter},
Channel, chunk::Chunk,
Handle,
Listener,
Transmitter
},
chunk::WebmStream, chunk::WebmStream,
error::WebmetroError, error::WebmetroError,
fixers::{ fixers::{ChunkStream, ChunkTimecodeFixer},
ChunkStream, stream_parser::StreamEbml,
ChunkTimecodeFixer, };
},
stream_parser::StreamEbml
, chunk::Chunk};
const BUFFER_LIMIT: usize = 2 * 1024 * 1024; const BUFFER_LIMIT: usize = 2 * 1024 * 1024;
fn get_stream(channel: Handle) -> impl Stream<Item = Result<Bytes, WebmetroError>> { fn get_stream(channel: Handle) -> impl Stream<Item = Result<Bytes, WebmetroError>> {
let mut timecode_fixer = ChunkTimecodeFixer::new(); let mut timecode_fixer = ChunkTimecodeFixer::new();
Listener::new(channel).map(|c| Result::<Chunk, WebmetroError>::Ok(c)) Listener::new(channel)
.map(|c| Result::<Chunk, WebmetroError>::Ok(c))
.map_ok(move |chunk| timecode_fixer.process(chunk)) .map_ok(move |chunk| timecode_fixer.process(chunk))
.find_starting_point() .find_starting_point()
.map_ok(|webm_chunk| iter(webm_chunk).map(Result::<Bytes, WebmetroError>::Ok)) .map_ok(|webm_chunk| iter(webm_chunk).map(Result::<Bytes, WebmetroError>::Ok))
.try_flatten() .try_flatten()
} }
fn post_stream(channel: Handle, stream: impl Stream<Item = Result<impl Buf, warp::Error>> + Unpin) -> impl Stream<Item = Result<Bytes, WebmetroError>> { fn post_stream(
channel: Handle,
stream: impl Stream<Item = Result<impl Buf, warp::Error>> + Unpin,
) -> impl Stream<Item = Result<Bytes, WebmetroError>> {
let channel = Transmitter::new(channel); let channel = Transmitter::new(channel);
stream stream
.map_err(WebmetroError::from) .map_err(WebmetroError::from)
.parse_ebml().with_soft_limit(BUFFER_LIMIT) .parse_ebml()
.chunk_webm().with_soft_limit(BUFFER_LIMIT) .with_soft_limit(BUFFER_LIMIT)
.chunk_webm()
.with_soft_limit(BUFFER_LIMIT)
.map_ok(move |chunk| { .map_ok(move |chunk| {
channel.send(chunk); channel.send(chunk);
Bytes::new() Bytes::new()
}) })
.inspect_err(|err| { .inspect_err(|err| warn!("{}", err))
warn!("{}", err)
})
} }
fn media_response(body: Body) -> Response<Body> { fn media_response(body: Body) -> Response<Body> {
@ -80,18 +59,19 @@ fn media_response(body: Body) -> Response<Body> {
.unwrap() .unwrap()
} }
pub fn options() -> App<'static, 'static> { /// Hosts an HTTP-based relay server
SubCommand::with_name("relay") #[derive(Args, Debug)]
.about("Hosts an HTTP-based relay server") pub struct RelayArgs {
.arg(Arg::with_name("listen") /// The address:port to listen to
.help("The address:port to listen to") listen: String,
.required(true))
} }
#[tokio::main] #[tokio::main]
pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> { pub async fn run(args: RelayArgs) -> Result<(), WebmetroError> {
let channel_map = Arc::new(Mutex::new(WeakValueHashMap::<String, Weak<Mutex<Channel>>>::new())); let channel_map = Arc::new(Mutex::new(
let addr_str = args.value_of("listen").ok_or("Listen address wasn't provided")?; WeakValueHashMap::<String, Weak<Mutex<Channel>>>::new(),
));
let addr_str = args.listen;
let addrs = addr_str.to_socket_addrs()?; let addrs = addr_str.to_socket_addrs()?;
info!("Binding to {:?}", addrs); info!("Binding to {:?}", addrs);
@ -100,37 +80,40 @@ pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> {
} }
let channel = path!("live" / String).map(move |name: String| { let channel = path!("live" / String).map(move |name: String| {
let channel = channel_map.lock().unwrap() let channel = channel_map
.lock()
.unwrap()
.entry(name.clone()) .entry(name.clone())
.or_insert_with(|| Channel::new(name.clone())); .or_insert_with(|| Channel::new(name.clone()));
(channel, name) (channel, name)
}); });
let head = channel.clone().and(warp::head()) let head = channel.clone().and(warp::head()).map(|(_, name)| {
.map(|(_, name)| {
info!("HEAD Request For Channel {}", name); info!("HEAD Request For Channel {}", name);
media_response(Body::empty()) media_response(Body::empty())
}); });
let get = channel.clone().and(warp::get()) let get = channel.clone().and(warp::get()).map(|(channel, name)| {
.map(|(channel, name)| {
info!("Listener Connected On Channel {}", name); info!("Listener Connected On Channel {}", name);
media_response(Body::wrap_stream(get_stream(channel))) media_response(Body::wrap_stream(get_stream(channel)))
}); });
let post_put = channel.clone().and(warp::post().or(warp::put()).unify()) let post_put = channel
.and(warp::body::stream()).map(|(channel, name), stream| { .clone()
.and(warp::post().or(warp::put()).unify())
.and(warp::body::stream())
.map(|(channel, name), stream| {
info!("Source Connected On Channel {}", name); info!("Source Connected On Channel {}", name);
Response::new(Body::wrap_stream(post_stream(channel, stream))) Response::new(Body::wrap_stream(post_stream(channel, stream)))
}); });
let routes = head let routes = head.or(get).or(post_put);
.or(get)
.or(post_put);
let mut server_futures: FuturesUnordered<_> = addrs.map(|addr| warp::serve(routes.clone()).try_bind(addr)).collect(); let mut server_futures: FuturesUnordered<_> = addrs
.map(|addr| warp::serve(routes.clone()).try_bind(addr))
.collect();
while let Some(_) = server_futures.next().await {}; while let Some(_) = server_futures.next().await {}
Ok(()) Ok(())
} }

View file

@ -1,8 +1,12 @@
use bytes::Bytes; use bytes::Bytes;
use clap::{App, Arg, ArgMatches, SubCommand}; use clap::Args;
use futures::prelude::*; use futures::prelude::*;
use hyper::{client::HttpConnector, Body, Client, Request}; use hyper::{client::HttpConnector, Body, Client, Request};
use std::{io::{stdout, Write}, pin::Pin}; use std::{
io::{stdout, Write},
pin::Pin,
time::Duration,
};
use stream::iter; use stream::iter;
use super::{parse_time, stdin_stream}; use super::{parse_time, stdin_stream};
@ -13,39 +17,30 @@ use webmetro::{
stream_parser::StreamEbml, stream_parser::StreamEbml,
}; };
pub fn options() -> App<'static, 'static> {
SubCommand::with_name("send")
.about("PUTs WebM from stdin to a relay server.")
.arg(Arg::with_name("url")
.help("The location to upload to")
.required(true))
.arg(Arg::with_name("throttle")
.long("throttle")
.help("Slow down upload to \"real time\" speed as determined by the timestamps (useful for streaming static files)"))
.arg(Arg::with_name("skip")
.takes_value(true)
.short("s")
.long("skip")
.help("Skip approximately n seconds of content before uploading or throttling"))
.arg(Arg::with_name("take")
.takes_value(true)
.short("t")
.long("take")
.help("Stop uploading after approximately n seconds of content"))
}
type BoxedChunkStream = Pin<Box<dyn Stream<Item = Result<Chunk, WebmetroError>> + Send + Sync>>; type BoxedChunkStream = Pin<Box<dyn Stream<Item = Result<Chunk, WebmetroError>> + Send + Sync>>;
#[tokio::main] /// PUTs WebM from stdin to a relay server.
pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> { #[derive(Args, Debug)]
// parse args pub struct SendArgs {
let url_str = match args.value_of("url") { /// The location to upload to
Some(url) => String::from(url), url: String,
_ => return Err("Listen address wasn't provided".into()), /// Slow down upload to "real time" speed as determined by the timestamps (useful for streaming static files)
}; #[clap(long)]
throttle: bool,
/// Skip approximately n seconds of content before uploading or throttling
#[clap(long, short, parse(try_from_str = parse_time))]
skip: Option<Duration>,
/// Stop uploading after approximately n seconds of content
#[clap(long, short, parse(try_from_str = parse_time))]
take: Option<Duration>,
}
let start_time = parse_time(args.value_of("skip"))?.map_or(0, |s| s.as_millis()); #[tokio::main]
let stop_time = parse_time(args.value_of("take"))?.map_or(std::u128::MAX, |t| t.as_millis() + start_time); pub async fn run(args: SendArgs) -> Result<(), WebmetroError> {
let start_time = args.skip.map_or(0, |s| s.as_millis());
let stop_time = args
.take
.map_or(std::u128::MAX, |t| t.as_millis() + start_time);
// build pipeline // build pipeline
let mut timecode_fixer = ChunkTimecodeFixer::new(); let mut timecode_fixer = ChunkTimecodeFixer::new();
@ -57,7 +52,7 @@ pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> {
.try_filter(move |chunk| future::ready(chunk.overlaps(start_time, stop_time))), .try_filter(move |chunk| future::ready(chunk.overlaps(start_time, stop_time))),
); );
if args.is_present("throttle") { if args.throttle {
chunk_stream = Box::pin(Throttle::new(chunk_stream)); chunk_stream = Box::pin(Throttle::new(chunk_stream));
} }
@ -71,7 +66,7 @@ pub async fn run(args: &ArgMatches) -> Result<(), WebmetroError> {
let request_payload = Body::wrap_stream(chunk_stream); let request_payload = Body::wrap_stream(chunk_stream);
let request = Request::put(url_str).body(request_payload)?; let request = Request::put(args.url).body(request_payload)?;
let client = Client::builder().build(HttpConnector::new()); let client = Client::builder().build(HttpConnector::new());
let response = client.request(request).await?; let response = client.request(request).await?;

View file

@ -1,44 +1,37 @@
#[macro_use]
#[macro_use] extern crate log; extern crate log;
mod commands; mod commands;
use clap::{App, AppSettings, crate_version}; use clap::{Parser, Subcommand};
use crate::commands::{ /// Utilities for broadcasting & relaying live WebM video/audio streams
relay, #[derive(Parser, Debug)]
filter, #[clap(version)]
send, struct Args {
dump #[clap(subcommand)]
}; command: Command,
}
fn options() -> App<'static, 'static> { #[derive(Subcommand, Debug)]
App::new("webmetro") enum Command {
.version(crate_version!()) Dump(commands::dump::DumpArgs),
.about("Utilities for broadcasting & relaying live WebM video/audio streams") Filter(commands::filter::FilterArgs),
.setting(AppSettings::DisableHelpSubcommand) Relay(commands::relay::RelayArgs),
.setting(AppSettings::VersionlessSubcommands) Send(commands::send::SendArgs),
.subcommand(relay::options())
.subcommand(filter::options())
.subcommand(send::options())
.subcommand(dump::options())
} }
fn main() { fn main() {
env_logger::init(); env_logger::init();
let args = options().get_matches(); let args = Args::parse();
match args.subcommand() { match args.command {
("filter", Some(sub_args)) => filter::run(sub_args), Command::Dump(args) => commands::dump::run(args),
("relay", Some(sub_args)) => relay::run(sub_args), Command::Filter(args) => commands::filter::run(args),
("send", Some(sub_args)) => send::run(sub_args), Command::Relay(args) => commands::relay::run(args),
("dump", Some(sub_args)) => dump::run(sub_args), Command::Send(args) => commands::send::run(args),
_ => {
options().print_help().unwrap();
println!("");
Ok(())
} }
}.unwrap_or_else(|err| { .unwrap_or_else(|err| {
error!("{}", err); error!("{}", err);
}); });
} }