use std::collections::HashMap;
use serde_json::value::{Value, to_value};
use slug;
use url::percent_encoding::{utf8_percent_encode, EncodeSet};
use errors::{TeraResult, TeraError};
use regex::{Regex, Captures};
lazy_static! {
static ref STRIPTAGS_RE: Regex = Regex::new(r"(<!--.*?-->|<[^>]*>)").unwrap();
static ref WORDS_RE: Regex = Regex::new(r"\b(?P<first>\w)(?P<rest>\w*)\b").unwrap();
}
pub fn upper(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("upper", "value", String, value);
Ok(to_value(&s.to_uppercase()))
}
pub fn lower(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("lower", "value", String, value);
Ok(to_value(&s.to_lowercase()))
}
pub fn trim(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("trim", "value", String, value);
Ok(to_value(&s.trim()))
}
pub fn truncate(value: Value, args: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("truncate", "value", String, value);
let length = match args.get("length") {
Some(l) => try_get_value!("truncate", "length", usize, l.clone()),
None => 255
};
if length > s.len() {
return Ok(to_value(&s));
}
let result = s[..s.char_indices().nth(length).unwrap().0].to_string() + "…";
Ok(to_value(&result))
}
pub fn wordcount(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("wordcount", "value", String, value);
Ok(to_value(&s.split_whitespace().count()))
}
pub fn replace(value: Value, args: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("replace", "value", String, value);
let from = match args.get("from") {
Some(val) => try_get_value!("replace", "from", String, val.clone()),
None => {
return Err(TeraError::FilterMissingArg("replace".to_string(), "from".to_string()));
}
};
let to = match args.get("to") {
Some(val) => try_get_value!("replace", "to", String, val.clone()),
None => {
return Err(TeraError::FilterMissingArg("replace".to_string(), "to".to_string()));
}
};
Ok(to_value(&s.replace(&from, &to)))
}
pub fn capitalize(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("capitalize", "value", String, value);
let mut chars = s.chars();
match chars.next() {
None => Ok(to_value("")),
Some(f) => {
let res = f.to_uppercase().collect::<String>() + &chars.as_str().to_lowercase();
Ok(to_value(&res))
}
}
}
#[derive(Clone)]
struct UrlEncodeSet(String);
impl UrlEncodeSet {
fn safe_bytes(&self) -> &[u8] {
let &UrlEncodeSet(ref safe) = self;
safe.as_bytes()
}
}
impl EncodeSet for UrlEncodeSet {
fn contains(&self, byte: u8) -> bool {
if byte >= 48 && byte <= 57 { false
} else if byte >= 65 && byte <= 90 { false
} else if byte >= 97 && byte <= 122 { false
} else if byte == 45 || byte == 46 || byte == 95 { false
} else if self.safe_bytes().contains(&byte) {
false
} else {
true
}
}
}
pub fn urlencode(value: Value, args: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("urlencode", "value", String, value);
let safe = match args.get("safe") {
Some(l) => try_get_value!("urlencode", "safe", String, l.clone()),
None => "/".to_string(),
};
let encoded = utf8_percent_encode(s.as_str(), UrlEncodeSet(safe)).collect::<String>();
Ok(to_value(&encoded))
}
pub fn addslashes(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("addslashes", "value", String, value);
Ok(to_value(&s.replace("\\","\\\\").replace("\"", "\\\"").replace("\'", "\\\'")))
}
pub fn slugify(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("slugify", "value", String, value);
Ok(to_value(&slug::slugify(s)))
}
pub fn title(value: Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("title", "value", String, value);
Ok(to_value(
&WORDS_RE.replace_all(&s, |caps: &Captures| {
let first = caps["first"].to_uppercase();
let rest = caps["rest"].to_lowercase();
format!("{}{}", first, rest)
})
))
}
pub fn striptags(value : Value, _: HashMap<String, Value>) -> TeraResult<Value> {
let s = try_get_value!("striptags", "value", String, value);
Ok(to_value(&STRIPTAGS_RE.replace_all(&s, "")))
}
#[cfg(test)]
mod tests {
use std::collections::HashMap;
use serde_json::value::{to_value};
use errors::TeraError::*;
use super::*;
#[test]
fn test_upper() {
let result = upper(to_value("hello"), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value("HELLO"));
}
#[test]
fn test_upper_error() {
let result = upper(to_value(&50), HashMap::new());
assert!(result.is_err());
assert_eq!(
result.err().unwrap(),
FilterIncorrectArgType("upper".to_string(), "value".to_string(), to_value(&50), "String".to_string())
);
}
#[test]
fn test_trim() {
let result = trim(to_value(" hello "), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value("hello"));
}
#[test]
fn test_truncate_smaller_than_length() {
let mut args = HashMap::new();
args.insert("length".to_string(), to_value(&255));
let result = truncate(to_value("hello"), args);
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value("hello"));
}
#[test]
fn test_truncate_when_required() {
let mut args = HashMap::new();
args.insert("length".to_string(), to_value(&2));
let result = truncate(to_value("日本語"), args);
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value("日本…"));
}
#[test]
fn test_lower() {
let result = lower(to_value("HELLO"), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value("hello"));
}
#[test]
fn test_wordcount() {
let result = wordcount(to_value("Joel is a slug"), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(&4));
}
#[test]
fn test_replace() {
let mut args = HashMap::new();
args.insert("from".to_string(), to_value(&"Hello"));
args.insert("to".to_string(), to_value(&"Goodbye"));
let result = replace(to_value(&"Hello world!"), args);
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value("Goodbye world!"));
}
#[test]
fn test_replace_missing_arg() {
let mut args = HashMap::new();
args.insert("from".to_string(), to_value(&"Hello"));
let result = replace(to_value(&"Hello world!"), args);
assert!(result.is_err());
assert_eq!(
result.err().unwrap(),
FilterMissingArg("replace".to_string(), "to".to_string())
);
}
#[test]
fn test_capitalize() {
let tests = vec![
("CAPITAL IZE", "Capital ize"),
("capital ize", "Capital ize"),
];
for (input, expected) in tests {
let result = capitalize(to_value(input), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(expected));
}
}
#[test]
fn test_addslashes() {
let tests = vec![
(r#"I'm so happy"#, r#"I\'m so happy"#),
(r#"Let "me" help you"#, r#"Let \"me\" help you"#),
(r#"<a>'"#, r#"<a>\'"#),
(r#""double quotes" and \'single quotes\'"#, r#"\"double quotes\" and \\\'single quotes\\\'"#),
(r#"\ : backslashes too"#, r#"\\ : backslashes too"#)
];
for (input, expected) in tests {
let result = addslashes(to_value(input), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(expected));
}
}
#[test]
fn test_slugify() {
let tests = vec![
(r#"Hello world"#, r#"hello-world"#),
(r#"Hello 世界"#, r#"hello-shi-jie"#),
];
for (input, expected) in tests {
let result = slugify(to_value(input), HashMap::new());
println!("{:?} - {:?}", input, result);
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(expected));
}
}
#[test]
fn test_urlencode() {
let tests = vec![
(r#"https://www.example.org/foo?a=b&c=d"#, None, r#"https%3A//www.example.org/foo%3Fa%3Db%26c%3Dd"#),
(r#"https://www.example.org/"#, Some(""), r#"https%3A%2F%2Fwww.example.org%2F"#),
(r#"/test&"/me?/"#, None, r#"/test%26%22/me%3F/"#),
(r#"escape/slash"#, Some(""), r#"escape%2Fslash"#),
];
for (input, safe, expected) in tests {
let mut args = HashMap::new();
if let Some(safe) = safe {
args.insert("safe".to_string(), to_value(&safe));
}
let result = urlencode(to_value(input), args);
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(expected));
}
}
#[test]
fn test_title() {
let tests = vec![
("foo bar", "Foo Bar"),
("foo\tbar", "Foo\tBar"),
("foo bar", "Foo Bar"),
("f bar f", "F Bar F"),
("foo-bar", "Foo-Bar"),
("FOO\tBAR", "Foo\tBar"),
("foo (bar)", "Foo (Bar)"),
("foo (bar) ", "Foo (Bar) "),
("foo {bar}", "Foo {Bar}"),
("foo [bar]", "Foo [Bar]"),
("foo <bar>", "Foo <Bar>"),
(" foo bar", " Foo Bar"),
("\tfoo\tbar\t", "\tFoo\tBar\t"),
("foo bar ", "Foo Bar "),
("foo bar\t", "Foo Bar\t")
];
for (input, expected) in tests {
let result = title(to_value(input), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(expected));
}
}
#[test]
fn test_striptags() {
let tests = vec![
(r"<b>Joel</b> <button>is</button> a <span>slug</span>", "Joel is a slug"),
(r#"<p>just a small \n <a href="x"> example</a> link</p>\n<p>to a webpage</p><!-- <p>and some commented stuff</p> -->"#,
r#"just a small \n example link\nto a webpage"#),
(r"<p>See: 'é is an apostrophe followed by e acute</p>",r"See: 'é is an apostrophe followed by e acute"),
(r"<adf>a", "a"),
(r"</adf>a", "a"),
(r"<asdf><asdf>e", "e"),
(r"hi, <f x", "hi, <f x"),
("234<235, right?", "234<235, right?"),
("a4<a5 right?", "a4<a5 right?"),
("b7>b2!", "b7>b2!"),
("</fe", "</fe"),
("<x>b<y>", "b"),
(r#"a<p a >b</p>c"#, "abc"),
(r#"d<a:b c:d>e</p>f"#, "def"),
(r#"<strong>foo</strong><a href="http://example.com">bar</a>"#, "foobar"),
];
for (input, expected) in tests {
let result = striptags(to_value(input), HashMap::new());
assert!(result.is_ok());
assert_eq!(result.unwrap(), to_value(expected));
}
}
}