use reqwest::RequestBuilder;
use crate::error::{Error, Result};
pub trait Auth: Sync + Send {
fn wrap(&self, builder: RequestBuilder) -> Result<RequestBuilder>;
fn can_reload(&self) -> bool {
false
}
fn username(&self) -> String;
}
#[derive(Clone)]
pub struct BasicAuth {
username: String,
password: SensitiveString,
}
impl BasicAuth {
pub fn new(username: impl ToString, password: impl ToString) -> Self {
Self {
username: username.to_string(),
password: SensitiveString(password.to_string()),
}
}
}
impl Auth for BasicAuth {
fn wrap(&self, builder: RequestBuilder) -> Result<RequestBuilder> {
Ok(builder.basic_auth(&self.username, Some(self.password.inner())))
}
fn username(&self) -> String {
self.username.clone()
}
}
#[derive(Clone)]
pub struct AccessTokenAuth {
token: SensitiveString,
}
impl AccessTokenAuth {
pub fn new(token: impl ToString) -> Self {
Self {
token: SensitiveString::from(token.to_string()),
}
}
}
impl Auth for AccessTokenAuth {
fn wrap(&self, builder: RequestBuilder) -> Result<RequestBuilder> {
Ok(builder.bearer_auth(self.token.inner()))
}
fn username(&self) -> String {
"token".to_string()
}
}
#[derive(Clone)]
pub struct AccessTokenFileAuth {
token_file: String,
}
impl AccessTokenFileAuth {
pub fn new(token_file: impl ToString) -> Self {
let token_file = token_file.to_string();
Self { token_file }
}
}
impl Auth for AccessTokenFileAuth {
fn wrap(&self, builder: RequestBuilder) -> Result<RequestBuilder> {
let token = std::fs::read_to_string(&self.token_file).map_err(|e| {
Error::IO(format!(
"cannot read access token from file {}: {}",
self.token_file, e
))
})?;
Ok(builder.bearer_auth(token.trim()))
}
fn can_reload(&self) -> bool {
true
}
fn username(&self) -> String {
"token".to_string()
}
}
#[derive(::serde::Deserialize, ::serde::Serialize)]
#[serde(from = "String", into = "String")]
#[derive(Clone, Default, PartialEq, Eq)]
pub struct SensitiveString(String);
impl From<String> for SensitiveString {
fn from(value: String) -> Self {
Self(value)
}
}
impl From<&str> for SensitiveString {
fn from(value: &str) -> Self {
Self(value.to_string())
}
}
impl From<SensitiveString> for String {
fn from(value: SensitiveString) -> Self {
value.0
}
}
impl std::fmt::Display for SensitiveString {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "**REDACTED**")
}
}
impl std::fmt::Debug for SensitiveString {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "\"**REDACTED**\"")
}
}
impl SensitiveString {
#[must_use]
pub fn inner(&self) -> &str {
self.0.as_str()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn serialization() {
let json_value = "\"foo\"";
let value: SensitiveString = serde_json::from_str(json_value).unwrap();
let result: String = serde_json::to_string(&value).unwrap();
assert_eq!(result, json_value);
}
#[test]
fn hide_content() {
let value = SensitiveString("hello world".to_string());
let display = format!("{value}");
assert_eq!(display, "**REDACTED**");
let debug = format!("{value:?}");
assert_eq!(debug, "\"**REDACTED**\"");
}
}