2
0
Fork 0
mirror of https://git.asonix.dog/asonix/pict-rs synced 2025-01-18 15:35:50 +00:00
pict-rs/src/config/primitives.rs

321 lines
7.1 KiB
Rust
Raw Normal View History

2022-03-28 04:27:07 +00:00
use crate::magick::ValidInputType;
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,
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 ImageFormat {
2023-06-21 22:05:35 +00:00
Avif,
2022-03-28 00:10:06 +00:00
Jpeg,
2023-06-21 22:05:35 +00:00
Jxl,
2022-03-28 00:10:06 +00:00
Png,
2023-06-21 22:05:35 +00:00
Webp,
2022-03-28 00:10:06 +00:00
}
#[derive(
Clone,
Copy,
Debug,
PartialEq,
Eq,
PartialOrd,
Ord,
Hash,
serde::Deserialize,
serde::Serialize,
ValueEnum,
)]
#[serde(rename_all = "snake_case")]
pub(crate) enum VideoCodec {
H264,
H265,
Av1,
Vp8,
Vp9,
}
#[derive(
Clone,
Copy,
Debug,
PartialEq,
Eq,
PartialOrd,
Ord,
Hash,
serde::Deserialize,
serde::Serialize,
ValueEnum,
)]
#[serde(rename_all = "snake_case")]
pub(crate) enum AudioCodec {
Aac,
Opus,
Vorbis,
}
2022-03-28 00:10:06 +00:00
#[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>,
}
#[derive(Clone, Debug, serde::Deserialize, serde::Serialize)]
#[serde(rename_all = "snake_case")]
#[serde(tag = "type")]
pub(crate) enum Store {
Filesystem(Filesystem),
ObjectStorage(ObjectStorage),
}
impl ImageFormat {
pub(crate) fn as_hint(self) -> ValidInputType {
ValidInputType::from_format(self)
2022-03-28 04:27:07 +00:00
}
pub(crate) fn as_magick_format(self) -> &'static str {
match self {
2023-06-21 22:05:35 +00:00
Self::Avif => "AVIF",
2022-03-28 04:27:07 +00:00
Self::Jpeg => "JPEG",
2023-06-21 22:05:35 +00:00
Self::Jxl => "JXL",
2022-03-28 04:27:07 +00:00
Self::Png => "PNG",
Self::Webp => "WEBP",
}
}
pub(crate) fn as_ext(self) -> &'static str {
match self {
2023-06-21 22:05:35 +00:00
Self::Avif => ".avif",
Self::Jpeg => ".jpeg",
2023-06-21 22:05:35 +00:00
Self::Jxl => ".jxl",
Self::Png => ".png",
Self::Webp => ".webp",
}
}
2022-03-28 04:27:07 +00:00
}
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 ImageFormat {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s.to_lowercase().as_str() {
2023-06-21 22:05:35 +00:00
"avif" => Ok(Self::Avif),
"jpeg" | "jpg" => Ok(Self::Jpeg),
2023-06-21 22:05:35 +00:00
"jxl" => Ok(Self::Jxl),
"png" => Ok(Self::Png),
"webp" => Ok(Self::Webp),
2023-01-29 17:57:59 +00:00
other => Err(format!("Invalid variant: {other}")),
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 ImageFormat {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
self.to_possible_value()
.expect("no values are skipped")
.get_name()
.fmt(f)
}
}
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");
}
}