mirror of
https://gitee.com/openharmony/third_party_rust_nom
synced 2024-11-27 09:40:54 +00:00
ab7211e2fc
list of removed parsers: eol, alpha, digit, hex_digit, oct_digit, alphanumeric, space, multispace Those parsers were aliases of other parsers, like alpha being an alias of alias1. So they are still present, this is just reducing the number of functions to choose from
222 lines
5.3 KiB
Rust
222 lines
5.3 KiB
Rust
#[macro_use]
|
|
extern crate nom;
|
|
|
|
use nom::{
|
|
IResult,
|
|
bytes::complete::take_while,
|
|
sequence::delimited,
|
|
combinator::map_res,
|
|
character::complete::{char, alphanumeric1 as alphanumeric, multispace0 as multispace, space0 as space}
|
|
};
|
|
|
|
use std::str;
|
|
use std::collections::HashMap;
|
|
|
|
fn category(i: &[u8]) -> IResult<&[u8], &str> {
|
|
map_res(delimited(char('['), take_while(|c| c != b']'), char(']')), str::from_utf8)(i)
|
|
}
|
|
|
|
fn complete_byte_slice_to_str<'a>(s: &'a[u8]) -> Result<&'a str, str::Utf8Error> {
|
|
str::from_utf8(s)
|
|
}
|
|
|
|
named!(key_value <&[u8],(&str,&str)>,
|
|
do_parse!(
|
|
key: map_res!(alphanumeric, complete_byte_slice_to_str)
|
|
>> opt!(space)
|
|
>> char!('=')
|
|
>> opt!(space)
|
|
>> val: map_res!(
|
|
take_while!(call!(|c| c != b'\n' && c != b';')),
|
|
complete_byte_slice_to_str
|
|
)
|
|
>> opt!(pair!(char!(';'), take_while!(call!(|c| c != b'\n'))))
|
|
>> (key, val)
|
|
)
|
|
);
|
|
|
|
named!(keys_and_values<&[u8], HashMap<&str, &str> >,
|
|
map!(
|
|
many0!(terminated!(key_value, opt!(multispace))),
|
|
|vec: Vec<_>| vec.into_iter().collect()
|
|
)
|
|
);
|
|
|
|
named!(category_and_keys<&[u8],(&str,HashMap<&str,&str>)>,
|
|
do_parse!(
|
|
category: category >>
|
|
opt!(multispace) >>
|
|
keys: keys_and_values >>
|
|
(category, keys)
|
|
)
|
|
);
|
|
|
|
named!(categories<&[u8], HashMap<&str, HashMap<&str,&str> > >,
|
|
map!(
|
|
many0!(
|
|
separated_pair!(
|
|
category,
|
|
opt!(multispace),
|
|
map!(
|
|
many0!(terminated!(key_value, opt!(multispace))),
|
|
|vec: Vec<_>| vec.into_iter().collect()
|
|
)
|
|
)
|
|
),
|
|
|vec: Vec<_>| vec.into_iter().collect()
|
|
)
|
|
);
|
|
|
|
#[test]
|
|
fn parse_category_test() {
|
|
let ini_file = &b"[category]
|
|
|
|
parameter=value
|
|
key = value2"[..];
|
|
|
|
let ini_without_category = &b"\n\nparameter=value
|
|
key = value2"[..];
|
|
|
|
let res = category(ini_file);
|
|
println!("{:?}", res);
|
|
match res {
|
|
Ok((i, o)) => println!("i: {:?} | o: {:?}", str::from_utf8(i), o),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
assert_eq!(res, Ok((ini_without_category, "category")));
|
|
}
|
|
|
|
#[test]
|
|
fn parse_key_value_test() {
|
|
let ini_file = &b"parameter=value
|
|
key = value2"[..];
|
|
|
|
let ini_without_key_value = &b"\nkey = value2"[..];
|
|
|
|
let res = key_value(ini_file);
|
|
println!("{:?}", res);
|
|
match res {
|
|
Ok((i, (o1, o2))) => println!("i: {:?} | o: ({:?},{:?})", str::from_utf8(i), o1, o2),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
assert_eq!(res, Ok((ini_without_key_value, ("parameter", "value"))));
|
|
}
|
|
|
|
#[test]
|
|
fn parse_key_value_with_space_test() {
|
|
let ini_file = &b"parameter = value
|
|
key = value2"[..];
|
|
|
|
let ini_without_key_value = &b"\nkey = value2"[..];
|
|
|
|
let res = key_value(ini_file);
|
|
println!("{:?}", res);
|
|
match res {
|
|
Ok((i, (o1, o2))) => println!("i: {:?} | o: ({:?},{:?})", str::from_utf8(i), o1, o2),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
assert_eq!(res, Ok((ini_without_key_value, ("parameter", "value"))));
|
|
}
|
|
|
|
#[test]
|
|
fn parse_key_value_with_comment_test() {
|
|
let ini_file = &b"parameter=value;abc
|
|
key = value2"[..];
|
|
|
|
let ini_without_key_value = &b"\nkey = value2"[..];
|
|
|
|
let res = key_value(ini_file);
|
|
println!("{:?}", res);
|
|
match res {
|
|
Ok((i, (o1, o2))) => println!("i: {:?} | o: ({:?},{:?})", str::from_utf8(i), o1, o2),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
assert_eq!(res, Ok((ini_without_key_value, ("parameter", "value"))));
|
|
}
|
|
|
|
#[test]
|
|
fn parse_multiple_keys_and_values_test() {
|
|
let ini_file = &b"parameter=value;abc
|
|
|
|
key = value2
|
|
|
|
[category]"[..];
|
|
|
|
let ini_without_key_value = &b"[category]"[..];
|
|
|
|
let res = keys_and_values(ini_file);
|
|
println!("{:?}", res);
|
|
match res {
|
|
Ok((i, ref o)) => println!("i: {:?} | o: {:?}", str::from_utf8(i), o),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
let mut expected: HashMap<&str, &str> = HashMap::new();
|
|
expected.insert("parameter", "value");
|
|
expected.insert("key", "value2");
|
|
assert_eq!(res, Ok((ini_without_key_value, expected)));
|
|
}
|
|
|
|
#[test]
|
|
fn parse_category_then_multiple_keys_and_values_test() {
|
|
//FIXME: there can be an empty line or a comment line after a category
|
|
let ini_file = &b"[abcd]
|
|
parameter=value;abc
|
|
|
|
key = value2
|
|
|
|
[category]"[..];
|
|
|
|
let ini_after_parser = &b"[category]"[..];
|
|
|
|
let res = category_and_keys(ini_file);
|
|
println!("{:?}", res);
|
|
match res {
|
|
Ok((i, ref o)) => println!("i: {:?} | o: {:?}", str::from_utf8(i), o),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
let mut expected_h: HashMap<&str, &str> = HashMap::new();
|
|
expected_h.insert("parameter", "value");
|
|
expected_h.insert("key", "value2");
|
|
assert_eq!(res, Ok((ini_after_parser, ("abcd", expected_h))));
|
|
}
|
|
|
|
#[test]
|
|
fn parse_multiple_categories_test() {
|
|
let ini_file = &b"[abcd]
|
|
|
|
parameter=value;abc
|
|
|
|
key = value2
|
|
|
|
[category]
|
|
parameter3=value3
|
|
key4 = value4
|
|
"[..];
|
|
|
|
let ini_after_parser = &b""[..];
|
|
|
|
let res = categories(ini_file);
|
|
//println!("{:?}", res);
|
|
match res {
|
|
Ok((i, ref o)) => println!("i: {:?} | o: {:?}", str::from_utf8(i), o),
|
|
_ => println!("error"),
|
|
}
|
|
|
|
let mut expected_1: HashMap<&str, &str> = HashMap::new();
|
|
expected_1.insert("parameter", "value");
|
|
expected_1.insert("key", "value2");
|
|
let mut expected_2: HashMap<&str, &str> = HashMap::new();
|
|
expected_2.insert("parameter3", "value3");
|
|
expected_2.insert("key4", "value4");
|
|
let mut expected_h: HashMap<&str, HashMap<&str, &str>> = HashMap::new();
|
|
expected_h.insert("abcd", expected_1);
|
|
expected_h.insert("category", expected_2);
|
|
assert_eq!(res, Ok((ini_after_parser, expected_h)));
|
|
}
|