veilid/veilid-core/src/routing_table/node_ref.rs

324 lines
11 KiB
Rust
Raw Normal View History

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-06-25 14:57:33 +00:00
entry: Arc<BucketEntry>,
2022-04-19 15:23:44 +00:00
filter: Option<DialInfoFilter>,
2022-05-25 15:12:19 +00:00
#[cfg(feature = "tracking")]
track_id: usize,
2021-11-22 16:28:30 +00:00
}
impl NodeRef {
2022-04-19 15:23:44 +00:00
pub fn new(
routing_table: RoutingTable,
2022-06-25 14:57:33 +00:00
node_id: DHTKey,
entry: Arc<BucketEntry>,
2022-04-19 15:23:44 +00:00
filter: Option<DialInfoFilter>,
) -> Self {
2022-06-25 14:57:33 +00:00
entry.ref_count.fetch_add(1u32, Ordering::Relaxed);
2022-05-25 15:12:19 +00:00
2021-11-22 16:28:30 +00:00
Self {
2021-11-26 15:39:43 +00:00
routing_table,
2022-06-25 14:57:33 +00:00
node_id,
entry,
2022-04-19 15:23:44 +00:00
filter,
2022-05-25 15:12:19 +00:00
#[cfg(feature = "tracking")]
track_id: entry.track(),
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() {
2022-04-25 15:29:02 +00:00
let mut dif = self.filter.clone().unwrap_or_default();
2022-04-21 00:49:16 +00:00
dif.protocol_set &= protocol_set;
self.filter = Some(dif);
}
self.filter
2022-04-25 15:29:02 +00:00
.as_ref()
2022-04-21 00:49:16 +00:00
.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
2022-06-25 14:57:33 +00:00
F: FnOnce(&BucketEntryInner) -> T,
2021-11-22 16:28:30 +00:00
{
2022-06-25 14:57:33 +00:00
self.entry.with(f)
}
pub fn operate_mut<T, F>(&self, f: F) -> T
where
F: FnOnce(&mut BucketEntryInner) -> T,
{
self.entry.with_mut(f)
2021-11-22 16:28:30 +00:00
}
2022-05-11 01:49:42 +00:00
pub fn peer_info(&self) -> Option<PeerInfo> {
2022-04-16 15:18:54 +00:00
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) {
2022-06-25 14:57:33 +00:00
self.operate_mut(|e| e.set_seen_our_node_info(true));
2021-11-22 16:28:30 +00:00
}
2022-05-11 01:49:42 +00:00
pub fn network_class(&self) -> Option<NetworkClass> {
self.operate(|e| e.node_info().map(|n| n.network_class))
2022-04-21 00:49:16 +00:00
}
2022-05-11 01:49:42 +00:00
pub fn outbound_protocols(&self) -> Option<ProtocolSet> {
self.operate(|e| e.node_info().map(|n| n.outbound_protocols))
2022-04-21 00:49:16 +00:00
}
pub fn relay(&self) -> Option<NodeRef> {
2022-05-11 01:49:42 +00:00
let target_rpi = self.operate(|e| e.node_info().map(|n| n.relay_peer_info))?;
target_rpi.and_then(|t| {
// If relay is ourselves, then return None, because we can't relay through ourselves
// and to contact this node we should have had an existing inbound connection
if t.node_id.key == self.routing_table.node_id() {
return None;
}
// Register relay node and return noderef
2022-05-11 01:49:42 +00:00
self.routing_table
.register_node_with_signed_node_info(t.node_id.key, t.signed_node_info)
.map(|mut nr| {
nr.set_filter(self.filter_ref().cloned());
nr
})
})
2022-04-21 00:49:16 +00:00
}
pub fn first_filtered_dial_info_detail(
&self,
routing_domain: Option<RoutingDomain>,
) -> Option<DialInfoDetail> {
2022-04-19 15:23:44 +00:00
self.operate(|e| {
2022-04-25 15:29:02 +00:00
// Prefer local dial info first unless it is filtered out
if (routing_domain == None || routing_domain == Some(RoutingDomain::LocalNetwork))
&& matches!(
2022-04-25 15:29:02 +00:00
self.filter
.as_ref()
.map(|f| f.peer_scope)
.unwrap_or(PeerScope::All),
PeerScope::All | PeerScope::Local
)
{
2022-05-11 01:49:42 +00:00
e.local_node_info().and_then(|l| {
l.first_filtered_dial_info(|di| {
2022-04-25 15:29:02 +00:00
if let Some(filter) = self.filter.as_ref() {
di.matches_filter(filter)
} else {
true
}
})
.map(|di| DialInfoDetail {
class: DialInfoClass::Direct,
dial_info: di,
})
2022-05-11 01:49:42 +00:00
})
2022-04-19 15:23:44 +00:00
} else {
None
}
.or_else(|| {
if (routing_domain == None || routing_domain == Some(RoutingDomain::PublicInternet))
&& matches!(
2022-04-25 15:29:02 +00:00
self.filter
.as_ref()
.map(|f| f.peer_scope)
.unwrap_or(PeerScope::All),
PeerScope::All | PeerScope::Global
)
{
2022-05-11 01:49:42 +00:00
e.node_info().and_then(|n| {
n.first_filtered_dial_info_detail(|did| {
if let Some(filter) = self.filter.as_ref() {
did.matches_filter(filter)
} else {
true
}
})
2022-04-19 15:23:44 +00:00
})
} else {
None
}
})
})
}
pub fn all_filtered_dial_info_details<F>(
&self,
routing_domain: Option<RoutingDomain>,
) -> Vec<DialInfoDetail> {
2022-04-19 15:23:44 +00:00
let mut out = Vec::new();
self.operate(|e| {
2022-04-25 15:29:02 +00:00
// Prefer local dial info first unless it is filtered out
if (routing_domain == None || routing_domain == Some(RoutingDomain::LocalNetwork))
&& matches!(
2022-04-25 15:29:02 +00:00
self.filter
.as_ref()
.map(|f| f.peer_scope)
.unwrap_or(PeerScope::All),
PeerScope::All | PeerScope::Local
)
{
2022-05-11 01:49:42 +00:00
if let Some(lni) = e.local_node_info() {
for di in lni.all_filtered_dial_info(|di| {
if let Some(filter) = self.filter.as_ref() {
di.matches_filter(filter)
} else {
true
}
}) {
out.push(DialInfoDetail {
class: DialInfoClass::Direct,
dial_info: di,
});
2022-04-19 15:23:44 +00:00
}
}
2022-04-19 15:23:44 +00:00
}
if (routing_domain == None || routing_domain == Some(RoutingDomain::PublicInternet))
&& matches!(
2022-04-25 15:29:02 +00:00
self.filter
.as_ref()
.map(|f| f.peer_scope)
.unwrap_or(PeerScope::All),
PeerScope::All | PeerScope::Global
)
{
2022-05-11 01:49:42 +00:00
if let Some(ni) = e.node_info() {
out.append(&mut ni.all_filtered_dial_info_details(|did| {
if let Some(filter) = self.filter.as_ref() {
did.matches_filter(filter)
} else {
true
}
}))
}
2022-04-19 15:23:44 +00:00
}
});
out.remove_duplicates();
2022-04-19 15:23:44 +00:00
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_ts) = e.peer_stats().rpc_stats.last_seen_ts {
Some((last_connection, u64::max(last_seen_ts, connection_ts)))
2022-04-17 23:10:10 +00:00
} else {
Some((last_connection, connection_ts))
}
} else {
None
}
})?;
2022-05-28 14:07:57 +00:00
// Verify this connection matches the noderef filter
if let Some(filter) = &self.filter {
if !last_connection.matches_filter(filter) {
return None;
}
}
2022-04-17 23:10:10 +00:00
// 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 {
2022-05-11 01:49:42 +00:00
self.operate(|e| {
e.node_info()
.map(|n| n.has_any_dial_info())
.unwrap_or(false)
|| e.local_node_info()
.map(|l| l.has_dial_info())
.unwrap_or(false)
})
2022-04-17 17:28:39 +00:00
}
2021-11-22 16:28:30 +00:00
}
impl Clone for NodeRef {
fn clone(&self) -> Self {
2022-06-25 14:57:33 +00:00
self.entry.ref_count.fetch_add(1u32, Ordering::Relaxed);
2022-05-25 15:12:19 +00:00
2022-06-25 14:57:33 +00:00
Self {
routing_table: self.routing_table.clone(),
node_id: self.node_id,
entry: self.entry.clone(),
filter: self.filter.clone(),
#[cfg(feature = "tracking")]
track_id: e.track(),
}
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 {}
2022-05-25 15:12:19 +00:00
impl fmt::Display for NodeRef {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.node_id.encode())
}
}
2021-11-22 16:28:30 +00:00
impl fmt::Debug for NodeRef {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2022-05-25 15:12:19 +00:00
f.debug_struct("NodeRef")
.field("node_id", &self.node_id)
.field("filter", &self.filter)
.finish()
2021-11-22 16:28:30 +00:00
}
}
impl Drop for NodeRef {
fn drop(&mut self) {
2022-05-25 15:12:19 +00:00
#[cfg(feature = "tracking")]
self.operate(|e| e.untrack(self.track_id));
2022-06-25 14:57:33 +00:00
// drop the noderef and queue a bucket kick if it was the last one
let new_ref_count = self.entry.ref_count.fetch_sub(1u32, Ordering::Relaxed) - 1;
if new_ref_count == 0 {
self.routing_table.queue_bucket_kick(self.node_id);
}
2021-11-22 16:28:30 +00:00
}
}