use percent_encoding::percent_decode_str;
use std::borrow::Cow;
pub type QueryParamRange = [Option<usize>; 2];
pub struct QueryParser<'a> {
input: &'a str,
from: usize,
}
fn decode(input: &str) -> Cow<'_, str> {
percent_decode_str(input).decode_utf8_lossy()
}
fn take_name(input: &str, from: usize) -> (usize, Option<Cow<'_, str>>) {
let len = input.len();
let at = take_while(input, from, |byte| byte == b'&').map(|start| {
match take_while(input, start, |byte| byte != b'=') {
Some(end) => (start, end),
None => (start, len),
}
});
match at {
Some((start, end)) => (end, input.get(start..end).map(decode)),
None => (len, None),
}
}
fn take_value(input: &str, from: usize) -> (usize, Option<QueryParamRange>) {
let len = input.len();
let at = take_while(input, from, |byte| byte == b'=').map(|start| {
match take_while(input, start, |byte| byte != b'&') {
Some(end) => [Some(start), Some(end)],
None => [Some(start), None],
}
});
match at {
Some([_, Some(end)]) => (end, at),
Some([_, None]) => (len, at),
None => (len, None),
}
}
fn take_while(input: &str, from: usize, f: impl Fn(u8) -> bool) -> Option<usize> {
let rest = input.get(from..)?;
rest.bytes()
.enumerate()
.find_map(|(to, byte)| if !f(byte) { from.checked_add(to) } else { None })
}
impl<'a> QueryParser<'a> {
pub fn new(input: &'a str) -> Self {
Self { input, from: 0 }
}
}
impl<'a> Iterator for QueryParser<'a> {
type Item = (Cow<'a, str>, Option<QueryParamRange>);
fn next(&mut self) -> Option<Self::Item> {
let (start, name) = take_name(self.input, self.from);
let (end, at) = take_value(self.input, start);
self.from = end;
name.zip(Some(at))
}
}
#[cfg(test)]
mod tests {
use super::QueryParser;
#[test]
fn parse_query_params_test() {
let query_strings = vec![
"query=books&category=fiction&sort=asc",
"query=hello%20world&category=%E2%9C%93",
"category=books&category=electronics&category=clothing",
"query=books&category=",
"query=100%25%20organic&category=food",
"query=books&filter={\"price\":\"low\",\"rating\":5}",
"items[]=book&items[]=pen&items[]=notebook",
"",
"data={\"name\":\"John\",\"age\":30,\"city\":\"New York\"}",
"query=books&category=fiction#section2",
"query=books&&category=fiction", "query==books&category=fiction", "query=books&category", "query=books&=fiction", "query=books&category=fiction&", "qu%65ry=books&ca%74egory=fiction",
"%71uery=books&%63ategory=fiction",
"query=books&ca%74egory=fic%74ion",
"query=books&category=%80%80%80",
"query=books&category=%C3%28",
"query%C3%28=books&category=%C3%28",
];
let expected_results = vec![
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(21), Some(28)])),
("sort", Some([Some(34), None])),
],
vec![
("query", Some([Some(6), Some(19)])),
("category", Some([Some(29), None])),
],
vec![
("category", Some([Some(9), Some(14)])),
("category", Some([Some(24), Some(35)])),
("category", Some([Some(45), None])),
],
vec![("query", Some([Some(6), Some(11)])), ("category", None)],
vec![
("query", Some([Some(6), Some(22)])),
("category", Some([Some(32), None])),
],
vec![
("query", Some([Some(6), Some(11)])),
("filter", Some([Some(19), None])),
],
vec![
("items[]", Some([Some(8), Some(12)])),
("items[]", Some([Some(21), Some(24)])),
("items[]", Some([Some(33), None])),
],
vec![],
vec![("data", Some([Some(5), None]))],
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(21), None])),
],
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(22), None])),
],
vec![
("query", Some([Some(7), Some(12)])),
("category", Some([Some(22), None])),
],
vec![("query", Some([Some(6), Some(11)])), ("category", None)],
vec![
("query", Some([Some(6), Some(11)])),
("", Some([Some(13), None])),
],
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(21), Some(28)])),
],
vec![
("query", Some([Some(8), Some(13)])),
("category", Some([Some(25), None])),
],
vec![
("query", Some([Some(8), Some(13)])),
("category", Some([Some(25), None])),
],
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(23), None])),
],
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(21), None])),
],
vec![
("query", Some([Some(6), Some(11)])),
("category", Some([Some(21), None])),
],
vec![
("query�(", Some([Some(12), Some(17)])),
("category", Some([Some(27), None])),
],
];
for (expected_result_index, query) in query_strings.iter().enumerate() {
let mut actual_result = vec![];
let expected_result = &expected_results[expected_result_index];
actual_result.extend(QueryParser::new(query));
assert_eq!(
actual_result.len(),
expected_result.len(),
"Expected {} to have {} entries. Got {} instead. Query: '{}'.",
expected_result_index,
expected_result.len(),
actual_result.len(),
query
);
for (entry_index, (name, at)) in actual_result.into_iter().enumerate() {
let expect = &expected_result[entry_index];
assert_eq!(
name, expect.0,
"Expected name at index [{}, {}] to be {}. Got {} instead.",
expected_result_index, entry_index, expect.0, name
);
assert_eq!(
at, expect.1,
"Expected range at index [{}, {}] to be {:?}. Got {:?} instead.",
expected_result_index, entry_index, expect.1, at
);
}
}
}
}