Rewrite age stanza parser to be much simpler

This commit is contained in:
Jack Grigg 2020-02-09 02:16:43 +00:00
parent cfdf0bffba
commit a4ae701670

View file

@ -14,10 +14,9 @@ pub struct AgeStanza<'a> {
pub mod read { pub mod read {
use nom::{ use nom::{
bytes::streaming::tag, bytes::streaming::{tag, take_while1},
character::streaming::newline, character::streaming::newline,
combinator::{map, map_opt}, combinator::{map, map_opt, verify},
error::{make_error, ErrorKind},
multi::separated_nonempty_list, multi::separated_nonempty_list,
sequence::separated_pair, sequence::separated_pair,
IResult, IResult,
@ -30,8 +29,6 @@ pub mod read {
/// ... an arbitrary string is a sequence of ASCII characters with values 33 to 126. /// ... an arbitrary string is a sequence of ASCII characters with values 33 to 126.
/// ``` /// ```
pub fn arbitrary_string(input: &[u8]) -> IResult<&[u8], &str> { pub fn arbitrary_string(input: &[u8]) -> IResult<&[u8], &str> {
use nom::bytes::streaming::take_while1;
map(take_while1(|c| c >= 33 && c <= 126), |bytes| { map(take_while1(|c| c >= 33 && c <= 126), |bytes| {
std::str::from_utf8(bytes).expect("ASCII is valid UTF-8") std::str::from_utf8(bytes).expect("ASCII is valid UTF-8")
})(input) })(input)
@ -44,40 +41,14 @@ pub mod read {
/// ///
/// - Returns Failure on an empty slice. /// - Returns Failure on an empty slice.
/// - Returns Incomplete(1) if a LF is not found. /// - Returns Incomplete(1) if a LF is not found.
fn take_b64_line(config: base64::Config) -> impl Fn(&[u8]) -> IResult<&[u8], &[u8]> { fn take_b64_line(input: &[u8]) -> IResult<&[u8], &[u8]> {
move |input: &[u8]| { verify(take_while1(|c| c != b'\n'), |bytes: &[u8]| {
let mut end = 0; base64::decode_config(bytes, base64::STANDARD_NO_PAD).is_ok()
while end < input.len() { })(input)
let c = input[end];
if c == b'\n' {
break;
}
// Substitute the character in twice after AA, so that padding
// characters will also be detected as a valid if allowed.
if base64::decode_config_slice(&[65, 65, c, c], config, &mut [0, 0, 0]).is_err() {
end = 0;
break;
}
end += 1;
}
if !input.is_empty() && end == 0 {
Err(nom::Err::Error(make_error(input, ErrorKind::Eof)))
} else if end < input.len() {
Ok((&input[end..], &input[..end]))
} else {
Err(nom::Err::Incomplete(nom::Needed::Size(1)))
}
}
} }
fn wrapped_encoded_data(input: &[u8]) -> IResult<&[u8], Vec<u8>> { fn wrapped_encoded_data(input: &[u8]) -> IResult<&[u8], Vec<u8>> {
map_opt( map_opt(separated_nonempty_list(newline, take_b64_line), |chunks| {
separated_nonempty_list(newline, take_b64_line(base64::STANDARD_NO_PAD)),
|chunks| {
// Enforce that the only chunk allowed to be shorter than 64 characters // Enforce that the only chunk allowed to be shorter than 64 characters
// is the last chunk. // is the last chunk.
if chunks.iter().rev().skip(1).any(|s| s.len() != 64) if chunks.iter().rev().skip(1).any(|s| s.len() != 64)
@ -88,8 +59,7 @@ pub mod read {
let data: Vec<u8> = chunks.into_iter().flatten().cloned().collect(); let data: Vec<u8> = chunks.into_iter().flatten().cloned().collect();
base64::decode_config(&data, base64::STANDARD_NO_PAD).ok() base64::decode_config(&data, base64::STANDARD_NO_PAD).ok()
} }
}, })(input)
)(input)
} }
/// Reads an age stanza. /// Reads an age stanza.