fix ospf ipv4 map error when ipv4 conflicted and changed (#1359)
Some checks failed
EasyTier Core / pre_job (push) Has been cancelled
EasyTier Core / build_web (push) Has been cancelled
EasyTier Core / build (freebsd-13.2-x86_64, 13.2, ubuntu-22.04, x86_64-unknown-freebsd) (push) Has been cancelled
EasyTier Core / build (linux-aarch64, ubuntu-22.04, aarch64-unknown-linux-musl) (push) Has been cancelled
EasyTier Core / build (linux-arm, ubuntu-22.04, arm-unknown-linux-musleabi) (push) Has been cancelled
EasyTier Core / build (linux-armhf, ubuntu-22.04, arm-unknown-linux-musleabihf) (push) Has been cancelled
EasyTier Core / build (linux-armv7, ubuntu-22.04, armv7-unknown-linux-musleabi) (push) Has been cancelled
EasyTier Core / build (linux-armv7hf, ubuntu-22.04, armv7-unknown-linux-musleabihf) (push) Has been cancelled
EasyTier Core / build (linux-loongarch64, ubuntu-24.04, loongarch64-unknown-linux-musl) (push) Has been cancelled
EasyTier Core / build (linux-mips, ubuntu-22.04, mips-unknown-linux-musl) (push) Has been cancelled
EasyTier Core / build (linux-mipsel, ubuntu-22.04, mipsel-unknown-linux-musl) (push) Has been cancelled
EasyTier Core / build (linux-riscv64, ubuntu-22.04, riscv64gc-unknown-linux-musl) (push) Has been cancelled
EasyTier Core / build (linux-x86_64, ubuntu-22.04, x86_64-unknown-linux-musl) (push) Has been cancelled
EasyTier Core / build (macos-aarch64, macos-latest, aarch64-apple-darwin) (push) Has been cancelled
EasyTier Core / build (macos-x86_64, macos-latest, x86_64-apple-darwin) (push) Has been cancelled
EasyTier Core / build (windows-arm64, windows-latest, aarch64-pc-windows-msvc) (push) Has been cancelled
EasyTier Core / build (windows-i686, windows-latest, i686-pc-windows-msvc) (push) Has been cancelled
EasyTier Core / build (windows-x86_64, windows-latest, x86_64-pc-windows-msvc) (push) Has been cancelled
EasyTier Core / core-result (push) Has been cancelled
EasyTier Core / magisk_build (push) Has been cancelled
EasyTier GUI / pre_job (push) Has been cancelled
EasyTier GUI / build-gui (linux-aarch64, aarch64-unknown-linux-gnu, ubuntu-22.04, aarch64-unknown-linux-musl) (push) Has been cancelled
EasyTier GUI / build-gui (linux-x86_64, x86_64-unknown-linux-gnu, ubuntu-22.04, x86_64-unknown-linux-musl) (push) Has been cancelled
EasyTier GUI / build-gui (macos-aarch64, aarch64-apple-darwin, macos-latest, aarch64-apple-darwin) (push) Has been cancelled
EasyTier GUI / build-gui (macos-x86_64, x86_64-apple-darwin, macos-latest, x86_64-apple-darwin) (push) Has been cancelled
EasyTier GUI / build-gui (windows-arm64, aarch64-pc-windows-msvc, windows-latest, aarch64-pc-windows-msvc) (push) Has been cancelled
EasyTier GUI / build-gui (windows-i686, i686-pc-windows-msvc, windows-latest, i686-pc-windows-msvc) (push) Has been cancelled
EasyTier GUI / build-gui (windows-x86_64, x86_64-pc-windows-msvc, windows-latest, x86_64-pc-windows-msvc) (push) Has been cancelled
EasyTier GUI / gui-result (push) Has been cancelled
EasyTier Mobile / pre_job (push) Has been cancelled
EasyTier Mobile / build-mobile (android, ubuntu-22.04, android) (push) Has been cancelled
EasyTier Mobile / mobile-result (push) Has been cancelled
EasyTier OHOS / pre_job (push) Has been cancelled
EasyTier OHOS / build-ohos (push) Has been cancelled
EasyTier Test / pre_job (push) Has been cancelled
EasyTier Test / test (push) Has been cancelled

This commit is contained in:
Sijie.Sun
2025-09-13 08:48:50 +08:00
committed by GitHub
parent eb42086f9c
commit 793889c3b7

View File

@@ -720,15 +720,20 @@ struct NextHopInfo {
}
// dst_peer_id -> (next_hop_peer_id, cost, path_len)
type NextHopMap = DashMap<PeerId, NextHopInfo>;
#[derive(Debug, Clone, Copy)]
struct PeerIdAndVersion {
peer_id: PeerId,
version: Version,
}
// computed with SyncedRouteInfo. used to get next hop.
#[derive(Debug)]
struct RouteTable {
peer_infos: DashMap<PeerId, RoutePeerInfo>,
next_hop_map: NextHopMap,
ipv4_peer_id_map: DashMap<Ipv4Addr, PeerId>,
ipv6_peer_id_map: DashMap<Ipv6Addr, PeerId>,
cidr_peer_id_map: DashMap<cidr::IpCidr, PeerId>,
ipv4_peer_id_map: DashMap<Ipv4Addr, PeerIdAndVersion>,
ipv6_peer_id_map: DashMap<Ipv6Addr, PeerIdAndVersion>,
cidr_peer_id_map: DashMap<cidr::IpCidr, PeerIdAndVersion>,
next_hop_map_version: AtomicVersion,
}
@@ -834,15 +839,15 @@ impl RouteTable {
});
self.ipv4_peer_id_map.retain(|_, v| {
// 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| {
// 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| {
// 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());
let is_new_peer_better = |old_peer_id: PeerId| -> bool {
let old_next_hop = self.get_next_hop(old_peer_id);
let peer_id_and_version = PeerIdAndVersion {
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();
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
.entry(ipv4_addr.into())
.and_modify(|v| {
if *v != *peer_id && is_new_peer_better(*v) {
*v = *peer_id;
if is_new_peer_better(v) {
*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) {
self.ipv6_peer_id_map
.entry(ipv6_addr.into())
.and_modify(|v| {
if *v != *peer_id && is_new_peer_better(*v) {
*v = *peer_id;
if is_new_peer_better(v) {
*v = peer_id_and_version;
}
})
.or_insert(*peer_id);
.or_insert(peer_id_and_version);
}
for cidr in info.proxy_cidrs.iter() {
self.cidr_peer_id_map
.entry(cidr.parse().unwrap())
.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.
*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() {
let (k, v) = item.pair();
if k.contains(&ipv4) {
return Some(*v);
return Some(v.peer_id);
}
}
None
@@ -2376,8 +2392,8 @@ impl Route for PeerRoute {
async fn get_peer_id_by_ipv4(&self, ipv4_addr: &Ipv4Addr) -> Option<PeerId> {
let route_table = &self.service_impl.route_table;
if let Some(peer_id) = route_table.ipv4_peer_id_map.get(ipv4_addr) {
return Some(*peer_id);
if let Some(p) = route_table.ipv4_peer_id_map.get(ipv4_addr) {
return Some(p.peer_id);
}
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> {
let route_table = &self.service_impl.route_table;
if let Some(peer_id) = route_table.ipv6_peer_id_map.get(ipv6_addr) {
return Some(*peer_id);
if let Some(p) = route_table.ipv6_peer_id_map.get(ipv6_addr) {
return Some(p.peer_id);
}
// TODO: Add proxy support for IPv6 similar to IPv4
@@ -2493,6 +2509,7 @@ mod tests {
time::Duration,
};
use cidr::{Ipv4Cidr, Ipv4Inet, Ipv6Inet};
use dashmap::DashMap;
use prost_reflect::{DynamicMessage, ReflectMessage};
@@ -2504,7 +2521,7 @@ mod tests {
peer_manager::{PeerManager, RouteAlgoType},
peer_ospf_route::PeerRouteServiceImpl,
route_trait::{NextHopPolicy, Route, RouteCostCalculatorInterface},
tests::connect_peer_manager,
tests::{connect_peer_manager, create_mock_peer_manager},
},
proto::{
common::NatType,
@@ -2964,4 +2981,58 @@ mod tests {
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;
}
}