// 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 . //! 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::{Bytes, H256}; /// Errors found when decoding headers and verifying with basic constraints. #[derive(Debug, PartialEq)] pub enum BasicError { /// Wrong skip value: expected, found (if any). WrongSkip(u64, Option), /// Wrong start number. WrongStartNumber(u64, u64), /// Wrong start hash. WrongStartHash(H256, H256), /// Too many headers. TooManyHeaders(usize, usize), /// Decoder error. Decoder(DecoderError), } impl From for BasicError { fn from(err: DecoderError) -> Self { BasicError::Decoder(err) } } impl fmt::Display for BasicError { fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { try!(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, "{}", 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, 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)); Ok(headers) } 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> { headers.first().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> { headers.first().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(()) } } } #[cfg(test)] mod tests { use ethcore::header::Header; use light::request::Headers as HeadersRequest; use super::*; #[test] fn sequential_forward() { let request = HeadersRequest { start: 10.into(), max: 30, skip: 0, reverse: false, }; let mut parent_hash = None; let headers: Vec<_> = (0..25).map(|x| x + 10).map(|x| { let mut header = Header::default(); header.set_number(x); if let Some(parent_hash) = parent_hash { header.set_parent_hash(parent_hash); } parent_hash = Some(header.hash()); ::rlp::encode(&header).to_vec() }).collect(); assert!(decode_and_verify(&headers, &request).is_ok()); } #[test] fn sequential_backward() { let request = HeadersRequest { start: 34.into(), max: 30, skip: 0, reverse: true, }; let mut parent_hash = None; let headers: Vec<_> = (0..25).map(|x| x + 10).rev().map(|x| { let mut header = Header::default(); header.set_number(x); if let Some(parent_hash) = parent_hash { header.set_parent_hash(parent_hash); } parent_hash = Some(header.hash()); ::rlp::encode(&header).to_vec() }).collect(); assert!(decode_and_verify(&headers, &request).is_ok()); } #[test] fn too_many() { let request = HeadersRequest { start: 10.into(), max: 20, skip: 0, reverse: false, }; let mut parent_hash = None; let headers: Vec<_> = (0..25).map(|x| x + 10).map(|x| { let mut header = Header::default(); header.set_number(x); if let Some(parent_hash) = parent_hash { header.set_parent_hash(parent_hash); } parent_hash = Some(header.hash()); ::rlp::encode(&header).to_vec() }).collect(); assert_eq!(decode_and_verify(&headers, &request), Err(BasicError::TooManyHeaders(20, 25))); } #[test] fn wrong_skip() { let request = HeadersRequest { start: 10.into(), max: 30, skip: 5, reverse: false, }; let headers: Vec<_> = (0..25).map(|x| x * 3).map(|x| x + 10).map(|x| { let mut header = Header::default(); header.set_number(x); ::rlp::encode(&header).to_vec() }).collect(); assert_eq!(decode_and_verify(&headers, &request), Err(BasicError::WrongSkip(5, Some(2)))); } }