diff --git a/Cargo.lock b/Cargo.lock index 64079111..12d9f412 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -107,6 +107,12 @@ version = "3.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0d261e256854913907f67ed06efbc3338dfe6179796deefc1ff763fc1aee5535" +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + [[package]] name = "bytes" version = "1.4.0" @@ -119,6 +125,17 @@ version = "1.0.79" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "50d30906286121d95be3d479533b458f87493b30a4b5f79a607db8f5d11aa91f" +[[package]] +name = "cfb" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b390793e912300f1aa713429f7fd0c391024e6c18b988962558bc4f96a349b1f" +dependencies = [ + "byteorder", + "fnv", + "uuid", +] + [[package]] name = "cfg-if" version = "1.0.0" @@ -394,9 +411,9 @@ checksum = "34aa73646ffb006b8f5147f3dc182bd4bcb190227ce861fc4a4844bf8e3cb2c0" [[package]] name = "encoding_rs" -version = "0.8.32" +version = "0.8.33" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "071a31f4ee85403370b58aca746f01041ede6f0da2730960ad001edc2b71b394" +checksum = "7268b386296a025e474d5140678f75d6de9493ae55a5d709eeb9dd08149945e1" dependencies = [ "cfg-if", ] @@ -1222,10 +1239,12 @@ version = "0.19.0" dependencies = [ "anyhow", "backoff", + "cfb", "chrono", "colored", "dialoguer", "dirs", + "encoding_rs", "env_logger", "indicatif", "log", diff --git a/cli/Cargo.toml b/cli/Cargo.toml index ec8768d8..4a10e6c5 100644 --- a/cli/Cargo.toml +++ b/cli/Cargo.toml @@ -38,6 +38,8 @@ reinfer-client = { version = "0.19.0", path = "../api" } dialoguer = "0.10.4" scoped_threadpool = "0.1.9" backoff = "0.4.0" +cfb = "0.9.0" +encoding_rs = "0.8.33" [dev-dependencies] pretty_assertions = "1.3.0" diff --git a/cli/src/args.rs b/cli/src/args.rs index a3e8cabc..7c078aab 100644 --- a/cli/src/args.rs +++ b/cli/src/args.rs @@ -1,6 +1,6 @@ use crate::{ commands::{ - config::ConfigArgs, create::CreateArgs, delete::DeleteArgs, get::GetArgs, + config::ConfigArgs, create::CreateArgs, delete::DeleteArgs, get::GetArgs, parse::ParseArgs, update::UpdateArgs, }, printer::OutputFormat, @@ -104,6 +104,13 @@ pub enum Command { #[structopt(subcommand)] get_args: GetArgs, }, + + #[structopt(name = "parse")] + /// Upload data from various file types + Parse { + #[structopt(subcommand)] + parse_args: ParseArgs, + }, } #[derive(Debug)] diff --git a/cli/src/commands/mod.rs b/cli/src/commands/mod.rs index be824d0c..2d781a82 100644 --- a/cli/src/commands/mod.rs +++ b/cli/src/commands/mod.rs @@ -6,6 +6,7 @@ pub mod config; pub mod create; pub mod delete; pub mod get; +pub mod parse; pub mod update; pub fn ensure_uip_user_consents_to_ai_unit_charge(base_url: &Url) -> Result<()> { diff --git a/cli/src/commands/parse/mod.rs b/cli/src/commands/parse/mod.rs new file mode 100644 index 00000000..9523197d --- /dev/null +++ b/cli/src/commands/parse/mod.rs @@ -0,0 +1,21 @@ +mod msgs; + +use anyhow::Result; +use reinfer_client::Client; +use structopt::StructOpt; + +use self::msgs::ParseMsgArgs; + +#[derive(Debug, StructOpt)] +pub enum ParseArgs { + #[structopt(name = "msgs")] + /// Parse unicode msg files. Note: Currently the body is processed as plain text. + /// Html bodies are not supported. + Msgs(ParseMsgArgs), +} + +pub fn run(args: &ParseArgs, client: Client) -> Result<()> { + match args { + ParseArgs::Msgs(parse_msg_args) => msgs::parse(&client, parse_msg_args), + } +} diff --git a/cli/src/commands/parse/msgs.rs b/cli/src/commands/parse/msgs.rs new file mode 100644 index 00000000..8ebdf961 --- /dev/null +++ b/cli/src/commands/parse/msgs.rs @@ -0,0 +1,415 @@ +use anyhow::{anyhow, Context, Result}; +use cfb::CompoundFile; +use colored::Colorize; +use core::sync::atomic::Ordering; +use log::error; +use once_cell::sync::Lazy; +use regex::Regex; +use std::{ + fs::DirEntry, + io::Read, + sync::{atomic::AtomicUsize, Arc}, +}; + +use reinfer_client::{ + resources::{ + documents::{Document, RawEmail, RawEmailBody, RawEmailHeaders}, + email::AttachmentMetadata, + }, + Client, PropertyMap, Source, SourceIdentifier, TransformTag, +}; +use std::{fs::File, path::PathBuf}; +use structopt::StructOpt; + +use crate::{ + commands::ensure_uip_user_consents_to_ai_unit_charge, + progress::{Options as ProgressOptions, Progress}, +}; + +const UPLOAD_BATCH_SIZE: usize = 128; +const MSG_NAME_USER_PROPERTY_NAME: &str = "MSG NAME ID"; +const STREAM_PATH_ATTACHMENT_STORE_PREFIX: &str = "__attach_version1.0_#"; + +static CONTENT_TYPE_MIME_HEADER_RX: Lazy = + Lazy::new(|| Regex::new(r"Content-Type:((\s)+.+\n)*").unwrap()); +static STREAM_PATH_MESSAGE_BODY_PLAIN: Lazy = + Lazy::new(|| PathBuf::from("__substg1.0_1000001F")); +static STREAM_PATH_MESSAGE_HEADER: Lazy = + Lazy::new(|| PathBuf::from("__substg1.0_007d001F")); +static STREAM_PATH_ATTACHMENT_FILENAME: Lazy = + Lazy::new(|| PathBuf::from("__substg1.0_3707001F")); +static STREAM_PATH_ATTACHMENT_EXTENSION: Lazy = + Lazy::new(|| PathBuf::from("__substg1.0_3703001F")); +static STREAM_PATH_ATTACHMENT_DATA: Lazy = + Lazy::new(|| PathBuf::from("__substg1.0_37010102")); + +pub struct Statistics { + processed: AtomicUsize, + failed: AtomicUsize, + uploaded: AtomicUsize, +} + +impl Statistics { + fn new() -> Self { + Self { + processed: AtomicUsize::new(0), + failed: AtomicUsize::new(0), + uploaded: AtomicUsize::new(0), + } + } + + #[inline] + fn add_uploaded(&self, num_uploaded: usize) { + self.uploaded.fetch_add(num_uploaded, Ordering::SeqCst); + } + + #[inline] + fn increment_failed(&self) { + self.failed.fetch_add(1, Ordering::SeqCst); + } + + #[inline] + fn increment_processed(&self) { + self.processed.fetch_add(1, Ordering::SeqCst); + } + + #[inline] + fn num_uploaded(&self) -> usize { + self.uploaded.load(Ordering::SeqCst) + } + + #[inline] + fn num_failed(&self) -> usize { + self.failed.load(Ordering::SeqCst) + } + + #[inline] + fn num_processed(&self) -> usize { + self.processed.load(Ordering::SeqCst) + } +} + +#[derive(Debug, StructOpt)] +pub struct ParseMsgArgs { + #[structopt(short = "d", long = "dir", parse(from_os_str))] + /// Directory containing the msgs + directory: PathBuf, + + #[structopt(short = "s", long = "source")] + /// Source name or id + source: SourceIdentifier, + + #[structopt(long = "transform-tag")] + /// Transform tag to use. + transform_tag: TransformTag, + + #[structopt(short = "n", long = "no-charge")] + /// Whether to attempt to bypass billing (internal only) + no_charge: bool, + + #[structopt(short = "y", long = "yes")] + /// Consent to ai unit charge. Suppresses confirmation prompt. + yes: bool, +} + +fn read_stream(stream_path: PathBuf, compound_file: &mut CompoundFile) -> Result> { + let data = { + let mut stream = compound_file.open_stream(stream_path)?; + let mut buffer = Vec::new(); + stream.read_to_end(&mut buffer)?; + buffer + }; + + Ok(data) +} + +fn read_unicode_stream_to_string( + stream_path: PathBuf, + compound_file: &mut CompoundFile, +) -> Result { + if !compound_file.is_stream(stream_path.clone()) { + return Err(anyhow!( + "Could not find stream {}. Please check that you are using unicode msgs", + stream_path.to_string_lossy() + )); + } + + let data = read_stream(stream_path, compound_file)?; + + // Stream data is a UTF16 string encoded as Vec[u8] + let mut buffer: String = String::with_capacity(data.len() * 2); + let mut decoder = encoding_rs::UTF_16LE.new_decoder(); + let (coder_result, _, _) = decoder.decode_to_string(&data, &mut buffer, true); + if coder_result != encoding_rs::CoderResult::InputEmpty { + Err(anyhow!("Unexpected coder result")) + } else { + Ok(buffer) + } +} + +fn get_attachment_store_path(attachment_number: usize) -> PathBuf { + PathBuf::from(format!( + "{}{:08}", + STREAM_PATH_ATTACHMENT_STORE_PREFIX, attachment_number + )) +} + +fn read_attachment( + attachment_path: PathBuf, + compound_file: &mut CompoundFile, +) -> Result { + let mut attachment_name_path = attachment_path.clone(); + attachment_name_path.push(STREAM_PATH_ATTACHMENT_FILENAME.clone()); + + let mut content_type_path = attachment_path.clone(); + content_type_path.push(STREAM_PATH_ATTACHMENT_EXTENSION.clone()); + + let mut data_path = attachment_path.clone(); + data_path.push(STREAM_PATH_ATTACHMENT_DATA.clone()); + + let name = read_unicode_stream_to_string(attachment_name_path.clone(), compound_file)?; + let content_type = read_unicode_stream_to_string(content_type_path, compound_file)?; + let data = read_stream(data_path, compound_file)?; + + Ok(AttachmentMetadata { + name, + content_type, + size: data.len() as u64, + }) +} + +fn remove_content_type_header(headers_string: String) -> Result { + Ok(CONTENT_TYPE_MIME_HEADER_RX + .clone() + .replace(&headers_string, "") + .to_string()) +} + +fn read_msg_to_document(path: &PathBuf) -> Result { + if !path.is_file() { + return Err(anyhow!("No such file: {:?}", path)); + } + + let mut compound_file = cfb::open(path)?; + + // Headers + let headers_string = + read_unicode_stream_to_string(STREAM_PATH_MESSAGE_HEADER.clone(), &mut compound_file)?; + + // As the content type won't match the parsed value from the body in the msg + let headers_string_no_content_type = remove_content_type_header(headers_string)?; + + let plain_body_string = + read_unicode_stream_to_string(STREAM_PATH_MESSAGE_BODY_PLAIN.clone(), &mut compound_file)?; + + // Attachments + let mut attachment_number = 0; + let mut attachments = Vec::new(); + loop { + let attachment_path = get_attachment_store_path(attachment_number); + + if compound_file.is_storage(attachment_path.clone()) { + attachments.push(read_attachment(attachment_path, &mut compound_file)?); + } else { + break; + } + + attachment_number += 1; + } + + // User Properties + let mut user_properties = PropertyMap::new(); + user_properties.insert_string( + MSG_NAME_USER_PROPERTY_NAME.to_string(), + path.file_name() + .context("Could not get file name")? + .to_string_lossy() + .to_string(), + ); + + Ok(Document { + raw_email: RawEmail { + body: RawEmailBody::Plain(plain_body_string), + headers: RawEmailHeaders::Raw(headers_string_no_content_type), + attachments, + }, + user_properties, + comment_id: None, + }) +} + +fn upload_batch_of_documents( + client: &Client, + source: &Source, + documents: &Vec, + transform_tag: &TransformTag, + no_charge: bool, + statistics: &Arc, +) -> Result<()> { + client.sync_raw_emails( + &source.full_name(), + documents, + transform_tag, + false, + no_charge, + )?; + statistics.add_uploaded(documents.len()); + Ok(()) +} + +pub fn get_msgs_in_directory(directory: &PathBuf) -> Result> { + Ok(std::fs::read_dir(directory)? + .filter_map(|path| { + if let Ok(path) = path { + if !path.path().extension().is_some_and(|msg| msg == "msg") { + None + } else { + Some(path) + } + } else { + None + } + }) + .collect()) +} + +pub fn parse(client: &Client, args: &ParseMsgArgs) -> Result<()> { + let ParseMsgArgs { + directory, + source, + transform_tag, + no_charge, + yes, + } = args; + + if !no_charge && !yes { + ensure_uip_user_consents_to_ai_unit_charge(client.base_url())?; + } + + let msg_paths = get_msgs_in_directory(directory)?; + let statistics = Arc::new(Statistics::new()); + let _progress = get_progress_bar(msg_paths.len() as u64, &statistics); + let source = client.get_source(source.clone())?; + + let mut documents = Vec::new(); + let mut errors = Vec::new(); + for path in msg_paths { + match read_msg_to_document(&path.path()) { + Ok(document) => { + documents.push(document); + + if documents.len() >= UPLOAD_BATCH_SIZE { + upload_batch_of_documents( + client, + &source, + &documents, + transform_tag, + *no_charge, + &statistics, + )?; + documents.clear(); + } + statistics.increment_processed(); + } + Err(error) => { + errors.push(format!( + "Failed to process file {}: {}", + path.file_name().to_string_lossy(), + error + )); + statistics.increment_failed(); + statistics.increment_processed(); + } + } + } + + upload_batch_of_documents( + client, + &source, + &documents, + transform_tag, + *no_charge, + &statistics, + )?; + + for error in errors { + error!("{}", error); + } + + Ok(()) +} + +fn get_progress_bar(total_bytes: u64, statistics: &Arc) -> Progress { + Progress::new( + move |statistic| { + let num_processed = statistic.num_processed(); + let num_failed = statistic.num_failed(); + let num_uploaded = statistic.num_uploaded(); + ( + num_processed as u64, + format!( + "{} {} {} {} {} {}", + num_processed.to_string().bold(), + "processed".dimmed(), + num_failed.to_string().bold(), + "failed".dimmed(), + num_uploaded.to_string().bold(), + "uploaded".dimmed() + ), + ) + }, + statistics, + Some(total_bytes), + ProgressOptions { bytes_units: false }, + ) +} + +#[cfg(test)] +mod tests { + use super::*; + use pretty_assertions::assert_eq; + + #[test] + fn test_read_msg_to_document_non_unicode() { + let result = read_msg_to_document(&PathBuf::from("tests/samples/non-unicode.msg")); + + assert_eq!(result.expect_err("Expected Error Result").to_string(), "Could not find stream __substg1.0_007d001F. Please check that you are using unicode msgs"); + } + + #[test] + fn test_read_msg_to_document_unicode() { + let mut expected_user_properties = PropertyMap::new(); + expected_user_properties + .insert_string("MSG NAME ID".to_string(), "unicode.msg".to_string()); + + let expected_headers = "Received: from DB8PR02MB5883.eurprd02.prod.outlook.com (2603:10a6:10:116::17)\r\n by AM6PR02MB4215.eurprd02.prod.outlook.com with HTTPS; Wed, 25 Oct 2023\r\n 17:03:35 +0000\r\nAuthentication-Results: dkim=none (message not signed)\r\n header.d=none;dmarc=none action=none header.from=uipath.com;\r\nReceived: from AM9PR02MB6642.eurprd02.prod.outlook.com (2603:10a6:20b:2d2::18)\r\n by DB8PR02MB5883.eurprd02.prod.outlook.com (2603:10a6:10:116::17) with\r\n Microsoft SMTP Server (version=TLS1_2,\r\n cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.6907.33; Wed, 25 Oct\r\n 2023 17:03:23 +0000\r\nReceived: from AM9PR02MB6642.eurprd02.prod.outlook.com\r\n ([fe80::fb6e:4d16:d9ba:33ae]) by AM9PR02MB6642.eurprd02.prod.outlook.com\r\n ([fe80::fb6e:4d16:d9ba:33ae%6]) with mapi id 15.20.6933.019; Wed, 25 Oct 2023\r\n 17:03:23 +0000\r\nContent-Transfer-Encoding: binary\r\nFrom: Joe Prosser \r\nTo: Andra Buica \r\nSubject: Re: Testing the CLI!!\r\nThread-Topic: Testing the CLI!!\r\nThread-Index: AQHaB2N76HBc3H6u4Eeu9hAZAXjmE7BauHazgAAAIR2AAAJRsw==\r\nDate: Wed, 25 Oct 2023 17:03:22 +0000\r\nMessage-ID:\r\n \r\nReferences:\r\n \r\n \r\n \r\nIn-Reply-To:\r\n \r\nAccept-Language: en-GB, en-US\r\nContent-Language: en-GB\r\nX-MS-Has-Attach: yes\r\nX-MS-Exchange-Organization-SCL: 1\r\nX-MS-TNEF-Correlator:\r\n \r\nmsip_labels:\r\nMIME-Version: 1.0\r\nX-MS-Exchange-Organization-MessageDirectionality: Originating\r\nX-MS-Exchange-Organization-AuthSource: AM9PR02MB6642.eurprd02.prod.outlook.com\r\nX-MS-Exchange-Organization-AuthAs: Internal\r\nX-MS-Exchange-Organization-AuthMechanism: 04\r\nX-MS-Exchange-Organization-Network-Message-Id:\r\n 178907c2-2871-4221-a75a-08dbd57c4bf2\r\nX-MS-PublicTrafficType: Email\r\nX-MS-TrafficTypeDiagnostic:\r\n AM9PR02MB6642:EE_|DB8PR02MB5883:EE_|AM6PR02MB4215:EE_\r\nReturn-Path: joe.prosser@uipath.com\r\nX-MS-Exchange-Organization-ExpirationStartTime: 25 Oct 2023 17:03:23.2098\r\n (UTC)\r\nX-MS-Exchange-Organization-ExpirationStartTimeReason: OriginalSubmit\r\nX-MS-Exchange-Organization-ExpirationInterval: 1:00:00:00.0000000\r\nX-MS-Exchange-Organization-ExpirationIntervalReason: OriginalSubmit\r\nX-MS-Office365-Filtering-Correlation-Id: 178907c2-2871-4221-a75a-08dbd57c4bf2\r\nX-MS-Exchange-AtpMessageProperties: SA|SL\r\nX-Microsoft-Antispam: BCL:0;\r\nX-Forefront-Antispam-Report:\r\n CIP:255.255.255.255;CTRY:;LANG:en;SCL:1;SRV:;IPV:NLI;SFV:NSPM;H:AM9PR02MB6642.eurprd02.prod.outlook.com;PTR:;CAT:NONE;SFS:;DIR:INT;\r\nX-MS-Exchange-CrossTenant-OriginalArrivalTime: 25 Oct 2023 17:03:22.9433\r\n (UTC)\r\nX-MS-Exchange-CrossTenant-FromEntityHeader: Hosted\r\nX-MS-Exchange-CrossTenant-Id: d8353d2a-b153-4d17-8827-902c51f72357\r\nX-MS-Exchange-CrossTenant-AuthSource: AM9PR02MB6642.eurprd02.prod.outlook.com\r\nX-MS-Exchange-CrossTenant-AuthAs: Internal\r\nX-MS-Exchange-CrossTenant-Network-Message-Id: 178907c2-2871-4221-a75a-08dbd57c4bf2\r\nX-MS-Exchange-CrossTenant-MailboxType: HOSTED\r\nX-MS-Exchange-CrossTenant-UserPrincipalName: ++FDvUGfWQ/f3Ycjz3vJlna939o3V0zI8K8BPWCQ7aaQDhAJYe5XMf5Tr6B6e/5ucWri5/sUCnvrb1GDymXZmA==\r\nX-MS-Exchange-Transport-CrossTenantHeadersStamped: DB8PR02MB5883\r\nX-MS-Exchange-Transport-EndToEndLatency: 00:00:12.3444456\r\nX-MS-Exchange-Processed-By-BccFoldering: 15.20.6907.032\r\nX-Microsoft-Antispam-Mailbox-Delivery:\r\n\tucf:0;jmr:0;auth:0;dest:I;ENG:(910001)(944506478)(944626604)(920097)(425001)(930097)(140003)(1420103);\r\nX-Microsoft-Antispam-Message-Info:\r\n\thxMOcmClDYIwXiLHDHfW3mj4bIuPkLWLYD9z839jTLrsdaQbyDGWXui95ou9iKuUSUHzubxtiSwgj9OVTVInYaJ6SuN1imGY/n99Cn1vAx4VAh1xSYBUsm2bHu5atiwnMZTQaWg7BA03Yj6BSrOYqh49oLJrh9blZmtlkMs4uCi1Y5Y7YpEMpzLi7ye2fg+gPqELrNLHIKfziazCdrPNLKQ+9tlbb6UHbxeX1YVGY2ebnYXPwRilEP++cljm3hV8abvlgC1GF+nsuIS5XJwhkHmgTyetr7iZE3GWR9XC0NsB9w5bQgg82r75ozlGKWVKkuev1IlRjStpKaJbOoPBvo/6SCqSbWaQinyUEfPDXiJOYHW3D0xsf4uCpbFpb1D+TpQr+dFaCxBNdmFjBrd/SCHoyyl9QAO1nz0W5cUAwjSDKN7Pv7iKIUlx24nkDCeYeVqKhNcqulwIlEP6ewBBL2BTSplNPApIbliiHhh/Z6mes1xx3dfB5T4tIUv6wINSH3G2Ddec2fLzeTHknuyaOA9Wj8ks+JIjE+i5/CPidxfH4ACoggwbdnLwwnwniGcNoZKdyM+G4whogPe2oKXPggX9er/44bUOlKEcK5DsPFEZX2xKzsg7JwPPLcgO/lbP2iN/yFww6I6vri27P29a86np21iNSGOb51giFj5wgSq4iZLeRe64cEj4+i4K1KosNLBgDYTj/WGrioHl5Xe9ww==\r\n"; + + let expected_body = "Hey, \r\n\r\nWe should check that attachments work to ✅\r\n\r\nJoe \r\n________________________________\r\n\r\nFrom: Joe Prosser \r\nSent: 25 October 2023 17:53\r\nTo: Andra Buica \r\nSubject: Re: Testing the CLI!! \r\n \r\nHey,\r\n\r\nFingers cross 🤞 \r\n\r\nJoe\r\n________________________________\r\n\r\nFrom: Andra Buica \r\nSent: 25 October 2023 17:52\r\nTo: Joe Prosser \r\nSubject: Re: Testing the CLI!! \r\n \r\n\r\nHey, \r\n\r\n \r\n\r\nHopefully it will work! \r\n\r\n \r\n\r\nAndra\r\n\r\n \r\n\r\nFrom: Joe Prosser \r\nDate: Wednesday, 25 October 2023 at 17:52\r\nTo: Andra Buica \r\nSubject: Testing the CLI!!\r\n\r\nHey,\r\n\r\n \r\n\r\nDo you think it will work?\r\n\r\n \r\n\r\nJoe \r\n\r\n"; + + let expected_document = Document { + comment_id: None, + raw_email: RawEmail { + body: RawEmailBody::Plain(expected_body.to_string()), + headers: RawEmailHeaders::Raw(expected_headers.to_string()), + attachments: vec![ + AttachmentMetadata { + name: "hello.txt".to_string(), + size: 12, + content_type: ".txt".to_string(), + }, + AttachmentMetadata { + name: "world.pdf".to_string(), + size: 7302, + content_type: ".pdf".to_string(), + }, + ], + }, + user_properties: expected_user_properties, + }; + + let actual_document = read_msg_to_document(&PathBuf::from("tests/samples/unicode.msg")) + .expect("Failed to read msg"); + + assert_eq!(expected_document, actual_document); + } +} diff --git a/cli/src/main.rs b/cli/src/main.rs index 13830be2..45585bf1 100644 --- a/cli/src/main.rs +++ b/cli/src/main.rs @@ -19,7 +19,7 @@ use structopt::{clap::Shell as ClapShell, StructOpt}; use crate::{ args::{Args, Command, Shell}, - commands::{config as config_command, create, delete, get, update}, + commands::{config as config_command, create, delete, get, parse, update}, config::ReinferConfig, printer::Printer, }; @@ -71,6 +71,7 @@ fn run(args: Args) -> Result<()> { Command::Update { update_args } => { update::run(update_args, client_from_args(&args, &config)?, &printer) } + Command::Parse { parse_args } => parse::run(parse_args, client_from_args(&args, &config)?), } } diff --git a/cli/tests/samples/non-unicode.msg b/cli/tests/samples/non-unicode.msg new file mode 100644 index 00000000..782017a8 Binary files /dev/null and b/cli/tests/samples/non-unicode.msg differ diff --git a/cli/tests/samples/unicode.msg b/cli/tests/samples/unicode.msg new file mode 100644 index 00000000..ff28342c Binary files /dev/null and b/cli/tests/samples/unicode.msg differ