2024-06-24 01:30:47 +00:00
|
|
|
use cuprate_pruning::{PruningError, PruningSeed};
|
|
|
|
use cuprate_wire::{CoreSyncData, PeerListEntryBase};
|
2023-12-08 15:03:01 +00:00
|
|
|
|
|
|
|
use crate::{
|
|
|
|
client::InternalPeerID, handles::ConnectionHandle, NetZoneAddress, NetworkAddressIncorrectZone,
|
|
|
|
NetworkZone,
|
|
|
|
};
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A request to the service that keeps track of peers sync states.
|
2024-05-02 22:58:22 +00:00
|
|
|
pub enum PeerSyncRequest<N: NetworkZone> {
|
|
|
|
/// Request some peers to sync from.
|
|
|
|
///
|
|
|
|
/// This takes in the current cumulative difficulty of our chain and will return peers that
|
|
|
|
/// claim to have a higher cumulative difficulty.
|
|
|
|
PeersToSyncFrom {
|
|
|
|
current_cumulative_difficulty: u128,
|
2024-08-06 23:48:53 +00:00
|
|
|
block_needed: Option<usize>,
|
2024-05-02 22:58:22 +00:00
|
|
|
},
|
2024-07-04 20:05:22 +00:00
|
|
|
/// Add/update a peer's core sync data.
|
2024-05-02 22:58:22 +00:00
|
|
|
IncomingCoreSyncData(InternalPeerID<N::Addr>, ConnectionHandle, CoreSyncData),
|
2023-12-08 15:03:01 +00:00
|
|
|
}
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A response from the service that keeps track of peers sync states.
|
2024-05-02 22:58:22 +00:00
|
|
|
pub enum PeerSyncResponse<N: NetworkZone> {
|
|
|
|
/// The return value of [`PeerSyncRequest::PeersToSyncFrom`].
|
|
|
|
PeersToSyncFrom(Vec<InternalPeerID<N::Addr>>),
|
|
|
|
/// A generic ok response.
|
2023-12-08 15:03:01 +00:00
|
|
|
Ok,
|
|
|
|
}
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A request to the core sync service for our node's [`CoreSyncData`].
|
2024-05-02 22:58:22 +00:00
|
|
|
pub struct CoreSyncDataRequest;
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A response from the core sync service containing our [`CoreSyncData`].
|
2024-05-02 22:58:22 +00:00
|
|
|
pub struct CoreSyncDataResponse(pub CoreSyncData);
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A [`NetworkZone`] specific [`PeerListEntryBase`].
|
|
|
|
///
|
|
|
|
/// Using this type instead of [`PeerListEntryBase`] in the address book makes
|
|
|
|
/// usage easier for the rest of the P2P code as we can guarantee only the correct addresses will be stored and returned.
|
2023-12-08 15:03:01 +00:00
|
|
|
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
|
|
|
|
#[cfg_attr(
|
|
|
|
feature = "borsh",
|
|
|
|
derive(borsh::BorshSerialize, borsh::BorshDeserialize)
|
|
|
|
)]
|
|
|
|
pub struct ZoneSpecificPeerListEntryBase<A: NetZoneAddress> {
|
|
|
|
pub adr: A,
|
|
|
|
pub id: u64,
|
|
|
|
pub last_seen: i64,
|
|
|
|
pub pruning_seed: PruningSeed,
|
|
|
|
pub rpc_port: u16,
|
|
|
|
pub rpc_credits_per_hash: u32,
|
|
|
|
}
|
|
|
|
|
2024-06-24 01:30:47 +00:00
|
|
|
impl<A: NetZoneAddress> From<ZoneSpecificPeerListEntryBase<A>> for cuprate_wire::PeerListEntryBase {
|
2023-12-08 15:03:01 +00:00
|
|
|
fn from(value: ZoneSpecificPeerListEntryBase<A>) -> Self {
|
|
|
|
Self {
|
|
|
|
adr: value.adr.into(),
|
|
|
|
id: value.id,
|
|
|
|
last_seen: value.last_seen,
|
2024-03-15 22:11:27 +00:00
|
|
|
pruning_seed: value.pruning_seed.compress(),
|
2023-12-08 15:03:01 +00:00
|
|
|
rpc_port: value.rpc_port,
|
|
|
|
rpc_credits_per_hash: value.rpc_credits_per_hash,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// An error converting a [`PeerListEntryBase`] into a [`ZoneSpecificPeerListEntryBase`].
|
2023-12-08 15:03:01 +00:00
|
|
|
#[derive(Debug, thiserror::Error)]
|
|
|
|
pub enum PeerListConversionError {
|
|
|
|
#[error("Address is in incorrect zone")]
|
|
|
|
Address(#[from] NetworkAddressIncorrectZone),
|
|
|
|
#[error("Pruning seed error: {0}")]
|
|
|
|
PruningSeed(#[from] PruningError),
|
|
|
|
}
|
|
|
|
|
2024-06-24 01:30:47 +00:00
|
|
|
impl<A: NetZoneAddress> TryFrom<cuprate_wire::PeerListEntryBase>
|
2023-12-08 15:03:01 +00:00
|
|
|
for ZoneSpecificPeerListEntryBase<A>
|
|
|
|
{
|
|
|
|
type Error = PeerListConversionError;
|
|
|
|
|
|
|
|
fn try_from(value: PeerListEntryBase) -> Result<Self, Self::Error> {
|
|
|
|
Ok(Self {
|
|
|
|
adr: value.adr.try_into()?,
|
|
|
|
id: value.id,
|
|
|
|
last_seen: value.last_seen,
|
2024-03-15 22:11:27 +00:00
|
|
|
pruning_seed: PruningSeed::decompress_p2p_rules(value.pruning_seed)?,
|
2023-12-08 15:03:01 +00:00
|
|
|
rpc_port: value.rpc_port,
|
|
|
|
rpc_credits_per_hash: value.rpc_credits_per_hash,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A request to the address book service.
|
2023-12-08 15:03:01 +00:00
|
|
|
pub enum AddressBookRequest<Z: NetworkZone> {
|
2024-03-20 20:58:12 +00:00
|
|
|
/// Tells the address book that we have connected or received a connection from a peer.
|
2023-12-08 15:03:01 +00:00
|
|
|
NewConnection {
|
2024-03-20 20:58:12 +00:00
|
|
|
/// The [`InternalPeerID`] of this connection.
|
2023-12-08 15:03:01 +00:00
|
|
|
internal_peer_id: InternalPeerID<Z::Addr>,
|
2024-03-20 20:58:12 +00:00
|
|
|
/// The public address of the peer, if this peer has a reachable public address.
|
|
|
|
public_address: Option<Z::Addr>,
|
|
|
|
/// The [`ConnectionHandle`] to this peer.
|
2023-12-08 15:03:01 +00:00
|
|
|
handle: ConnectionHandle,
|
2024-03-20 20:58:12 +00:00
|
|
|
/// An ID the peer assigned itself.
|
2023-12-08 15:03:01 +00:00
|
|
|
id: u64,
|
2024-03-20 20:58:12 +00:00
|
|
|
/// The peers [`PruningSeed`].
|
2023-12-08 15:03:01 +00:00
|
|
|
pruning_seed: PruningSeed,
|
2024-03-20 20:58:12 +00:00
|
|
|
/// The peers rpc port.
|
2023-12-08 15:03:01 +00:00
|
|
|
rpc_port: u16,
|
|
|
|
/// The peers rpc credits per hash
|
|
|
|
rpc_credits_per_hash: u32,
|
|
|
|
},
|
2024-03-20 20:58:12 +00:00
|
|
|
/// Tells the address book about a peer list received from a peer.
|
2023-12-08 15:03:01 +00:00
|
|
|
IncomingPeerList(Vec<ZoneSpecificPeerListEntryBase<Z::Addr>>),
|
2024-03-20 20:58:12 +00:00
|
|
|
/// Takes a random white peer from the peer list. If height is specified
|
2023-12-08 15:03:01 +00:00
|
|
|
/// then the peer list should retrieve a peer that should have a full
|
|
|
|
/// block at that height according to it's pruning seed
|
2024-08-06 23:48:53 +00:00
|
|
|
TakeRandomWhitePeer { height: Option<usize> },
|
2024-03-20 20:58:12 +00:00
|
|
|
/// Takes a random gray peer from the peer list. If height is specified
|
2023-12-08 15:03:01 +00:00
|
|
|
/// then the peer list should retrieve a peer that should have a full
|
|
|
|
/// block at that height according to it's pruning seed
|
2024-08-06 23:48:53 +00:00
|
|
|
TakeRandomGrayPeer { height: Option<usize> },
|
2024-03-20 20:58:12 +00:00
|
|
|
/// Takes a random peer from the peer list. If height is specified
|
|
|
|
/// then the peer list should retrieve a peer that should have a full
|
|
|
|
/// block at that height according to it's pruning seed.
|
|
|
|
///
|
|
|
|
/// The address book will look in the white peer list first, then the gray
|
|
|
|
/// one if no peer is found.
|
2024-08-06 23:48:53 +00:00
|
|
|
TakeRandomPeer { height: Option<usize> },
|
2024-03-20 20:58:12 +00:00
|
|
|
/// Gets the specified number of white peers, or less if we don't have enough.
|
2023-12-08 15:03:01 +00:00
|
|
|
GetWhitePeers(usize),
|
2024-06-04 17:19:25 +00:00
|
|
|
/// Checks if the given peer is banned.
|
|
|
|
IsPeerBanned(Z::Addr),
|
2023-12-08 15:03:01 +00:00
|
|
|
}
|
|
|
|
|
2024-07-04 20:05:22 +00:00
|
|
|
/// A response from the address book service.
|
2023-12-08 15:03:01 +00:00
|
|
|
pub enum AddressBookResponse<Z: NetworkZone> {
|
|
|
|
Ok,
|
|
|
|
Peer(ZoneSpecificPeerListEntryBase<Z::Addr>),
|
|
|
|
Peers(Vec<ZoneSpecificPeerListEntryBase<Z::Addr>>),
|
2024-06-04 17:19:25 +00:00
|
|
|
/// Contains `true` if the peer is banned.
|
|
|
|
IsPeerBanned(bool),
|
2023-12-08 15:03:01 +00:00
|
|
|
}
|