mirror of
https://github.com/str4d/rage.git
synced 2025-04-05 20:07:42 +03:00
Rewrite age stanza parser to be much simpler
This commit is contained in:
parent
cfdf0bffba
commit
a4ae701670
1 changed files with 18 additions and 48 deletions
|
@ -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.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue