2022-09-28 23:23:41 +00:00
|
|
|
use clap::ValueEnum;
|
2022-03-28 04:27:07 +00:00
|
|
|
use std::{fmt::Display, path::PathBuf, str::FromStr};
|
|
|
|
use tracing::Level;
|
2022-09-24 19:18:49 +00:00
|
|
|
use url::Url;
|
2022-03-28 00:10:06 +00:00
|
|
|
|
|
|
|
#[derive(
|
|
|
|
Clone,
|
2022-03-28 04:27:07 +00:00
|
|
|
Copy,
|
2022-03-28 00:10:06 +00:00
|
|
|
Debug,
|
|
|
|
PartialEq,
|
|
|
|
Eq,
|
|
|
|
PartialOrd,
|
|
|
|
Ord,
|
|
|
|
Hash,
|
|
|
|
serde::Deserialize,
|
|
|
|
serde::Serialize,
|
2022-09-28 23:23:41 +00:00
|
|
|
ValueEnum,
|
2022-03-28 00:10:06 +00:00
|
|
|
)]
|
2022-03-28 04:27:07 +00:00
|
|
|
#[serde(rename_all = "snake_case")]
|
2022-03-28 00:10:06 +00:00
|
|
|
pub(crate) enum LogFormat {
|
|
|
|
Compact,
|
|
|
|
Json,
|
|
|
|
Normal,
|
|
|
|
Pretty,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub(crate) struct Targets {
|
|
|
|
pub(crate) targets: tracing_subscriber::filter::Targets,
|
|
|
|
}
|
|
|
|
|
2022-03-28 04:27:07 +00:00
|
|
|
/// Configuration for filesystem media storage
|
|
|
|
#[derive(Clone, Debug, serde::Deserialize, serde::Serialize, clap::Parser)]
|
|
|
|
#[serde(rename_all = "snake_case")]
|
|
|
|
pub(crate) struct Filesystem {
|
|
|
|
/// Path to store media
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-03-28 04:27:07 +00:00
|
|
|
pub(crate) path: PathBuf,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Configuration for object media storage
|
|
|
|
#[derive(Clone, Debug, serde::Deserialize, serde::Serialize, clap::Parser)]
|
|
|
|
#[serde(rename_all = "snake_case")]
|
|
|
|
pub(crate) struct ObjectStorage {
|
2022-09-24 19:18:49 +00:00
|
|
|
/// The base endpoint for the object storage
|
|
|
|
///
|
|
|
|
/// Examples:
|
|
|
|
/// - `http://localhost:9000`
|
|
|
|
/// - `https://s3.dualstack.eu-west-1.amazonaws.com`
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-09-24 19:18:49 +00:00
|
|
|
pub(crate) endpoint: Url,
|
|
|
|
|
|
|
|
/// Determines whether to use path style or virtualhost style for accessing objects
|
|
|
|
///
|
|
|
|
/// When this is true, objects will be fetched from {endpoint}/{bucket_name}/{object}
|
|
|
|
/// When false, objects will be fetched from {bucket_name}.{endpoint}/{object}
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-09-24 19:18:49 +00:00
|
|
|
pub(crate) use_path_style: bool,
|
|
|
|
|
2022-03-28 04:27:07 +00:00
|
|
|
/// The bucket in which to store media
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-03-28 04:27:07 +00:00
|
|
|
pub(crate) bucket_name: String,
|
|
|
|
|
|
|
|
/// The region the bucket is located in
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-09-24 19:18:49 +00:00
|
|
|
pub(crate) region: String,
|
2022-03-28 04:27:07 +00:00
|
|
|
|
|
|
|
/// The Access Key for the user accessing the bucket
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-03-28 04:27:07 +00:00
|
|
|
pub(crate) access_key: String,
|
|
|
|
|
|
|
|
/// The secret key for the user accessing the bucket
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(short, long)]
|
2022-03-28 04:27:07 +00:00
|
|
|
pub(crate) secret_key: String,
|
|
|
|
|
|
|
|
/// The session token for accessing the bucket
|
2022-09-28 23:23:41 +00:00
|
|
|
#[arg(long)]
|
2022-03-28 04:27:07 +00:00
|
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
|
|
pub(crate) session_token: Option<String>,
|
2023-07-11 18:01:58 +00:00
|
|
|
|
|
|
|
/// How long signatures for object storage requests are valid (in seconds)
|
|
|
|
///
|
|
|
|
/// This defaults to 15 seconds
|
|
|
|
#[arg(long)]
|
|
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
|
|
pub(crate) signature_duration: Option<u64>,
|
|
|
|
|
|
|
|
/// How long a client can wait on an object storage request before giving up (in seconds)
|
|
|
|
///
|
|
|
|
/// This defaults to 30 seconds
|
|
|
|
#[arg(long)]
|
|
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
|
|
pub(crate) client_timeout: Option<u64>,
|
2023-07-14 19:53:37 +00:00
|
|
|
|
|
|
|
/// Base endpoint at which object storage images are publicly accessible
|
|
|
|
#[arg(long)]
|
|
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
|
|
pub(crate) public_endpoint: Option<Url>,
|
2022-03-28 04:27:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug, serde::Deserialize, serde::Serialize)]
|
|
|
|
#[serde(rename_all = "snake_case")]
|
|
|
|
#[serde(tag = "type")]
|
2023-07-11 18:01:58 +00:00
|
|
|
// allow large enum variant - this is an instantiated-once config
|
|
|
|
#[allow(clippy::large_enum_variant)]
|
2022-03-28 04:27:07 +00:00
|
|
|
pub(crate) enum Store {
|
|
|
|
Filesystem(Filesystem),
|
|
|
|
|
|
|
|
ObjectStorage(ObjectStorage),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Filesystem> for Store {
|
|
|
|
fn from(f: Filesystem) -> Self {
|
|
|
|
Self::Filesystem(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<ObjectStorage> for Store {
|
|
|
|
fn from(o: ObjectStorage) -> Self {
|
|
|
|
Self::ObjectStorage(o)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-28 00:10:06 +00:00
|
|
|
impl FromStr for Targets {
|
|
|
|
type Err = <tracing_subscriber::filter::Targets as FromStr>::Err;
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
Ok(Targets {
|
|
|
|
targets: s.parse()?,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Display for Targets {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
let targets = self
|
|
|
|
.targets
|
|
|
|
.iter()
|
2023-01-29 17:57:59 +00:00
|
|
|
.map(|(path, level)| format!("{path}={level}"))
|
2022-03-28 00:10:06 +00:00
|
|
|
.collect::<Vec<_>>()
|
|
|
|
.join(",");
|
|
|
|
|
2022-03-28 04:27:07 +00:00
|
|
|
let max_level = [
|
|
|
|
Level::TRACE,
|
|
|
|
Level::DEBUG,
|
|
|
|
Level::INFO,
|
|
|
|
Level::WARN,
|
|
|
|
Level::ERROR,
|
|
|
|
]
|
|
|
|
.iter()
|
|
|
|
.fold(None, |found, level| {
|
|
|
|
if found.is_none()
|
|
|
|
&& self
|
|
|
|
.targets
|
|
|
|
.would_enable("not_a_real_target_so_nothing_can_conflict", level)
|
|
|
|
{
|
|
|
|
Some(level.to_string().to_lowercase())
|
|
|
|
} else {
|
|
|
|
found
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
if let Some(level) = max_level {
|
|
|
|
if !targets.is_empty() {
|
2023-01-29 17:57:59 +00:00
|
|
|
write!(f, "{level},{targets}")
|
2022-03-28 04:27:07 +00:00
|
|
|
} else {
|
2023-01-29 17:57:59 +00:00
|
|
|
write!(f, "{level}")
|
2022-03-28 04:27:07 +00:00
|
|
|
}
|
|
|
|
} else if !targets.is_empty() {
|
2023-01-29 17:57:59 +00:00
|
|
|
write!(f, "{targets}")
|
2022-03-28 04:27:07 +00:00
|
|
|
} else {
|
|
|
|
Ok(())
|
|
|
|
}
|
2022-03-28 00:10:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FromStr for LogFormat {
|
|
|
|
type Err = String;
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
for variant in Self::value_variants() {
|
|
|
|
if variant.to_possible_value().unwrap().matches(s, false) {
|
|
|
|
return Ok(*variant);
|
|
|
|
}
|
|
|
|
}
|
2023-01-29 17:57:59 +00:00
|
|
|
Err(format!("Invalid variant: {s}"))
|
2022-03-28 00:10:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Display for LogFormat {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
|
|
self.to_possible_value()
|
|
|
|
.expect("no values are skipped")
|
|
|
|
.get_name()
|
|
|
|
.fmt(f)
|
|
|
|
}
|
|
|
|
}
|
2022-03-28 04:27:07 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2022-09-24 19:18:49 +00:00
|
|
|
use super::Targets;
|
|
|
|
use crate::serde_str::Serde;
|
2022-03-28 04:27:07 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn builds_info_targets() {
|
|
|
|
let t: Serde<Targets> = "info".parse().unwrap();
|
|
|
|
|
|
|
|
println!("{:?}", t);
|
|
|
|
|
|
|
|
assert_eq!(t.to_string(), "info");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn builds_specific_targets() {
|
|
|
|
let t: Serde<Targets> = "pict_rs=info".parse().unwrap();
|
|
|
|
|
|
|
|
assert_eq!(t.to_string(), "pict_rs=info");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn builds_warn_and_specific_targets() {
|
|
|
|
let t: Serde<Targets> = "warn,pict_rs=info".parse().unwrap();
|
|
|
|
|
|
|
|
assert_eq!(t.to_string(), "warn,pict_rs=info");
|
|
|
|
}
|
|
|
|
}
|