checkpoint

This commit is contained in:
John Smith 2022-11-09 22:27:37 -05:00
parent e672ae0319
commit 592c83d83a
10 changed files with 299 additions and 325 deletions

View File

@ -1210,18 +1210,16 @@ impl NetworkManager {
}; };
// Node A is our own node // Node A is our own node
let node_a = routing_table.get_own_node_info(routing_domain); let peer_a = routing_table.get_own_peer_info(routing_domain);
let node_a_id = routing_table.node_id();
// Node B is the target node // Node B is the target node
let node_b = match target_node_ref.node_info(routing_domain) { let peer_b = match target_node_ref.make_peer_info(routing_domain) {
Some(ni) => ni, Some(ni) => ni,
None => { None => {
log_net!("no node info for node {:?}", target_node_ref); log_net!("no node info for node {:?}", target_node_ref);
return Ok(NodeContactMethod::Unreachable); return Ok(NodeContactMethod::Unreachable);
} }
}; };
let node_b_id = target_node_ref.node_id();
// Dial info filter comes from the target node ref // Dial info filter comes from the target node ref
let dial_info_filter = target_node_ref.dial_info_filter(); let dial_info_filter = target_node_ref.dial_info_filter();
@ -1229,10 +1227,8 @@ impl NetworkManager {
let cm = routing_table.get_contact_method( let cm = routing_table.get_contact_method(
routing_domain, routing_domain,
&node_a_id, &peer_a,
&node_a, &peer_b,
&node_b_id,
&node_b,
dial_info_filter, dial_info_filter,
sequencing, sequencing,
); );

View File

@ -348,24 +348,30 @@ impl RoutingTable {
.node_info_is_valid_in_routing_domain(routing_domain, node_info) .node_info_is_valid_in_routing_domain(routing_domain, node_info)
} }
pub fn signed_node_info_is_valid_in_routing_domain(
&self,
routing_domain: RoutingDomain,
signed_node_info: &SignedNodeInfo,
) -> bool {
self.inner
.read()
.signed_node_info_is_valid_in_routing_domain(routing_domain, signed_node_info)
}
/// Look up the best way for two nodes to reach each other over a specific routing domain /// Look up the best way for two nodes to reach each other over a specific routing domain
#[instrument(level = "trace", skip(self), ret)] #[instrument(level = "trace", skip(self), ret)]
pub fn get_contact_method( pub fn get_contact_method(
&self, &self,
routing_domain: RoutingDomain, routing_domain: RoutingDomain,
node_a_id: &DHTKey, peer_a: &PeerInfo,
node_a: &NodeInfo, peer_b: &PeerInfo,
node_b_id: &DHTKey,
node_b: &NodeInfo,
dial_info_filter: DialInfoFilter, dial_info_filter: DialInfoFilter,
sequencing: Sequencing, sequencing: Sequencing,
) -> ContactMethod { ) -> ContactMethod {
self.inner.read().get_contact_method( self.inner.read().get_contact_method(
routing_domain, routing_domain,
node_a_id, peer_a,
node_a, peer_b,
node_b_id,
node_b,
dial_info_filter, dial_info_filter,
sequencing, sequencing,
) )
@ -381,16 +387,6 @@ impl RoutingTable {
self.inner.read().get_own_peer_info(routing_domain) self.inner.read().get_own_peer_info(routing_domain)
} }
/// Return a copy of our node's signednodeinfo
pub fn get_own_signed_node_info(&self, routing_domain: RoutingDomain) -> SignedDirectNodeInfo {
self.inner.read().get_own_signed_node_info(routing_domain)
}
/// Return a copy of our node's nodeinfo
pub fn get_own_node_info(&self, routing_domain: RoutingDomain) -> NodeInfo {
self.inner.read().get_own_node_info(routing_domain)
}
/// If we have a valid network class in this routing domain, then our 'NodeInfo' is valid /// If we have a valid network class in this routing domain, then our 'NodeInfo' is valid
pub fn has_valid_own_node_info(&self, routing_domain: RoutingDomain) -> bool { pub fn has_valid_own_node_info(&self, routing_domain: RoutingDomain) -> bool {
self.inner.read().has_valid_own_node_info(routing_domain) self.inner.read().has_valid_own_node_info(routing_domain)
@ -525,7 +521,7 @@ impl RoutingTable {
&self, &self,
routing_domain: RoutingDomain, routing_domain: RoutingDomain,
node_id: DHTKey, node_id: DHTKey,
signed_node_info: SignedDirectNodeInfo, signed_node_info: SignedNodeInfo,
allow_invalid: bool, allow_invalid: bool,
) -> Option<NodeRef> { ) -> Option<NodeRef> {
self.inner.write().register_node_with_signed_node_info( self.inner.write().register_node_with_signed_node_info(

View File

@ -116,10 +116,10 @@ pub trait NodeRefBase: Sized {
e.update_node_status(node_status); e.update_node_status(node_status);
}); });
} }
fn min_max_version(&self) -> Option<(u8, u8)> { fn min_max_version(&self) -> Option<VersionRange> {
self.operate(|_rti, e| e.min_max_version()) self.operate(|_rti, e| e.min_max_version())
} }
fn set_min_max_version(&self, min_max_version: (u8, u8)) { fn set_min_max_version(&self, min_max_version: VersionRange) {
self.operate_mut(|_rti, e| e.set_min_max_version(min_max_version)) self.operate_mut(|_rti, e| e.set_min_max_version(min_max_version))
} }
fn state(&self, cur_ts: u64) -> BucketEntryState { fn state(&self, cur_ts: u64) -> BucketEntryState {
@ -170,26 +170,24 @@ pub trait NodeRefBase: Sized {
} }
fn relay(&self, routing_domain: RoutingDomain) -> Option<NodeRef> { fn relay(&self, routing_domain: RoutingDomain) -> Option<NodeRef> {
self.operate_mut(|rti, e| { self.operate_mut(|rti, e| {
let opt_target_rpi = e e.signed_node_info(routing_domain)
.node_info(routing_domain) .and_then(|n| n.relay_peer_info())
.map(|n| n.relay_peer_info.as_ref().map(|pi| pi.as_ref().clone())) .and_then(|t| {
.flatten(); // If relay is ourselves, then return None, because we can't relay through ourselves
opt_target_rpi.and_then(|t| { // and to contact this node we should have had an existing inbound connection
// If relay is ourselves, then return None, because we can't relay through ourselves if t.node_id.key == rti.unlocked_inner.node_id {
// and to contact this node we should have had an existing inbound connection return None;
if t.node_id.key == rti.unlocked_inner.node_id { }
return None;
}
// Register relay node and return noderef // Register relay node and return noderef
rti.register_node_with_signed_node_info( rti.register_node_with_signed_node_info(
self.routing_table(), self.routing_table(),
routing_domain, routing_domain,
t.node_id.key, t.node_id.key,
t.signed_node_info, t.signed_node_info,
false, false,
) )
}) })
}) })
} }
@ -301,8 +299,8 @@ pub trait NodeRefBase: Sized {
fn has_any_dial_info(&self) -> bool { fn has_any_dial_info(&self) -> bool {
self.operate(|_rti, e| { self.operate(|_rti, e| {
for rtd in RoutingDomain::all() { for rtd in RoutingDomain::all() {
if let Some(ni) = e.node_info(rtd) { if let Some(sni) = e.signed_node_info(rtd) {
if ni.has_any_dial_info() { if sni.has_any_dial_info() {
return true; return true;
} }
} }

View File

@ -114,10 +114,10 @@ fn route_hops_to_hop_cache(hops: &[DHTKey]) -> Vec<u8> {
} }
/// get the hop cache key for a particular route permutation /// get the hop cache key for a particular route permutation
fn route_permutation_to_hop_cache(nodes: &[(DHTKey, NodeInfo)], perm: &[usize]) -> Vec<u8> { fn route_permutation_to_hop_cache(nodes: &[PeerInfo], perm: &[usize]) -> Vec<u8> {
let mut cache: Vec<u8> = Vec::with_capacity(perm.len() * DHT_KEY_LENGTH); let mut cache: Vec<u8> = Vec::with_capacity(perm.len() * DHT_KEY_LENGTH);
for n in perm { for n in perm {
cache.extend_from_slice(&nodes[*n].0.bytes) cache.extend_from_slice(&nodes[*n].node_id.key.bytes)
} }
cache cache
} }
@ -422,12 +422,12 @@ impl RouteSpecStore {
// Exclude nodes with no publicinternet nodeinfo, or incompatible nodeinfo or node status won't route // Exclude nodes with no publicinternet nodeinfo, or incompatible nodeinfo or node status won't route
v.with(rti, move |_rti, e| { v.with(rti, move |_rti, e| {
let node_info_ok = if let Some(ni) = e.node_info(RoutingDomain::PublicInternet) let node_info_ok =
{ if let Some(sni) = e.signed_node_info(RoutingDomain::PublicInternet) {
ni.has_sequencing_matched_dial_info(sequencing) sni.has_sequencing_matched_dial_info(sequencing)
} else { } else {
false false
}; };
let node_status_ok = let node_status_ok =
if let Some(ns) = e.node_status(RoutingDomain::PublicInternet) { if let Some(ns) = e.node_status(RoutingDomain::PublicInternet) {
ns.will_route() ns.will_route()
@ -495,20 +495,15 @@ impl RouteSpecStore {
}); });
cmpout cmpout
}; };
let transform = |rti: &RoutingTableInner, let transform =
k: DHTKey, |rti: &RoutingTableInner, k: DHTKey, v: Option<Arc<BucketEntry>>| -> PeerInfo {
v: Option<Arc<BucketEntry>>| // Return the peerinfo for that key
-> (DHTKey, NodeInfo) {
// Return the key and the nodeinfo for that key
(
k,
v.unwrap().with(rti, |_rti, e| { v.unwrap().with(rti, |_rti, e| {
e.node_info(RoutingDomain::PublicInternet.into()) e.make_peer_info(k, RoutingDomain::PublicInternet.into())
.unwrap() .unwrap()
.clone() .clone()
}), })
) };
};
// Pull the whole routing table in sorted order // Pull the whole routing table in sorted order
let node_count = rti.get_entry_count( let node_count = rti.get_entry_count(
@ -536,18 +531,15 @@ impl RouteSpecStore {
// Ensure this route is viable by checking that each node can contact the next one // Ensure this route is viable by checking that each node can contact the next one
if directions.contains(Direction::Outbound) { if directions.contains(Direction::Outbound) {
let our_node_info = rti.get_own_node_info(RoutingDomain::PublicInternet); let our_peer_info = rti.get_own_peer_info(RoutingDomain::PublicInternet);
let our_node_id = rti.node_id(); let mut previous_node = &our_peer_info;
let mut previous_node = &(our_node_id, our_node_info);
let mut reachable = true; let mut reachable = true;
for n in permutation { for n in permutation {
let current_node = nodes.get(*n).unwrap(); let current_node = nodes.get(*n).unwrap();
let cm = rti.get_contact_method( let cm = rti.get_contact_method(
RoutingDomain::PublicInternet, RoutingDomain::PublicInternet,
&previous_node.0, previous_node,
&previous_node.1, current_node,
&current_node.0,
&current_node.1,
DialInfoFilter::all(), DialInfoFilter::all(),
sequencing, sequencing,
); );
@ -562,18 +554,15 @@ impl RouteSpecStore {
} }
} }
if directions.contains(Direction::Inbound) { if directions.contains(Direction::Inbound) {
let our_node_info = rti.get_own_node_info(RoutingDomain::PublicInternet); let our_peer_info = rti.get_own_peer_info(RoutingDomain::PublicInternet);
let our_node_id = rti.node_id(); let mut next_node = &our_peer_info;
let mut next_node = &(our_node_id, our_node_info);
let mut reachable = true; let mut reachable = true;
for n in permutation.iter().rev() { for n in permutation.iter().rev() {
let current_node = nodes.get(*n).unwrap(); let current_node = nodes.get(*n).unwrap();
let cm = rti.get_contact_method( let cm = rti.get_contact_method(
RoutingDomain::PublicInternet, RoutingDomain::PublicInternet,
&next_node.0, next_node,
&next_node.1, current_node,
&current_node.0,
&current_node.1,
DialInfoFilter::all(), DialInfoFilter::all(),
sequencing, sequencing,
); );
@ -609,11 +598,11 @@ impl RouteSpecStore {
} }
// Got a unique route, lets build the detail, register it, and return it // Got a unique route, lets build the detail, register it, and return it
let hops = route_nodes.iter().map(|v| nodes[*v].0).collect(); let hops: Vec<DHTKey> = route_nodes.iter().map(|v| nodes[*v].node_id.key).collect();
let hop_node_refs = route_nodes let hop_node_refs = hops
.iter() .iter()
.map(|v| { .map(|k| {
rti.lookup_node_ref(self.unlocked_inner.routing_table.clone(), nodes[*v].0) rti.lookup_node_ref(self.unlocked_inner.routing_table.clone(), *k)
.unwrap() .unwrap()
}) })
.collect(); .collect();

View File

@ -111,18 +111,31 @@ impl RoutingDomainDetailCommon {
dial_info_detail_list: self.dial_info_details.clone(), dial_info_detail_list: self.dial_info_details.clone(),
}; };
let relay_peer_info = self let relay_info = self
.relay_node .relay_node
.as_ref() .as_ref()
.and_then(|rn| rn.locked(rti).make_peer_info(self.routing_domain)); .and_then(|rn| {
let opt_relay_pi = rn.locked(rti).make_peer_info(self.routing_domain);
if let Some(relay_pi) = opt_relay_pi {
match relay_pi.signed_node_info {
SignedNodeInfo::Direct(d) => Some((relay_pi.node_id, d)),
SignedNodeInfo::Relayed(_) => {
warn!("relay node should not have a relay itself! if this happens, a relay updated its signed node info and became a relay, which should cause the relay to be dropped");
None
},
}
} else {
None
}
});
let signed_node_info = match relay_peer_info { let signed_node_info = match relay_info {
Some(relay_pi) => SignedNodeInfo::Relayed( Some((relay_id, relay_sdni)) => SignedNodeInfo::Relayed(
SignedRelayedNodeInfo::with_secret( SignedRelayedNodeInfo::with_secret(
NodeId::new(rti.unlocked_inner.node_id), NodeId::new(rti.unlocked_inner.node_id),
node_info, node_info,
relay_pi.node_id, relay_id,
relay_pi.signed_node_info, relay_sdni,
&rti.unlocked_inner.node_id_secret, &rti.unlocked_inner.node_id_secret,
) )
.unwrap(), .unwrap(),
@ -133,7 +146,7 @@ impl RoutingDomainDetailCommon {
node_info, node_info,
&rti.unlocked_inner.node_id_secret, &rti.unlocked_inner.node_id_secret,
) )
.unwrap(), .unwrap()
), ),
}; };
@ -147,27 +160,8 @@ impl RoutingDomainDetailCommon {
let mut cpi = self.cached_peer_info.lock(); let mut cpi = self.cached_peer_info.lock();
if cpi.is_none() { if cpi.is_none() {
// Regenerate peer info // Regenerate peer info
let pi = PeerInfo::new( let pi = self.make_peer_info(rti);
NodeId::new(rti.unlocked_inner.node_id),
SignedDirectNodeInfo::with_secret(
NodeInfo {
network_class: self.network_class.unwrap_or(NetworkClass::Invalid),
outbound_protocols: self.outbound_protocols,
address_types: self.address_types,
min_version: MIN_CRYPTO_VERSION,
max_version: MAX_CRYPTO_VERSION,
dial_info_detail_list: self.dial_info_details.clone(),
relay_peer_info: self.relay_node.as_ref().and_then(|rn| {
rn.locked(rti)
.make_peer_info(self.routing_domain)
.map(Box::new)
}),
},
NodeId::new(rti.unlocked_inner.node_id),
&rti.unlocked_inner.node_id_secret,
)
.unwrap(),
);
// Cache the peer info // Cache the peer info
*cpi = Some(pi); *cpi = Some(pi);
} }
@ -204,10 +198,8 @@ pub trait RoutingDomainDetail {
fn get_contact_method( fn get_contact_method(
&self, &self,
rti: &RoutingTableInner, rti: &RoutingTableInner,
node_a_id: &DHTKey, peer_a: &PeerInfo,
node_a: &NodeInfo, peer_b: &PeerInfo,
node_b_id: &DHTKey,
node_b: &NodeInfo,
dial_info_filter: DialInfoFilter, dial_info_filter: DialInfoFilter,
sequencing: Sequencing, sequencing: Sequencing,
) -> ContactMethod; ) -> ContactMethod;
@ -280,13 +272,15 @@ impl RoutingDomainDetail for PublicInternetRoutingDomainDetail {
fn get_contact_method( fn get_contact_method(
&self, &self,
_rti: &RoutingTableInner, _rti: &RoutingTableInner,
node_a_id: &DHTKey, peer_a: &PeerInfo,
node_a: &NodeInfo, peer_b: &PeerInfo,
node_b_id: &DHTKey,
node_b: &NodeInfo,
dial_info_filter: DialInfoFilter, dial_info_filter: DialInfoFilter,
sequencing: Sequencing, sequencing: Sequencing,
) -> ContactMethod { ) -> ContactMethod {
// Get the nodeinfos for convenience
let node_a = peer_a.signed_node_info.node_info();
let node_b = peer_b.signed_node_info.node_info();
// Get the best match dial info for node B if we have it // Get the best match dial info for node B if we have it
if let Some(target_did) = if let Some(target_did) =
first_filtered_dial_info_detail(node_a, node_b, &dial_info_filter, sequencing) first_filtered_dial_info_detail(node_a, node_b, &dial_info_filter, sequencing)
@ -298,17 +292,18 @@ impl RoutingDomainDetail for PublicInternetRoutingDomainDetail {
} }
// Get the target's inbound relay, it must have one or it is not reachable // Get the target's inbound relay, it must have one or it is not reachable
if let Some(inbound_relay) = &node_b.relay_peer_info { if let Some(node_b_relay) = peer_b.signed_node_info.relay_info() {
let node_b_relay_id = peer_b.signed_node_info.relay_id().unwrap();
// Note that relay_peer_info could be node_a, in which case a connection already exists // Note that relay_peer_info could be node_a, in which case a connection already exists
// and we shouldn't have even gotten here // and we shouldn't have even gotten here
if inbound_relay.node_id.key == *node_a_id { if node_b_relay_id.key == peer_a.node_id.key {
return ContactMethod::Existing; return ContactMethod::Existing;
} }
// Can node A reach the inbound relay directly? // Can node A reach the inbound relay directly?
if first_filtered_dial_info_detail( if first_filtered_dial_info_detail(
node_a, node_a,
&inbound_relay.signed_node_info.node_info, node_b_relay,
&dial_info_filter, &dial_info_filter,
sequencing, sequencing,
) )
@ -332,8 +327,8 @@ impl RoutingDomainDetail for PublicInternetRoutingDomainDetail {
// Can we receive a direct reverse connection? // Can we receive a direct reverse connection?
if !reverse_did.class.requires_signal() { if !reverse_did.class.requires_signal() {
return ContactMethod::SignalReverse( return ContactMethod::SignalReverse(
inbound_relay.node_id.key, node_b_relay_id.key,
*node_b_id, peer_b.node_id.key,
); );
} }
} }
@ -364,8 +359,8 @@ impl RoutingDomainDetail for PublicInternetRoutingDomainDetail {
{ {
// The target and ourselves have a udp dialinfo that they can reach // The target and ourselves have a udp dialinfo that they can reach
return ContactMethod::SignalHolePunch( return ContactMethod::SignalHolePunch(
inbound_relay.node_id.key, node_b_relay_id.key,
*node_b_id, peer_a.node_id.key,
); );
} }
} }
@ -373,28 +368,30 @@ impl RoutingDomainDetail for PublicInternetRoutingDomainDetail {
// Otherwise we have to inbound relay // Otherwise we have to inbound relay
} }
return ContactMethod::InboundRelay(inbound_relay.node_id.key); return ContactMethod::InboundRelay(node_b_relay_id.key);
} }
} }
} }
// If the node B has no direct dial info, it needs to have an inbound relay // If the node B has no direct dial info, it needs to have an inbound relay
else if let Some(inbound_relay) = &node_b.relay_peer_info { else if let Some(node_b_relay) = peer_b.signed_node_info.relay_info() {
let node_b_relay_id = peer_b.signed_node_info.relay_id().unwrap();
// Can we reach the full relay? // Can we reach the full relay?
if first_filtered_dial_info_detail( if first_filtered_dial_info_detail(
node_a, node_a,
&inbound_relay.signed_node_info.node_info, &node_b_relay,
&dial_info_filter, &dial_info_filter,
sequencing, sequencing,
) )
.is_some() .is_some()
{ {
return ContactMethod::InboundRelay(inbound_relay.node_id.key); return ContactMethod::InboundRelay(node_b_relay_id.key);
} }
} }
// If node A can't reach the node by other means, it may need to use its own relay // If node A can't reach the node by other means, it may need to use its own relay
if let Some(outbound_relay) = &node_a.relay_peer_info { if let Some(node_a_relay_id) = peer_a.signed_node_info.relay_id() {
return ContactMethod::OutboundRelay(outbound_relay.node_id.key); return ContactMethod::OutboundRelay(node_a_relay_id.key);
} }
ContactMethod::Unreachable ContactMethod::Unreachable
@ -450,18 +447,16 @@ impl RoutingDomainDetail for LocalNetworkRoutingDomainDetail {
fn get_contact_method( fn get_contact_method(
&self, &self,
_rti: &RoutingTableInner, _rti: &RoutingTableInner,
_node_a_id: &DHTKey, peer_a: &PeerInfo,
node_a: &NodeInfo, peer_b: &PeerInfo,
_node_b_id: &DHTKey,
node_b: &NodeInfo,
dial_info_filter: DialInfoFilter, dial_info_filter: DialInfoFilter,
sequencing: Sequencing, sequencing: Sequencing,
) -> ContactMethod { ) -> ContactMethod {
// Scope the filter down to protocols node A can do outbound // Scope the filter down to protocols node A can do outbound
let dial_info_filter = dial_info_filter.filtered( let dial_info_filter = dial_info_filter.filtered(
&DialInfoFilter::all() &DialInfoFilter::all()
.with_address_type_set(node_a.address_types) .with_address_type_set(peer_a.signed_node_info.node_info().address_types)
.with_protocol_type_set(node_a.outbound_protocols), .with_protocol_type_set(peer_a.signed_node_info.node_info().outbound_protocols),
); );
// Get first filtered dialinfo // Get first filtered dialinfo
@ -485,7 +480,7 @@ impl RoutingDomainDetail for LocalNetworkRoutingDomainDetail {
let filter = |did: &DialInfoDetail| did.matches_filter(&dial_info_filter); let filter = |did: &DialInfoDetail| did.matches_filter(&dial_info_filter);
let opt_target_did = node_b.first_filtered_dial_info_detail(sort, filter); let opt_target_did = peer_b.signed_node_info.node_info().first_filtered_dial_info_detail(sort, filter);
if let Some(target_did) = opt_target_did { if let Some(target_did) = opt_target_did {
return ContactMethod::Direct(target_did.dial_info); return ContactMethod::Direct(target_did.dial_info);
} }

View File

@ -191,9 +191,20 @@ impl RoutingTableInner {
return false; return false;
} }
} }
true
}
pub fn signed_node_info_is_valid_in_routing_domain(
&self,
routing_domain: RoutingDomain,
signed_node_info: &SignedNodeInfo,
) -> bool {
if !self.node_info_is_valid_in_routing_domain(routing_domain, signed_node_info.node_info())
{
return false;
}
// Ensure the relay is also valid in this routing domain if it is provided // Ensure the relay is also valid in this routing domain if it is provided
if let Some(relay_peer_info) = node_info.relay_peer_info.as_ref() { if let Some(relay_ni) = signed_node_info.relay_info() {
let relay_ni = &relay_peer_info.signed_node_info.node_info;
if !self.node_info_is_valid_in_routing_domain(routing_domain, relay_ni) { if !self.node_info_is_valid_in_routing_domain(routing_domain, relay_ni) {
return false; return false;
} }
@ -205,23 +216,13 @@ impl RoutingTableInner {
pub fn get_contact_method( pub fn get_contact_method(
&self, &self,
routing_domain: RoutingDomain, routing_domain: RoutingDomain,
node_a_id: &DHTKey, peer_a: &PeerInfo,
node_a: &NodeInfo, peer_b: &PeerInfo,
node_b_id: &DHTKey,
node_b: &NodeInfo,
dial_info_filter: DialInfoFilter, dial_info_filter: DialInfoFilter,
sequencing: Sequencing, sequencing: Sequencing,
) -> ContactMethod { ) -> ContactMethod {
self.with_routing_domain(routing_domain, |rdd| { self.with_routing_domain(routing_domain, |rdd| {
rdd.get_contact_method( rdd.get_contact_method(self, peer_a, peer_b, dial_info_filter, sequencing)
self,
node_a_id,
node_a,
node_b_id,
node_b,
dial_info_filter,
sequencing,
)
}) })
} }
@ -252,22 +253,6 @@ impl RoutingTableInner {
}) })
} }
/// Return a copy of our node's signednodeinfo
pub fn get_own_signed_node_info(&self, routing_domain: RoutingDomain) -> SignedDirectNodeInfo {
self.with_routing_domain(routing_domain, |rdd| {
rdd.common()
.with_peer_info(self, |pi| pi.signed_node_info.clone())
})
}
/// Return a copy of our node's nodeinfo
pub fn get_own_node_info(&self, routing_domain: RoutingDomain) -> NodeInfo {
self.with_routing_domain(routing_domain, |rdd| {
rdd.common()
.with_peer_info(self, |pi| pi.signed_node_info.node_info.clone())
})
}
/// Return our currently registered network class /// Return our currently registered network class
pub fn has_valid_own_node_info(&self, routing_domain: RoutingDomain) -> bool { pub fn has_valid_own_node_info(&self, routing_domain: RoutingDomain) -> bool {
self.with_routing_domain(routing_domain, |rdd| rdd.common().has_valid_own_node_info()) self.with_routing_domain(routing_domain, |rdd| rdd.common().has_valid_own_node_info())
@ -662,7 +647,7 @@ impl RoutingTableInner {
outer_self: RoutingTable, outer_self: RoutingTable,
routing_domain: RoutingDomain, routing_domain: RoutingDomain,
node_id: DHTKey, node_id: DHTKey,
signed_node_info: SignedDirectNodeInfo, signed_node_info: SignedNodeInfo,
allow_invalid: bool, allow_invalid: bool,
) -> Option<NodeRef> { ) -> Option<NodeRef> {
// validate signed node info is not something malicious // validate signed node info is not something malicious
@ -670,8 +655,8 @@ impl RoutingTableInner {
log_rtab!(debug "can't register own node id in routing table"); log_rtab!(debug "can't register own node id in routing table");
return None; return None;
} }
if let Some(rpi) = &signed_node_info.node_info.relay_peer_info { if let Some(relay_id) = signed_node_info.relay_id() {
if rpi.node_id.key == node_id { if relay_id.key == node_id {
log_rtab!(debug "node can not be its own relay"); log_rtab!(debug "node can not be its own relay");
return None; return None;
} }
@ -683,8 +668,7 @@ impl RoutingTableInner {
return None; return None;
} }
// verify signed node info is valid in this routing domain // verify signed node info is valid in this routing domain
if !self if !self.signed_node_info_is_valid_in_routing_domain(routing_domain, &signed_node_info)
.node_info_is_valid_in_routing_domain(routing_domain, &signed_node_info.node_info)
{ {
log_rtab!(debug "signed node info for {} not valid in the {:?} routing domain", node_id, routing_domain); log_rtab!(debug "signed node info for {} not valid in the {:?} routing domain", node_id, routing_domain);
return None; return None;

View File

@ -336,10 +336,14 @@ impl RPCProcessor {
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////
/// Determine if a NodeInfo can be placed into the specified routing domain /// Determine if a SignedNodeInfo can be placed into the specified routing domain
fn filter_node_info(&self, routing_domain: RoutingDomain, node_info: &NodeInfo) -> bool { fn filter_node_info(
&self,
routing_domain: RoutingDomain,
signed_node_info: &SignedNodeInfo,
) -> bool {
let routing_table = self.routing_table(); let routing_table = self.routing_table();
routing_table.node_info_is_valid_in_routing_domain(routing_domain, &node_info) routing_table.signed_node_info_is_valid_in_routing_domain(routing_domain, &signed_node_info)
} }
////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////
@ -619,7 +623,7 @@ impl RPCProcessor {
// routing table caching when it is okay to do so // routing table caching when it is okay to do so
// This is only done in the PublicInternet routing domain because // This is only done in the PublicInternet routing domain because
// as far as we can tell this is the only domain that will really benefit // as far as we can tell this is the only domain that will really benefit
fn get_sender_signed_node_info(&self, dest: &Destination) -> Option<SignedDirectNodeInfo> { fn get_sender_signed_node_info(&self, dest: &Destination) -> Option<SignedNodeInfo> {
// Don't do this if the sender is to remain private // Don't do this if the sender is to remain private
// Otherwise we would be attaching the original sender's identity to the final destination, // Otherwise we would be attaching the original sender's identity to the final destination,
// thus defeating the purpose of the safety route entirely :P // thus defeating the purpose of the safety route entirely :P

View File

@ -50,10 +50,7 @@ impl RPCProcessor {
// Verify peers are in the correct peer scope // Verify peers are in the correct peer scope
for peer_info in &find_node_a.peers { for peer_info in &find_node_a.peers {
if !self.filter_node_info( if !self.filter_node_info(RoutingDomain::PublicInternet, &peer_info.signed_node_info) {
RoutingDomain::PublicInternet,
&peer_info.signed_node_info.node_info,
) {
return Err(RPCError::invalid_format( return Err(RPCError::invalid_format(
"find_node response has invalid peer scope", "find_node response has invalid peer scope",
)); ));

View File

@ -11,7 +11,8 @@ impl RPCProcessor {
// Get the signed node info for the desired routing domain to send update with // Get the signed node info for the desired routing domain to send update with
let signed_node_info = self let signed_node_info = self
.routing_table() .routing_table()
.get_own_signed_node_info(routing_domain); .get_own_peer_info(routing_domain)
.signed_node_info;
let node_info_update = RPCOperationNodeInfoUpdate { signed_node_info }; let node_info_update = RPCOperationNodeInfoUpdate { signed_node_info };
let statement = RPCStatement::new(RPCStatementDetail::NodeInfoUpdate(node_info_update)); let statement = RPCStatement::new(RPCStatementDetail::NodeInfoUpdate(node_info_update));
@ -50,7 +51,7 @@ impl RPCProcessor {
}; };
// Update our routing table with signed node info // Update our routing table with signed node info
if !self.filter_node_info(routing_domain, &node_info_update.signed_node_info.node_info) { if !self.filter_node_info(routing_domain, &node_info_update.signed_node_info) {
log_rpc!(debug "node info doesn't belong in {:?} routing domain: {}", routing_domain, sender_node_id); log_rpc!(debug "node info doesn't belong in {:?} routing domain: {}", routing_domain, sender_node_id);
return Ok(()); return Ok(());
} }

View File

@ -775,6 +775,68 @@ pub struct NodeInfo {
pub dial_info_detail_list: Vec<DialInfoDetail>, pub dial_info_detail_list: Vec<DialInfoDetail>,
} }
impl NodeInfo {
pub fn first_filtered_dial_info_detail<S, F>(
&self,
sort: Option<S>,
filter: F,
) -> Option<DialInfoDetail>
where
S: Fn(&DialInfoDetail, &DialInfoDetail) -> std::cmp::Ordering,
F: Fn(&DialInfoDetail) -> bool,
{
if let Some(sort) = sort {
let mut dids = self.dial_info_detail_list.clone();
dids.sort_by(sort);
for did in dids {
if filter(&did) {
return Some(did);
}
}
} else {
for did in &self.dial_info_detail_list {
if filter(did) {
return Some(did.clone());
}
}
};
None
}
pub fn all_filtered_dial_info_details<S, F>(
&self,
sort: Option<S>,
filter: F,
) -> Vec<DialInfoDetail>
where
S: Fn(&DialInfoDetail, &DialInfoDetail) -> std::cmp::Ordering,
F: Fn(&DialInfoDetail) -> bool,
{
let mut dial_info_detail_list = Vec::new();
if let Some(sort) = sort {
let mut dids = self.dial_info_detail_list.clone();
dids.sort_by(sort);
for did in dids {
if filter(&did) {
dial_info_detail_list.push(did);
}
}
} else {
for did in &self.dial_info_detail_list {
if filter(did) {
dial_info_detail_list.push(did.clone());
}
}
};
dial_info_detail_list
}
pub fn has_direct_dial_info(&self) -> bool {
!self.dial_info_detail_list.is_empty()
}
}
#[allow(clippy::derive_hash_xor_eq)] #[allow(clippy::derive_hash_xor_eq)]
#[derive( #[derive(
Debug, Debug,
@ -2046,6 +2108,53 @@ impl SignedNodeInfo {
SignedNodeInfo::Relayed(r) => Some(&r.relay_info.node_info), SignedNodeInfo::Relayed(r) => Some(&r.relay_info.node_info),
} }
} }
pub fn relay_peer_info(&self) -> Option<PeerInfo> {
match self {
SignedNodeInfo::Direct(d) => None,
SignedNodeInfo::Relayed(r) => Some(PeerInfo::new(
r.relay_id.clone(),
SignedNodeInfo::Direct(r.relay_info.clone()),
)),
}
}
pub fn has_any_dial_info(&self) -> bool {
self.node_info().has_direct_dial_info()
|| self
.relay_info()
.map(|relay_ni| relay_ni.has_direct_dial_info())
.unwrap_or_default()
}
pub fn has_sequencing_matched_dial_info(&self, sequencing: Sequencing) -> bool {
// Check our dial info
for did in &self.node_info().dial_info_detail_list {
match sequencing {
Sequencing::NoPreference | Sequencing::PreferOrdered => return true,
Sequencing::EnsureOrdered => {
if did.dial_info.protocol_type().is_connection_oriented() {
return true;
}
}
}
}
// Check our relay if we have one
return self
.relay_info()
.map(|relay_ni| {
for did in &relay_ni.dial_info_detail_list {
match sequencing {
Sequencing::NoPreference | Sequencing::PreferOrdered => return true,
Sequencing::EnsureOrdered => {
if did.dial_info.protocol_type().is_connection_oriented() {
return true;
}
}
}
}
false
})
.unwrap_or_default();
}
} }
#[derive(Clone, Debug, Serialize, Deserialize, RkyvArchive, RkyvSerialize, RkyvDeserialize)] #[derive(Clone, Debug, Serialize, Deserialize, RkyvArchive, RkyvSerialize, RkyvDeserialize)]
@ -2066,157 +2175,62 @@ impl PeerInfo {
impl PeerInfo { impl PeerInfo {
/* /*
pub fn first_filtered_dial_info_detail<S, F>(
&self,
sort: Option<S>,
filter: F,
) -> Option<DialInfoDetail>
where
S: Fn(&DialInfoDetail, &DialInfoDetail) -> std::cmp::Ordering,
F: Fn(&DialInfoDetail) -> bool,
{
if let Some(sort) = sort {
let mut dids = self.dial_info_detail_list.clone();
dids.sort_by(sort);
for did in dids {
if filter(&did) {
return Some(did);
}
}
} else {
for did in &self.dial_info_detail_list {
if filter(did) {
return Some(did.clone());
}
}
};
None
}
pub fn all_filtered_dial_info_details<S, F>( xxx move these back to NodeInfo
&self,
sort: Option<S>,
filter: F,
) -> Vec<DialInfoDetail>
where
S: Fn(&DialInfoDetail, &DialInfoDetail) -> std::cmp::Ordering,
F: Fn(&DialInfoDetail) -> bool,
{
let mut dial_info_detail_list = Vec::new();
if let Some(sort) = sort { pub fn has_direct_dial_info(&self) -> bool {
let mut dids = self.dial_info_detail_list.clone(); !self.dial_info_detail_list.is_empty()
dids.sort_by(sort);
for did in dids {
if filter(&did) {
dial_info_detail_list.push(did);
}
}
} else {
for did in &self.dial_info_detail_list {
if filter(did) {
dial_info_detail_list.push(did.clone());
}
}
};
dial_info_detail_list
}
pub fn has_any_dial_info(&self) -> bool {
!self.dial_info_detail_list.is_empty()
|| !self
.relay_peer_info
.as_ref()
.map(|rpi| rpi.signed_node_info.node_info.has_direct_dial_info())
.unwrap_or_default()
}
pub fn has_sequencing_matched_dial_info(&self, sequencing: Sequencing) -> bool {
// Check our dial info
for did in &self.dial_info_detail_list {
match sequencing {
Sequencing::NoPreference | Sequencing::PreferOrdered => return true,
Sequencing::EnsureOrdered => {
if did.dial_info.protocol_type().is_connection_oriented() {
return true;
}
}
}
} }
// Check our relay if we have one
return self // Is some relay required either for signal or inbound relay or outbound relay?
.relay_peer_info pub fn requires_relay(&self) -> bool {
.as_ref() match self.network_class {
.map(|rpi| { NetworkClass::InboundCapable => {
let relay_ni = &rpi.signed_node_info.node_info; for did in &self.dial_info_detail_list {
for did in &relay_ni.dial_info_detail_list { if did.class.requires_relay() {
match sequencing { return true;
Sequencing::NoPreference | Sequencing::PreferOrdered => return true,
Sequencing::EnsureOrdered => {
if did.dial_info.protocol_type().is_connection_oriented() {
return true;
}
} }
} }
} }
false NetworkClass::OutboundOnly => {
}) return true;
.unwrap_or_default();
}
pub fn has_direct_dial_info(&self) -> bool {
!self.dial_info_detail_list.is_empty()
}
// Is some relay required either for signal or inbound relay or outbound relay?
pub fn requires_relay(&self) -> bool {
match self.network_class {
NetworkClass::InboundCapable => {
for did in &self.dial_info_detail_list {
if did.class.requires_relay() {
return true;
}
} }
NetworkClass::WebApp => {
return true;
}
NetworkClass::Invalid => {}
} }
NetworkClass::OutboundOnly => { false
return true;
}
NetworkClass::WebApp => {
return true;
}
NetworkClass::Invalid => {}
} }
false
}
// Can this node assist with signalling? Yes but only if it doesn't require signalling, itself. // Can this node assist with signalling? Yes but only if it doesn't require signalling, itself.
pub fn can_signal(&self) -> bool { pub fn can_signal(&self) -> bool {
// Must be inbound capable // Must be inbound capable
if !matches!(self.network_class, NetworkClass::InboundCapable) { if !matches!(self.network_class, NetworkClass::InboundCapable) {
return false;
}
// Do any of our dial info require signalling? if so, we can't offer signalling
for did in &self.dial_info_detail_list {
if did.class.requires_signal() {
return false; return false;
} }
// Do any of our dial info require signalling? if so, we can't offer signalling
for did in &self.dial_info_detail_list {
if did.class.requires_signal() {
return false;
}
}
true
} }
true
}
// Can this node relay be an inbound relay? // Can this node relay be an inbound relay?
pub fn can_inbound_relay(&self) -> bool { pub fn can_inbound_relay(&self) -> bool {
// For now this is the same // For now this is the same
self.can_signal() self.can_signal()
} }
// Is this node capable of validating dial info // Is this node capable of validating dial info
pub fn can_validate_dial_info(&self) -> bool { pub fn can_validate_dial_info(&self) -> bool {
// For now this is the same // For now this is the same
self.can_signal() self.can_signal()
} }
*/ */
} }
#[derive( #[derive(