df40700ddd
## Description The `eth2_libp2p` crate was originally named and designed to incorporate a simple libp2p integration into lighthouse. Since its origins the crates purpose has expanded dramatically. It now houses a lot more sophistication that is specific to lighthouse and no longer just a libp2p integration. As of this writing it currently houses the following high-level lighthouse-specific logic: - Lighthouse's implementation of the eth2 RPC protocol and specific encodings/decodings - Integration and handling of ENRs with respect to libp2p and eth2 - Lighthouse's discovery logic, its integration with discv5 and logic about searching and handling peers. - Lighthouse's peer manager - This is a large module handling various aspects of Lighthouse's network, such as peer scoring, handling pings and metadata, connection maintenance and recording, etc. - Lighthouse's peer database - This is a collection of information stored for each individual peer which is specific to lighthouse. We store connection state, sync state, last seen ips and scores etc. The data stored for each peer is designed for various elements of the lighthouse code base such as syncing and the http api. - Gossipsub scoring - This stores a collection of gossipsub 1.1 scoring mechanisms that are continuously analyssed and updated based on the ethereum 2 networks and how Lighthouse performs on these networks. - Lighthouse specific types for managing gossipsub topics, sync status and ENR fields - Lighthouse's network HTTP API metrics - A collection of metrics for lighthouse network monitoring - Lighthouse's custom configuration of all networking protocols, RPC, gossipsub, discovery, identify and libp2p. Therefore it makes sense to rename the crate to be more akin to its current purposes, simply that it manages the majority of Lighthouse's network stack. This PR renames this crate to `lighthouse_network` Co-authored-by: Paul Hauner <paul@paulhauner.com>
594 lines
21 KiB
Rust
594 lines
21 KiB
Rust
use super::methods::*;
|
|
use crate::rpc::{
|
|
codec::{base::BaseInboundCodec, ssz_snappy::SSZSnappyInboundCodec, InboundCodec},
|
|
methods::{MaxErrorLen, ResponseTermination, MAX_ERROR_LEN},
|
|
MaxRequestBlocks, MAX_REQUEST_BLOCKS,
|
|
};
|
|
use futures::future::BoxFuture;
|
|
use futures::prelude::{AsyncRead, AsyncWrite};
|
|
use futures::{FutureExt, StreamExt};
|
|
use libp2p::core::{InboundUpgrade, ProtocolName, UpgradeInfo};
|
|
use ssz::Encode;
|
|
use ssz_types::VariableList;
|
|
use std::io;
|
|
use std::marker::PhantomData;
|
|
use std::sync::Arc;
|
|
use std::time::Duration;
|
|
use strum::{AsStaticRef, AsStaticStr};
|
|
use tokio_io_timeout::TimeoutStream;
|
|
use tokio_util::{
|
|
codec::Framed,
|
|
compat::{Compat, FuturesAsyncReadCompatExt},
|
|
};
|
|
use types::{
|
|
BeaconBlock, BeaconBlockAltair, BeaconBlockBase, EthSpec, ForkContext, Hash256, MainnetEthSpec,
|
|
Signature, SignedBeaconBlock,
|
|
};
|
|
|
|
lazy_static! {
|
|
// Note: Hardcoding the `EthSpec` type for `SignedBeaconBlock` as min/max values is
|
|
// same across different `EthSpec` implementations.
|
|
pub static ref SIGNED_BEACON_BLOCK_BASE_MIN: usize = SignedBeaconBlock::<MainnetEthSpec>::from_block(
|
|
BeaconBlock::Base(BeaconBlockBase::<MainnetEthSpec>::empty(&MainnetEthSpec::default_spec())),
|
|
Signature::empty(),
|
|
)
|
|
.as_ssz_bytes()
|
|
.len();
|
|
pub static ref SIGNED_BEACON_BLOCK_BASE_MAX: usize = SignedBeaconBlock::<MainnetEthSpec>::from_block(
|
|
BeaconBlock::Base(BeaconBlockBase::full(&MainnetEthSpec::default_spec())),
|
|
Signature::empty(),
|
|
)
|
|
.as_ssz_bytes()
|
|
.len();
|
|
|
|
pub static ref SIGNED_BEACON_BLOCK_ALTAIR_MIN: usize = SignedBeaconBlock::<MainnetEthSpec>::from_block(
|
|
BeaconBlock::Altair(BeaconBlockAltair::<MainnetEthSpec>::empty(&MainnetEthSpec::default_spec())),
|
|
Signature::empty(),
|
|
)
|
|
.as_ssz_bytes()
|
|
.len();
|
|
pub static ref SIGNED_BEACON_BLOCK_ALTAIR_MAX: usize = SignedBeaconBlock::<MainnetEthSpec>::from_block(
|
|
BeaconBlock::Altair(BeaconBlockAltair::full(&MainnetEthSpec::default_spec())),
|
|
Signature::empty(),
|
|
)
|
|
.as_ssz_bytes()
|
|
.len();
|
|
pub static ref BLOCKS_BY_ROOT_REQUEST_MIN: usize =
|
|
VariableList::<Hash256, MaxRequestBlocks>::from(Vec::<Hash256>::new())
|
|
.as_ssz_bytes()
|
|
.len();
|
|
pub static ref BLOCKS_BY_ROOT_REQUEST_MAX: usize =
|
|
VariableList::<Hash256, MaxRequestBlocks>::from(vec![
|
|
Hash256::zero();
|
|
MAX_REQUEST_BLOCKS
|
|
as usize
|
|
])
|
|
.as_ssz_bytes()
|
|
.len();
|
|
pub static ref ERROR_TYPE_MIN: usize =
|
|
VariableList::<u8, MaxErrorLen>::from(Vec::<u8>::new())
|
|
.as_ssz_bytes()
|
|
.len();
|
|
pub static ref ERROR_TYPE_MAX: usize =
|
|
VariableList::<u8, MaxErrorLen>::from(vec![
|
|
0u8;
|
|
MAX_ERROR_LEN
|
|
as usize
|
|
])
|
|
.as_ssz_bytes()
|
|
.len();
|
|
|
|
}
|
|
|
|
/// The maximum bytes that can be sent across the RPC.
|
|
const MAX_RPC_SIZE: usize = 1_048_576; // 1M
|
|
/// The protocol prefix the RPC protocol id.
|
|
const PROTOCOL_PREFIX: &str = "/eth2/beacon_chain/req";
|
|
/// Time allowed for the first byte of a request to arrive before we time out (Time To First Byte).
|
|
const TTFB_TIMEOUT: u64 = 5;
|
|
/// The number of seconds to wait for the first bytes of a request once a protocol has been
|
|
/// established before the stream is terminated.
|
|
const REQUEST_TIMEOUT: u64 = 15;
|
|
|
|
/// Protocol names to be used.
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
|
pub enum Protocol {
|
|
/// The Status protocol name.
|
|
Status,
|
|
/// The Goodbye protocol name.
|
|
Goodbye,
|
|
/// The `BlocksByRange` protocol name.
|
|
BlocksByRange,
|
|
/// The `BlocksByRoot` protocol name.
|
|
BlocksByRoot,
|
|
/// The `Ping` protocol name.
|
|
Ping,
|
|
/// The `MetaData` protocol name.
|
|
MetaData,
|
|
}
|
|
|
|
/// RPC Versions
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
pub enum Version {
|
|
/// Version 1 of RPC
|
|
V1,
|
|
/// Version 2 of RPC
|
|
V2,
|
|
}
|
|
|
|
/// RPC Encondings supported.
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
pub enum Encoding {
|
|
SSZSnappy,
|
|
}
|
|
|
|
impl std::fmt::Display for Protocol {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let repr = match self {
|
|
Protocol::Status => "status",
|
|
Protocol::Goodbye => "goodbye",
|
|
Protocol::BlocksByRange => "beacon_blocks_by_range",
|
|
Protocol::BlocksByRoot => "beacon_blocks_by_root",
|
|
Protocol::Ping => "ping",
|
|
Protocol::MetaData => "metadata",
|
|
};
|
|
f.write_str(repr)
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for Encoding {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let repr = match self {
|
|
Encoding::SSZSnappy => "ssz_snappy",
|
|
};
|
|
f.write_str(repr)
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for Version {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let repr = match self {
|
|
Version::V1 => "1",
|
|
Version::V2 => "2",
|
|
};
|
|
f.write_str(repr)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct RPCProtocol<TSpec: EthSpec> {
|
|
pub fork_context: Arc<ForkContext>,
|
|
pub phantom: PhantomData<TSpec>,
|
|
}
|
|
|
|
impl<TSpec: EthSpec> UpgradeInfo for RPCProtocol<TSpec> {
|
|
type Info = ProtocolId;
|
|
type InfoIter = Vec<Self::Info>;
|
|
|
|
/// The list of supported RPC protocols for Lighthouse.
|
|
fn protocol_info(&self) -> Self::InfoIter {
|
|
vec![
|
|
ProtocolId::new(Protocol::Status, Version::V1, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::Goodbye, Version::V1, Encoding::SSZSnappy),
|
|
// V2 variants have higher preference then V1
|
|
ProtocolId::new(Protocol::BlocksByRange, Version::V2, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::BlocksByRange, Version::V1, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::BlocksByRoot, Version::V2, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::BlocksByRoot, Version::V1, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::Ping, Version::V1, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::MetaData, Version::V2, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::MetaData, Version::V1, Encoding::SSZSnappy),
|
|
]
|
|
}
|
|
}
|
|
|
|
/// Represents the ssz length bounds for RPC messages.
|
|
#[derive(Debug, PartialEq)]
|
|
pub struct RpcLimits {
|
|
pub min: usize,
|
|
pub max: usize,
|
|
}
|
|
|
|
impl RpcLimits {
|
|
pub fn new(min: usize, max: usize) -> Self {
|
|
Self { min, max }
|
|
}
|
|
|
|
/// Returns true if the given length is out of bounds, false otherwise.
|
|
pub fn is_out_of_bounds(&self, length: usize) -> bool {
|
|
length > self.max || length < self.min
|
|
}
|
|
}
|
|
|
|
/// Tracks the types in a protocol id.
|
|
#[derive(Clone, Debug)]
|
|
pub struct ProtocolId {
|
|
/// The RPC message type/name.
|
|
pub message_name: Protocol,
|
|
|
|
/// The version of the RPC.
|
|
pub version: Version,
|
|
|
|
/// The encoding of the RPC.
|
|
pub encoding: Encoding,
|
|
|
|
/// The protocol id that is formed from the above fields.
|
|
protocol_id: String,
|
|
}
|
|
|
|
impl ProtocolId {
|
|
/// Returns min and max size for messages of given protocol id requests.
|
|
pub fn rpc_request_limits(&self) -> RpcLimits {
|
|
match self.message_name {
|
|
Protocol::Status => RpcLimits::new(
|
|
<StatusMessage as Encode>::ssz_fixed_len(),
|
|
<StatusMessage as Encode>::ssz_fixed_len(),
|
|
),
|
|
Protocol::Goodbye => RpcLimits::new(
|
|
<GoodbyeReason as Encode>::ssz_fixed_len(),
|
|
<GoodbyeReason as Encode>::ssz_fixed_len(),
|
|
),
|
|
Protocol::BlocksByRange => RpcLimits::new(
|
|
<BlocksByRangeRequest as Encode>::ssz_fixed_len(),
|
|
<BlocksByRangeRequest as Encode>::ssz_fixed_len(),
|
|
),
|
|
Protocol::BlocksByRoot => {
|
|
RpcLimits::new(*BLOCKS_BY_ROOT_REQUEST_MIN, *BLOCKS_BY_ROOT_REQUEST_MAX)
|
|
}
|
|
Protocol::Ping => RpcLimits::new(
|
|
<Ping as Encode>::ssz_fixed_len(),
|
|
<Ping as Encode>::ssz_fixed_len(),
|
|
),
|
|
Protocol::MetaData => RpcLimits::new(0, 0), // Metadata requests are empty
|
|
}
|
|
}
|
|
|
|
/// Returns min and max size for messages of given protocol id responses.
|
|
pub fn rpc_response_limits<T: EthSpec>(&self) -> RpcLimits {
|
|
match self.message_name {
|
|
Protocol::Status => RpcLimits::new(
|
|
<StatusMessage as Encode>::ssz_fixed_len(),
|
|
<StatusMessage as Encode>::ssz_fixed_len(),
|
|
),
|
|
Protocol::Goodbye => RpcLimits::new(0, 0), // Goodbye request has no response
|
|
Protocol::BlocksByRange => RpcLimits::new(
|
|
std::cmp::min(
|
|
*SIGNED_BEACON_BLOCK_ALTAIR_MIN,
|
|
*SIGNED_BEACON_BLOCK_BASE_MIN,
|
|
),
|
|
std::cmp::max(
|
|
*SIGNED_BEACON_BLOCK_ALTAIR_MAX,
|
|
*SIGNED_BEACON_BLOCK_BASE_MAX,
|
|
),
|
|
),
|
|
Protocol::BlocksByRoot => RpcLimits::new(
|
|
std::cmp::min(
|
|
*SIGNED_BEACON_BLOCK_ALTAIR_MIN,
|
|
*SIGNED_BEACON_BLOCK_BASE_MIN,
|
|
),
|
|
std::cmp::max(
|
|
*SIGNED_BEACON_BLOCK_ALTAIR_MAX,
|
|
*SIGNED_BEACON_BLOCK_BASE_MAX,
|
|
),
|
|
),
|
|
|
|
Protocol::Ping => RpcLimits::new(
|
|
<Ping as Encode>::ssz_fixed_len(),
|
|
<Ping as Encode>::ssz_fixed_len(),
|
|
),
|
|
Protocol::MetaData => RpcLimits::new(
|
|
<MetaDataV1<T> as Encode>::ssz_fixed_len(),
|
|
<MetaDataV2<T> as Encode>::ssz_fixed_len(),
|
|
),
|
|
}
|
|
}
|
|
|
|
/// Returns `true` if the given `ProtocolId` should expect `context_bytes` in the
|
|
/// beginning of the stream, else returns `false`.
|
|
pub fn has_context_bytes(&self) -> bool {
|
|
if self.version == Version::V2 {
|
|
match self.message_name {
|
|
Protocol::BlocksByRange | Protocol::BlocksByRoot => return true,
|
|
_ => return false,
|
|
}
|
|
}
|
|
false
|
|
}
|
|
}
|
|
|
|
/// An RPC protocol ID.
|
|
impl ProtocolId {
|
|
pub fn new(message_name: Protocol, version: Version, encoding: Encoding) -> Self {
|
|
let protocol_id = format!(
|
|
"{}/{}/{}/{}",
|
|
PROTOCOL_PREFIX, message_name, version, encoding
|
|
);
|
|
|
|
ProtocolId {
|
|
message_name,
|
|
version,
|
|
encoding,
|
|
protocol_id,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ProtocolName for ProtocolId {
|
|
fn protocol_name(&self) -> &[u8] {
|
|
self.protocol_id.as_bytes()
|
|
}
|
|
}
|
|
|
|
/* Inbound upgrade */
|
|
|
|
// The inbound protocol reads the request, decodes it and returns the stream to the protocol
|
|
// handler to respond to once ready.
|
|
|
|
pub type InboundOutput<TSocket, TSpec> = (InboundRequest<TSpec>, InboundFramed<TSocket, TSpec>);
|
|
pub type InboundFramed<TSocket, TSpec> =
|
|
Framed<std::pin::Pin<Box<TimeoutStream<Compat<TSocket>>>>, InboundCodec<TSpec>>;
|
|
|
|
impl<TSocket, TSpec> InboundUpgrade<TSocket> for RPCProtocol<TSpec>
|
|
where
|
|
TSocket: AsyncRead + AsyncWrite + Unpin + Send + 'static,
|
|
TSpec: EthSpec,
|
|
{
|
|
type Output = InboundOutput<TSocket, TSpec>;
|
|
type Error = RPCError;
|
|
type Future = BoxFuture<'static, Result<Self::Output, Self::Error>>;
|
|
|
|
fn upgrade_inbound(self, socket: TSocket, protocol: ProtocolId) -> Self::Future {
|
|
async move {
|
|
let protocol_name = protocol.message_name;
|
|
// convert the socket to tokio compatible socket
|
|
let socket = socket.compat();
|
|
let codec = match protocol.encoding {
|
|
Encoding::SSZSnappy => {
|
|
let ssz_snappy_codec = BaseInboundCodec::new(SSZSnappyInboundCodec::new(
|
|
protocol,
|
|
MAX_RPC_SIZE,
|
|
self.fork_context.clone(),
|
|
));
|
|
InboundCodec::SSZSnappy(ssz_snappy_codec)
|
|
}
|
|
};
|
|
let mut timed_socket = TimeoutStream::new(socket);
|
|
timed_socket.set_read_timeout(Some(Duration::from_secs(TTFB_TIMEOUT)));
|
|
|
|
let socket = Framed::new(Box::pin(timed_socket), codec);
|
|
|
|
// MetaData requests should be empty, return the stream
|
|
match protocol_name {
|
|
Protocol::MetaData => Ok((InboundRequest::MetaData(PhantomData), socket)),
|
|
_ => {
|
|
match tokio::time::timeout(
|
|
Duration::from_secs(REQUEST_TIMEOUT),
|
|
socket.into_future(),
|
|
)
|
|
.await
|
|
{
|
|
Err(e) => Err(RPCError::from(e)),
|
|
Ok((Some(Ok(request)), stream)) => Ok((request, stream)),
|
|
Ok((Some(Err(e)), _)) => Err(e),
|
|
Ok((None, _)) => Err(RPCError::IncompleteStream),
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.boxed()
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
pub enum InboundRequest<TSpec: EthSpec> {
|
|
Status(StatusMessage),
|
|
Goodbye(GoodbyeReason),
|
|
BlocksByRange(BlocksByRangeRequest),
|
|
BlocksByRoot(BlocksByRootRequest),
|
|
Ping(Ping),
|
|
MetaData(PhantomData<TSpec>),
|
|
}
|
|
|
|
impl<TSpec: EthSpec> UpgradeInfo for InboundRequest<TSpec> {
|
|
type Info = ProtocolId;
|
|
type InfoIter = Vec<Self::Info>;
|
|
|
|
// add further protocols as we support more encodings/versions
|
|
fn protocol_info(&self) -> Self::InfoIter {
|
|
self.supported_protocols()
|
|
}
|
|
}
|
|
|
|
/// Implements the encoding per supported protocol for `RPCRequest`.
|
|
impl<TSpec: EthSpec> InboundRequest<TSpec> {
|
|
pub fn supported_protocols(&self) -> Vec<ProtocolId> {
|
|
match self {
|
|
// add more protocols when versions/encodings are supported
|
|
InboundRequest::Status(_) => vec![ProtocolId::new(
|
|
Protocol::Status,
|
|
Version::V1,
|
|
Encoding::SSZSnappy,
|
|
)],
|
|
InboundRequest::Goodbye(_) => vec![ProtocolId::new(
|
|
Protocol::Goodbye,
|
|
Version::V1,
|
|
Encoding::SSZSnappy,
|
|
)],
|
|
InboundRequest::BlocksByRange(_) => vec![
|
|
// V2 has higher preference when negotiating a stream
|
|
ProtocolId::new(Protocol::BlocksByRange, Version::V2, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::BlocksByRange, Version::V1, Encoding::SSZSnappy),
|
|
],
|
|
InboundRequest::BlocksByRoot(_) => vec![
|
|
// V2 has higher preference when negotiating a stream
|
|
ProtocolId::new(Protocol::BlocksByRoot, Version::V2, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::BlocksByRoot, Version::V1, Encoding::SSZSnappy),
|
|
],
|
|
InboundRequest::Ping(_) => vec![ProtocolId::new(
|
|
Protocol::Ping,
|
|
Version::V1,
|
|
Encoding::SSZSnappy,
|
|
)],
|
|
InboundRequest::MetaData(_) => vec![
|
|
ProtocolId::new(Protocol::MetaData, Version::V2, Encoding::SSZSnappy),
|
|
ProtocolId::new(Protocol::MetaData, Version::V1, Encoding::SSZSnappy),
|
|
],
|
|
}
|
|
}
|
|
|
|
/* These functions are used in the handler for stream management */
|
|
|
|
/// Number of responses expected for this request.
|
|
pub fn expected_responses(&self) -> u64 {
|
|
match self {
|
|
InboundRequest::Status(_) => 1,
|
|
InboundRequest::Goodbye(_) => 0,
|
|
InboundRequest::BlocksByRange(req) => req.count,
|
|
InboundRequest::BlocksByRoot(req) => req.block_roots.len() as u64,
|
|
InboundRequest::Ping(_) => 1,
|
|
InboundRequest::MetaData(_) => 1,
|
|
}
|
|
}
|
|
|
|
/// Gives the corresponding `Protocol` to this request.
|
|
pub fn protocol(&self) -> Protocol {
|
|
match self {
|
|
InboundRequest::Status(_) => Protocol::Status,
|
|
InboundRequest::Goodbye(_) => Protocol::Goodbye,
|
|
InboundRequest::BlocksByRange(_) => Protocol::BlocksByRange,
|
|
InboundRequest::BlocksByRoot(_) => Protocol::BlocksByRoot,
|
|
InboundRequest::Ping(_) => Protocol::Ping,
|
|
InboundRequest::MetaData(_) => Protocol::MetaData,
|
|
}
|
|
}
|
|
|
|
/// Returns the `ResponseTermination` type associated with the request if a stream gets
|
|
/// terminated.
|
|
pub fn stream_termination(&self) -> ResponseTermination {
|
|
match self {
|
|
// this only gets called after `multiple_responses()` returns true. Therefore, only
|
|
// variants that have `multiple_responses()` can have values.
|
|
InboundRequest::BlocksByRange(_) => ResponseTermination::BlocksByRange,
|
|
InboundRequest::BlocksByRoot(_) => ResponseTermination::BlocksByRoot,
|
|
InboundRequest::Status(_) => unreachable!(),
|
|
InboundRequest::Goodbye(_) => unreachable!(),
|
|
InboundRequest::Ping(_) => unreachable!(),
|
|
InboundRequest::MetaData(_) => unreachable!(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Error in RPC Encoding/Decoding.
|
|
#[derive(Debug, Clone, PartialEq, AsStaticStr)]
|
|
#[strum(serialize_all = "snake_case")]
|
|
pub enum RPCError {
|
|
/// Error when decoding the raw buffer from ssz.
|
|
// NOTE: in the future a ssz::DecodeError should map to an InvalidData error
|
|
#[strum(serialize = "decode_error")]
|
|
SSZDecodeError(ssz::DecodeError),
|
|
/// IO Error.
|
|
IoError(String),
|
|
/// The peer returned a valid response but the response indicated an error.
|
|
ErrorResponse(RPCResponseErrorCode, String),
|
|
/// Timed out waiting for a response.
|
|
StreamTimeout,
|
|
/// Peer does not support the protocol.
|
|
UnsupportedProtocol,
|
|
/// Stream ended unexpectedly.
|
|
IncompleteStream,
|
|
/// Peer sent invalid data.
|
|
InvalidData,
|
|
/// An error occurred due to internal reasons. Ex: timer failure.
|
|
InternalError(&'static str),
|
|
/// Negotiation with this peer timed out.
|
|
NegotiationTimeout,
|
|
/// Handler rejected this request.
|
|
HandlerRejected,
|
|
/// We have intentionally disconnected.
|
|
Disconnected,
|
|
}
|
|
|
|
impl From<ssz::DecodeError> for RPCError {
|
|
#[inline]
|
|
fn from(err: ssz::DecodeError) -> Self {
|
|
RPCError::SSZDecodeError(err)
|
|
}
|
|
}
|
|
impl From<tokio::time::error::Elapsed> for RPCError {
|
|
fn from(_: tokio::time::error::Elapsed) -> Self {
|
|
RPCError::StreamTimeout
|
|
}
|
|
}
|
|
|
|
impl From<io::Error> for RPCError {
|
|
fn from(err: io::Error) -> Self {
|
|
RPCError::IoError(err.to_string())
|
|
}
|
|
}
|
|
|
|
// Error trait is required for `ProtocolsHandler`
|
|
impl std::fmt::Display for RPCError {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match *self {
|
|
RPCError::SSZDecodeError(ref err) => write!(f, "Error while decoding ssz: {:?}", err),
|
|
RPCError::InvalidData => write!(f, "Peer sent unexpected data"),
|
|
RPCError::IoError(ref err) => write!(f, "IO Error: {}", err),
|
|
RPCError::ErrorResponse(ref code, ref reason) => write!(
|
|
f,
|
|
"RPC response was an error: {} with reason: {}",
|
|
code, reason
|
|
),
|
|
RPCError::StreamTimeout => write!(f, "Stream Timeout"),
|
|
RPCError::UnsupportedProtocol => write!(f, "Peer does not support the protocol"),
|
|
RPCError::IncompleteStream => write!(f, "Stream ended unexpectedly"),
|
|
RPCError::InternalError(ref err) => write!(f, "Internal error: {}", err),
|
|
RPCError::NegotiationTimeout => write!(f, "Negotiation timeout"),
|
|
RPCError::HandlerRejected => write!(f, "Handler rejected the request"),
|
|
RPCError::Disconnected => write!(f, "Gracefully Disconnected"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::error::Error for RPCError {
|
|
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
|
|
match *self {
|
|
// NOTE: this does have a source
|
|
RPCError::SSZDecodeError(_) => None,
|
|
RPCError::IoError(_) => None,
|
|
RPCError::StreamTimeout => None,
|
|
RPCError::UnsupportedProtocol => None,
|
|
RPCError::IncompleteStream => None,
|
|
RPCError::InvalidData => None,
|
|
RPCError::InternalError(_) => None,
|
|
RPCError::ErrorResponse(_, _) => None,
|
|
RPCError::NegotiationTimeout => None,
|
|
RPCError::HandlerRejected => None,
|
|
RPCError::Disconnected => None,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<TSpec: EthSpec> std::fmt::Display for InboundRequest<TSpec> {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
InboundRequest::Status(status) => write!(f, "Status Message: {}", status),
|
|
InboundRequest::Goodbye(reason) => write!(f, "Goodbye: {}", reason),
|
|
InboundRequest::BlocksByRange(req) => write!(f, "Blocks by range: {}", req),
|
|
InboundRequest::BlocksByRoot(req) => write!(f, "Blocks by root: {:?}", req),
|
|
InboundRequest::Ping(ping) => write!(f, "Ping: {}", ping.data),
|
|
InboundRequest::MetaData(_) => write!(f, "MetaData request"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl RPCError {
|
|
/// Get a `str` representation of the error.
|
|
/// Used for metrics.
|
|
pub fn as_static_str(&self) -> &'static str {
|
|
match self {
|
|
RPCError::ErrorResponse(ref code, ..) => code.as_static(),
|
|
e => e.as_static(),
|
|
}
|
|
}
|
|
}
|