Added peers details to ethcore_netPeers RPC (#2580)
* added peers details to ethcore_netPeers RPC * fixed build (traits autoimplemented) * - documentation fixes - spaces -> tabs - Rust-way Option's handling * prepare for new protocols in ethcore_netPeers * commas & documentation
This commit is contained in:
parent
693b0ec402
commit
c9ce25c8f3
@ -165,13 +165,16 @@ impl<C, M, S: ?Sized, F> Ethcore for EthcoreClient<C, M, S, F> where
|
|||||||
fn net_peers(&self) -> Result<Peers, Error> {
|
fn net_peers(&self) -> Result<Peers, Error> {
|
||||||
try!(self.active());
|
try!(self.active());
|
||||||
|
|
||||||
let sync_status = take_weak!(self.sync).status();
|
let sync = take_weak!(self.sync);
|
||||||
|
let sync_status = sync.status();
|
||||||
let net_config = take_weak!(self.net).network_config();
|
let net_config = take_weak!(self.net).network_config();
|
||||||
|
let peers = sync.peers().into_iter().map(Into::into).collect();
|
||||||
|
|
||||||
Ok(Peers {
|
Ok(Peers {
|
||||||
active: sync_status.num_active_peers,
|
active: sync_status.num_active_peers,
|
||||||
connected: sync_status.num_peers,
|
connected: sync_status.num_peers,
|
||||||
max: sync_status.current_max_peers(net_config.min_peers, net_config.max_peers),
|
max: sync_status.current_max_peers(net_config.min_peers, net_config.max_peers),
|
||||||
|
peers: peers
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
//! Test implementation of SyncProvider.
|
//! Test implementation of SyncProvider.
|
||||||
|
|
||||||
use util::{RwLock, U256};
|
use util::{RwLock, U256};
|
||||||
use ethsync::{SyncProvider, SyncStatus, SyncState};
|
use ethsync::{SyncProvider, SyncStatus, SyncState, PeerInfo};
|
||||||
|
|
||||||
/// TestSyncProvider config.
|
/// TestSyncProvider config.
|
||||||
pub struct Config {
|
pub struct Config {
|
||||||
@ -60,5 +60,30 @@ impl SyncProvider for TestSyncProvider {
|
|||||||
fn status(&self) -> SyncStatus {
|
fn status(&self) -> SyncStatus {
|
||||||
self.status.read().clone()
|
self.status.read().clone()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn peers(&self) -> Vec<PeerInfo> {
|
||||||
|
vec![
|
||||||
|
PeerInfo {
|
||||||
|
id: Some("node1".to_owned()),
|
||||||
|
client_version: "Parity/1".to_owned(),
|
||||||
|
capabilities: vec!["eth/62".to_owned(), "eth/63".to_owned()],
|
||||||
|
remote_address: "127.0.0.1:7777".to_owned(),
|
||||||
|
local_address: "127.0.0.1:8888".to_owned(),
|
||||||
|
eth_version: 62,
|
||||||
|
eth_difficulty: Some(40.into()),
|
||||||
|
eth_head: 50.into()
|
||||||
|
},
|
||||||
|
PeerInfo {
|
||||||
|
id: None,
|
||||||
|
client_version: "Parity/2".to_owned(),
|
||||||
|
capabilities: vec!["eth/63".to_owned(), "eth/64".to_owned()],
|
||||||
|
remote_address: "Handshake".to_owned(),
|
||||||
|
local_address: "127.0.0.1:3333".to_owned(),
|
||||||
|
eth_version: 64,
|
||||||
|
eth_difficulty: None,
|
||||||
|
eth_head: 60.into()
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -208,7 +208,12 @@ fn rpc_ethcore_net_peers() {
|
|||||||
io.add_delegate(ethcore_client(&client, &miner, &sync, &net).to_delegate());
|
io.add_delegate(ethcore_client(&client, &miner, &sync, &net).to_delegate());
|
||||||
|
|
||||||
let request = r#"{"jsonrpc": "2.0", "method": "ethcore_netPeers", "params":[], "id": 1}"#;
|
let request = r#"{"jsonrpc": "2.0", "method": "ethcore_netPeers", "params":[], "id": 1}"#;
|
||||||
let response = r#"{"jsonrpc":"2.0","result":{"active":0,"connected":120,"max":50},"id":1}"#;
|
let response = "{\"jsonrpc\":\"2.0\",\"result\":{\"active\":0,\"connected\":120,\"max\":50,\"peers\":[{\"caps\":[\"eth/62\",\"eth/63\"],\
|
||||||
|
\"id\":\"node1\",\"name\":\"Parity/1\",\"network\":{\"localAddress\":\"127.0.0.1:8888\",\"remoteAddress\":\"127.0.0.1:7777\"}\
|
||||||
|
,\"protocols\":{\"eth\":{\"difficulty\":\"0x28\",\"head\":\"0000000000000000000000000000000000000000000000000000000000000032\"\
|
||||||
|
,\"version\":62}}},{\"caps\":[\"eth/63\",\"eth/64\"],\"id\":null,\"name\":\"Parity/2\",\"network\":{\"localAddress\":\
|
||||||
|
\"127.0.0.1:3333\",\"remoteAddress\":\"Handshake\"},\"protocols\":{\"eth\":{\"difficulty\":null,\"head\":\
|
||||||
|
\"000000000000000000000000000000000000000000000000000000000000003c\",\"version\":64}}}]},\"id\":1}";
|
||||||
|
|
||||||
assert_eq!(io.handle_request_sync(request), Some(response.to_owned()));
|
assert_eq!(io.handle_request_sync(request), Some(response.to_owned()));
|
||||||
}
|
}
|
||||||
|
@ -42,7 +42,7 @@ pub use self::filter::{Filter, FilterChanges};
|
|||||||
pub use self::hash::{H64, H160, H256, H512, H520, H2048};
|
pub use self::hash::{H64, H160, H256, H512, H520, H2048};
|
||||||
pub use self::index::Index;
|
pub use self::index::Index;
|
||||||
pub use self::log::Log;
|
pub use self::log::Log;
|
||||||
pub use self::sync::{SyncStatus, SyncInfo, Peers};
|
pub use self::sync::{SyncStatus, SyncInfo, Peers, PeerInfo, PeerNetworkInfo, PeerProtocolsInfo, PeerEthereumProtocolInfo};
|
||||||
pub use self::transaction::Transaction;
|
pub use self::transaction::Transaction;
|
||||||
pub use self::transaction_request::TransactionRequest;
|
pub use self::transaction_request::TransactionRequest;
|
||||||
pub use self::receipt::Receipt;
|
pub use self::receipt::Receipt;
|
||||||
|
@ -14,6 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
use ethsync::PeerInfo as SyncPeerInfo;
|
||||||
use serde::{Serialize, Serializer};
|
use serde::{Serialize, Serializer};
|
||||||
use v1::types::U256;
|
use v1::types::U256;
|
||||||
|
|
||||||
@ -32,7 +33,7 @@ pub struct SyncInfo {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Peers info
|
/// Peers info
|
||||||
#[derive(Default, Debug, Serialize, PartialEq)]
|
#[derive(Default, Debug, Serialize)]
|
||||||
pub struct Peers {
|
pub struct Peers {
|
||||||
/// Number of active peers
|
/// Number of active peers
|
||||||
pub active: usize,
|
pub active: usize,
|
||||||
@ -40,6 +41,52 @@ pub struct Peers {
|
|||||||
pub connected: usize,
|
pub connected: usize,
|
||||||
/// Max number of peers
|
/// Max number of peers
|
||||||
pub max: u32,
|
pub max: u32,
|
||||||
|
/// Detailed information on peers
|
||||||
|
pub peers: Vec<PeerInfo>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Peer connection information
|
||||||
|
#[derive(Default, Debug, Serialize)]
|
||||||
|
pub struct PeerInfo {
|
||||||
|
/// Public node id
|
||||||
|
pub id: Option<String>,
|
||||||
|
/// Node client ID
|
||||||
|
pub name: String,
|
||||||
|
/// Capabilities
|
||||||
|
pub caps: Vec<String>,
|
||||||
|
/// Network information
|
||||||
|
pub network: PeerNetworkInfo,
|
||||||
|
/// Protocols information
|
||||||
|
pub protocols: PeerProtocolsInfo,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Peer network information
|
||||||
|
#[derive(Default, Debug, Serialize)]
|
||||||
|
pub struct PeerNetworkInfo {
|
||||||
|
/// Remote endpoint address
|
||||||
|
#[serde(rename="remoteAddress")]
|
||||||
|
pub remote_address: String,
|
||||||
|
/// Local endpoint address
|
||||||
|
#[serde(rename="localAddress")]
|
||||||
|
pub local_address: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Peer protocols information
|
||||||
|
#[derive(Default, Debug, Serialize)]
|
||||||
|
pub struct PeerProtocolsInfo {
|
||||||
|
/// Ethereum protocol information
|
||||||
|
pub eth: Option<PeerEthereumProtocolInfo>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Peer Ethereum protocol information
|
||||||
|
#[derive(Default, Debug, Serialize)]
|
||||||
|
pub struct PeerEthereumProtocolInfo {
|
||||||
|
/// Negotiated ethereum protocol version
|
||||||
|
pub version: u32,
|
||||||
|
/// Peer total difficulty if known
|
||||||
|
pub difficulty: Option<U256>,
|
||||||
|
/// SHA3 of peer best block hash
|
||||||
|
pub head: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Sync status
|
/// Sync status
|
||||||
@ -61,6 +108,27 @@ impl Serialize for SyncStatus {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl From<SyncPeerInfo> for PeerInfo {
|
||||||
|
fn from(p: SyncPeerInfo) -> PeerInfo {
|
||||||
|
PeerInfo {
|
||||||
|
id: p.id,
|
||||||
|
name: p.client_version,
|
||||||
|
caps: p.capabilities,
|
||||||
|
network: PeerNetworkInfo {
|
||||||
|
remote_address: p.remote_address,
|
||||||
|
local_address: p.local_address,
|
||||||
|
},
|
||||||
|
protocols: PeerProtocolsInfo {
|
||||||
|
eth: Some(PeerEthereumProtocolInfo {
|
||||||
|
version: p.eth_version,
|
||||||
|
difficulty: p.eth_difficulty.map(|d| d.into()),
|
||||||
|
head: p.eth_head.hex(),
|
||||||
|
})
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use serde_json;
|
use serde_json;
|
||||||
@ -77,7 +145,7 @@ mod tests {
|
|||||||
fn test_serialize_peers() {
|
fn test_serialize_peers() {
|
||||||
let t = Peers::default();
|
let t = Peers::default();
|
||||||
let serialized = serde_json::to_string(&t).unwrap();
|
let serialized = serde_json::to_string(&t).unwrap();
|
||||||
assert_eq!(serialized, r#"{"active":0,"connected":0,"max":0}"#);
|
assert_eq!(serialized, r#"{"active":0,"connected":0,"max":0,"peers":[]}"#);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -61,6 +61,30 @@ binary_fixed_size!(SyncStatus);
|
|||||||
pub trait SyncProvider: Send + Sync {
|
pub trait SyncProvider: Send + Sync {
|
||||||
/// Get sync status
|
/// Get sync status
|
||||||
fn status(&self) -> SyncStatus;
|
fn status(&self) -> SyncStatus;
|
||||||
|
|
||||||
|
/// Get peers information
|
||||||
|
fn peers(&self) -> Vec<PeerInfo>;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Peer connection information
|
||||||
|
#[derive(Debug, Binary)]
|
||||||
|
pub struct PeerInfo {
|
||||||
|
/// Public node id
|
||||||
|
pub id: Option<String>,
|
||||||
|
/// Node client ID
|
||||||
|
pub client_version: String,
|
||||||
|
/// Capabilities
|
||||||
|
pub capabilities: Vec<String>,
|
||||||
|
/// Remote endpoint address
|
||||||
|
pub remote_address: String,
|
||||||
|
/// Local endpoint address
|
||||||
|
pub local_address: String,
|
||||||
|
/// Ethereum protocol version
|
||||||
|
pub eth_version: u32,
|
||||||
|
/// SHA3 of peer best block hash
|
||||||
|
pub eth_head: H256,
|
||||||
|
/// Peer total difficulty if known
|
||||||
|
pub eth_difficulty: Option<U256>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Ethereum network protocol handler
|
/// Ethereum network protocol handler
|
||||||
@ -94,6 +118,14 @@ impl SyncProvider for EthSync {
|
|||||||
fn status(&self) -> SyncStatus {
|
fn status(&self) -> SyncStatus {
|
||||||
self.handler.sync.write().status()
|
self.handler.sync.write().status()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get sync peers
|
||||||
|
fn peers(&self) -> Vec<PeerInfo> {
|
||||||
|
self.network.with_context_eval(self.subprotocol_name, |context| {
|
||||||
|
let sync_io = NetSyncIo::new(context, &*self.handler.chain, &*self.handler.snapshot_service);
|
||||||
|
self.handler.sync.write().peers(&sync_io)
|
||||||
|
}).unwrap_or(Vec::new())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct SyncProtocolHandler {
|
struct SyncProtocolHandler {
|
||||||
|
@ -102,6 +102,7 @@ use super::SyncConfig;
|
|||||||
use blocks::BlockCollection;
|
use blocks::BlockCollection;
|
||||||
use snapshot::{Snapshot, ChunkType};
|
use snapshot::{Snapshot, ChunkType};
|
||||||
use rand::{thread_rng, Rng};
|
use rand::{thread_rng, Rng};
|
||||||
|
use api::PeerInfo as PeerInfoDigest;
|
||||||
|
|
||||||
known_heap_size!(0, PeerInfo);
|
known_heap_size!(0, PeerInfo);
|
||||||
|
|
||||||
@ -346,7 +347,7 @@ impl ChainSync {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// @returns Synchonization status
|
/// Returns synchonization status
|
||||||
pub fn status(&self) -> SyncStatus {
|
pub fn status(&self) -> SyncStatus {
|
||||||
SyncStatus {
|
SyncStatus {
|
||||||
state: self.state.clone(),
|
state: self.state.clone(),
|
||||||
@ -368,6 +369,25 @@ impl ChainSync {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns information on peers connections
|
||||||
|
pub fn peers(&self, io: &SyncIo) -> Vec<PeerInfoDigest> {
|
||||||
|
self.peers.iter()
|
||||||
|
.filter_map(|(&peer_id, ref peer_data)|
|
||||||
|
io.peer_session_info(peer_id).map(|session_info|
|
||||||
|
PeerInfoDigest {
|
||||||
|
id: session_info.id.map(|id| id.hex()),
|
||||||
|
client_version: session_info.client_version,
|
||||||
|
capabilities: session_info.peer_capabilities.into_iter().map(|c| c.to_string()).collect(),
|
||||||
|
remote_address: session_info.remote_address,
|
||||||
|
local_address: session_info.local_address,
|
||||||
|
eth_version: peer_data.protocol_version,
|
||||||
|
eth_difficulty: peer_data.difficulty,
|
||||||
|
eth_head: peer_data.latest_hash,
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
/// Abort all sync activity
|
/// Abort all sync activity
|
||||||
pub fn abort(&mut self, io: &mut SyncIo) {
|
pub fn abort(&mut self, io: &mut SyncIo) {
|
||||||
self.restart(io);
|
self.restart(io);
|
||||||
|
@ -60,7 +60,7 @@ mod api {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub use api::{EthSync, SyncProvider, SyncClient, NetworkManagerClient, ManageNetwork, SyncConfig,
|
pub use api::{EthSync, SyncProvider, SyncClient, NetworkManagerClient, ManageNetwork, SyncConfig,
|
||||||
ServiceConfiguration, NetworkConfiguration};
|
ServiceConfiguration, NetworkConfiguration, PeerInfo};
|
||||||
pub use chain::{SyncStatus, SyncState};
|
pub use chain::{SyncStatus, SyncState};
|
||||||
pub use network::{is_valid_node_url, NonReservedPeerMode, NetworkError};
|
pub use network::{is_valid_node_url, NonReservedPeerMode, NetworkError};
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
use network::{NetworkContext, PeerId, PacketId, NetworkError};
|
use network::{NetworkContext, PeerId, PacketId, NetworkError, SessionInfo};
|
||||||
use ethcore::client::BlockChainClient;
|
use ethcore::client::BlockChainClient;
|
||||||
use ethcore::snapshot::SnapshotService;
|
use ethcore::snapshot::SnapshotService;
|
||||||
|
|
||||||
@ -34,10 +34,12 @@ pub trait SyncIo {
|
|||||||
fn chain(&self) -> &BlockChainClient;
|
fn chain(&self) -> &BlockChainClient;
|
||||||
/// Get the snapshot service.
|
/// Get the snapshot service.
|
||||||
fn snapshot_service(&self) -> &SnapshotService;
|
fn snapshot_service(&self) -> &SnapshotService;
|
||||||
/// Returns peer client identifier string
|
/// Returns peer identifier string
|
||||||
fn peer_info(&self, peer_id: PeerId) -> String {
|
fn peer_info(&self, peer_id: PeerId) -> String {
|
||||||
peer_id.to_string()
|
peer_id.to_string()
|
||||||
}
|
}
|
||||||
|
/// Returns information on p2p session
|
||||||
|
fn peer_session_info(&self, peer_id: PeerId) -> Option<SessionInfo>;
|
||||||
/// Maximum mutuallt supported ETH protocol version
|
/// Maximum mutuallt supported ETH protocol version
|
||||||
fn eth_protocol_version(&self, peer_id: PeerId) -> u8;
|
fn eth_protocol_version(&self, peer_id: PeerId) -> u8;
|
||||||
/// Returns if the chain block queue empty
|
/// Returns if the chain block queue empty
|
||||||
@ -91,8 +93,8 @@ impl<'s, 'h> SyncIo for NetSyncIo<'s, 'h> {
|
|||||||
self.snapshot_service
|
self.snapshot_service
|
||||||
}
|
}
|
||||||
|
|
||||||
fn peer_info(&self, peer_id: PeerId) -> String {
|
fn peer_session_info(&self, peer_id: PeerId) -> Option<SessionInfo> {
|
||||||
self.network.peer_info(peer_id)
|
self.network.session_info(peer_id)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn is_expired(&self) -> bool {
|
fn is_expired(&self) -> bool {
|
||||||
|
@ -83,6 +83,10 @@ impl<'p> SyncIo for TestIo<'p> {
|
|||||||
self.snapshot_service
|
self.snapshot_service
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn peer_session_info(&self, _peer_id: PeerId) -> Option<SessionInfo> {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
fn eth_protocol_version(&self, _peer: PeerId) -> u8 {
|
fn eth_protocol_version(&self, _peer: PeerId) -> u8 {
|
||||||
64
|
64
|
||||||
}
|
}
|
||||||
|
@ -191,6 +191,11 @@ impl Connection {
|
|||||||
self.socket.peer_addr().map(|a| a.to_string()).unwrap_or_else(|_| "Unknown".to_owned())
|
self.socket.peer_addr().map(|a| a.to_string()).unwrap_or_else(|_| "Unknown".to_owned())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get local peer address string
|
||||||
|
pub fn local_addr_str(&self) -> String {
|
||||||
|
self.socket.local_addr().map(|a| a.to_string()).unwrap_or_else(|_| "Unknown".to_owned())
|
||||||
|
}
|
||||||
|
|
||||||
/// Clone this connection. Clears the receiving buffer of the returned connection.
|
/// Clone this connection. Clears the receiving buffer of the returned connection.
|
||||||
pub fn try_clone(&self) -> io::Result<Self> {
|
pub fn try_clone(&self) -> io::Result<Self> {
|
||||||
Ok(Connection {
|
Ok(Connection {
|
||||||
|
@ -31,7 +31,7 @@ use util::hash::*;
|
|||||||
use util::Hashable;
|
use util::Hashable;
|
||||||
use util::version;
|
use util::version;
|
||||||
use rlp::*;
|
use rlp::*;
|
||||||
use session::{Session, SessionData};
|
use session::{Session, SessionInfo, SessionData};
|
||||||
use error::*;
|
use error::*;
|
||||||
use io::*;
|
use io::*;
|
||||||
use {NetworkProtocolHandler, NonReservedPeerMode, PROTOCOL_VERSION};
|
use {NetworkProtocolHandler, NonReservedPeerMode, PROTOCOL_VERSION};
|
||||||
@ -280,12 +280,13 @@ impl<'s> NetworkContext<'s> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Returns peer identification string
|
/// Returns peer identification string
|
||||||
pub fn peer_info(&self, peer: PeerId) -> String {
|
pub fn peer_client_version(&self, peer: PeerId) -> String {
|
||||||
let session = self.resolve_session(peer);
|
self.resolve_session(peer).map_or("unknown".to_owned(), |s| s.lock().info.client_version.clone())
|
||||||
if let Some(session) = session {
|
}
|
||||||
return session.lock().info.client_version.clone()
|
|
||||||
}
|
/// Returns information on p2p session
|
||||||
"unknown".to_owned()
|
pub fn session_info(&self, peer: PeerId) -> Option<SessionInfo> {
|
||||||
|
self.resolve_session(peer).map(|s| s.lock().info.clone())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns max version for a given protocol.
|
/// Returns max version for a given protocol.
|
||||||
@ -918,6 +919,13 @@ impl Host {
|
|||||||
let context = NetworkContext::new(io, protocol, None, self.sessions.clone(), &reserved);
|
let context = NetworkContext::new(io, protocol, None, self.sessions.clone(), &reserved);
|
||||||
action(&context);
|
action(&context);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn with_context_eval<F, T>(&self, protocol: ProtocolId, io: &IoContext<NetworkIoMessage>, action: F) -> T where F: Fn(&NetworkContext) -> T {
|
||||||
|
let reserved = { self.reserved_nodes.read() };
|
||||||
|
|
||||||
|
let context = NetworkContext::new(io, protocol, None, self.sessions.clone(), &reserved);
|
||||||
|
action(&context)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl IoHandler<NetworkIoMessage> for Host {
|
impl IoHandler<NetworkIoMessage> for Host {
|
||||||
|
@ -99,6 +99,7 @@ pub use host::NetworkIoMessage;
|
|||||||
pub use error::NetworkError;
|
pub use error::NetworkError;
|
||||||
pub use host::NetworkConfiguration;
|
pub use host::NetworkConfiguration;
|
||||||
pub use stats::NetworkStats;
|
pub use stats::NetworkStats;
|
||||||
|
pub use session::SessionInfo;
|
||||||
|
|
||||||
use io::TimerToken;
|
use io::TimerToken;
|
||||||
pub use node_table::is_valid_node_url;
|
pub use node_table::is_valid_node_url;
|
||||||
|
@ -178,6 +178,13 @@ impl NetworkService {
|
|||||||
host.with_context(protocol, &io, action);
|
host.with_context(protocol, &io, action);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Evaluates function in the network context
|
||||||
|
pub fn with_context_eval<F, T>(&self, protocol: ProtocolId, action: F) -> Option<T> where F: Fn(&NetworkContext) -> T {
|
||||||
|
let io = IoContext::new(self.io_service.channel(), 0);
|
||||||
|
let host = self.host.read();
|
||||||
|
host.as_ref().map(|ref host| host.with_context_eval(protocol, &io, action))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl MayPanic for NetworkService {
|
impl MayPanic for NetworkService {
|
||||||
|
@ -72,6 +72,7 @@ pub enum SessionData {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Shared session information
|
/// Shared session information
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
pub struct SessionInfo {
|
pub struct SessionInfo {
|
||||||
/// Peer public key
|
/// Peer public key
|
||||||
pub id: Option<NodeId>,
|
pub id: Option<NodeId>,
|
||||||
@ -79,15 +80,21 @@ pub struct SessionInfo {
|
|||||||
pub client_version: String,
|
pub client_version: String,
|
||||||
/// Peer RLPx protocol version
|
/// Peer RLPx protocol version
|
||||||
pub protocol_version: u32,
|
pub protocol_version: u32,
|
||||||
|
/// Session protocol capabilities
|
||||||
|
pub capabilities: Vec<SessionCapabilityInfo>,
|
||||||
/// Peer protocol capabilities
|
/// Peer protocol capabilities
|
||||||
capabilities: Vec<SessionCapabilityInfo>,
|
pub peer_capabilities: Vec<PeerCapabilityInfo>,
|
||||||
/// Peer ping delay in milliseconds
|
/// Peer ping delay in milliseconds
|
||||||
pub ping_ms: Option<u64>,
|
pub ping_ms: Option<u64>,
|
||||||
/// True if this session was originated by us.
|
/// True if this session was originated by us.
|
||||||
pub originated: bool,
|
pub originated: bool,
|
||||||
|
/// Remote endpoint address of the session
|
||||||
|
pub remote_address: String,
|
||||||
|
/// Local endpoint address of the session
|
||||||
|
pub local_address: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Eq)]
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
pub struct PeerCapabilityInfo {
|
pub struct PeerCapabilityInfo {
|
||||||
pub protocol: ProtocolId,
|
pub protocol: ProtocolId,
|
||||||
pub version: u8,
|
pub version: u8,
|
||||||
@ -109,8 +116,14 @@ impl Decodable for PeerCapabilityInfo {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
impl ToString for PeerCapabilityInfo {
|
||||||
struct SessionCapabilityInfo {
|
fn to_string(&self) -> String {
|
||||||
|
format!("{}/{}", str::from_utf8(&self.protocol[..]).unwrap_or("???"), self.version)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct SessionCapabilityInfo {
|
||||||
pub protocol: [u8; 3],
|
pub protocol: [u8; 3],
|
||||||
pub version: u8,
|
pub version: u8,
|
||||||
pub packet_count: u8,
|
pub packet_count: u8,
|
||||||
@ -134,6 +147,7 @@ impl Session {
|
|||||||
where Message: Send + Clone {
|
where Message: Send + Clone {
|
||||||
let originated = id.is_some();
|
let originated = id.is_some();
|
||||||
let mut handshake = Handshake::new(token, id, socket, nonce, stats).expect("Can't create handshake");
|
let mut handshake = Handshake::new(token, id, socket, nonce, stats).expect("Can't create handshake");
|
||||||
|
let local_addr = handshake.connection.local_addr_str();
|
||||||
try!(handshake.start(io, host, originated));
|
try!(handshake.start(io, host, originated));
|
||||||
Ok(Session {
|
Ok(Session {
|
||||||
state: State::Handshake(handshake),
|
state: State::Handshake(handshake),
|
||||||
@ -143,8 +157,11 @@ impl Session {
|
|||||||
client_version: String::new(),
|
client_version: String::new(),
|
||||||
protocol_version: 0,
|
protocol_version: 0,
|
||||||
capabilities: Vec::new(),
|
capabilities: Vec::new(),
|
||||||
|
peer_capabilities: Vec::new(),
|
||||||
ping_ms: None,
|
ping_ms: None,
|
||||||
originated: originated,
|
originated: originated,
|
||||||
|
remote_address: "Handshake".to_owned(),
|
||||||
|
local_address: local_addr,
|
||||||
},
|
},
|
||||||
ping_time_ns: 0,
|
ping_time_ns: 0,
|
||||||
pong_time_ns: None,
|
pong_time_ns: None,
|
||||||
@ -155,6 +172,7 @@ impl Session {
|
|||||||
fn complete_handshake<Message>(&mut self, io: &IoContext<Message>, host: &HostInfo) -> Result<(), NetworkError> where Message: Send + Sync + Clone {
|
fn complete_handshake<Message>(&mut self, io: &IoContext<Message>, host: &HostInfo) -> Result<(), NetworkError> where Message: Send + Sync + Clone {
|
||||||
let connection = if let State::Handshake(ref mut h) = self.state {
|
let connection = if let State::Handshake(ref mut h) = self.state {
|
||||||
self.info.id = Some(h.id.clone());
|
self.info.id = Some(h.id.clone());
|
||||||
|
self.info.remote_address = h.connection.remote_addr_str();
|
||||||
try!(EncryptedConnection::new(h))
|
try!(EncryptedConnection::new(h))
|
||||||
} else {
|
} else {
|
||||||
panic!("Unexpected state");
|
panic!("Unexpected state");
|
||||||
@ -431,8 +449,10 @@ impl Session {
|
|||||||
i += 1;
|
i += 1;
|
||||||
}
|
}
|
||||||
trace!(target: "network", "Hello: {} v{} {} {:?}", client_version, protocol, id, caps);
|
trace!(target: "network", "Hello: {} v{} {} {:?}", client_version, protocol, id, caps);
|
||||||
|
self.info.protocol_version = protocol;
|
||||||
self.info.client_version = client_version;
|
self.info.client_version = client_version;
|
||||||
self.info.capabilities = caps;
|
self.info.capabilities = caps;
|
||||||
|
self.info.peer_capabilities = peer_caps;
|
||||||
if self.info.capabilities.is_empty() {
|
if self.info.capabilities.is_empty() {
|
||||||
trace!(target: "network", "No common capabilities with peer.");
|
trace!(target: "network", "No common capabilities with peer.");
|
||||||
return Err(From::from(self.disconnect(io, DisconnectReason::UselessPeer)));
|
return Err(From::from(self.disconnect(io, DisconnectReason::UselessPeer)));
|
||||||
|
@ -69,7 +69,7 @@ impl NetworkProtocolHandler for TestProtocol {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn connected(&self, io: &NetworkContext, peer: &PeerId) {
|
fn connected(&self, io: &NetworkContext, peer: &PeerId) {
|
||||||
assert!(io.peer_info(*peer).contains("Parity"));
|
assert!(io.peer_client_version(*peer).contains("Parity"));
|
||||||
if self.drop_session {
|
if self.drop_session {
|
||||||
io.disconnect_peer(*peer)
|
io.disconnect_peer(*peer)
|
||||||
} else {
|
} else {
|
||||||
|
Loading…
Reference in New Issue
Block a user