Improve variable-length integer robustness
Move the maximum length check above the left shift, so we don't try and
shift more than 64 bits into the accumulator.
Return serde:🇩🇪:Errors from the invalid branches of the Uint
deserialization implementation rather than panicking on bogus or
truncated input.
Also clean up a Clippy lint in the Uint serialize implementation, using
the .take(i) iterator adapter rather than a ranged for-loop.
This commit is contained in:
parent
5681bc1aeb
commit
6695663b5b
53
src/lib.rs
53
src/lib.rs
@ -57,8 +57,8 @@ impl serde::ser::Serialize for Uint {
|
|||||||
i += 1;
|
i += 1;
|
||||||
|
|
||||||
let mut s = serializer.serialize_tuple(usize::MAX)?;
|
let mut s = serializer.serialize_tuple(usize::MAX)?;
|
||||||
for j in 0..i {
|
for b in buf.iter().take(i) {
|
||||||
s.serialize_element(&buf[j])?;
|
s.serialize_element(&b)?;
|
||||||
}
|
}
|
||||||
s.end()
|
s.end()
|
||||||
}
|
}
|
||||||
@ -75,8 +75,8 @@ impl<'de> serde::de::Deserialize<'de> for Uint {
|
|||||||
impl<'de> serde::de::Visitor<'de> for UintVisitor {
|
impl<'de> serde::de::Visitor<'de> for UintVisitor {
|
||||||
type Value = Uint;
|
type Value = Uint;
|
||||||
|
|
||||||
fn expecting(&self, _formatter: &mut fmt::Formatter) -> fmt::Result {
|
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
|
||||||
unreachable!()
|
write!(formatter, "a BARE encoded variable-length integer")
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_seq<A>(self, mut seq: A) -> std::result::Result<Self::Value, A::Error>
|
fn visit_seq<A>(self, mut seq: A) -> std::result::Result<Self::Value, A::Error>
|
||||||
@ -86,16 +86,28 @@ impl<'de> serde::de::Deserialize<'de> for Uint {
|
|||||||
let mut x = 0u64;
|
let mut x = 0u64;
|
||||||
let mut s = 0usize;
|
let mut s = 0usize;
|
||||||
for i in 0.. {
|
for i in 0.. {
|
||||||
// SeqAccess::next_element should never return None
|
let b = seq.next_element::<u8>()?;
|
||||||
let b = seq.next_element::<u8>()?.unwrap();
|
if let Some(b) = b {
|
||||||
if b < 0x80 {
|
|
||||||
if i > 9 || i == 9 && b > 1 {
|
if i > 9 || i == 9 && b > 1 {
|
||||||
todo!("error out");
|
// No more than 10 bytes can be in a BARE uint/int
|
||||||
|
return Err(serde::de::Error::custom(
|
||||||
|
"continuation bit indicated an invalid variable-length integer",
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
if b < 0x80 {
|
||||||
|
// No continuation bit is set
|
||||||
return Ok(Uint(x | (b as u64) << s));
|
return Ok(Uint(x | (b as u64) << s));
|
||||||
}
|
}
|
||||||
x |= ((b & 0x7f) as u64) << s;
|
x |= ((b & 0x7f) as u64) << s;
|
||||||
s += 7;
|
s += 7;
|
||||||
|
} else {
|
||||||
|
// Since we're calling next_element for u8 it's probably impossible to
|
||||||
|
// enter this branch without having raised an io::Error earlier, but better
|
||||||
|
// to handle it anyway instead of introducing a potential panic
|
||||||
|
return Err(serde::de::Error::custom(
|
||||||
|
"expected further bytes in variable-length integer",
|
||||||
|
));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
unreachable!()
|
unreachable!()
|
||||||
}
|
}
|
||||||
@ -169,10 +181,7 @@ mod test {
|
|||||||
(0, &[0]),
|
(0, &[0]),
|
||||||
(1, &[1]),
|
(1, &[1]),
|
||||||
(275, &[147, 2]),
|
(275, &[147, 2]),
|
||||||
(
|
(u64::MAX, &[255, 255, 255, 255, 255, 255, 255, 255, 255, 1]),
|
||||||
18446744073709551615,
|
|
||||||
&[255, 255, 255, 255, 255, 255, 255, 255, 255, 1],
|
|
||||||
),
|
|
||||||
];
|
];
|
||||||
for &(n, bytes) in CASES {
|
for &(n, bytes) in CASES {
|
||||||
println!("testing {}", n);
|
println!("testing {}", n);
|
||||||
@ -183,4 +192,24 @@ mod test {
|
|||||||
assert_eq!(got_int, int);
|
assert_eq!(got_int, int);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_uint_too_long() {
|
||||||
|
// Too many bytes
|
||||||
|
let bytes: &'static [u8] = &[255, 255, 255, 255, 255, 255, 255, 255, 255, 255, 1];
|
||||||
|
let result = from_slice::<Uint>(&bytes);
|
||||||
|
assert!(result.is_err());
|
||||||
|
|
||||||
|
// Too many bits of precision (effectively u64::MAX + 1)
|
||||||
|
let bytes: &'static [u8] = &[255, 255, 255, 255, 255, 255, 255, 255, 255, 2];
|
||||||
|
let result = from_slice::<Uint>(&bytes);
|
||||||
|
assert!(result.is_err());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_uint_too_short() {
|
||||||
|
let bytes: &'static [u8] = &[255, 255, 255];
|
||||||
|
let result = from_slice::<Uint>(&bytes);
|
||||||
|
assert!(result.is_err());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user