Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support reading args from a config file #165

Merged
merged 11 commits into from
Aug 2, 2021
38 changes: 35 additions & 3 deletions src/cli.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,19 @@ use std::convert::TryFrom;
use std::env;
use std::ffi::OsString;
use std::fmt;
use std::fs;
use std::io::Write;
use std::mem;
use std::path::PathBuf;
use std::str::FromStr;
use std::time::Duration;

use reqwest::{Method, Url};
use serde::{Deserialize, Serialize};
use structopt::clap::{self, arg_enum, AppSettings, Error, ErrorKind, Result};
use structopt::StructOpt;

use crate::{buffer::Buffer, request_items::RequestItem};
use crate::{buffer::Buffer, request_items::RequestItem, utils::config_dir};

// Some doc comments were copy-pasted from HTTPie

Expand All @@ -33,6 +35,7 @@ use crate::{buffer::Buffer, request_items::RequestItem};
AppSettings::DeriveDisplayOrder,
AppSettings::UnifiedHelpMessage,
AppSettings::ColoredHelp,
AppSettings::AllArgsOverrideSelf,
blyxxyz marked this conversation as resolved.
Show resolved Hide resolved
],
)]
pub struct Cli {
Expand Down Expand Up @@ -297,8 +300,17 @@ const NEGATION_FLAGS: &[&str] = &[
];

impl Cli {
pub fn from_args() -> Self {
Cli::from_iter(std::env::args_os())
pub fn parse() -> Self {
if let Some(default_args) = default_cli_args() {
let mut args = std::env::args_os();
Cli::from_iter(
std::iter::once(args.next().unwrap_or_else(|| "xh".into()))
.chain(default_args.iter().map(Into::into))
ducaale marked this conversation as resolved.
Show resolved Hide resolved
.chain(args),
)
} else {
Cli::from_iter(std::env::args_os())
}
}

pub fn from_iter<I>(iter: I) -> Self
Expand Down Expand Up @@ -481,6 +493,26 @@ impl Cli {
}
}

#[derive(Serialize, Deserialize)]
struct Config {
default_options: Vec<String>,
}

fn default_cli_args() -> Option<Vec<String>> {
let content = fs::read_to_string(config_dir()?.join("config.json")).ok()?;
ducaale marked this conversation as resolved.
Show resolved Hide resolved
match serde_json::from_str::<Config>(&content) {
Ok(config) => Some(config.default_options),
Err(error) => {
eprintln!(
"\n{}: warning: Unable to read config file: {}\n",
env!("CARGO_PKG_NAME"),
error
);
None
}
}
}

fn parse_method(method: &str) -> Option<Method> {
// This unfortunately matches "localhost"
if !method.is_empty() && method.chars().all(|c| c.is_ascii_alphabetic()) {
Expand Down
4 changes: 2 additions & 2 deletions src/main.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
#![allow(clippy::bool_assert_comparison)]
#![allow(clippy::needless_borrow)]
blyxxyz marked this conversation as resolved.
Show resolved Hide resolved
mod auth;
mod buffer;
mod cli;
Expand Down Expand Up @@ -43,7 +43,7 @@ fn get_user_agent() -> &'static str {

#[exit_status::main]
fn main() -> Result<i32> {
let args = Cli::from_args();
let args = Cli::parse();

if args.curl {
to_curl::print_curl_translation(args)?;
Expand Down
59 changes: 58 additions & 1 deletion src/printer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ use crate::{
buffer::Buffer,
cli::{Pretty, Theme},
formatting::{get_json_formatter, Highlighter},
utils::{copy_largebuf, get_content_type, test_mode, valid_json, ContentType, BUFFER_SIZE},
utils::{copy_largebuf, test_mode, BUFFER_SIZE},
};

const BINARY_SUPPRESSOR: &str = concat!(
Expand Down Expand Up @@ -387,6 +387,63 @@ impl Printer {
}
}

pub enum ContentType {
blyxxyz marked this conversation as resolved.
Show resolved Hide resolved
Json,
Html,
Xml,
JavaScript,
Css,
Text,
UrlencodedForm,
Multipart,
Unknown,
}

impl ContentType {
pub fn is_text(&self) -> bool {
!matches!(
self,
ContentType::Unknown | ContentType::UrlencodedForm | ContentType::Multipart
)
}
}

pub fn get_content_type(headers: &HeaderMap) -> ContentType {
headers
.get(CONTENT_TYPE)
.and_then(|value| value.to_str().ok())
.and_then(|content_type| {
if content_type.contains("json") {
Some(ContentType::Json)
} else if content_type.contains("html") {
Some(ContentType::Html)
} else if content_type.contains("xml") {
Some(ContentType::Xml)
} else if content_type.contains("multipart") {
Some(ContentType::Multipart)
} else if content_type.contains("x-www-form-urlencoded") {
Some(ContentType::UrlencodedForm)
} else if content_type.contains("javascript") {
Some(ContentType::JavaScript)
} else if content_type.contains("css") {
Some(ContentType::Css)
} else if content_type.contains("text") {
// We later check if this one's JSON
// HTTPie checks for "json", "javascript" and "text" in one place:
// https://github.com/httpie/httpie/blob/a32ad344dd/httpie/output/formatters/json.py#L14
// We have it more spread out but it behaves more or less the same
Some(ContentType::Text)
} else {
None
}
})
.unwrap_or(ContentType::Unknown)
}

pub fn valid_json(text: &str) -> bool {
serde_json::from_str::<serde::de::IgnoredAny>(text).is_ok()
}

/// Decode a streaming response in a way that matches `.text()`.
///
/// Note that in practice this seems to behave like String::from_utf8_lossy(),
Expand Down
62 changes: 6 additions & 56 deletions src/utils.rs
Original file line number Diff line number Diff line change
@@ -1,10 +1,9 @@
use std::{
env::var_os,
io::{self, Write},
path::PathBuf,
};

use reqwest::header::{HeaderMap, CONTENT_TYPE};

/// Whether to make some things more deterministic for the benefit of tests
pub fn test_mode() -> bool {
// In integration tests the binary isn't compiled with cfg(test), so we
Expand All @@ -21,59 +20,14 @@ pub fn test_default_color() -> bool {
var_os("XH_TEST_MODE_COLOR").is_some()
}

pub enum ContentType {
Json,
Html,
Xml,
JavaScript,
Css,
Text,
UrlencodedForm,
Multipart,
Unknown,
}

impl ContentType {
pub fn is_text(&self) -> bool {
!matches!(
self,
ContentType::Unknown | ContentType::UrlencodedForm | ContentType::Multipart
)
pub fn config_dir() -> Option<PathBuf> {
if let Some(dir) = std::env::var_os("XH_CONFIG_DIR") {
Some(dir.into())
} else {
dirs::config_dir().map(|dir| dir.join("xh"))
}
}

pub fn get_content_type(headers: &HeaderMap) -> ContentType {
headers
.get(CONTENT_TYPE)
.and_then(|value| value.to_str().ok())
.and_then(|content_type| {
if content_type.contains("json") {
Some(ContentType::Json)
} else if content_type.contains("html") {
Some(ContentType::Html)
} else if content_type.contains("xml") {
Some(ContentType::Xml)
} else if content_type.contains("multipart") {
Some(ContentType::Multipart)
} else if content_type.contains("x-www-form-urlencoded") {
Some(ContentType::UrlencodedForm)
} else if content_type.contains("javascript") {
Some(ContentType::JavaScript)
} else if content_type.contains("css") {
Some(ContentType::Css)
} else if content_type.contains("text") {
// We later check if this one's JSON
// HTTPie checks for "json", "javascript" and "text" in one place:
// https://github.com/httpie/httpie/blob/a32ad344dd/httpie/output/formatters/json.py#L14
// We have it more spread out but it behaves more or less the same
Some(ContentType::Text)
} else {
None
}
})
.unwrap_or(ContentType::Unknown)
}

// https://stackoverflow.com/a/45145246/5915221
#[macro_export]
macro_rules! vec_of_strings {
Expand Down Expand Up @@ -121,7 +75,3 @@ pub fn copy_largebuf(reader: &mut impl io::Read, writer: &mut impl Write) -> io:
}
}
}

pub fn valid_json(text: &str) -> bool {
serde_json::from_str::<serde::de::IgnoredAny>(text).is_ok()
}