use inflector::Inflector;
use quote::__private::TokenStream;
pub trait Escaper {
fn escape_path(&self) -> String;
fn escape_test_name(&self) -> String;
fn escape_module_name(&self) -> String;
}
fn escape_str(s: &str) -> String {
match s.chars().next() {
None => "_".to_string(),
Some(c) => {
if c.is_numeric() {
format!("_{}", s.to_snake_case())
} else {
s.to_snake_case()
}
}
}
}
impl Escaper for &str {
fn escape_path(&self) -> String {
escape_str(self)
}
fn escape_test_name(&self) -> String {
format!("r#{}", escape_str(self))
}
fn escape_module_name(&self) -> String {
format!("r#{}", escape_str(self))
}
}
impl Escaper for String {
fn escape_path(&self) -> String {
self.as_str().escape_path()
}
fn escape_test_name(&self) -> String {
self.as_str().escape_test_name()
}
fn escape_module_name(&self) -> String {
self.as_str().escape_module_name()
}
}
#[inline]
pub fn escape_fn_code(ts: TokenStream) -> String {
ts.to_string().replace("\\n", "\n")
}
#[cfg(test)]
mod test {
use crate::util::Escaper;
#[test]
fn escaping_letters_and_whitespace() {
assert_eq!("a B c \t D \n e_f_G".escape_path(), "a_b_c_d_e_f_g");
assert_eq!("a B c \t D \n e_f_G".escape_test_name(), "r#a_b_c_d_e_f_g");
assert_eq!(
"a B c \t D \n e_f_G".escape_module_name(),
"r#a_b_c_d_e_f_g"
);
}
#[test]
fn escaping_letters_numbers_other_chars() {
assert_eq!(
"a B c 1 2 3 e f G !?#$%*!(".escape_path(),
"a_b_c_1_2_3_e_f_g"
);
assert_eq!(
"a B c 1 2 3 e f G !?#$%*!(".escape_test_name(),
"r#a_b_c_1_2_3_e_f_g"
);
assert_eq!(
"a B c 1 2 3 e f G !?#$%*!(".escape_module_name(),
"r#a_b_c_1_2_3_e_f_g"
);
}
}