use alloc::string::{String, ToString};
use alloc::vec::Vec;
use core::fmt;
use super::nip01::{self, Coordinate};
use super::util::{take_and_parse_from_str, take_and_parse_optional_coordinate, take_string};
use crate::event::tag::{Tag, TagCodec, TagCodecError, impl_tag_codec_conversions};
use crate::types::url::{self, Url};
const EMOJI: &str = "emoji";
#[derive(Debug, PartialEq)]
pub enum Error {
Nip01(nip01::Error),
Url(url::ParseError),
Codec(TagCodecError),
InvalidShortcode,
}
impl core::error::Error for Error {}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Nip01(e) => e.fmt(f),
Self::Url(e) => e.fmt(f),
Self::Codec(e) => e.fmt(f),
Self::InvalidShortcode => f.write_str("Invalid shortcode"),
}
}
}
impl From<nip01::Error> for Error {
fn from(e: nip01::Error) -> Self {
Self::Nip01(e)
}
}
impl From<url::ParseError> for Error {
fn from(e: url::ParseError) -> Self {
Self::Url(e)
}
}
impl From<TagCodecError> for Error {
fn from(e: TagCodecError) -> Self {
Self::Codec(e)
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum Nip30Tag {
Emoji {
shortcode: String,
image_url: Url,
emoji_set: Option<Coordinate>,
},
}
impl TagCodec for Nip30Tag {
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() {
EMOJI => {
let (shortcode, image_url, emoji_set) = parse_emoji_tag(iter)?;
Ok(Self::Emoji {
shortcode,
image_url,
emoji_set,
})
}
_ => Err(TagCodecError::Unknown.into()),
}
}
fn to_tag(&self) -> Tag {
match self {
Self::Emoji {
shortcode,
image_url,
emoji_set,
} => {
let mut tag: Vec<String> = Vec::with_capacity(3 + emoji_set.is_some() as usize);
tag.push(String::from(EMOJI));
tag.push(shortcode.clone());
tag.push(image_url.to_string());
if let Some(emoji_set) = emoji_set {
tag.push(emoji_set.to_string());
}
Tag::new(tag)
}
}
}
}
impl_tag_codec_conversions!(Nip30Tag);
fn parse_emoji_tag<T, S>(mut iter: T) -> Result<(String, Url, Option<Coordinate>), Error>
where
T: Iterator<Item = S>,
S: AsRef<str>,
{
let shortcode: String = take_string(&mut iter, "shortcode")?;
if !is_valid_shortcode(&shortcode) {
return Err(Error::InvalidShortcode);
}
let image_url: Url = take_and_parse_from_str::<_, _, _, Error>(&mut iter, "image URL")?;
let emoji_set: Option<Coordinate> = take_and_parse_optional_coordinate(&mut iter)?;
Ok((shortcode, image_url, emoji_set))
}
fn is_valid_shortcode(shortcode: &str) -> bool {
!shortcode.is_empty()
&& shortcode
.chars()
.all(|c| c.is_ascii_alphanumeric() || c == '-' || c == '_')
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{Kind, PublicKey};
#[test]
fn test_nip30_emoji_tag() {
let image_url = Url::parse("https://example.com/emoji.png").unwrap();
let tag = vec!["emoji", "soapbox", "https://example.com/emoji.png"];
let parsed = Nip30Tag::parse(&tag).unwrap();
assert_eq!(
parsed,
Nip30Tag::Emoji {
shortcode: String::from("soapbox"),
image_url,
emoji_set: None,
}
);
assert_eq!(parsed.to_tag(), Tag::parse(tag).unwrap());
}
#[test]
fn test_nip30_emoji_tag_with_set() {
let image_url = Url::parse("https://example.com/emoji.png").unwrap();
let emoji_set = Coordinate::new(
Kind::EmojiSet,
PublicKey::from_hex("79c2cae114ea28a981e7559b4fe7854a473521a8d22a66bbab9fa248eb820ff6")
.unwrap(),
)
.identifier("blobcats");
let tag = vec![
"emoji",
"soapbox",
"https://example.com/emoji.png",
"30030:79c2cae114ea28a981e7559b4fe7854a473521a8d22a66bbab9fa248eb820ff6:blobcats",
];
let parsed = Nip30Tag::parse(&tag).unwrap();
assert_eq!(
parsed,
Nip30Tag::Emoji {
shortcode: String::from("soapbox"),
image_url,
emoji_set: Some(emoji_set),
}
);
assert_eq!(parsed.to_tag(), Tag::parse(tag).unwrap());
}
#[test]
fn test_nip30_invalid_shortcode() {
let tag = vec!["emoji", "soap box", "https://example.com/emoji.png"];
assert_eq!(Nip30Tag::parse(&tag).unwrap_err(), Error::InvalidShortcode);
}
}