use std::borrow::Borrow;
use std::fmt;
use std::ops::Deref;
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Eq)]
#[non_exhaustive]
pub enum KeywordError {
Empty,
InvalidChar(char),
ReservedPrefix,
}
impl fmt::Display for KeywordError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
KeywordError::Empty => f.write_str("keyword must not be empty"),
KeywordError::InvalidChar(c) => write!(f, "keyword contains invalid character {:?}", c),
KeywordError::ReservedPrefix => f.write_str(
"custom keywords must not begin with '$' (reserved for system keywords)",
),
}
}
}
impl std::error::Error for KeywordError {}
const SYSTEM_KEYWORDS: &[&str] = &[
"$draft",
"$seen",
"$flagged",
"$answered",
"$forwarded",
"$phishing",
"$junk",
"$notjunk",
];
#[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)]
#[serde(transparent)]
pub struct Keyword(String);
impl Keyword {
pub fn new(s: impl Into<String>) -> Self {
Keyword(s.into())
}
pub fn try_new(s: impl Into<String>) -> Result<Self, KeywordError> {
let s: String = s.into();
if s.is_empty() {
return Err(KeywordError::Empty);
}
for ch in s.chars() {
let b = ch as u32;
if !(0x21..=0x7E).contains(&b) {
return Err(KeywordError::InvalidChar(ch));
}
}
if s.starts_with('$') {
let lower = s.to_ascii_lowercase();
if !SYSTEM_KEYWORDS.contains(&lower.as_str()) {
return Err(KeywordError::ReservedPrefix);
}
}
Ok(Keyword(s))
}
}
impl From<&str> for Keyword {
fn from(s: &str) -> Self {
Keyword(s.to_owned())
}
}
impl From<String> for Keyword {
fn from(s: String) -> Self {
Keyword(s)
}
}
impl AsRef<str> for Keyword {
fn as_ref(&self) -> &str {
&self.0
}
}
impl Borrow<str> for Keyword {
fn borrow(&self) -> &str {
&self.0
}
}
impl Deref for Keyword {
type Target = str;
fn deref(&self) -> &str {
&self.0
}
}
impl fmt::Display for Keyword {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.write_str(&self.0)
}
}
pub const DRAFT: &str = "$draft";
pub const SEEN: &str = "$seen";
pub const FLAGGED: &str = "$flagged";
pub const ANSWERED: &str = "$answered";
pub const FORWARDED: &str = "$forwarded";
pub const PHISHING: &str = "$phishing";
pub const JUNK: &str = "$junk";
pub const NOT_JUNK: &str = "$notjunk";
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn keyword_wire_strings() {
assert_eq!(DRAFT, "$draft");
assert_eq!(SEEN, "$seen");
assert_eq!(FLAGGED, "$flagged");
assert_eq!(ANSWERED, "$answered");
assert_eq!(FORWARDED, "$forwarded");
assert_eq!(PHISHING, "$phishing");
assert_eq!(JUNK, "$junk");
assert_eq!(NOT_JUNK, "$notjunk");
}
#[test]
fn keyword_usable_as_hashmap_key() {
let mut keywords = std::collections::HashMap::new();
keywords.insert(Keyword::from(SEEN), true);
keywords.insert(Keyword::from(FLAGGED), true);
assert!(keywords.contains_key(SEEN));
assert!(keywords.contains_key(FLAGGED));
assert!(!keywords.contains_key(DRAFT));
}
#[test]
fn keyword_roundtrips_as_json_string() {
let kw = Keyword::from(SEEN);
let json = serde_json::to_string(&kw).expect("serialize");
assert_eq!(json, "\"$seen\"");
let back: Keyword = serde_json::from_str(&json).expect("deserialize");
assert_eq!(back, kw);
}
#[test]
fn keyword_construction() {
let a = Keyword::new("$seen");
let b = Keyword::from("$seen");
let c = Keyword::from("$seen".to_owned());
assert_eq!(a, b);
assert_eq!(b, c);
assert_eq!(a.as_ref(), "$seen");
assert_eq!(a.to_string(), "$seen");
}
#[test]
fn try_new_empty_fails() {
assert_eq!(Keyword::try_new(""), Err(KeywordError::Empty));
}
#[test]
fn try_new_space_fails() {
let err = Keyword::try_new("has space").unwrap_err();
assert_eq!(err, KeywordError::InvalidChar(' '));
}
#[test]
fn try_new_tab_fails() {
let err = Keyword::try_new("has\ttab").unwrap_err();
assert_eq!(err, KeywordError::InvalidChar('\t'));
}
#[test]
fn try_new_control_char_fails() {
let err = Keyword::try_new("has\x01ctrl").unwrap_err();
assert_eq!(err, KeywordError::InvalidChar('\x01'));
}
#[test]
fn try_new_unknown_dollar_prefix_fails() {
let err = Keyword::try_new("$unknown").unwrap_err();
assert_eq!(err, KeywordError::ReservedPrefix);
}
#[test]
fn try_new_all_system_keywords_succeed() {
for kw in &[
"$draft",
"$seen",
"$flagged",
"$answered",
"$forwarded",
"$phishing",
"$junk",
"$notjunk",
] {
Keyword::try_new(*kw)
.unwrap_or_else(|e| panic!("system keyword {kw:?} must succeed: {e}"));
}
}
#[test]
fn try_new_system_keyword_case_insensitive() {
Keyword::try_new("$SEEN").expect("$SEEN (case variant of $seen) must succeed");
}
#[test]
fn try_new_custom_keyword_succeeds() {
let kw = Keyword::try_new("project-alpha").expect("valid custom keyword must succeed");
assert_eq!(kw.as_ref(), "project-alpha");
}
#[test]
fn keyword_error_implements_error() {
let e = Keyword::try_new("").unwrap_err();
let _: &dyn std::error::Error = &e;
assert!(!e.to_string().is_empty());
}
}