use crate::sql::error::IResult; use nom::bytes::complete::escaped; use nom::bytes::complete::is_not; use nom::character::complete::anychar; use nom::character::complete::char; use revision::revisioned; use serde::{ de::{self, Visitor}, Deserialize, Deserializer, Serialize, Serializer, }; use std::cmp::Ordering; use std::fmt::Debug; use std::fmt::{self, Display, Formatter}; use std::hash::{Hash, Hasher}; use std::str; use std::str::FromStr; pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Regex"; #[derive(Clone)] #[revisioned(revision = 1)] pub struct Regex(pub regex::Regex); impl Regex { // Deref would expose `regex::Regex::as_str` which wouldn't have the '/' delimiters. pub fn regex(&self) -> ®ex::Regex { &self.0 } } impl FromStr for Regex { type Err = ::Err; fn from_str(s: &str) -> Result { if s.contains('\0') { Err(regex::Error::Syntax("regex contained NUL byte".to_owned())) } else { regex::Regex::new(&s.replace("\\/", "/")).map(Self) } } } impl PartialEq for Regex { fn eq(&self, other: &Self) -> bool { self.0.as_str().eq(other.0.as_str()) } } impl Eq for Regex {} impl Ord for Regex { fn cmp(&self, other: &Self) -> Ordering { self.0.as_str().cmp(other.0.as_str()) } } impl PartialOrd for Regex { fn partial_cmp(&self, other: &Self) -> Option { Some(self.cmp(other)) } } impl Hash for Regex { fn hash(&self, state: &mut H) { self.0.as_str().hash(state); } } impl Debug for Regex { fn fmt(&self, f: &mut Formatter) -> fmt::Result { Display::fmt(self, f) } } impl Display for Regex { fn fmt(&self, f: &mut Formatter) -> fmt::Result { write!(f, "/{}/", &self.0) } } impl Serialize for Regex { fn serialize(&self, serializer: S) -> Result where S: Serializer, { serializer.serialize_newtype_struct(TOKEN, self.0.as_str()) } } impl<'de> Deserialize<'de> for Regex { fn deserialize(deserializer: D) -> Result where D: Deserializer<'de>, { struct RegexNewtypeVisitor; impl<'de> Visitor<'de> for RegexNewtypeVisitor { type Value = Regex; fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { formatter.write_str("a regex newtype") } fn visit_newtype_struct(self, deserializer: D) -> Result where D: Deserializer<'de>, { struct RegexVisitor; impl<'de> Visitor<'de> for RegexVisitor { type Value = Regex; fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { formatter.write_str("a regex str") } fn visit_str(self, value: &str) -> Result where E: de::Error, { Regex::from_str(value).map_err(|_| de::Error::custom("invalid regex")) } } deserializer.deserialize_str(RegexVisitor) } } deserializer.deserialize_newtype_struct(TOKEN, RegexNewtypeVisitor) } } pub fn regex(i: &str) -> IResult<&str, Regex> { let (i, _) = char('/')(i)?; let (i, v) = escaped(is_not("\\/"), '\\', anychar)(i)?; let (i, _) = char('/')(i)?; let regex = v.parse().map_err(|_| nom::Err::Error(crate::sql::Error::Parser(v)))?; Ok((i, regex)) } #[cfg(test)] mod tests { use super::*; #[test] fn regex_simple() { let sql = "/test/"; let res = regex(sql); let out = res.unwrap().1; assert_eq!("/test/", format!("{}", out)); assert_eq!(out, "test".parse().unwrap()); } #[test] fn regex_complex() { let sql = r"/(?i)test\/[a-z]+\/\s\d\w{1}.*/"; let res = regex(sql); let out = res.unwrap().1; assert_eq!(r"/(?i)test/[a-z]+/\s\d\w{1}.*/", format!("{}", out)); assert_eq!(out, r"(?i)test/[a-z]+/\s\d\w{1}.*".parse().unwrap()); } }