light: response decoding and verification module
This commit is contained in:
parent
359d433292
commit
887bcfb88a
152
sync/src/light_sync/response.rs
Normal file
152
sync/src/light_sync/response.rs
Normal file
@ -0,0 +1,152 @@
|
||||
// Copyright 2015, 2016 Parity Technologies (UK) Ltd.
|
||||
// This file is part of Parity.
|
||||
|
||||
// Parity is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
|
||||
// Parity is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
//! Helpers for decoding and verifying responses for headers.
|
||||
|
||||
use std::fmt;
|
||||
|
||||
use ethcore::header::Header;
|
||||
use light::request::{HashOrNumber, Headers as HeadersRequest};
|
||||
use rlp::{DecoderError, UntrustedRlp, View};
|
||||
use util::H256;
|
||||
|
||||
/// Errors found when decoding headers and verifying with basic constraints.
|
||||
#[derive(Debug, Clone)]
|
||||
pub enum BasicError {
|
||||
/// Wrong skip value: expected, found (if any).
|
||||
WrongSkip(u64, Option<u64>),
|
||||
/// Wrong start number.
|
||||
WrongStartNumber(u64, u64),
|
||||
/// Wrong start hash.
|
||||
WrongStartHash(H256, H256),
|
||||
/// Too many headers.
|
||||
TooManyHeaders(usize, usize),
|
||||
/// Decoder error.
|
||||
Decoder(DecoderError),
|
||||
}
|
||||
|
||||
impl From<DecoderError> for BasicError {
|
||||
fn from(err: DecoderError) -> Self {
|
||||
BasicError::Decoder(err)
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Display for BasicError {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
write!(f, "Header response verification error: ");
|
||||
|
||||
match *self {
|
||||
BasicError::WrongSkip(ref exp, ref got)
|
||||
=> write!(f, "wrong skip (expected {}, got {:?})", exp, got),
|
||||
BasicError::WrongStartNumber(ref exp, ref got)
|
||||
=> write!(f, "wrong start number (expected {}, got {})", exp, got),
|
||||
BasicError::WrongStartHash(ref exp, ref got)
|
||||
=> write!(f, "wrong start hash (expected {}, got {})", exp, got),
|
||||
BasicError::TooManyHeaders(ref max, ref got)
|
||||
=> write!(f, "too many headers (max {}, got {})", max, got),
|
||||
BasicError::Decoder(ref err)
|
||||
=> write!(f, "invalid encoding ({})", err),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Request verification constraint.
|
||||
pub trait Constraint {
|
||||
type Error;
|
||||
|
||||
/// Verify headers against this.
|
||||
fn verify(&self, headers: &[Header], reverse: bool) -> Result<(), Self::Error>;
|
||||
}
|
||||
|
||||
/// Decode a response and do basic verification against a request.
|
||||
pub fn decode_and_verify(headers: &[Bytes], request: &HeadersRequest) -> Result<Vec<Header>, BasicError> {
|
||||
let headers: Vec<_> = try!!(headers.iter().map(|x| UntrustedRlp::new(&x).as_val()).collect());
|
||||
|
||||
let reverse = request.reverse;
|
||||
|
||||
try!(Max(request.max).verify(&headers, reverse));
|
||||
match request.start {
|
||||
HashOrNumber::Number(ref num) => try!(StartsAtNumber(*num).verify(&headers, reverse)),
|
||||
HashOrNumber::Hash(ref hash) => try!(StartsAtHash(*hash).verify(&headers, reverse)),
|
||||
}
|
||||
|
||||
try!(SkipsBetween(request.skip).verify(&headers, reverse));
|
||||
}
|
||||
|
||||
struct StartsAtNumber(u64);
|
||||
struct StartsAtHash(H256);
|
||||
struct SkipsBetween(u64);
|
||||
struct Max(usize);
|
||||
|
||||
impl Constraint for StartsAtNumber {
|
||||
type Error = BasicError;
|
||||
|
||||
fn verify(&self, headers: &[Header], reverse: bool) -> Result<(), BasicError> {
|
||||
let earliest = if reverse { headers.last() } else { headers.first() };
|
||||
|
||||
earliest.map_or(Ok(()), |h| {
|
||||
if h.number() == self.0 {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(BasicError::WrongStartNumber(self.0, h.number()))
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl Constraint for StartsAtHash {
|
||||
type Error = BasicError;
|
||||
|
||||
fn verify(&self, headers: &[Header], reverse: bool) -> Result<(), BasicError> {
|
||||
let earliest = if reverse { headers.last() } else { headers.first() };
|
||||
|
||||
earliest.map_or(Ok(()), |h| {
|
||||
if h.hash() == self.0 {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(BasicError::WrongStartHash(self.0, h.hash()))
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl Constraint for SkipsBetween {
|
||||
type Error = BasicError;
|
||||
|
||||
fn verify(&self, headers: &[Header], reverse: bool) -> Result<(), BasicError> {
|
||||
for pair in headers.windows(2) {
|
||||
let (low, high) = if reverse { (&pair[1], &pair[0]) } else { (&pair[0], &pair[1]) };
|
||||
if low.number() >= high.number() { return Err(BasicError::WrongSkip(self.0, None)) }
|
||||
|
||||
let skip = (high.number() - low.number()) - 1;
|
||||
if skip != self.0 { return Err(BasicError::WrongSkip(self.0, Some(skip))) }
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl Constraint for Max {
|
||||
type Error = BasicError;
|
||||
|
||||
fn verify(&self, headers: &[Header], _reverse: bool) -> Result<(), BasicError> {
|
||||
match headers.len() > self.0 {
|
||||
true => Err(BasicError::TooManyHeaders(self.0, headers.len())),
|
||||
false => Ok(())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user