use crate::{error::Error, warning};
use serde::{Deserialize, Serialize, de, ser};
use std::{fmt, str::FromStr};
#[derive(Clone, Debug, Eq, Hash, PartialEq, PartialOrd, Ord)]
#[non_exhaustive]
pub enum Informational {
Notice,
Unmaintained,
Unsound,
Other(String),
}
impl Informational {
pub fn as_str(&self) -> &str {
match self {
Self::Notice => "notice",
Self::Unmaintained => "unmaintained",
Self::Unsound => "unsound",
Self::Other(other) => other,
}
}
pub fn is_notice(&self) -> bool {
*self == Self::Notice
}
pub fn is_unmaintained(&self) -> bool {
*self == Self::Unmaintained
}
pub fn is_unsound(&self) -> bool {
*self == Self::Unsound
}
pub fn is_other(&self) -> bool {
matches!(self, Self::Other(_))
}
pub fn warning_kind(&self) -> Option<warning::WarningKind> {
match self {
Self::Notice => Some(warning::WarningKind::Notice),
Self::Unmaintained => Some(warning::WarningKind::Unmaintained),
Self::Unsound => Some(warning::WarningKind::Unsound),
Self::Other(_) => None,
}
}
}
impl fmt::Display for Informational {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.as_str())
}
}
impl FromStr for Informational {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Error> {
Ok(match s {
"notice" => Self::Notice,
"unmaintained" => Self::Unmaintained,
"unsound" => Self::Unsound,
other => Self::Other(other.to_owned()),
})
}
}
impl<'de> Deserialize<'de> for Informational {
fn deserialize<D: de::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
use de::Error;
let string = String::deserialize(deserializer)?;
string.parse().map_err(D::Error::custom)
}
}
impl Serialize for Informational {
fn serialize<S: ser::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
self.to_string().serialize(serializer)
}
}
#[cfg(test)]
mod tests {
use super::Informational;
#[test]
fn parse_notice() {
let notice = "notice".parse::<Informational>().unwrap();
assert_eq!(Informational::Notice, notice);
assert_eq!("notice", notice.as_str());
}
#[test]
fn parse_unmaintained() {
let unmaintained = "unmaintained".parse::<Informational>().unwrap();
assert_eq!(Informational::Unmaintained, unmaintained);
assert_eq!("unmaintained", unmaintained.as_str());
}
#[test]
fn parse_other() {
let other = "foobar".parse::<Informational>().unwrap();
assert_eq!(Informational::Other("foobar".to_owned()), other);
assert_eq!("foobar", other.as_str());
}
}