From 10d5652fc8c4a83ed45389059338dfc896497af9 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Fri, 5 Jul 2024 17:20:10 +0200 Subject: [PATCH 1/8] Configure `scouting/*/autoconnect` with a sequence --- DEFAULT_CONFIG.json5 | 25 ++++--- commons/zenoh-protocol/src/core/whatami.rs | 81 +++++++++------------- 2 files changed, 44 insertions(+), 62 deletions(-) diff --git a/DEFAULT_CONFIG.json5 b/DEFAULT_CONFIG.json5 index 1e9921bbe3..e0f2c8c28e 100644 --- a/DEFAULT_CONFIG.json5 +++ b/DEFAULT_CONFIG.json5 @@ -14,7 +14,7 @@ /// The node's metadata (name, location, DNS name, etc.) Arbitrary JSON data not interpreted by zenohd and available in admin space @/router/ metadata: { name: "strawberry", - location: "Penny Lane" + location: "Penny Lane", }, /// Which endpoints to connect to. E.g. tcp/localhost:7447. @@ -27,7 +27,7 @@ /// or different values for router, peer and client (e.g. timeout_ms: { router: -1, peer: -1, client: 0 }). timeout_ms: { router: -1, peer: -1, client: 0 }, - /// The list of endpoints to connect to. + /// The list of endpoints to connect to. /// Accepts a single list (e.g. endpoints: ["tcp/10.10.10.10:7447", "tcp/11.11.11.11:7447"]) /// or different lists for router, peer and client (e.g. endpoints: { router: ["tcp/10.10.10.10:7447"], peer: ["tcp/11.11.11.11:7447"] }). endpoints: [ @@ -64,7 +64,7 @@ /// or different values for router, peer and client (e.g. timeout_ms: { router: -1, peer: -1, client: 0 }). timeout_ms: 0, - /// The list of endpoints to listen on. + /// The list of endpoints to listen on. /// Accepts a single list (e.g. endpoints: ["tcp/[::]:7447", "udp/[::]:7447"]) /// or different lists for router, peer and client (e.g. endpoints: { router: ["tcp/[::]:7447"], peer: ["tcp/[::]:0"] }). endpoints: { router: ["tcp/[::]:7447"], peer: ["tcp/[::]:0"] }, @@ -104,10 +104,10 @@ /// The time-to-live on multicast scouting packets ttl: 1, /// Which type of Zenoh instances to automatically establish sessions with upon discovery on UDP multicast. - /// Accepts a single value (e.g. autoconnect: "router|peer") - /// or different values for router, peer and client (e.g. autoconnect: { router: "", peer: "router|peer" }). + /// Accepts a single value (e.g. autoconnect: ["router", "peer"]) + /// or different values for router, peer and client (e.g. autoconnect: { router: [], peer: ["router", "peer"] }). /// Each value is bit-or-like combinations of "peer", "router" and "client". - autoconnect: { router: "", peer: "router|peer" }, + autoconnect: { router: "", peer: ["router", "peer"] }, /// Whether or not to listen for scout messages on UDP multicast and reply to them. listen: true, }, @@ -122,10 +122,10 @@ /// direct connectivity with each other. multihop: false, /// Which type of Zenoh instances to automatically establish sessions with upon discovery on gossip. - /// Accepts a single value (e.g. autoconnect: "router|peer") - /// or different values for router, peer and client (e.g. autoconnect: { router: "", peer: "router|peer" }). + /// Accepts a single value (e.g. autoconnect: ["router", "peer"]) + /// or different values for router, peer and client (e.g. autoconnect: { router: [], peer: ["router", "peer"] }). /// Each value is bit-or-like combinations of "peer", "router" and "client". - autoconnect: { router: "", peer: "router|peer" }, + autoconnect: { router: [], peer: ["router", "peer"] }, }, }, @@ -208,7 +208,7 @@ // "interfaces": [ // "lo0" // ], - // /// Subjects can be cert_common_names when using TLS or Quic + // /// Subjects can be cert_common_names when using TLS or Quic // "cert_common_names": [ // "example.zenoh.io" // ], @@ -238,7 +238,7 @@ /// NOTE: Currently, the LowLatency transport doesn't preserve QoS prioritization. /// NOTE: Due to the note above, 'lowlatency' is incompatible with 'qos' option, so in order to /// enable 'lowlatency' you need to explicitly disable 'qos'. - /// NOTE: LowLatency transport does not support the fragmentation, so the message size should be + /// NOTE: LowLatency transport does not support the fragmentation, so the message size should be /// smaller than the tx batch_size. lowlatency: false, /// Enables QoS on unicast communications. @@ -317,7 +317,7 @@ /// Using CongestionControl::Drop the message might be dropped, depending on conditions configured here. congestion_control: { /// The maximum time in microseconds to wait for an available batch before dropping the message if still no batch is available. - wait_before_drop: 1000 + wait_before_drop: 1000, }, /// The initial exponential backoff time in nanoseconds to allow the batching to eventually progress. /// Higher values lead to a more aggressive batching but it will introduce additional latency. @@ -539,5 +539,4 @@ // __config__: "./plugins/zenoh-plugin-storage-manager/config.json5", // } // }, - } diff --git a/commons/zenoh-protocol/src/core/whatami.rs b/commons/zenoh-protocol/src/core/whatami.rs index 9eb9628e3f..3dcef00d87 100644 --- a/commons/zenoh-protocol/src/core/whatami.rs +++ b/commons/zenoh-protocol/src/core/whatami.rs @@ -142,10 +142,12 @@ impl WhatAmIMatcher { Self::U8_R => WhatAmI::STR_R, Self::U8_P => WhatAmI::STR_P, Self::U8_C => WhatAmI::STR_C, - Self::U8_R_P => formatcp!("{}|{}", WhatAmI::STR_R, WhatAmI::STR_P), - Self::U8_R_C => formatcp!("{}|{}", WhatAmI::STR_R, WhatAmI::STR_C), - Self::U8_P_C => formatcp!("{}|{}", WhatAmI::STR_P, WhatAmI::STR_C), - Self::U8_R_P_C => formatcp!("{}|{}|{}", WhatAmI::STR_R, WhatAmI::STR_P, WhatAmI::STR_C), + Self::U8_R_P => formatcp!("[{},{}]", WhatAmI::STR_R, WhatAmI::STR_P), + Self::U8_R_C => formatcp!("[{},{}]", WhatAmI::STR_R, WhatAmI::STR_C), + Self::U8_P_C => formatcp!("[{},{}]", WhatAmI::STR_P, WhatAmI::STR_C), + Self::U8_R_P_C => { + formatcp!("[{},{},{}]", WhatAmI::STR_R, WhatAmI::STR_P, WhatAmI::STR_C) + } _ => unreachable!(), } } @@ -186,24 +188,6 @@ impl TryFrom for WhatAmIMatcher { } } -impl FromStr for WhatAmIMatcher { - type Err = (); - - fn from_str(s: &str) -> Result { - let mut inner = 0; - for s in s.split('|') { - match s.trim() { - "" => {} - WhatAmI::STR_R => inner |= WhatAmI::U8_R, - WhatAmI::STR_P => inner |= WhatAmI::U8_P, - WhatAmI::STR_C => inner |= WhatAmI::U8_C, - _ => return Err(()), - } - } - Self::try_from(inner) - } -} - impl fmt::Display for WhatAmIMatcher { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.write_str(self.to_str()) @@ -329,41 +313,40 @@ impl<'de> serde::de::Visitor<'de> for WhatAmIMatcherVisitor { fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { write!( formatter, - "a | separated list of whatami variants ('{}', '{}', '{}')", + "a list of whatami variants ('{}', '{}', '{}')", WhatAmI::STR_R, WhatAmI::STR_P, WhatAmI::STR_C ) } - fn visit_str(self, v: &str) -> Result + fn visit_seq(self, mut seq: A) -> Result where - E: serde::de::Error, + A: serde::de::SeqAccess<'de>, { - v.parse().map_err(|_| { - serde::de::Error::invalid_value( - serde::de::Unexpected::Str(v), - &formatcp!( - "a | separated list of whatami variants ('{}', '{}', '{}')", - WhatAmI::STR_R, - WhatAmI::STR_P, - WhatAmI::STR_C - ), - ) - }) - } + let mut inner = 0; - fn visit_borrowed_str(self, v: &'de str) -> Result - where - E: serde::de::Error, - { - self.visit_str(v) - } - fn visit_string(self, v: String) -> Result - where - E: serde::de::Error, - { - self.visit_str(&v) + while let Some(s) = seq.next_element::()? { + match s.as_str() { + WhatAmI::STR_R => inner |= WhatAmI::U8_R, + WhatAmI::STR_P => inner |= WhatAmI::U8_P, + WhatAmI::STR_C => inner |= WhatAmI::U8_C, + _ => { + return Err(serde::de::Error::invalid_value( + serde::de::Unexpected::Str(&s), + &formatcp!( + "one of ('{}', '{}', '{}')", + WhatAmI::STR_R, + WhatAmI::STR_P, + WhatAmI::STR_C + ), + )) + } + } + } + + Ok(WhatAmIMatcher::try_from(inner) + .expect("`WhatAmIMatcher` should be valid by construction")) } } @@ -372,6 +355,6 @@ impl<'de> serde::Deserialize<'de> for WhatAmIMatcher { where D: serde::Deserializer<'de>, { - deserializer.deserialize_str(WhatAmIMatcherVisitor) + deserializer.deserialize_seq(WhatAmIMatcherVisitor) } } From fdb0b8d2e761df8ee3f4e6ecc4e5090879175311 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Fri, 5 Jul 2024 17:27:19 +0200 Subject: [PATCH 2/8] Fix `zenoh/tests/routing.rs` --- zenoh/tests/routing.rs | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/zenoh/tests/routing.rs b/zenoh/tests/routing.rs index fd680ae545..c11d6dec47 100644 --- a/zenoh/tests/routing.rs +++ b/zenoh/tests/routing.rs @@ -445,9 +445,7 @@ async fn static_failover_brokering() -> Result<()> { config .scouting .gossip - .set_autoconnect(Some(ModeDependentValue::Unique( - WhatAmIMatcher::from_str("").unwrap(), - ))) + .set_autoconnect(Some(ModeDependentValue::Unique(WhatAmIMatcher::empty()))) .unwrap(); Some(config) }; From dedb36323f0a3c452a95c3b78d459b3a87603147 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Fri, 5 Jul 2024 17:40:47 +0200 Subject: [PATCH 3/8] Fix `zenoh/tests/routing.rs` (again) --- zenoh/tests/routing.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/zenoh/tests/routing.rs b/zenoh/tests/routing.rs index c11d6dec47..efeedc835d 100644 --- a/zenoh/tests/routing.rs +++ b/zenoh/tests/routing.rs @@ -12,7 +12,6 @@ // ZettaScale Zenoh Team, // use std::{ - str::FromStr, sync::{ atomic::{AtomicUsize, Ordering}, Arc, From 2fe0a05a78b12e89349c71d302e33d2660e40936 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Tue, 9 Jul 2024 16:02:11 +0200 Subject: [PATCH 4/8] Fix zenoh-config tests --- commons/zenoh-config/src/lib.rs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/commons/zenoh-config/src/lib.rs b/commons/zenoh-config/src/lib.rs index e239ac8b7a..56e35d025c 100644 --- a/commons/zenoh-config/src/lib.rs +++ b/commons/zenoh-config/src/lib.rs @@ -571,7 +571,7 @@ fn config_deser() { scouting: { multicast: { enabled: false, - autoconnect: "peer|router" + autoconnect: ["peer", "router"] } } }"#, @@ -598,7 +598,7 @@ fn config_deser() { scouting: { multicast: { enabled: false, - autoconnect: {router: "", peer: "peer|router"} + autoconnect: {router: [], peer: ["peer", "router"]} } } }"#, From e43714d63bb3f2305d75c3dd34ea52b0d966e182 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Tue, 9 Jul 2024 16:03:14 +0200 Subject: [PATCH 5/8] Fix `scouting/multicast/autoconnect/router` in default config --- DEFAULT_CONFIG.json5 | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/DEFAULT_CONFIG.json5 b/DEFAULT_CONFIG.json5 index e0f2c8c28e..bf9fdd3f22 100644 --- a/DEFAULT_CONFIG.json5 +++ b/DEFAULT_CONFIG.json5 @@ -107,7 +107,7 @@ /// Accepts a single value (e.g. autoconnect: ["router", "peer"]) /// or different values for router, peer and client (e.g. autoconnect: { router: [], peer: ["router", "peer"] }). /// Each value is bit-or-like combinations of "peer", "router" and "client". - autoconnect: { router: "", peer: ["router", "peer"] }, + autoconnect: { router: [], peer: ["router", "peer"] }, /// Whether or not to listen for scout messages on UDP multicast and reply to them. listen: true, }, From efc7e08d3b11c7fc6743454704fcf2aaebb86fdb Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Tue, 16 Jul 2024 14:11:16 +0200 Subject: [PATCH 6/8] Keep string representation --- commons/zenoh-protocol/src/core/whatami.rs | 29 +++++++++++++++++----- 1 file changed, 23 insertions(+), 6 deletions(-) diff --git a/commons/zenoh-protocol/src/core/whatami.rs b/commons/zenoh-protocol/src/core/whatami.rs index 3dcef00d87..9ae0690382 100644 --- a/commons/zenoh-protocol/src/core/whatami.rs +++ b/commons/zenoh-protocol/src/core/whatami.rs @@ -142,12 +142,11 @@ impl WhatAmIMatcher { Self::U8_R => WhatAmI::STR_R, Self::U8_P => WhatAmI::STR_P, Self::U8_C => WhatAmI::STR_C, - Self::U8_R_P => formatcp!("[{},{}]", WhatAmI::STR_R, WhatAmI::STR_P), - Self::U8_R_C => formatcp!("[{},{}]", WhatAmI::STR_R, WhatAmI::STR_C), - Self::U8_P_C => formatcp!("[{},{}]", WhatAmI::STR_P, WhatAmI::STR_C), - Self::U8_R_P_C => { - formatcp!("[{},{},{}]", WhatAmI::STR_R, WhatAmI::STR_P, WhatAmI::STR_C) - } + Self::U8_R_P => formatcp!("{}|{}", WhatAmI::STR_R, WhatAmI::STR_P), + Self::U8_R_C => formatcp!("{}|{}", WhatAmI::STR_R, WhatAmI::STR_C), + Self::U8_P_C => formatcp!("{}|{}", WhatAmI::STR_P, WhatAmI::STR_C), + Self::U8_R_P_C => formatcp!("{}|{}|{}", WhatAmI::STR_R, WhatAmI::STR_P, WhatAmI::STR_C), + _ => unreachable!(), } } @@ -188,6 +187,24 @@ impl TryFrom for WhatAmIMatcher { } } +impl FromStr for WhatAmIMatcher { + type Err = (); + + fn from_str(s: &str) -> Result { + let mut inner = 0; + for s in s.split('|') { + match s.trim() { + "" => {} + WhatAmI::STR_R => inner |= WhatAmI::U8_R, + WhatAmI::STR_P => inner |= WhatAmI::U8_P, + WhatAmI::STR_C => inner |= WhatAmI::U8_C, + _ => return Err(()), + } + } + Self::try_from(inner) + } +} + impl fmt::Display for WhatAmIMatcher { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.write_str(self.to_str()) From 7bc61d78dd51a397a066d061f8dcd2a723cd5500 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Thu, 18 Jul 2024 15:02:58 +0200 Subject: [PATCH 7/8] Fix `ModeDependentValue` de impl --- commons/zenoh-config/src/mode_dependent.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/commons/zenoh-config/src/mode_dependent.rs b/commons/zenoh-config/src/mode_dependent.rs index 6a06f967ba..6576161473 100644 --- a/commons/zenoh-config/src/mode_dependent.rs +++ b/commons/zenoh-config/src/mode_dependent.rs @@ -255,12 +255,12 @@ impl<'a> serde::Deserialize<'a> for ModeDependentValue { formatter.write_str("WhatAmIMatcher or mode dependent WhatAmIMatcher") } - fn visit_str(self, value: &str) -> Result + fn visit_seq(self, seq: A) -> Result where - E: de::Error, + A: de::SeqAccess<'de>, { WhatAmIMatcherVisitor {} - .visit_str(value) + .visit_seq(seq) .map(ModeDependentValue::Unique) } From 9316ac5c1f98f11c59448ad13d1222f8e9281d77 Mon Sep 17 00:00:00 2001 From: Mahmoud Mazouz Date: Thu, 18 Jul 2024 15:04:33 +0200 Subject: [PATCH 8/8] Remove `#[serde(deserialize_with = "treat_error_as_none")]` This attribute makes it hard to debug certain errors and was only present on autoconnect fields. --- commons/zenoh-config/src/lib.rs | 11 ----------- 1 file changed, 11 deletions(-) diff --git a/commons/zenoh-config/src/lib.rs b/commons/zenoh-config/src/lib.rs index 56e35d025c..fec1a1cf8d 100644 --- a/commons/zenoh-config/src/lib.rs +++ b/commons/zenoh-config/src/lib.rs @@ -197,15 +197,6 @@ fn config_keys() { dbg!(c.keys()); } -fn treat_error_as_none<'a, T, D>(deserializer: D) -> Result, D::Error> -where - T: serde::de::Deserialize<'a>, - D: serde::de::Deserializer<'a>, -{ - let value: Value = serde::de::Deserialize::deserialize(deserializer)?; - Ok(T::deserialize(value).ok()) -} - validated_struct::validator! { /// The main configuration structure for Zenoh. /// @@ -264,7 +255,6 @@ validated_struct::validator! { /// The time-to-live on multicast scouting packets. (default: 1) pub ttl: Option, /// Which type of Zenoh instances to automatically establish sessions with upon discovery through UDP multicast. - #[serde(deserialize_with = "treat_error_as_none")] autoconnect: Option>, /// Whether or not to listen for scout messages on UDP multicast and reply to them. listen: Option>, @@ -281,7 +271,6 @@ validated_struct::validator! { /// direct connectivity with each other. multihop: Option, /// Which type of Zenoh instances to automatically establish sessions with upon discovery through gossip. - #[serde(deserialize_with = "treat_error_as_none")] autoconnect: Option>, }, },