mirror of
https://mirror.suhoan.cn/https://github.com/EasyTier/EasyTier.git
synced 2025-12-12 20:57:26 +08:00
fix ospf ipv4 map error when ipv4 conflicted and changed (#1359)
This commit is contained in:
@@ -720,15 +720,20 @@ struct NextHopInfo {
|
|||||||
}
|
}
|
||||||
// dst_peer_id -> (next_hop_peer_id, cost, path_len)
|
// dst_peer_id -> (next_hop_peer_id, cost, path_len)
|
||||||
type NextHopMap = DashMap<PeerId, NextHopInfo>;
|
type NextHopMap = DashMap<PeerId, NextHopInfo>;
|
||||||
|
#[derive(Debug, Clone, Copy)]
|
||||||
|
struct PeerIdAndVersion {
|
||||||
|
peer_id: PeerId,
|
||||||
|
version: Version,
|
||||||
|
}
|
||||||
|
|
||||||
// computed with SyncedRouteInfo. used to get next hop.
|
// computed with SyncedRouteInfo. used to get next hop.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
struct RouteTable {
|
struct RouteTable {
|
||||||
peer_infos: DashMap<PeerId, RoutePeerInfo>,
|
peer_infos: DashMap<PeerId, RoutePeerInfo>,
|
||||||
next_hop_map: NextHopMap,
|
next_hop_map: NextHopMap,
|
||||||
ipv4_peer_id_map: DashMap<Ipv4Addr, PeerId>,
|
ipv4_peer_id_map: DashMap<Ipv4Addr, PeerIdAndVersion>,
|
||||||
ipv6_peer_id_map: DashMap<Ipv6Addr, PeerId>,
|
ipv6_peer_id_map: DashMap<Ipv6Addr, PeerIdAndVersion>,
|
||||||
cidr_peer_id_map: DashMap<cidr::IpCidr, PeerId>,
|
cidr_peer_id_map: DashMap<cidr::IpCidr, PeerIdAndVersion>,
|
||||||
next_hop_map_version: AtomicVersion,
|
next_hop_map_version: AtomicVersion,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -834,15 +839,15 @@ impl RouteTable {
|
|||||||
});
|
});
|
||||||
self.ipv4_peer_id_map.retain(|_, v| {
|
self.ipv4_peer_id_map.retain(|_, v| {
|
||||||
// remove ipv4 map for peers we cannot reach.
|
// remove ipv4 map for peers we cannot reach.
|
||||||
self.next_hop_map.contains_key(v)
|
self.next_hop_map.contains_key(&v.peer_id)
|
||||||
});
|
});
|
||||||
self.ipv6_peer_id_map.retain(|_, v| {
|
self.ipv6_peer_id_map.retain(|_, v| {
|
||||||
// remove ipv6 map for peers we cannot reach.
|
// remove ipv6 map for peers we cannot reach.
|
||||||
self.next_hop_map.contains_key(v)
|
self.next_hop_map.contains_key(&v.peer_id)
|
||||||
});
|
});
|
||||||
self.cidr_peer_id_map.retain(|_, v| {
|
self.cidr_peer_id_map.retain(|_, v| {
|
||||||
// remove cidr map for peers we cannot reach.
|
// remove cidr map for peers we cannot reach.
|
||||||
self.next_hop_map.contains_key(v)
|
self.next_hop_map.contains_key(&v.peer_id)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -957,8 +962,19 @@ impl RouteTable {
|
|||||||
|
|
||||||
self.peer_infos.insert(*peer_id, info.clone());
|
self.peer_infos.insert(*peer_id, info.clone());
|
||||||
|
|
||||||
let is_new_peer_better = |old_peer_id: PeerId| -> bool {
|
let peer_id_and_version = PeerIdAndVersion {
|
||||||
let old_next_hop = self.get_next_hop(old_peer_id);
|
peer_id: *peer_id,
|
||||||
|
version,
|
||||||
|
};
|
||||||
|
|
||||||
|
let is_new_peer_better = |old_peer: &PeerIdAndVersion| -> bool {
|
||||||
|
if peer_id_and_version.version > old_peer.version {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if peer_id_and_version.peer_id == old_peer.peer_id {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
let old_next_hop = self.get_next_hop(old_peer.peer_id);
|
||||||
let new_next_hop = item.value();
|
let new_next_hop = item.value();
|
||||||
old_next_hop.is_none() || new_next_hop.path_len < old_next_hop.unwrap().path_len
|
old_next_hop.is_none() || new_next_hop.path_len < old_next_hop.unwrap().path_len
|
||||||
};
|
};
|
||||||
@@ -967,34 +983,34 @@ impl RouteTable {
|
|||||||
self.ipv4_peer_id_map
|
self.ipv4_peer_id_map
|
||||||
.entry(ipv4_addr.into())
|
.entry(ipv4_addr.into())
|
||||||
.and_modify(|v| {
|
.and_modify(|v| {
|
||||||
if *v != *peer_id && is_new_peer_better(*v) {
|
if is_new_peer_better(v) {
|
||||||
*v = *peer_id;
|
*v = peer_id_and_version;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.or_insert(*peer_id);
|
.or_insert(peer_id_and_version);
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(ipv6_addr) = info.ipv6_addr.and_then(|x| x.address) {
|
if let Some(ipv6_addr) = info.ipv6_addr.and_then(|x| x.address) {
|
||||||
self.ipv6_peer_id_map
|
self.ipv6_peer_id_map
|
||||||
.entry(ipv6_addr.into())
|
.entry(ipv6_addr.into())
|
||||||
.and_modify(|v| {
|
.and_modify(|v| {
|
||||||
if *v != *peer_id && is_new_peer_better(*v) {
|
if is_new_peer_better(v) {
|
||||||
*v = *peer_id;
|
*v = peer_id_and_version;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.or_insert(*peer_id);
|
.or_insert(peer_id_and_version);
|
||||||
}
|
}
|
||||||
|
|
||||||
for cidr in info.proxy_cidrs.iter() {
|
for cidr in info.proxy_cidrs.iter() {
|
||||||
self.cidr_peer_id_map
|
self.cidr_peer_id_map
|
||||||
.entry(cidr.parse().unwrap())
|
.entry(cidr.parse().unwrap())
|
||||||
.and_modify(|v| {
|
.and_modify(|v| {
|
||||||
if *v != *peer_id && is_new_peer_better(*v) {
|
if is_new_peer_better(v) {
|
||||||
// if the next hop is not set or the new next hop is better, update it.
|
// if the next hop is not set or the new next hop is better, update it.
|
||||||
*v = *peer_id;
|
*v = peer_id_and_version;
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.or_insert(*peer_id);
|
.or_insert(peer_id_and_version);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1004,7 +1020,7 @@ impl RouteTable {
|
|||||||
for item in self.cidr_peer_id_map.iter() {
|
for item in self.cidr_peer_id_map.iter() {
|
||||||
let (k, v) = item.pair();
|
let (k, v) = item.pair();
|
||||||
if k.contains(&ipv4) {
|
if k.contains(&ipv4) {
|
||||||
return Some(*v);
|
return Some(v.peer_id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
None
|
None
|
||||||
@@ -2376,8 +2392,8 @@ impl Route for PeerRoute {
|
|||||||
|
|
||||||
async fn get_peer_id_by_ipv4(&self, ipv4_addr: &Ipv4Addr) -> Option<PeerId> {
|
async fn get_peer_id_by_ipv4(&self, ipv4_addr: &Ipv4Addr) -> Option<PeerId> {
|
||||||
let route_table = &self.service_impl.route_table;
|
let route_table = &self.service_impl.route_table;
|
||||||
if let Some(peer_id) = route_table.ipv4_peer_id_map.get(ipv4_addr) {
|
if let Some(p) = route_table.ipv4_peer_id_map.get(ipv4_addr) {
|
||||||
return Some(*peer_id);
|
return Some(p.peer_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(peer_id) = route_table.get_peer_id_for_proxy(ipv4_addr) {
|
if let Some(peer_id) = route_table.get_peer_id_for_proxy(ipv4_addr) {
|
||||||
@@ -2390,8 +2406,8 @@ impl Route for PeerRoute {
|
|||||||
|
|
||||||
async fn get_peer_id_by_ipv6(&self, ipv6_addr: &Ipv6Addr) -> Option<PeerId> {
|
async fn get_peer_id_by_ipv6(&self, ipv6_addr: &Ipv6Addr) -> Option<PeerId> {
|
||||||
let route_table = &self.service_impl.route_table;
|
let route_table = &self.service_impl.route_table;
|
||||||
if let Some(peer_id) = route_table.ipv6_peer_id_map.get(ipv6_addr) {
|
if let Some(p) = route_table.ipv6_peer_id_map.get(ipv6_addr) {
|
||||||
return Some(*peer_id);
|
return Some(p.peer_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Add proxy support for IPv6 similar to IPv4
|
// TODO: Add proxy support for IPv6 similar to IPv4
|
||||||
@@ -2493,6 +2509,7 @@ mod tests {
|
|||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use cidr::{Ipv4Cidr, Ipv4Inet, Ipv6Inet};
|
||||||
use dashmap::DashMap;
|
use dashmap::DashMap;
|
||||||
use prost_reflect::{DynamicMessage, ReflectMessage};
|
use prost_reflect::{DynamicMessage, ReflectMessage};
|
||||||
|
|
||||||
@@ -2504,7 +2521,7 @@ mod tests {
|
|||||||
peer_manager::{PeerManager, RouteAlgoType},
|
peer_manager::{PeerManager, RouteAlgoType},
|
||||||
peer_ospf_route::PeerRouteServiceImpl,
|
peer_ospf_route::PeerRouteServiceImpl,
|
||||||
route_trait::{NextHopPolicy, Route, RouteCostCalculatorInterface},
|
route_trait::{NextHopPolicy, Route, RouteCostCalculatorInterface},
|
||||||
tests::connect_peer_manager,
|
tests::{connect_peer_manager, create_mock_peer_manager},
|
||||||
},
|
},
|
||||||
proto::{
|
proto::{
|
||||||
common::NatType,
|
common::NatType,
|
||||||
@@ -2964,4 +2981,58 @@ mod tests {
|
|||||||
|
|
||||||
assert_eq!(req, req2);
|
assert_eq!(req, req2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::test]
|
||||||
|
async fn test_peer_id_map_override() {
|
||||||
|
let p_a = create_mock_peer_manager().await;
|
||||||
|
let p_b = create_mock_peer_manager().await;
|
||||||
|
let p_c = create_mock_peer_manager().await;
|
||||||
|
|
||||||
|
connect_peer_manager(p_a.clone(), p_b.clone()).await;
|
||||||
|
connect_peer_manager(p_b.clone(), p_c.clone()).await;
|
||||||
|
|
||||||
|
let ip: Ipv4Inet = "10.0.0.1/24".parse().unwrap();
|
||||||
|
let ipv6: Ipv6Inet = "2001:db8::1/64".parse().unwrap();
|
||||||
|
let proxy: Ipv4Cidr = "10.3.0.0/24".parse().unwrap();
|
||||||
|
let check_route_peer_id = async |p: Arc<PeerManager>| {
|
||||||
|
let p = p.clone();
|
||||||
|
wait_for_condition(
|
||||||
|
|| async {
|
||||||
|
p_a.get_route().get_peer_id_by_ipv4(&ip.address()).await == Some(p.my_peer_id())
|
||||||
|
&& p_a.get_route().get_peer_id_by_ipv6(&ipv6.address()).await
|
||||||
|
== Some(p.my_peer_id())
|
||||||
|
&& p_a
|
||||||
|
.get_route()
|
||||||
|
.get_peer_id_by_ipv4(&proxy.first_address())
|
||||||
|
.await
|
||||||
|
== Some(p.my_peer_id())
|
||||||
|
},
|
||||||
|
Duration::from_secs(5),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
};
|
||||||
|
|
||||||
|
p_c.get_global_ctx().set_ipv4(Some(ip));
|
||||||
|
p_c.get_global_ctx().set_ipv6(Some(ipv6));
|
||||||
|
p_c.get_global_ctx()
|
||||||
|
.config
|
||||||
|
.add_proxy_cidr(proxy, None)
|
||||||
|
.unwrap();
|
||||||
|
check_route_peer_id(p_c.clone()).await;
|
||||||
|
|
||||||
|
p_b.get_global_ctx().set_ipv4(Some(ip));
|
||||||
|
p_b.get_global_ctx().set_ipv6(Some(ipv6));
|
||||||
|
p_b.get_global_ctx()
|
||||||
|
.config
|
||||||
|
.add_proxy_cidr(proxy, None)
|
||||||
|
.unwrap();
|
||||||
|
check_route_peer_id(p_b.clone()).await;
|
||||||
|
|
||||||
|
p_b.get_global_ctx()
|
||||||
|
.set_ipv4(Some("10.0.0.2/24".parse().unwrap()));
|
||||||
|
p_b.get_global_ctx()
|
||||||
|
.set_ipv6(Some("2001:db8::2/64".parse().unwrap()));
|
||||||
|
p_b.get_global_ctx().config.remove_proxy_cidr(proxy);
|
||||||
|
check_route_peer_id(p_c.clone()).await;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user