use alloc::string::{String, ToString};
use alloc::vec::Vec;
use core::fmt;
use core::str::FromStr;
use crate::event::tag::{Tag, TagCodec, TagCodecError, impl_tag_codec_conversions};
use crate::nips::util::take_relay_url;
use crate::types::url;
use crate::{Event, RelayUrl};
const RELAY_METADATA: &str = "r";
#[derive(Debug, PartialEq, Eq)]
pub enum Error {
Url(url::Error),
Codec(TagCodecError),
InvalidRelayMetadata,
}
impl core::error::Error for Error {}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Url(e) => e.fmt(f),
Self::Codec(e) => e.fmt(f),
Self::InvalidRelayMetadata => f.write_str("Invalid relay metadata"),
}
}
}
impl From<url::Error> for Error {
fn from(e: url::Error) -> Self {
Self::Url(e)
}
}
impl From<TagCodecError> for Error {
fn from(e: TagCodecError) -> Self {
Self::Codec(e)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum RelayMetadata {
Read,
Write,
}
impl RelayMetadata {
pub fn as_str(&self) -> &str {
match self {
Self::Read => "read",
Self::Write => "write",
}
}
#[inline]
pub fn is_read(&self) -> bool {
matches!(self, Self::Read)
}
#[inline]
pub fn is_write(&self) -> bool {
matches!(self, Self::Write)
}
}
impl fmt::Display for RelayMetadata {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(self.as_str())
}
}
impl FromStr for RelayMetadata {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"read" => Ok(Self::Read),
"write" => Ok(Self::Write),
_ => Err(Error::InvalidRelayMetadata),
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum Nip65Tag {
RelayMetadata {
relay_url: RelayUrl,
metadata: Option<RelayMetadata>,
},
}
impl TagCodec for Nip65Tag {
type Error = Error;
fn parse<I, S>(tag: I) -> Result<Self, Self::Error>
where
I: IntoIterator<Item = S>,
S: AsRef<str>,
{
let mut iter = tag.into_iter();
let kind: S = iter.next().ok_or(TagCodecError::missing_tag_kind())?;
match kind.as_ref() {
RELAY_METADATA => {
let relay_url: RelayUrl = take_relay_url::<_, _, Error>(&mut iter)?;
let metadata: Option<RelayMetadata> = match iter.next() {
Some(metadata) => Some(RelayMetadata::from_str(metadata.as_ref())?),
None => None,
};
Ok(Self::RelayMetadata {
relay_url,
metadata,
})
}
_ => Err(TagCodecError::Unknown.into()),
}
}
fn to_tag(&self) -> Tag {
match self {
Self::RelayMetadata {
relay_url,
metadata,
} => {
let mut tag: Vec<String> = Vec::with_capacity(2 + metadata.is_some() as usize);
tag.push(String::from(RELAY_METADATA));
tag.push(relay_url.to_string());
if let Some(metadata) = metadata {
tag.push(metadata.to_string());
}
Tag::new(tag)
}
}
}
}
impl_tag_codec_conversions!(Nip65Tag);
#[inline]
pub fn extract_relay_list(
event: &Event,
) -> impl Iterator<Item = (RelayUrl, Option<RelayMetadata>)> + '_ {
event
.tags
.iter()
.filter_map(|tag| match Nip65Tag::try_from(tag) {
Ok(Nip65Tag::RelayMetadata {
relay_url,
metadata,
}) => Some((relay_url, metadata)),
_ => None,
})
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_standardized_relay_metadata_tag() {
let relay_url = RelayUrl::parse("wss://relay.damus.io").unwrap();
let tag = vec!["r".to_string(), relay_url.to_string(), String::from("read")];
let parsed = Nip65Tag::parse(&tag).unwrap();
assert_eq!(
parsed,
Nip65Tag::RelayMetadata {
relay_url: relay_url.clone(),
metadata: Some(RelayMetadata::Read),
}
);
assert_eq!(parsed.to_tag(), Tag::parse(tag).unwrap());
}
#[test]
fn test_standardized_relay_metadata_tag_without_marker() {
let relay_url = RelayUrl::parse("wss://relay.damus.io").unwrap();
let tag = vec!["r".to_string(), relay_url.to_string()];
let parsed = Nip65Tag::parse(&tag).unwrap();
assert_eq!(
parsed,
Nip65Tag::RelayMetadata {
relay_url: relay_url.clone(),
metadata: None,
}
);
assert_eq!(parsed.to_tag(), Tag::parse(tag).unwrap());
}
}