Skip to main content

reth_rpc/
admin.rs

1use std::sync::Arc;
2
3use alloy_genesis::ChainConfig;
4use alloy_rpc_types_admin::{
5    EthInfo, EthPeerInfo, EthProtocolInfo, NodeInfo, PeerInfo, PeerNetworkInfo, PeerProtocolInfo,
6    Ports, ProtocolInfo,
7};
8use async_trait::async_trait;
9use jsonrpsee::core::RpcResult;
10use reth_chainspec::{EthChainSpec, EthereumHardfork, EthereumHardforks, ForkCondition};
11use reth_network_api::{NetworkInfo, Peers};
12use reth_network_peers::{AnyNode, NodeRecord};
13use reth_network_types::PeerKind;
14use reth_rpc_api::AdminApiServer;
15use reth_rpc_server_types::ToRpcResult;
16use reth_transaction_pool::TransactionPool;
17use revm_primitives::keccak256;
18
19/// `admin` API implementation.
20///
21/// This type provides the functionality for handling `admin` related requests.
22pub struct AdminApi<N, ChainSpec, Pool> {
23    /// An interface to interact with the network
24    network: N,
25    /// The specification of the blockchain's configuration.
26    chain_spec: Arc<ChainSpec>,
27    /// The transaction pool
28    pool: Pool,
29}
30
31impl<N, ChainSpec, Pool> AdminApi<N, ChainSpec, Pool> {
32    /// Creates a new instance of `AdminApi`.
33    pub const fn new(network: N, chain_spec: Arc<ChainSpec>, pool: Pool) -> Self {
34        Self { network, chain_spec, pool }
35    }
36}
37
38#[async_trait]
39impl<N, ChainSpec, Pool> AdminApiServer for AdminApi<N, ChainSpec, Pool>
40where
41    N: NetworkInfo + Peers + 'static,
42    ChainSpec: EthChainSpec + EthereumHardforks + Send + Sync + 'static,
43    Pool: TransactionPool + 'static,
44{
45    /// Handler for `admin_addPeer`
46    fn add_peer(&self, record: NodeRecord) -> RpcResult<bool> {
47        self.network.add_peer_with_udp(record.id, record.tcp_addr(), record.udp_addr());
48        Ok(true)
49    }
50
51    /// Handler for `admin_removePeer`
52    fn remove_peer(&self, record: AnyNode) -> RpcResult<bool> {
53        self.network.remove_peer(record.peer_id(), PeerKind::Basic);
54        Ok(true)
55    }
56
57    /// Handler for `admin_addTrustedPeer`
58    fn add_trusted_peer(&self, record: AnyNode) -> RpcResult<bool> {
59        if let Some(trusted) = record.trusted_peer().cloned() {
60            self.network.add_trusted_peer_node(trusted);
61        } else {
62            if let Some(record) = record.node_record() {
63                self.network.add_trusted_peer_with_udp(
64                    record.id,
65                    record.tcp_addr(),
66                    record.udp_addr(),
67                )
68            }
69            self.network.add_trusted_peer_id(record.peer_id());
70        }
71        Ok(true)
72    }
73
74    /// Handler for `admin_removeTrustedPeer`
75    fn remove_trusted_peer(&self, record: AnyNode) -> RpcResult<bool> {
76        self.network.remove_peer(record.peer_id(), PeerKind::Trusted);
77        Ok(true)
78    }
79
80    /// Handler for `admin_peers`
81    async fn peers(&self) -> RpcResult<Vec<PeerInfo>> {
82        let peers = self.network.get_all_peers().await.to_rpc_result()?;
83        let mut infos = Vec::with_capacity(peers.len());
84
85        for peer in peers {
86            infos.push(PeerInfo {
87                id: alloy_primitives::hex::encode(keccak256(peer.remote_id.as_slice())),
88                name: peer.client_version.to_string(),
89                enode: peer.enode,
90                enr: peer.enr,
91                caps: peer.capabilities.capabilities().iter().map(|cap| cap.to_string()).collect(),
92                network: PeerNetworkInfo {
93                    remote_address: peer.remote_addr,
94                    local_address: peer.local_addr.unwrap_or_else(|| self.network.local_addr()),
95                    inbound: peer.direction.is_incoming(),
96                    trusted: peer.kind.is_trusted(),
97                    static_node: peer.kind.is_static(),
98                },
99                protocols: PeerProtocolInfo {
100                    eth: Some(EthPeerInfo::Info(EthInfo { version: peer.status.version as u64 })),
101                    snap: None,
102                    other: Default::default(),
103                },
104            })
105        }
106
107        Ok(infos)
108    }
109
110    /// Handler for `admin_nodeInfo`
111    async fn node_info(&self) -> RpcResult<NodeInfo> {
112        let enode = self.network.local_node_record();
113        let status = self.network.network_status().await.to_rpc_result()?;
114        let mut config = ChainConfig {
115            chain_id: self.chain_spec.chain().id(),
116            terminal_total_difficulty_passed: self
117                .chain_spec
118                .final_paris_total_difficulty()
119                .is_some(),
120            terminal_total_difficulty: self
121                .chain_spec
122                .ethereum_fork_activation(EthereumHardfork::Paris)
123                .ttd(),
124            deposit_contract_address: self.chain_spec.deposit_contract().map(|dc| dc.address),
125            ..self.chain_spec.genesis().config.clone()
126        };
127
128        // helper macro to set the block or time for a hardfork if known
129        macro_rules! set_block_or_time {
130            ($config:expr, [$( $field:ident => $fork:ident,)*]) => {
131                $(
132                    // don't overwrite if already set
133                    if $config.$field.is_none() {
134                        $config.$field = match self.chain_spec.ethereum_fork_activation(EthereumHardfork::$fork) {
135                            ForkCondition::Block(block) => Some(block),
136                            ForkCondition::TTD { fork_block, .. } => fork_block,
137                            ForkCondition::Timestamp(ts) => Some(ts),
138                            ForkCondition::Never => None,
139                        };
140                    }
141                )*
142            };
143        }
144
145        set_block_or_time!(config, [
146            homestead_block => Homestead,
147            dao_fork_block => Dao,
148            eip150_block => Tangerine,
149            eip155_block => SpuriousDragon,
150            eip158_block => SpuriousDragon,
151            byzantium_block => Byzantium,
152            constantinople_block => Constantinople,
153            petersburg_block => Petersburg,
154            istanbul_block => Istanbul,
155            muir_glacier_block => MuirGlacier,
156            berlin_block => Berlin,
157            london_block => London,
158            arrow_glacier_block => ArrowGlacier,
159            gray_glacier_block => GrayGlacier,
160            shanghai_time => Shanghai,
161            cancun_time => Cancun,
162            prague_time => Prague,
163        ]);
164
165        Ok(NodeInfo {
166            id: alloy_primitives::hex::encode(keccak256(enode.id.as_slice())),
167            name: status.client_version,
168            enode: enode.to_string(),
169            enr: self.network.local_enr().to_string(),
170            ip: enode.address,
171            ports: Ports { discovery: enode.udp_port, listener: enode.tcp_port },
172            listen_addr: enode.tcp_addr(),
173            #[expect(deprecated)]
174            protocols: ProtocolInfo {
175                eth: Some(EthProtocolInfo {
176                    network: status.eth_protocol_info.network,
177                    genesis: status.eth_protocol_info.genesis,
178                    config,
179                    head: status.eth_protocol_info.head,
180                    difficulty: None,
181                }),
182                snap: None,
183            },
184        })
185    }
186
187    /// Handler for `admin_peerEvents`
188    async fn subscribe_peer_events(
189        &self,
190        _pending: jsonrpsee::PendingSubscriptionSink,
191    ) -> jsonrpsee::core::SubscriptionResult {
192        Err("admin_peerEvents is not implemented yet".into())
193    }
194
195    /// Handler for `admin_clearTxpool`
196    async fn clear_txpool(&self) -> RpcResult<u64> {
197        let all_hashes = self.pool.all_transaction_hashes();
198        let count = all_hashes.len() as u64;
199        let _ = self.pool.remove_transactions(all_hashes);
200        Ok(count)
201    }
202}
203
204impl<N, ChainSpec, Pool> std::fmt::Debug for AdminApi<N, ChainSpec, Pool> {
205    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
206        f.debug_struct("AdminApi").finish_non_exhaustive()
207    }
208}