use alloc::string::{String, ToString};
use alloc::vec;
use core::fmt;
use core::str::FromStr;
use super::util::{take_and_parse_from_str, take_event_id, take_public_key};
use crate::event::tag::{Tag, TagCodec, TagCodecError, impl_tag_codec_conversions};
use crate::{EventId, PublicKey, event, key};
#[derive(Debug, PartialEq)]
pub enum Error {
Keys(key::Error),
Event(event::Error),
Codec(TagCodecError),
UnknownReportType,
}
impl core::error::Error for Error {}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Self::Keys(e) => e.fmt(f),
Self::Event(e) => e.fmt(f),
Self::Codec(e) => e.fmt(f),
Self::UnknownReportType => f.write_str("Unknown report type"),
}
}
}
impl From<key::Error> for Error {
fn from(e: key::Error) -> Self {
Self::Keys(e)
}
}
impl From<event::Error> for Error {
fn from(e: event::Error) -> Self {
Self::Event(e)
}
}
impl From<TagCodecError> for Error {
fn from(e: TagCodecError) -> Self {
Self::Codec(e)
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum Report {
Nudity,
Malware,
Profanity,
Illegal,
Spam,
Impersonation,
Other,
}
impl fmt::Display for Report {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.write_str(self.as_str())
}
}
impl Report {
pub fn as_str(&self) -> &str {
match self {
Self::Nudity => "nudity",
Self::Malware => "malware",
Self::Profanity => "profanity",
Self::Illegal => "illegal",
Self::Spam => "spam",
Self::Impersonation => "impersonation",
Self::Other => "other",
}
}
}
impl FromStr for Report {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"nudity" => Ok(Self::Nudity),
"malware" => Ok(Self::Malware),
"profanity" => Ok(Self::Profanity),
"illegal" => Ok(Self::Illegal),
"spam" => Ok(Self::Spam),
"impersonation" => Ok(Self::Impersonation),
"other" => Ok(Self::Other),
_ => Err(Error::UnknownReportType),
}
}
}
#[derive(Debug, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub enum Nip56Tag {
Event {
id: EventId,
report: Report,
},
PublicKey {
public_key: PublicKey,
report: Report,
},
}
impl TagCodec for Nip56Tag {
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() {
"e" => {
let (id, report) = parse_e_tag(iter)?;
Ok(Self::Event { id, report })
}
"p" => {
let (public_key, report) = parse_p_tag(iter)?;
Ok(Self::PublicKey { public_key, report })
}
_ => Err(TagCodecError::Unknown.into()),
}
}
fn to_tag(&self) -> Tag {
match self {
Self::Event { id, report } => {
Tag::new(vec![String::from("e"), id.to_hex(), report.to_string()])
}
Self::PublicKey { public_key, report } => Tag::new(vec![
String::from("p"),
public_key.to_hex(),
report.to_string(),
]),
}
}
}
impl_tag_codec_conversions!(Nip56Tag);
fn parse_e_tag<T, S>(mut iter: T) -> Result<(EventId, Report), Error>
where
T: Iterator<Item = S>,
S: AsRef<str>,
{
let id: EventId = take_event_id::<_, _, Error>(&mut iter)?;
let report: Report = take_and_parse_from_str::<_, _, _, Error>(&mut iter, "report")?;
Ok((id, report))
}
fn parse_p_tag<T, S>(mut iter: T) -> Result<(PublicKey, Report), Error>
where
T: Iterator<Item = S>,
S: AsRef<str>,
{
let public_key: PublicKey = take_public_key::<_, _, Error>(&mut iter)?;
let report: Report = take_and_parse_from_str::<_, _, _, Error>(&mut iter, "report")?;
Ok((public_key, report))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_report_e_tag() {
let tag = vec![
"e",
"378f145897eea948952674269945e88612420db35791784abf0616b4fed56ef7",
"nudity",
];
let parsed = Nip56Tag::parse(&tag).unwrap();
assert_eq!(
parsed,
Nip56Tag::Event {
id: EventId::from_hex(
"378f145897eea948952674269945e88612420db35791784abf0616b4fed56ef7"
)
.unwrap(),
report: Report::Nudity,
}
);
assert_eq!(parsed.to_tag(), Tag::parse(tag).unwrap());
}
#[test]
fn test_report_p_tag() {
let tag = vec![
"p",
"13adc511de7e1cfcf1c6b7f6365fb5a03442d7bcacf565ea57fa7770912c023d",
"impersonation",
];
let parsed = Nip56Tag::parse(&tag).unwrap();
assert_eq!(
parsed,
Nip56Tag::PublicKey {
public_key: PublicKey::from_hex(
"13adc511de7e1cfcf1c6b7f6365fb5a03442d7bcacf565ea57fa7770912c023d"
)
.unwrap(),
report: Report::Impersonation,
}
);
assert_eq!(parsed.to_tag(), Tag::parse(tag).unwrap());
}
#[test]
fn test_missing_report() {
let tag = vec![
"p",
"13adc511de7e1cfcf1c6b7f6365fb5a03442d7bcacf565ea57fa7770912c023d",
];
assert!(matches!(
Nip56Tag::parse(&tag).unwrap_err(),
Error::Codec(TagCodecError::Missing("report"))
));
let tag = vec![
"e",
"378f145897eea948952674269945e88612420db35791784abf0616b4fed56ef7",
];
assert!(matches!(
Nip56Tag::parse(&tag).unwrap_err(),
Error::Codec(TagCodecError::Missing("report"))
));
}
#[test]
fn test_empty_report() {
let tag = vec![
"p",
"13adc511de7e1cfcf1c6b7f6365fb5a03442d7bcacf565ea57fa7770912c023d",
"",
];
assert!(matches!(
Nip56Tag::parse(&tag).unwrap_err(),
Error::UnknownReportType
));
let tag = vec![
"e",
"378f145897eea948952674269945e88612420db35791784abf0616b4fed56ef7",
"",
];
assert!(matches!(
Nip56Tag::parse(&tag).unwrap_err(),
Error::UnknownReportType
));
}
}