2021-11-22 16:28:30 +00:00
|
|
|
use super::*;
|
|
|
|
use crate::dht::*;
|
|
|
|
use alloc::fmt;
|
|
|
|
|
2022-04-17 23:10:10 +00:00
|
|
|
// Connectionless protocols like UDP are dependent on a NAT translation timeout
|
|
|
|
// We should ping them with some frequency and 30 seconds is typical timeout
|
|
|
|
const CONNECTIONLESS_TIMEOUT_SECS: u32 = 29;
|
|
|
|
|
2021-11-22 16:28:30 +00:00
|
|
|
pub struct NodeRef {
|
|
|
|
routing_table: RoutingTable,
|
|
|
|
node_id: DHTKey,
|
2022-04-19 15:23:44 +00:00
|
|
|
filter: Option<DialInfoFilter>,
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl NodeRef {
|
2022-04-19 15:23:44 +00:00
|
|
|
pub fn new(
|
|
|
|
routing_table: RoutingTable,
|
|
|
|
key: DHTKey,
|
|
|
|
entry: &mut BucketEntry,
|
|
|
|
filter: Option<DialInfoFilter>,
|
|
|
|
) -> Self {
|
2021-11-22 16:28:30 +00:00
|
|
|
entry.ref_count += 1;
|
|
|
|
Self {
|
2021-11-26 15:39:43 +00:00
|
|
|
routing_table,
|
2021-11-22 16:28:30 +00:00
|
|
|
node_id: key,
|
2022-04-19 15:23:44 +00:00
|
|
|
filter,
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn node_id(&self) -> DHTKey {
|
|
|
|
self.node_id
|
|
|
|
}
|
|
|
|
|
2022-04-21 00:49:16 +00:00
|
|
|
pub fn filter_ref(&self) -> Option<&DialInfoFilter> {
|
|
|
|
self.filter.as_ref()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn take_filter(&mut self) -> Option<DialInfoFilter> {
|
|
|
|
self.filter.take()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_filter(&mut self, filter: Option<DialInfoFilter>) {
|
|
|
|
self.filter = filter
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns true if some protocols can still pass the filter and false if no protocols remain
|
|
|
|
pub fn filter_protocols(&mut self, protocol_set: ProtocolSet) -> bool {
|
|
|
|
if protocol_set != ProtocolSet::all() {
|
|
|
|
let mut dif = self.filter.unwrap_or_default();
|
|
|
|
dif.protocol_set &= protocol_set;
|
|
|
|
self.filter = Some(dif);
|
|
|
|
}
|
|
|
|
self.filter
|
|
|
|
.map(|f| !f.protocol_set.is_empty())
|
|
|
|
.unwrap_or(true)
|
|
|
|
}
|
|
|
|
|
2021-11-22 16:28:30 +00:00
|
|
|
pub fn operate<T, F>(&self, f: F) -> T
|
|
|
|
where
|
|
|
|
F: FnOnce(&mut BucketEntry) -> T,
|
|
|
|
{
|
|
|
|
self.routing_table.operate_on_bucket_entry(self.node_id, f)
|
|
|
|
}
|
|
|
|
|
2022-04-16 15:18:54 +00:00
|
|
|
pub fn peer_info(&self) -> PeerInfo {
|
|
|
|
self.operate(|e| e.peer_info(self.node_id()))
|
|
|
|
}
|
|
|
|
pub fn has_seen_our_node_info(&self) -> bool {
|
|
|
|
self.operate(|e| e.has_seen_our_node_info())
|
2022-03-25 02:07:55 +00:00
|
|
|
}
|
2022-04-16 15:18:54 +00:00
|
|
|
pub fn set_seen_our_node_info(&self) {
|
|
|
|
self.operate(|e| e.set_seen_our_node_info(true));
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
2022-04-19 15:23:44 +00:00
|
|
|
|
2022-04-21 00:49:16 +00:00
|
|
|
pub fn network_class(&self) -> NetworkClass {
|
|
|
|
self.operate(|e| e.node_info().network_class)
|
|
|
|
}
|
|
|
|
pub fn outbound_protocols(&self) -> ProtocolSet {
|
|
|
|
self.operate(|e| e.node_info().outbound_protocols)
|
|
|
|
}
|
|
|
|
pub fn relay(&self) -> Option<NodeRef> {
|
|
|
|
let target_rpi = self.operate(|e| e.node_info().relay_peer_info)?;
|
|
|
|
|
|
|
|
self.routing_table
|
|
|
|
.register_node_with_node_info(target_rpi.node_id.key, target_rpi.node_info)
|
|
|
|
.map_err(logthru_rtab!(error))
|
|
|
|
.ok()
|
|
|
|
.map(|nr| {
|
|
|
|
nr.set_filter(self.filter_ref().cloned());
|
|
|
|
nr
|
|
|
|
})
|
|
|
|
}
|
2022-04-19 15:23:44 +00:00
|
|
|
pub fn first_filtered_dial_info(&self) -> Option<DialInfo> {
|
|
|
|
self.operate(|e| {
|
|
|
|
if matches!(
|
|
|
|
self.filter.map(|f| f.peer_scope).unwrap_or(PeerScope::All),
|
2022-04-21 00:49:16 +00:00
|
|
|
PeerScope::All | PeerScope::Local
|
2022-04-19 15:23:44 +00:00
|
|
|
) {
|
2022-04-21 00:49:16 +00:00
|
|
|
e.local_node_info().first_filtered_dial_info(|di| {
|
2022-04-19 15:23:44 +00:00
|
|
|
if let Some(filter) = self.filter {
|
|
|
|
di.matches_filter(&filter)
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
.or_else(|| {
|
|
|
|
if matches!(
|
|
|
|
self.filter.map(|f| f.peer_scope).unwrap_or(PeerScope::All),
|
2022-04-21 00:49:16 +00:00
|
|
|
PeerScope::All | PeerScope::Global
|
2022-04-19 15:23:44 +00:00
|
|
|
) {
|
2022-04-21 00:49:16 +00:00
|
|
|
e.node_info().first_filtered_dial_info(|di| {
|
2022-04-19 15:23:44 +00:00
|
|
|
if let Some(filter) = self.filter {
|
|
|
|
di.matches_filter(&filter)
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn all_filtered_dial_info<F>(&self) -> Vec<DialInfo> {
|
|
|
|
let mut out = Vec::new();
|
|
|
|
self.operate(|e| {
|
|
|
|
if matches!(
|
|
|
|
self.filter.map(|f| f.peer_scope).unwrap_or(PeerScope::All),
|
|
|
|
PeerScope::All | PeerScope::Global
|
|
|
|
) {
|
|
|
|
out.append(&mut e.node_info().all_filtered_dial_info(|di| {
|
|
|
|
if let Some(filter) = self.filter {
|
|
|
|
di.matches_filter(&filter)
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
if matches!(
|
|
|
|
self.filter.map(|f| f.peer_scope).unwrap_or(PeerScope::All),
|
|
|
|
PeerScope::All | PeerScope::Local
|
|
|
|
) {
|
|
|
|
out.append(&mut e.local_node_info().all_filtered_dial_info(|di| {
|
|
|
|
if let Some(filter) = self.filter {
|
|
|
|
di.matches_filter(&filter)
|
|
|
|
} else {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
});
|
|
|
|
out
|
|
|
|
}
|
|
|
|
|
2022-04-17 23:10:10 +00:00
|
|
|
pub async fn last_connection(&self) -> Option<ConnectionDescriptor> {
|
|
|
|
// Get the last connection and the last time we saw anything with this connection
|
|
|
|
let (last_connection, last_seen) = self.operate(|e| {
|
|
|
|
if let Some((last_connection, connection_ts)) = e.last_connection() {
|
|
|
|
if let Some(last_seen) = e.peer_stats().last_seen {
|
|
|
|
Some((last_connection, u64::max(last_seen, connection_ts)))
|
|
|
|
} else {
|
|
|
|
Some((last_connection, connection_ts))
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
})?;
|
|
|
|
// Should we check the connection table?
|
|
|
|
if last_connection.protocol_type().is_connection_oriented() {
|
|
|
|
// Look the connection up in the connection manager and see if it's still there
|
|
|
|
let connection_manager = self.routing_table.network_manager().connection_manager();
|
|
|
|
connection_manager.get_connection(last_connection).await?;
|
|
|
|
} else {
|
|
|
|
// If this is not connection oriented, then we check our last seen time
|
|
|
|
// to see if this mapping has expired (beyond our timeout)
|
|
|
|
let cur_ts = intf::get_timestamp();
|
|
|
|
if (last_seen + (CONNECTIONLESS_TIMEOUT_SECS as u64 * 1_000_000u64)) < cur_ts {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(last_connection)
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
2022-04-17 23:10:10 +00:00
|
|
|
|
2022-04-17 17:28:39 +00:00
|
|
|
pub fn has_any_dial_info(&self) -> bool {
|
|
|
|
self.operate(|e| e.node_info().has_any_dial_info() || e.local_node_info().has_dial_info())
|
|
|
|
}
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Clone for NodeRef {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
self.operate(move |e| {
|
|
|
|
e.ref_count += 1;
|
|
|
|
});
|
|
|
|
Self {
|
|
|
|
routing_table: self.routing_table.clone(),
|
2021-11-26 15:39:43 +00:00
|
|
|
node_id: self.node_id,
|
2022-04-19 15:23:44 +00:00
|
|
|
filter: self.filter.clone(),
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-17 17:28:39 +00:00
|
|
|
impl PartialEq for NodeRef {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.node_id == other.node_id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for NodeRef {}
|
|
|
|
|
2021-11-22 16:28:30 +00:00
|
|
|
impl fmt::Debug for NodeRef {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
2022-04-16 15:18:54 +00:00
|
|
|
write!(f, "{}", self.node_id.encode())
|
2021-11-22 16:28:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for NodeRef {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.routing_table.drop_node_ref(self.node_id);
|
|
|
|
}
|
|
|
|
}
|