use serde::{
Deserialize, Deserializer, Serialize, Serializer,
de::{self, Visitor},
};
use super::{AgentId, IdentityError as Err, UserId};
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct Address {
user: UserId,
agent: AgentId,
}
impl TryFrom<&str> for Address {
type Error = Err;
fn try_from(value: &str) -> Result<Self, Err> {
Address::parse(value)
}
}
impl TryFrom<String> for Address {
type Error = Err;
fn try_from(value: String) -> Result<Self, Err> {
Address::parse(value)
}
}
impl Address {
pub fn new(user: UserId, agent: AgentId) -> Self {
Address { user, agent }
}
pub fn parse(input: impl AsRef<str>) -> Result<Self, Err> {
let input = input.as_ref();
if input.trim() != input {
return Err(Err::InvalidAddress);
}
let (user_str, agent_str) = input.rsplit_once('@').ok_or(Err::InvalidAddress)?;
let agent = AgentId::parse(agent_str)?;
let user = UserId::parse(user_str)?;
Ok(Address { user, agent })
}
pub fn canonical(&self) -> String {
format!("{}@{}", self.user.canonical(), self.agent.canonical())
}
pub fn user(&self) -> &UserId {
&self.user
}
pub fn agent(&self) -> &AgentId {
&self.agent
}
}
impl std::fmt::Display for Address {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.canonical())
}
}
impl Serialize for Address {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_str(&self.canonical())
}
}
impl<'de> Deserialize<'de> for Address {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
struct AddressVisitor;
impl<'de> Visitor<'de> for AddressVisitor {
type Value = Address;
fn expecting(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
f.write_str("a canonical Relay Mail address string")
}
fn visit_str<E>(self, value: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
Address::parse(value)
.map_err(|e| de::Error::custom(format!("invalid address `{}`: {}", value, e)))
}
fn visit_string<E>(self, value: String) -> Result<Self::Value, E>
where
E: de::Error,
{
self.visit_str(&value)
}
}
deserializer.deserialize_str(AddressVisitor)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn parses_and_canonicalizes() {
assert_eq!(
Address::parse("#Bob@example.org").unwrap().canonical(),
"#bob@example.org"
);
assert_eq!(
Address::parse("Bob@example.org").unwrap().canonical(),
"#bob@example.org"
);
assert_eq!(
Address::parse("wORk#Bob@example.ORG").unwrap().canonical(),
"work#bob@example.org"
);
}
#[test]
#[should_panic(expected = "called `Result::unwrap()` on an `Err` value: InvalidAddress")]
fn rejects_empty() {
let _ = Address::parse("").unwrap();
}
#[test]
#[should_panic(expected = "called `Result::unwrap()` on an `Err` value: InvalidAddress")]
fn rejects_invalid_chars() {
let _ = Address::parse("Invalid Address!").unwrap();
}
#[test]
#[should_panic(expected = "called `Result::unwrap()` on an `Err` value: InvalidAddress")]
fn reject_untrimmed() {
let _ = Address::parse(" trim_me ").unwrap();
}
#[test]
#[should_panic(expected = "called `Result::unwrap()` on an `Err` value: InvalidAddress")]
fn rejects_non_ascii() {
let _ = Address::parse("调试输出").unwrap();
}
#[test]
#[should_panic(expected = "called `Result::unwrap()` on an `Err` value: InvalidIdentityString")]
fn rejects_homoglyphs() {
let _ = Address::parse("wоrk#аlice@us.exaмple.org").unwrap(); }
}