use std::{
fmt::{self, Display, Formatter},
ops::Deref,
};
use serde::{Deserialize, Serialize};
use uuid::Uuid;
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Serialize, Clone)]
pub enum IdentifiedBy {
Uuid(Uuid),
Name(Name),
}
impl<'de> Deserialize<'de> for IdentifiedBy {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
#[derive(Deserialize)]
enum Stub {
Uuid(Uuid),
Name(Name),
}
#[derive(Deserialize)]
#[serde(untagged)]
enum Compat {
Uuid(Uuid), IdentifiedBy(Stub), }
if let Ok(id_compat) = <Compat as Deserialize>::deserialize(deserializer) {
match id_compat {
Compat::Uuid(uuid) => Ok(IdentifiedBy::Uuid(uuid)),
Compat::IdentifiedBy(stub) => match stub {
Stub::Uuid(uuid) => Ok(IdentifiedBy::Uuid(uuid)),
Stub::Name(name) => Ok(IdentifiedBy::Name(name)),
},
}
} else {
Err(serde::de::Error::custom(
"expected one of: a UUID, IdentifiedBy::Uuid or IdentifiedBy::Name",
))
}
}
}
impl From<Uuid> for IdentifiedBy {
fn from(value: Uuid) -> Self {
Self::Uuid(value)
}
}
impl From<Name> for IdentifiedBy {
fn from(value: Name) -> Self {
Self::Name(value)
}
}
impl From<String> for Name {
fn from(value: String) -> Self {
Self { inner: value }
}
}
impl Display for IdentifiedBy {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
match self {
IdentifiedBy::Uuid(uuid) => Display::fmt(uuid, f),
IdentifiedBy::Name(name) => Display::fmt(name, f),
}
}
}
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Serialize, Deserialize, Clone)]
#[serde(transparent)]
pub struct Name {
inner: String,
}
impl Name {
pub fn new_untrusted(name: &str) -> Self {
Self {
inner: name.to_owned(),
}
}
}
impl Deref for Name {
type Target = str;
fn deref(&self) -> &Self::Target {
&self.inner
}
}
const NAME_MAX_LEN: usize = 64;
pub struct InvalidNameError(pub String);
impl TryFrom<&str> for Name {
type Error = InvalidNameError;
fn try_from(value: &str) -> Result<Self, Self::Error> {
if !value.len() <= NAME_MAX_LEN {
return Err(InvalidNameError(format!(
"name length must be <= 64, got {}",
value.len()
)));
}
if value.is_empty() {
return Err(InvalidNameError(
"name must not be an empty string".to_string(),
));
}
if !value
.chars()
.all(|c| c.is_ascii_alphanumeric() || c == '_' || c == '-' || c == '/')
{
return Err(InvalidNameError(
"name must consist of only these allowed characters: A-Z a-z 0-9 _ - /".to_string(),
));
}
Ok(Name {
inner: value.to_string(),
})
}
}
impl Display for Name {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
Display::fmt(&self.inner, f)
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn name_validation() {
assert!(Name::try_from("alias").is_ok());
assert!(Name::try_from("foo/bar").is_ok());
assert!(Name::try_from(
"abcdefghijklmnopqrstuvqwxyzABCDEFGHIJKLMNOPQRSTUVQWXYZ0123456789-_/"
)
.is_ok());
assert!(Name::try_from(" leading-ws").is_err());
assert!(Name::try_from("trailing-ws ").is_err());
assert!(Name::try_from("punctuation%").is_err());
}
}