mirror of
https://github.com/sigp/lighthouse.git
synced 2026-03-03 00:31:50 +00:00
## Proposed Changes Implement the consensus changes necessary for the upcoming Altair hard fork. ## Additional Info This is quite a heavy refactor, with pivotal types like the `BeaconState` and `BeaconBlock` changing from structs to enums. This ripples through the whole codebase with field accesses changing to methods, e.g. `state.slot` => `state.slot()`. Co-authored-by: realbigsean <seananderson33@gmail.com>
433 lines
14 KiB
Rust
433 lines
14 KiB
Rust
//! Available RPC methods types and ids.
|
|
|
|
use crate::types::EnrBitfield;
|
|
use regex::bytes::Regex;
|
|
use serde::Serialize;
|
|
use ssz_derive::{Decode, Encode};
|
|
use ssz_types::{
|
|
typenum::{U1024, U256},
|
|
VariableList,
|
|
};
|
|
use std::ops::Deref;
|
|
use strum::AsStaticStr;
|
|
use types::{Epoch, EthSpec, Hash256, SignedBeaconBlock, Slot};
|
|
|
|
/// Maximum number of blocks in a single request.
|
|
pub type MaxRequestBlocks = U1024;
|
|
pub const MAX_REQUEST_BLOCKS: u64 = 1024;
|
|
|
|
/// Maximum length of error message.
|
|
pub type MaxErrorLen = U256;
|
|
pub const MAX_ERROR_LEN: u64 = 256;
|
|
|
|
/// Wrapper over SSZ List to represent error message in rpc responses.
|
|
#[derive(Debug, Clone)]
|
|
pub struct ErrorType(pub VariableList<u8, MaxErrorLen>);
|
|
|
|
impl From<String> for ErrorType {
|
|
fn from(s: String) -> Self {
|
|
Self(VariableList::from(s.as_bytes().to_vec()))
|
|
}
|
|
}
|
|
|
|
impl From<&str> for ErrorType {
|
|
fn from(s: &str) -> Self {
|
|
Self(VariableList::from(s.as_bytes().to_vec()))
|
|
}
|
|
}
|
|
|
|
impl Deref for ErrorType {
|
|
type Target = VariableList<u8, MaxErrorLen>;
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
impl ToString for ErrorType {
|
|
fn to_string(&self) -> String {
|
|
#[allow(clippy::invalid_regex)]
|
|
let re = Regex::new("\\p{C}").expect("Regex is valid");
|
|
String::from_utf8_lossy(&re.replace_all(self.0.deref(), &b""[..])).to_string()
|
|
}
|
|
}
|
|
|
|
/* Request/Response data structures for RPC methods */
|
|
|
|
/* Requests */
|
|
|
|
/// Identifier of a request.
|
|
///
|
|
// NOTE: The handler stores the `RequestId` to inform back of responses and errors, but it's execution
|
|
// is independent of the contents on this type.
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
|
pub enum RequestId {
|
|
Router,
|
|
Sync(usize),
|
|
Behaviour,
|
|
}
|
|
|
|
/// The STATUS request/response handshake message.
|
|
#[derive(Encode, Decode, Clone, Debug, PartialEq)]
|
|
pub struct StatusMessage {
|
|
/// The fork version of the chain we are broadcasting.
|
|
pub fork_digest: [u8; 4],
|
|
|
|
/// Latest finalized root.
|
|
pub finalized_root: Hash256,
|
|
|
|
/// Latest finalized epoch.
|
|
pub finalized_epoch: Epoch,
|
|
|
|
/// The latest block root.
|
|
pub head_root: Hash256,
|
|
|
|
/// The slot associated with the latest block root.
|
|
pub head_slot: Slot,
|
|
}
|
|
|
|
/// The PING request/response message.
|
|
#[derive(Encode, Decode, Clone, Debug, PartialEq)]
|
|
pub struct Ping {
|
|
/// The metadata sequence number.
|
|
pub data: u64,
|
|
}
|
|
|
|
/// The METADATA response structure.
|
|
#[derive(Encode, Decode, Clone, Debug, PartialEq, Serialize)]
|
|
#[serde(bound = "T: EthSpec")]
|
|
pub struct MetaData<T: EthSpec> {
|
|
/// A sequential counter indicating when data gets modified.
|
|
pub seq_number: u64,
|
|
/// The persistent subnet bitfield.
|
|
pub attnets: EnrBitfield<T>,
|
|
}
|
|
|
|
/// The reason given for a `Goodbye` message.
|
|
///
|
|
/// Note: any unknown `u64::into(n)` will resolve to `Goodbye::Unknown` for any unknown `n`,
|
|
/// however `GoodbyeReason::Unknown.into()` will go into `0_u64`. Therefore de-serializing then
|
|
/// re-serializing may not return the same bytes.
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
pub enum GoodbyeReason {
|
|
/// This node has shutdown.
|
|
ClientShutdown = 1,
|
|
|
|
/// Incompatible networks.
|
|
IrrelevantNetwork = 2,
|
|
|
|
/// Error/fault in the RPC.
|
|
Fault = 3,
|
|
|
|
/// Teku uses this code for not being able to verify a network.
|
|
UnableToVerifyNetwork = 128,
|
|
|
|
/// The node has too many connected peers.
|
|
TooManyPeers = 129,
|
|
|
|
/// Scored poorly.
|
|
BadScore = 250,
|
|
|
|
/// The peer is banned
|
|
Banned = 251,
|
|
|
|
/// Unknown reason.
|
|
Unknown = 0,
|
|
}
|
|
|
|
impl From<u64> for GoodbyeReason {
|
|
fn from(id: u64) -> GoodbyeReason {
|
|
match id {
|
|
1 => GoodbyeReason::ClientShutdown,
|
|
2 => GoodbyeReason::IrrelevantNetwork,
|
|
3 => GoodbyeReason::Fault,
|
|
128 => GoodbyeReason::UnableToVerifyNetwork,
|
|
129 => GoodbyeReason::TooManyPeers,
|
|
250 => GoodbyeReason::BadScore,
|
|
251 => GoodbyeReason::Banned,
|
|
_ => GoodbyeReason::Unknown,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Into<u64> for GoodbyeReason {
|
|
fn into(self) -> u64 {
|
|
self as u64
|
|
}
|
|
}
|
|
|
|
impl ssz::Encode for GoodbyeReason {
|
|
fn is_ssz_fixed_len() -> bool {
|
|
<u64 as ssz::Encode>::is_ssz_fixed_len()
|
|
}
|
|
|
|
fn ssz_fixed_len() -> usize {
|
|
<u64 as ssz::Encode>::ssz_fixed_len()
|
|
}
|
|
|
|
fn ssz_bytes_len(&self) -> usize {
|
|
0_u64.ssz_bytes_len()
|
|
}
|
|
|
|
fn ssz_append(&self, buf: &mut Vec<u8>) {
|
|
let conv: u64 = self.clone().into();
|
|
conv.ssz_append(buf)
|
|
}
|
|
}
|
|
|
|
impl ssz::Decode for GoodbyeReason {
|
|
fn is_ssz_fixed_len() -> bool {
|
|
<u64 as ssz::Decode>::is_ssz_fixed_len()
|
|
}
|
|
|
|
fn ssz_fixed_len() -> usize {
|
|
<u64 as ssz::Decode>::ssz_fixed_len()
|
|
}
|
|
|
|
fn from_ssz_bytes(bytes: &[u8]) -> Result<Self, ssz::DecodeError> {
|
|
u64::from_ssz_bytes(bytes).map(|n| n.into())
|
|
}
|
|
}
|
|
|
|
/// Request a number of beacon block roots from a peer.
|
|
#[derive(Encode, Decode, Clone, Debug, PartialEq)]
|
|
pub struct BlocksByRangeRequest {
|
|
/// The starting slot to request blocks.
|
|
pub start_slot: u64,
|
|
|
|
/// The number of blocks from the start slot.
|
|
pub count: u64,
|
|
|
|
/// The step increment to receive blocks.
|
|
///
|
|
/// A value of 1 returns every block.
|
|
/// A value of 2 returns every second block.
|
|
/// A value of 3 returns every third block and so on.
|
|
pub step: u64,
|
|
}
|
|
|
|
/// Request a number of beacon block bodies from a peer.
|
|
#[derive(Clone, Debug, PartialEq)]
|
|
pub struct BlocksByRootRequest {
|
|
/// The list of beacon block bodies being requested.
|
|
pub block_roots: VariableList<Hash256, MaxRequestBlocks>,
|
|
}
|
|
|
|
/* RPC Handling and Grouping */
|
|
// Collection of enums and structs used by the Codecs to encode/decode RPC messages
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
pub enum RPCResponse<T: EthSpec> {
|
|
/// A HELLO message.
|
|
Status(StatusMessage),
|
|
|
|
/// A response to a get BLOCKS_BY_RANGE request. A None response signifies the end of the
|
|
/// batch.
|
|
BlocksByRange(Box<SignedBeaconBlock<T>>),
|
|
|
|
/// A response to a get BLOCKS_BY_ROOT request.
|
|
BlocksByRoot(Box<SignedBeaconBlock<T>>),
|
|
|
|
/// A PONG response to a PING request.
|
|
Pong(Ping),
|
|
|
|
/// A response to a META_DATA request.
|
|
MetaData(MetaData<T>),
|
|
}
|
|
|
|
/// Indicates which response is being terminated by a stream termination response.
|
|
#[derive(Debug, Clone)]
|
|
pub enum ResponseTermination {
|
|
/// Blocks by range stream termination.
|
|
BlocksByRange,
|
|
|
|
/// Blocks by root stream termination.
|
|
BlocksByRoot,
|
|
}
|
|
|
|
/// The structured response containing a result/code indicating success or failure
|
|
/// and the contents of the response
|
|
#[derive(Debug, Clone)]
|
|
pub enum RPCCodedResponse<T: EthSpec> {
|
|
/// The response is a successful.
|
|
Success(RPCResponse<T>),
|
|
|
|
Error(RPCResponseErrorCode, ErrorType),
|
|
|
|
/// Received a stream termination indicating which response is being terminated.
|
|
StreamTermination(ResponseTermination),
|
|
}
|
|
|
|
/// The code assigned to an erroneous `RPCResponse`.
|
|
#[derive(Debug, Clone, Copy, PartialEq, AsStaticStr)]
|
|
#[strum(serialize_all = "snake_case")]
|
|
pub enum RPCResponseErrorCode {
|
|
RateLimited,
|
|
InvalidRequest,
|
|
ServerError,
|
|
/// Error spec'd to indicate that a peer does not have blocks on a requested range.
|
|
ResourceUnavailable,
|
|
Unknown,
|
|
}
|
|
|
|
impl<T: EthSpec> RPCCodedResponse<T> {
|
|
/// Used to encode the response in the codec.
|
|
pub fn as_u8(&self) -> Option<u8> {
|
|
match self {
|
|
RPCCodedResponse::Success(_) => Some(0),
|
|
RPCCodedResponse::Error(code, _) => Some(code.as_u8()),
|
|
RPCCodedResponse::StreamTermination(_) => None,
|
|
}
|
|
}
|
|
|
|
/// Tells the codec whether to decode as an RPCResponse or an error.
|
|
pub fn is_response(response_code: u8) -> bool {
|
|
matches!(response_code, 0)
|
|
}
|
|
|
|
/// Builds an RPCCodedResponse from a response code and an ErrorMessage
|
|
pub fn from_error(response_code: u8, err: ErrorType) -> Self {
|
|
let code = match response_code {
|
|
1 => RPCResponseErrorCode::InvalidRequest,
|
|
2 => RPCResponseErrorCode::ServerError,
|
|
3 => RPCResponseErrorCode::ResourceUnavailable,
|
|
139 => RPCResponseErrorCode::RateLimited,
|
|
_ => RPCResponseErrorCode::Unknown,
|
|
};
|
|
RPCCodedResponse::Error(code, err)
|
|
}
|
|
|
|
/// Specifies which response allows for multiple chunks for the stream handler.
|
|
pub fn multiple_responses(&self) -> bool {
|
|
match self {
|
|
RPCCodedResponse::Success(resp) => match resp {
|
|
RPCResponse::Status(_) => false,
|
|
RPCResponse::BlocksByRange(_) => true,
|
|
RPCResponse::BlocksByRoot(_) => true,
|
|
RPCResponse::Pong(_) => false,
|
|
RPCResponse::MetaData(_) => false,
|
|
},
|
|
RPCCodedResponse::Error(_, _) => true,
|
|
// Stream terminations are part of responses that have chunks
|
|
RPCCodedResponse::StreamTermination(_) => true,
|
|
}
|
|
}
|
|
|
|
/// Returns true if this response always terminates the stream.
|
|
pub fn close_after(&self) -> bool {
|
|
!matches!(self, RPCCodedResponse::Success(_))
|
|
}
|
|
}
|
|
|
|
impl RPCResponseErrorCode {
|
|
fn as_u8(&self) -> u8 {
|
|
match self {
|
|
RPCResponseErrorCode::InvalidRequest => 1,
|
|
RPCResponseErrorCode::ServerError => 2,
|
|
RPCResponseErrorCode::ResourceUnavailable => 3,
|
|
RPCResponseErrorCode::Unknown => 255,
|
|
RPCResponseErrorCode::RateLimited => 139,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for RPCResponseErrorCode {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let repr = match self {
|
|
RPCResponseErrorCode::InvalidRequest => "The request was invalid",
|
|
RPCResponseErrorCode::ResourceUnavailable => "Resource unavailable",
|
|
RPCResponseErrorCode::ServerError => "Server error occurred",
|
|
RPCResponseErrorCode::Unknown => "Unknown error occurred",
|
|
RPCResponseErrorCode::RateLimited => "Rate limited",
|
|
};
|
|
f.write_str(repr)
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for StatusMessage {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
write!(f, "Status Message: Fork Digest: {:?}, Finalized Root: {}, Finalized Epoch: {}, Head Root: {}, Head Slot: {}", self.fork_digest, self.finalized_root, self.finalized_epoch, self.head_root, self.head_slot)
|
|
}
|
|
}
|
|
|
|
impl<T: EthSpec> std::fmt::Display for RPCResponse<T> {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
RPCResponse::Status(status) => write!(f, "{}", status),
|
|
RPCResponse::BlocksByRange(block) => {
|
|
write!(f, "BlocksByRange: Block slot: {}", block.slot())
|
|
}
|
|
RPCResponse::BlocksByRoot(block) => {
|
|
write!(f, "BlocksByRoot: Block slot: {}", block.slot())
|
|
}
|
|
RPCResponse::Pong(ping) => write!(f, "Pong: {}", ping.data),
|
|
RPCResponse::MetaData(metadata) => write!(f, "Metadata: {}", metadata.seq_number),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T: EthSpec> std::fmt::Display for RPCCodedResponse<T> {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
RPCCodedResponse::Success(res) => write!(f, "{}", res),
|
|
RPCCodedResponse::Error(code, err) => write!(f, "{}: {}", code, err.to_string()),
|
|
RPCCodedResponse::StreamTermination(_) => write!(f, "Stream Termination"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for GoodbyeReason {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self {
|
|
GoodbyeReason::ClientShutdown => write!(f, "Client Shutdown"),
|
|
GoodbyeReason::IrrelevantNetwork => write!(f, "Irrelevant Network"),
|
|
GoodbyeReason::Fault => write!(f, "Fault"),
|
|
GoodbyeReason::UnableToVerifyNetwork => write!(f, "Unable to verify network"),
|
|
GoodbyeReason::TooManyPeers => write!(f, "Too many peers"),
|
|
GoodbyeReason::BadScore => write!(f, "Bad Score"),
|
|
GoodbyeReason::Banned => write!(f, "Banned"),
|
|
GoodbyeReason::Unknown => write!(f, "Unknown Reason"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Display for BlocksByRangeRequest {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
write!(
|
|
f,
|
|
"Start Slot: {}, Count: {}, Step: {}",
|
|
self.start_slot, self.count, self.step
|
|
)
|
|
}
|
|
}
|
|
|
|
impl slog::KV for StatusMessage {
|
|
fn serialize(
|
|
&self,
|
|
record: &slog::Record,
|
|
serializer: &mut dyn slog::Serializer,
|
|
) -> slog::Result {
|
|
use slog::Value;
|
|
serializer.emit_arguments("fork_digest", &format_args!("{:?}", self.fork_digest))?;
|
|
Value::serialize(&self.finalized_epoch, record, "finalized_epoch", serializer)?;
|
|
serializer.emit_arguments("finalized_root", &format_args!("{}", self.finalized_root))?;
|
|
Value::serialize(&self.head_slot, record, "head_slot", serializer)?;
|
|
serializer.emit_arguments("head_root", &format_args!("{}", self.head_root))?;
|
|
slog::Result::Ok(())
|
|
}
|
|
}
|
|
|
|
impl slog::Value for RequestId {
|
|
fn serialize(
|
|
&self,
|
|
record: &slog::Record,
|
|
key: slog::Key,
|
|
serializer: &mut dyn slog::Serializer,
|
|
) -> slog::Result {
|
|
match self {
|
|
RequestId::Behaviour => slog::Value::serialize("Behaviour", record, key, serializer),
|
|
RequestId::Router => slog::Value::serialize("Router", record, key, serializer),
|
|
RequestId::Sync(ref id) => slog::Value::serialize(id, record, key, serializer),
|
|
}
|
|
}
|
|
}
|