mirror of
https://github.com/sigp/lighthouse.git
synced 2026-07-04 21:34:36 +00:00
358 lines
14 KiB
Rust
358 lines
14 KiB
Rust
use crate::rpc::methods::*;
|
|
use crate::rpc::{
|
|
codec::base::OutboundCodec,
|
|
protocol::{Encoding, Protocol, ProtocolId, RPCError, Version},
|
|
};
|
|
use crate::rpc::{ErrorMessage, RPCErrorResponse, RPCRequest, RPCResponse};
|
|
use libp2p::bytes::BytesMut;
|
|
use snap::read::FrameDecoder;
|
|
use snap::write::FrameEncoder;
|
|
use ssz::{Decode, Encode};
|
|
use std::io::Cursor;
|
|
use std::io::ErrorKind;
|
|
use std::io::{Read, Write};
|
|
use std::marker::PhantomData;
|
|
use tokio::codec::{Decoder, Encoder};
|
|
use types::{EthSpec, SignedBeaconBlock};
|
|
use unsigned_varint::codec::Uvi;
|
|
|
|
/* Inbound Codec */
|
|
|
|
pub struct SSZSnappyInboundCodec<TSpec: EthSpec> {
|
|
protocol: ProtocolId,
|
|
inner: Uvi<usize>,
|
|
len: Option<usize>,
|
|
/// Maximum bytes that can be sent in one req/resp chunked responses.
|
|
max_packet_size: usize,
|
|
phantom: PhantomData<TSpec>,
|
|
}
|
|
|
|
impl<T: EthSpec> SSZSnappyInboundCodec<T> {
|
|
pub fn new(protocol: ProtocolId, max_packet_size: usize) -> Self {
|
|
let uvi_codec = Uvi::default();
|
|
// this encoding only applies to ssz_snappy.
|
|
debug_assert_eq!(protocol.encoding, Encoding::SSZSnappy);
|
|
|
|
SSZSnappyInboundCodec {
|
|
inner: uvi_codec,
|
|
protocol,
|
|
len: None,
|
|
phantom: PhantomData,
|
|
max_packet_size,
|
|
}
|
|
}
|
|
}
|
|
|
|
// Encoder for inbound streams: Encodes RPC Responses sent to peers.
|
|
impl<TSpec: EthSpec> Encoder for SSZSnappyInboundCodec<TSpec> {
|
|
type Item = RPCErrorResponse<TSpec>;
|
|
type Error = RPCError;
|
|
|
|
fn encode(&mut self, item: Self::Item, dst: &mut BytesMut) -> Result<(), Self::Error> {
|
|
let bytes = match item {
|
|
RPCErrorResponse::Success(resp) => match resp {
|
|
RPCResponse::Status(res) => res.as_ssz_bytes(),
|
|
RPCResponse::BlocksByRange(res) => res.as_ssz_bytes(),
|
|
RPCResponse::BlocksByRoot(res) => res.as_ssz_bytes(),
|
|
RPCResponse::Pong(res) => res.data.as_ssz_bytes(),
|
|
RPCResponse::MetaData(res) => res.as_ssz_bytes(),
|
|
},
|
|
RPCErrorResponse::InvalidRequest(err) => err.as_ssz_bytes(),
|
|
RPCErrorResponse::ServerError(err) => err.as_ssz_bytes(),
|
|
RPCErrorResponse::Unknown(err) => err.as_ssz_bytes(),
|
|
RPCErrorResponse::StreamTermination(_) => {
|
|
unreachable!("Code error - attempting to encode a stream termination")
|
|
}
|
|
};
|
|
// SSZ encoded bytes should be within `max_packet_size`
|
|
if bytes.len() > self.max_packet_size {
|
|
return Err(RPCError::Custom(
|
|
"attempting to encode data > max_packet_size".into(),
|
|
));
|
|
}
|
|
// Inserts the length prefix of the uncompressed bytes into dst
|
|
// encoded as a unsigned varint
|
|
self.inner
|
|
.encode(bytes.len(), dst)
|
|
.map_err(RPCError::from)?;
|
|
|
|
let mut writer = FrameEncoder::new(Vec::new());
|
|
writer.write_all(&bytes).map_err(RPCError::from)?;
|
|
writer.flush().map_err(RPCError::from)?;
|
|
|
|
// Write compressed bytes to `dst`
|
|
dst.extend_from_slice(writer.get_ref());
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// Decoder for inbound streams: Decodes RPC requests from peers
|
|
impl<TSpec: EthSpec> Decoder for SSZSnappyInboundCodec<TSpec> {
|
|
type Item = RPCRequest<TSpec>;
|
|
type Error = RPCError;
|
|
|
|
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
|
|
if self.len.is_none() {
|
|
// Decode the length of the uncompressed bytes from an unsigned varint
|
|
match self.inner.decode(src).map_err(RPCError::from)? {
|
|
Some(length) => {
|
|
self.len = Some(length);
|
|
}
|
|
None => return Ok(None), // need more bytes to decode length
|
|
}
|
|
};
|
|
|
|
let length = self.len.expect("length should be Some");
|
|
|
|
// Should not attempt to decode rpc chunks with length > max_packet_size
|
|
if length > self.max_packet_size {
|
|
return Err(RPCError::Custom(
|
|
"attempting to decode data > max_packet_size".into(),
|
|
));
|
|
}
|
|
let mut reader = FrameDecoder::new(Cursor::new(&src));
|
|
let mut decoded_buffer = vec![0; length];
|
|
|
|
match reader.read_exact(&mut decoded_buffer) {
|
|
Ok(()) => {
|
|
// `n` is how many bytes the reader read in the compressed stream
|
|
let n = reader.get_ref().position();
|
|
self.len = None;
|
|
src.split_to(n as usize);
|
|
match self.protocol.message_name {
|
|
Protocol::Status => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCRequest::Status(StatusMessage::from_ssz_bytes(
|
|
&decoded_buffer,
|
|
)?))),
|
|
},
|
|
Protocol::Goodbye => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCRequest::Goodbye(
|
|
GoodbyeReason::from_ssz_bytes(&decoded_buffer)?,
|
|
))),
|
|
},
|
|
Protocol::BlocksByRange => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCRequest::BlocksByRange(
|
|
BlocksByRangeRequest::from_ssz_bytes(&decoded_buffer)?,
|
|
))),
|
|
},
|
|
Protocol::BlocksByRoot => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCRequest::BlocksByRoot(BlocksByRootRequest {
|
|
block_roots: Vec::from_ssz_bytes(&decoded_buffer)?,
|
|
}))),
|
|
},
|
|
Protocol::Ping => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCRequest::Ping(Ping::from_ssz_bytes(
|
|
&decoded_buffer,
|
|
)?))),
|
|
},
|
|
Protocol::MetaData => match self.protocol.version {
|
|
Version::V1 => {
|
|
if decoded_buffer.len() > 0 {
|
|
Err(RPCError::Custom(
|
|
"Get metadata request should be empty".into(),
|
|
))
|
|
} else {
|
|
Ok(Some(RPCRequest::MetaData(PhantomData)))
|
|
}
|
|
}
|
|
},
|
|
}
|
|
}
|
|
Err(e) => match e.kind() {
|
|
// Haven't received enough bytes to decode yet
|
|
// TODO: check if this is the only Error variant where we return `Ok(None)`
|
|
ErrorKind::UnexpectedEof => {
|
|
return Ok(None);
|
|
}
|
|
_ => return Err(e).map_err(RPCError::from),
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Outbound Codec: Codec for initiating RPC requests */
|
|
pub struct SSZSnappyOutboundCodec<TSpec: EthSpec> {
|
|
inner: Uvi<usize>,
|
|
len: Option<usize>,
|
|
protocol: ProtocolId,
|
|
/// Maximum bytes that can be sent in one req/resp chunked responses.
|
|
max_packet_size: usize,
|
|
phantom: PhantomData<TSpec>,
|
|
}
|
|
|
|
impl<TSpec: EthSpec> SSZSnappyOutboundCodec<TSpec> {
|
|
pub fn new(protocol: ProtocolId, max_packet_size: usize) -> Self {
|
|
let uvi_codec = Uvi::default();
|
|
// this encoding only applies to ssz_snappy.
|
|
debug_assert_eq!(protocol.encoding, Encoding::SSZSnappy);
|
|
|
|
SSZSnappyOutboundCodec {
|
|
inner: uvi_codec,
|
|
protocol,
|
|
max_packet_size,
|
|
len: None,
|
|
phantom: PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
// Encoder for outbound streams: Encodes RPC Requests to peers
|
|
impl<TSpec: EthSpec> Encoder for SSZSnappyOutboundCodec<TSpec> {
|
|
type Item = RPCRequest<TSpec>;
|
|
type Error = RPCError;
|
|
|
|
fn encode(&mut self, item: Self::Item, dst: &mut BytesMut) -> Result<(), Self::Error> {
|
|
let bytes = match item {
|
|
RPCRequest::Status(req) => req.as_ssz_bytes(),
|
|
RPCRequest::Goodbye(req) => req.as_ssz_bytes(),
|
|
RPCRequest::BlocksByRange(req) => req.as_ssz_bytes(),
|
|
RPCRequest::BlocksByRoot(req) => req.block_roots.as_ssz_bytes(),
|
|
RPCRequest::Ping(req) => req.as_ssz_bytes(),
|
|
RPCRequest::MetaData(_) => return Ok(()), // no metadata to encode
|
|
};
|
|
// SSZ encoded bytes should be within `max_packet_size`
|
|
if bytes.len() > self.max_packet_size {
|
|
return Err(RPCError::Custom(
|
|
"attempting to encode data > max_packet_size".into(),
|
|
));
|
|
}
|
|
|
|
// Inserts the length prefix of the uncompressed bytes into dst
|
|
// encoded as a unsigned varint
|
|
self.inner
|
|
.encode(bytes.len(), dst)
|
|
.map_err(RPCError::from)?;
|
|
|
|
let mut writer = FrameEncoder::new(Vec::new());
|
|
writer.write_all(&bytes).map_err(RPCError::from)?;
|
|
writer.flush().map_err(RPCError::from)?;
|
|
|
|
// Write compressed bytes to `dst`
|
|
dst.extend_from_slice(writer.get_ref());
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// Decoder for outbound streams: Decodes RPC responses from peers.
|
|
//
|
|
// The majority of the decoding has now been pushed upstream due to the changing specification.
|
|
// We prefer to decode blocks and attestations with extra knowledge about the chain to perform
|
|
// faster verification checks before decoding entire blocks/attestations.
|
|
impl<TSpec: EthSpec> Decoder for SSZSnappyOutboundCodec<TSpec> {
|
|
type Item = RPCResponse<TSpec>;
|
|
type Error = RPCError;
|
|
|
|
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
|
|
if self.len.is_none() {
|
|
// Decode the length of the uncompressed bytes from an unsigned varint
|
|
match self.inner.decode(src).map_err(RPCError::from)? {
|
|
Some(length) => {
|
|
self.len = Some(length as usize);
|
|
}
|
|
None => return Ok(None), // need more bytes to decode length
|
|
}
|
|
};
|
|
|
|
let length = self.len.expect("length should be Some");
|
|
|
|
// Should not attempt to decode rpc chunks with length > max_packet_size
|
|
if length > self.max_packet_size {
|
|
return Err(RPCError::Custom(
|
|
"attempting to decode data > max_packet_size".into(),
|
|
));
|
|
}
|
|
let mut reader = FrameDecoder::new(Cursor::new(&src));
|
|
let mut decoded_buffer = vec![0; length];
|
|
match reader.read_exact(&mut decoded_buffer) {
|
|
Ok(()) => {
|
|
// `n` is how many bytes the reader read in the compressed stream
|
|
let n = reader.get_ref().position();
|
|
self.len = None;
|
|
src.split_to(n as usize);
|
|
match self.protocol.message_name {
|
|
Protocol::Status => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCResponse::Status(
|
|
StatusMessage::from_ssz_bytes(&decoded_buffer)?,
|
|
))),
|
|
},
|
|
Protocol::Goodbye => {
|
|
Err(RPCError::InvalidProtocol("GOODBYE doesn't have a response"))
|
|
}
|
|
Protocol::BlocksByRange => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCResponse::BlocksByRange(Box::new(
|
|
SignedBeaconBlock::from_ssz_bytes(&decoded_buffer)?,
|
|
)))),
|
|
},
|
|
Protocol::BlocksByRoot => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCResponse::BlocksByRoot(Box::new(
|
|
SignedBeaconBlock::from_ssz_bytes(&decoded_buffer)?,
|
|
)))),
|
|
},
|
|
Protocol::Ping => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCResponse::Pong(Ping {
|
|
data: u64::from_ssz_bytes(&decoded_buffer)?,
|
|
}))),
|
|
},
|
|
Protocol::MetaData => match self.protocol.version {
|
|
Version::V1 => Ok(Some(RPCResponse::MetaData(MetaData::from_ssz_bytes(
|
|
&decoded_buffer,
|
|
)?))),
|
|
},
|
|
}
|
|
}
|
|
Err(e) => match e.kind() {
|
|
// Haven't received enough bytes to decode yet
|
|
// TODO: check if this is the only Error variant where we return `Ok(None)`
|
|
ErrorKind::UnexpectedEof => {
|
|
return Ok(None);
|
|
}
|
|
_ => return Err(e).map_err(RPCError::from),
|
|
},
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<TSpec: EthSpec> OutboundCodec for SSZSnappyOutboundCodec<TSpec> {
|
|
type ErrorType = ErrorMessage;
|
|
|
|
fn decode_error(&mut self, src: &mut BytesMut) -> Result<Option<Self::ErrorType>, RPCError> {
|
|
if self.len.is_none() {
|
|
// Decode the length of the uncompressed bytes from an unsigned varint
|
|
match self.inner.decode(src).map_err(RPCError::from)? {
|
|
Some(length) => {
|
|
self.len = Some(length as usize);
|
|
}
|
|
None => return Ok(None), // need more bytes to decode length
|
|
}
|
|
};
|
|
|
|
let length = self.len.expect("length should be Some");
|
|
|
|
// Should not attempt to decode rpc chunks with length > max_packet_size
|
|
if length > self.max_packet_size {
|
|
return Err(RPCError::Custom(
|
|
"attempting to decode data > max_packet_size".into(),
|
|
));
|
|
}
|
|
let mut reader = FrameDecoder::new(Cursor::new(&src));
|
|
let mut decoded_buffer = vec![0; length];
|
|
match reader.read_exact(&mut decoded_buffer) {
|
|
Ok(()) => {
|
|
// `n` is how many bytes the reader read in the compressed stream
|
|
let n = reader.get_ref().position();
|
|
self.len = None;
|
|
src.split_to(n as usize);
|
|
Ok(Some(ErrorMessage::from_ssz_bytes(&decoded_buffer)?))
|
|
}
|
|
Err(e) => match e.kind() {
|
|
// Haven't received enough bytes to decode yet
|
|
// TODO: check if this is the only Error variant where we return `Ok(None)`
|
|
ErrorKind::UnexpectedEof => {
|
|
return Ok(None);
|
|
}
|
|
_ => return Err(e).map_err(RPCError::from),
|
|
},
|
|
}
|
|
}
|
|
}
|