surrealpatch/lib/src/sql/regex.rs

71 lines
1.5 KiB
Rust
Raw Normal View History

2022-01-16 20:31:50 +00:00
use crate::sql::error::IResult;
use nom::bytes::complete::escaped;
2020-06-29 15:36:01 +00:00
use nom::bytes::complete::is_not;
use nom::character::complete::anychar;
use nom::character::complete::char;
2020-06-29 15:36:01 +00:00
use serde::{Deserialize, Serialize};
use std::fmt;
use std::ops::Deref;
2020-06-29 15:36:01 +00:00
use std::str;
#[derive(Clone, Debug, Default, Eq, Ord, PartialEq, PartialOrd, Serialize, Deserialize)]
pub struct Regex(String);
2020-06-29 15:36:01 +00:00
2022-05-25 09:42:10 +00:00
impl From<&str> for Regex {
fn from(r: &str) -> Self {
Self(r.replace("\\/", "/"))
2020-06-29 15:36:01 +00:00
}
}
impl Deref for Regex {
type Target = String;
fn deref(&self) -> &Self::Target {
&self.0
2021-03-29 15:43:37 +00:00
}
}
impl fmt::Display for Regex {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "/{}/", &self.0)
2021-03-29 15:43:37 +00:00
}
}
impl Regex {
pub fn regex(&self) -> Option<regex::Regex> {
regex::Regex::new(&self.0).ok()
2020-06-29 15:36:01 +00:00
}
}
pub fn regex(i: &str) -> IResult<&str, Regex> {
let (i, _) = char('/')(i)?;
let (i, v) = escaped(is_not("\\/"), '\\', anychar)(i)?;
let (i, _) = char('/')(i)?;
2020-06-29 15:36:01 +00:00
Ok((i, Regex::from(v)))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn regex_simple() {
let sql = "/test/";
let res = regex(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!("/test/", format!("{}", out));
assert_eq!(out, Regex::from("test"));
}
#[test]
fn regex_complex() {
let sql = r"/(?i)test\/[a-z]+\/\s\d\w{1}.*/";
2020-06-29 15:36:01 +00:00
let res = regex(sql);
assert!(res.is_ok());
let out = res.unwrap().1;
assert_eq!(r"/(?i)test/[a-z]+/\s\d\w{1}.*/", format!("{}", out));
assert_eq!(out, Regex::from(r"(?i)test/[a-z]+/\s\d\w{1}.*"));
2020-06-29 15:36:01 +00:00
}
}