On Sun, May 17, 2026 at 11:20 PM Sragdhara Datta Chaudhuri <
[email protected]> wrote:
> Advertise address_set from one AZ to others via ovn-ic.
> The address sets from SB are advertised to the IC-SB Address_Set.
> The address sets from IC SB that belong to remote AZ are synced
> to the local NB Address_Set.
>
> A new table Address_Set has been added to IC-SB.
> "Address_Set": {
> "columns": {
> "name": {"type": "string"},
> "addresses": {"type": {"key": "string",
> "min": 0,
> "max": "unlimited"}},
> "availability_zone": {"type": {"key": {"type": "uuid",
> "refTable": "Availability_Zone"}}},
> "external_ids": {
> "type": {"key": "string", "value": "string",
> "min": 0, "max": "unlimited"}}},
> "indexes": [["name"]],
> "isRoot": true}
> }
>
> New "options" field has been added to NB Address_Set table.
> ic-adv - A boolean value that enables advertisement of this
> Address_Set to IC-SB. Default is false.
> New "options" field has been added to SB Address_Set table.
> The options from NB Address_Set gets synced to the SB entry.
>
> Two new nb_global options have been added:
> ic-as-adv - If this is set to true, the Address_Sets in this AZ that have
> option ic-adv set to true get advertised to the IC-SB.
> Default is false.
> ic-as-learn - If this is set to true, the Address_Sets in IC-SB get
> synced to the NB Address_Set table of this AZ. Default
> value of the option is false. The synced address_sets
> would have external-id "ic-learnt" set to true.
>
> The existing northd code that syncs Address_Set from NB to SB has been
> enhanced to sync "options".
>
> New logic has been added in ovn-ic to sync address sets with IC-SB.
> Advertise logic (from SB to IC-SB):
> - Each SB address set that needs to be advertised (ic-adv option set),
> check if it is already present in IC-SB. If not, create new entry in
> IC-SB. Otherwise sync addresses from local address set to IC-SB entry.
> - Delete extra address sets in IC-SB that were earlier learnt from this
> AZ, but is no longer present, or not enabled for advertisement.
>
> Learning logic (from IC-SB to NB):
> - For each NB Address set entries that were earlier learnt from IC-SB
> (external-id "ic-learnt" set to true), check if it is still present
> in IC-SB. If not, delete local entry in NB. If yes, sync addresses
> from IC-SB to NB.
> - Any remote address sets in IC-SB (AZ not same as local AZ) that is not
> present in local AZ, create local entry in NB with external-id
> "ic-learnt" set to true.
>
> Signed-off-by: Sragdhara Datta Chaudhuri <[email protected]>
> Acked-by: Priyankar Jain <[email protected]>
>
Hi Sragdhara,
What is the motivation for this patch ?
Technically speaking CMS can create the Address sets on the desired AZ NB
DBs right ?
Is the main motivation the operational ease for CMS ?
Thanks
Numan
---
> ic/inc-proc-ic.c | 12 ++-
> ic/ovn-ic.c | 188 ++++++++++++++++++++++++++++++++++++++++++++
> northd/en-sync-sb.c | 35 +++++++--
> ovn-ic-sb.ovsschema | 17 +++-
> ovn-ic-sb.xml | 26 ++++++
> ovn-nb.ovsschema | 9 ++-
> ovn-nb.xml | 31 ++++++++
> ovn-sb.ovsschema | 9 ++-
> ovn-sb.xml | 4 +
> tests/ovn-ic.at | 172 ++++++++++++++++++++++++++++++++++++++++
> 10 files changed, 486 insertions(+), 17 deletions(-)
>
> diff --git a/ic/inc-proc-ic.c b/ic/inc-proc-ic.c
> index 2c0420292..bbcbcdd17 100644
> --- a/ic/inc-proc-ic.c
> +++ b/ic/inc-proc-ic.c
> @@ -41,7 +41,8 @@ VLOG_DEFINE_THIS_MODULE(inc_proc_ic);
> NB_NODE(logical_switch, "logical_switch") \
> NB_NODE(logical_switch_port, "logical_switch_port") \
> NB_NODE(load_balancer, "load_balancer") \
> - NB_NODE(load_balancer_group, "load_balancer_group")
> + NB_NODE(load_balancer_group, "load_balancer_group") \
> + NB_NODE(address_set, "address_set")
>
> enum nb_engine_node {
> #define NB_NODE(NAME, NAME_STR) NB_##NAME,
> @@ -66,7 +67,8 @@ VLOG_DEFINE_THIS_MODULE(inc_proc_ic);
> SB_NODE(datapath_binding, "datapath_binding") \
> SB_NODE(port_binding, "port_binding") \
> SB_NODE(service_monitor, "service_monitor") \
> - SB_NODE(learned_route, "learned_route")
> + SB_NODE(learned_route, "learned_route") \
> + SB_NODE(address_set, "address_set")
>
> enum sb_engine_node {
> #define SB_NODE(NAME, NAME_STR) SB_##NAME,
> @@ -114,7 +116,8 @@ VLOG_DEFINE_THIS_MODULE(inc_proc_ic);
> ICSB_NODE(datapath_binding, "datapath_binding") \
> ICSB_NODE(encap, "encap") \
> ICSB_NODE(gateway, "gateway") \
> - ICSB_NODE(port_binding, "port_binding")
> + ICSB_NODE(port_binding, "port_binding") \
> + ICSB_NODE(address_set, "address_set")
>
> enum icsb_engine_node {
> #define ICSB_NODE(NAME, NAME_STR) ICSB_##NAME,
> @@ -176,6 +179,7 @@ void inc_proc_ic_init(struct ovsdb_idl_loop *nb,
> engine_add_input(&en_ic, &en_nb_logical_switch_port, NULL);
> engine_add_input(&en_ic, &en_nb_load_balancer, NULL);
> engine_add_input(&en_ic, &en_nb_load_balancer_group, NULL);
> + engine_add_input(&en_ic, &en_nb_address_set, NULL);
>
> engine_add_input(&en_ic, &en_sb_sb_global, NULL);
> engine_add_input(&en_ic, &en_sb_chassis, NULL);
> @@ -184,6 +188,7 @@ void inc_proc_ic_init(struct ovsdb_idl_loop *nb,
> engine_add_input(&en_ic, &en_sb_port_binding, NULL);
> engine_add_input(&en_ic, &en_sb_service_monitor, NULL);
> engine_add_input(&en_ic, &en_sb_learned_route, NULL);
> + engine_add_input(&en_ic, &en_sb_address_set, NULL);
>
> engine_add_input(&en_ic, &en_icnb_ic_nb_global, NULL);
> engine_add_input(&en_ic, &en_icnb_transit_switch, NULL);
> @@ -198,6 +203,7 @@ void inc_proc_ic_init(struct ovsdb_idl_loop *nb,
> engine_add_input(&en_ic, &en_icsb_gateway, NULL);
> engine_add_input(&en_ic, &en_icsb_route, NULL);
> engine_add_input(&en_ic, &en_icsb_datapath_binding, NULL);
> + engine_add_input(&en_ic, &en_icsb_address_set, NULL);
>
> struct engine_arg engine_arg = {
> .nb_idl = nb->idl,
> diff --git a/ic/ovn-ic.c b/ic/ovn-ic.c
> index ba9490658..c1718cca8 100644
> --- a/ic/ovn-ic.c
> +++ b/ic/ovn-ic.c
> @@ -526,6 +526,173 @@ sync_sb_gw_to_isb(struct ic_context *ctx,
> free(isb_encaps);
> }
>
> +static void
> +nb_addr_set_apply_diff(const void *arg, const char *item, bool add)
> +{
> + const struct nbrec_address_set *as = arg;
> + if (add) {
> + nbrec_address_set_update_addresses_addvalue(as, item);
> + } else {
> + nbrec_address_set_update_addresses_delvalue(as, item);
> + }
> +}
> +
> +static void
> +update_nb_addr_set(const struct icsbrec_address_set *icsb_as,
> + const struct nbrec_address_set *nb_as)
> +{
> + struct sorted_array nb_addrs =
> + sorted_array_from_dbrec(nb_as, addresses);
> + struct sorted_array icsb_addrs =
> + sorted_array_from_dbrec(icsb_as, addresses);
> + sorted_array_apply_diff(&icsb_addrs, &nb_addrs,
> + nb_addr_set_apply_diff, nb_as);
> + sorted_array_destroy(&icsb_addrs);
> + sorted_array_destroy(&nb_addrs);
> +}
> +
> +static void
> +icsb_addr_set_apply_diff(const void *arg, const char *item, bool add)
> +{
> + const struct icsbrec_address_set *as = arg;
> + if (add) {
> + icsbrec_address_set_update_addresses_addvalue(as, item);
> + } else {
> + icsbrec_address_set_update_addresses_delvalue(as, item);
> + }
> +}
> +
> +static void
> +update_icsb_addr_set(struct sorted_array *nb_addrs,
> + const struct icsbrec_address_set *icsb_as)
> +{
> + struct sorted_array icsb_addrs =
> + sorted_array_from_dbrec(icsb_as, addresses);
> + sorted_array_apply_diff(nb_addrs, &icsb_addrs,
> + icsb_addr_set_apply_diff, icsb_as);
> + sorted_array_destroy(&icsb_addrs);
> +}
> +
> +static void
> +sync_addr_set_to_icsb(struct ovsdb_idl_txn *ovnisb_txn,
> + const struct sbrec_address_set *sb_as,
> + const struct icsbrec_address_set *icsb_as,
> + const struct icsbrec_availability_zone *az)
> +{
> + struct sorted_array addrs =
> + sorted_array_from_dbrec(sb_as, addresses);
> + if (!icsb_as) {
> + icsb_as = icsbrec_address_set_insert(ovnisb_txn);
> + icsbrec_address_set_set_name(icsb_as, sb_as->name);
> + icsbrec_address_set_set_availability_zone(icsb_as, az);
> + icsbrec_address_set_set_addresses(icsb_as, addrs.arr, addrs.n);
> + } else {
> + update_icsb_addr_set(&addrs, icsb_as);
> + }
> + sorted_array_destroy(&addrs);
> +}
> +
> +static void
> +sync_addr_set_from_icsb(struct ovsdb_idl_txn *ovnnb_txn,
> + const struct icsbrec_address_set *icsb_as)
> +{
> + struct nbrec_address_set *nb_as;
> + struct sorted_array addrs =
> + sorted_array_from_dbrec(icsb_as, addresses);
> +
> + nb_as = nbrec_address_set_insert(ovnnb_txn);
> + nbrec_address_set_set_name(nb_as, icsb_as->name);
> + nbrec_address_set_update_external_ids_setkey(nb_as, "ic-learnt",
> "true");
> + nbrec_address_set_set_addresses(nb_as, addrs.arr, addrs.n);
> + sorted_array_destroy(&addrs);
> +}
> +
> +static void
> +address_set_run(struct ic_context *ctx)
> +{
> + if (!ctx->ovnisb_unlocked_txn || !ctx->ovnnb_txn || !ctx->ovnsb_txn) {
> + return;
> + }
> +
> + struct shash ic_local_as = SHASH_INITIALIZER(&ic_local_as);
> + struct shash ic_remote_as = SHASH_INITIALIZER(&ic_remote_as);
> + const struct icsbrec_address_set *ic_as;
> + ICSBREC_ADDRESS_SET_FOR_EACH (ic_as, ctx->ovnisb_unlocked_idl) {
> + if (ic_as->availability_zone == ctx->runned_az) {
> + shash_add(&ic_local_as, ic_as->name, ic_as);
> + } else {
> + shash_add(&ic_remote_as, ic_as->name, ic_as);
> + }
> + }
> +
> + const struct nbrec_nb_global *nb_global =
> + nbrec_nb_global_first(ctx->ovnnb_idl);
> + ovs_assert(nb_global);
> + bool global_learn = smap_get_bool(&nb_global->options, "ic-as-learn",
> + false);
> + bool global_adv = smap_get_bool(&nb_global->options, "ic-as-adv",
> false);
> +
> + /* Advertise address set - from SB to IC-SB:
> + * - Each SB address set that needs to be advertised (ic-adv option
> set),
> + * check if it is already present in IC-SB. If not, create new entry
> in
> + * IC-SB. Otherwise sync addresses from local address set to IC-SB
> entry.
> + * - Delete extra address sets in IC-SB that were earlier learnt from
> this
> + * AZ, but is no longer present, or not enabled for advertisement.
> + */
> + if (global_adv) {
> + const struct sbrec_address_set *sb_as;
> + SBREC_ADDRESS_SET_FOR_EACH (sb_as, ctx->ovnsb_idl) {
> + if (smap_get_bool(&sb_as->options, "ic-adv", false)) {
> + const struct icsbrec_address_set *icsb_as;
> + icsb_as = shash_find_and_delete(&ic_local_as,
> sb_as->name);
> + sync_addr_set_to_icsb(ctx->ovnisb_unlocked_txn, sb_as,
> icsb_as,
> + ctx->runned_az);
> + }
> + }
> + }
> + struct shash_node *node;
> + SHASH_FOR_EACH (node, &ic_local_as) {
> + icsbrec_address_set_delete(node->data);
> + }
> + shash_destroy(&ic_local_as);
> +
> + /* Learn address set - from IC-SB to NB:
> + * - For each NB Address set entries that were earlier learnt from
> IC-SB
> + * (external-id "ic-learnt" set to true), check if it is still present
> + * in IC-SB. If not, delete local entry in NB. If yes, sync addresses
> + * from IC-SB to NB.
> + * - Any remote address sets in IC-SB (AZ not same as local AZ) that
> is not
> + * present in local AZ, create local entry in NB with external-id
> + * "ic-learnt" set to true.
> + */
> + const struct nbrec_address_set *nb_as;
> + NBREC_ADDRESS_SET_FOR_EACH (nb_as, ctx->ovnnb_idl) {
> + const struct icsbrec_address_set *icsb_as;
> + icsb_as = shash_find_and_delete(&ic_remote_as, nb_as->name);
> + if (smap_get_bool(&nb_as->external_ids, "ic-learnt", false)) {
> + if (!icsb_as || !global_learn) {
> + nbrec_address_set_delete(nb_as);
> + } else {
> + update_nb_addr_set(icsb_as, nb_as);
> + }
> + }
> + }
> +
> + if (global_learn) {
> + SHASH_FOR_EACH (node, &ic_remote_as) {
> + /* In case local address-set with same name exists, we
> + * will not overwrite it because such address sets are already
> + * removed from ic_remote_as in the loop above.
> + */
> + if (node->data) {
> + sync_addr_set_from_icsb(ctx->ovnnb_txn, node->data);
> + }
> + }
> + }
> + shash_destroy(&ic_remote_as);
> +}
> +
> +
> static void
> gateway_run(struct ic_context *ctx)
> {
> @@ -3199,6 +3366,7 @@ ovn_db_run(struct ic_context *ctx)
> port_binding_run(ctx);
> route_run(ctx);
> sync_service_monitor(ctx);
> + address_set_run(ctx);
>
> ovn_destroy_tnlids(&dp_tnlids);
> shash_destroy(&isb_ts_dps);
> @@ -3536,6 +3704,17 @@ main(int argc, char *argv[])
> ovsdb_idl_track_add_column(ovnnb_idl_loop.idl,
>
> &nbrec_load_balancer_group_col_load_balancer);
>
> + ovsdb_idl_add_table(ovnnb_idl_loop.idl, &nbrec_table_address_set);
> + ovsdb_idl_track_add_column(ovnnb_idl_loop.idl,
> + &nbrec_address_set_col_name);
> + ovsdb_idl_track_add_column(ovnnb_idl_loop.idl,
> + &nbrec_address_set_col_addresses);
> + ovsdb_idl_track_add_column(ovnnb_idl_loop.idl,
> + &nbrec_address_set_col_options);
> + ovsdb_idl_track_add_column(ovnnb_idl_loop.idl,
> + &nbrec_address_set_col_external_ids);
> +
> +
> /* ovn-sb db. */
> struct ovsdb_idl_loop ovnsb_idl_loop = OVSDB_IDL_LOOP_INITIALIZER(
> ovsdb_idl_create(ovnsb_db, &sbrec_idl_class, false, true));
> @@ -3621,6 +3800,15 @@ main(int argc, char *argv[])
> &sbrec_learned_route_col_ip_prefix);
> ovsdb_idl_track_add_column(ovnsb_idl_loop.idl,
> &sbrec_learned_route_col_datapath);
> +
> + ovsdb_idl_add_table(ovnsb_idl_loop.idl, &sbrec_table_address_set);
> + ovsdb_idl_track_add_column(ovnsb_idl_loop.idl,
> + &sbrec_address_set_col_name);
> + ovsdb_idl_track_add_column(ovnsb_idl_loop.idl,
> + &sbrec_address_set_col_addresses);
> + ovsdb_idl_track_add_column(ovnsb_idl_loop.idl,
> + &sbrec_address_set_col_options);
> +
> /* Create IDL indexes */
> struct ovsdb_idl_index *nbrec_ls_by_name
> = ovsdb_idl_index_create1(ovnnb_idl_loop.idl,
> diff --git a/northd/en-sync-sb.c b/northd/en-sync-sb.c
> index 0b2a85b3b..b3d96af16 100644
> --- a/northd/en-sync-sb.c
> +++ b/northd/en-sync-sb.c
> @@ -43,7 +43,8 @@ VLOG_DEFINE_THIS_MODULE(en_sync_to_sb);
>
> static void sync_addr_set(struct ovsdb_idl_txn *ovnsb_txn, const char
> *name,
> struct sorted_array *addresses,
> - struct shash *sb_address_sets);
> + struct shash *sb_address_sets,
> + const struct smap *options);
> static void sync_addr_sets(struct ovsdb_idl_txn *ovnsb_txn,
> const struct nbrec_address_set_table *,
> const struct nbrec_port_group_table *,
> @@ -118,6 +119,18 @@ en_sync_to_sb_addr_set_cleanup(void *data OVS_UNUSED)
>
> }
>
> +static void
> +sync_address_set_options(const struct sbrec_address_set *sb_addr_set,
> + const struct smap *nb_options)
> +{
> + if (!smap_equal(&sb_addr_set->options, nb_options)) {
> + struct smap new_options;
> + smap_clone(&new_options, nb_options);
> + sbrec_address_set_set_options(sb_addr_set, &new_options);
> + smap_destroy(&new_options);
> + }
> +}
> +
> enum engine_input_handler_result
> sync_to_sb_addr_set_nb_address_set_handler(struct engine_node *node,
> void *data OVS_UNUSED)
> @@ -153,6 +166,7 @@ sync_to_sb_addr_set_nb_address_set_handler(struct
> engine_node *node,
> sorted_array_from_dbrec(nb_addr_set, addresses);
> update_sb_addr_set(&addrs, sb_addr_set);
> sorted_array_destroy(&addrs);
> + sync_address_set_options(sb_addr_set, &nb_addr_set->options);
> }
>
> return EN_HANDLED_UNCHANGED;
> @@ -449,7 +463,8 @@ sync_to_sb_pb_lr_stateful_handler(struct engine_node
> *node,
> static void
> sync_addr_set(struct ovsdb_idl_txn *ovnsb_txn, const char *name,
> struct sorted_array *addresses,
> - struct shash *sb_address_sets)
> + struct shash *sb_address_sets,
> + const struct smap *options)
> {
> const struct sbrec_address_set *sb_address_set;
> sb_address_set = shash_find_and_delete(sb_address_sets,
> @@ -462,6 +477,9 @@ sync_addr_set(struct ovsdb_idl_txn *ovnsb_txn, const
> char *name,
> } else {
> update_sb_addr_set(addresses, sb_address_set);
> }
> + if (options) {
> + sync_address_set_options(sb_address_set, options);
> + }
> }
>
> /* OVN_Southbound Address_Set table contains same records as in north
> @@ -500,7 +518,7 @@ sync_addr_sets(struct ovsdb_idl_txn *ovnsb_txn,
> const char *svc_macs[] = {svc_monitor_macp, svc_monitor_macp_dst};
> struct sorted_array svc =
> sorted_array_from_unsorted(svc_macs, ARRAY_SIZE(svc_macs), false);
> - sync_addr_set(ovnsb_txn, "svc_monitor_mac", &svc, &sb_address_sets);
> + sync_addr_set(ovnsb_txn, "svc_monitor_mac", &svc, &sb_address_sets,
> NULL);
> sorted_array_destroy(&svc);
>
> /* sync port group generated address sets first */
> @@ -519,9 +537,9 @@ sync_addr_sets(struct ovsdb_idl_txn *ovnsb_txn,
> sorted_array_from_svec(&ipv6_addrs);
>
> sync_addr_set(ovnsb_txn, ipv4_addrs_name,
> - &ipv4_addrs_sorted, &sb_address_sets);
> + &ipv4_addrs_sorted, &sb_address_sets, NULL);
> sync_addr_set(ovnsb_txn, ipv6_addrs_name,
> - &ipv6_addrs_sorted, &sb_address_sets);
> + &ipv6_addrs_sorted, &sb_address_sets, NULL);
> sorted_array_destroy(&ipv4_addrs_sorted);
> sorted_array_destroy(&ipv6_addrs_sorted);
> svec_destroy(&ipv4_addrs);
> @@ -544,7 +562,7 @@ sync_addr_sets(struct ovsdb_idl_txn *ovnsb_txn,
> &lr_stateful_rec->lb_ips->ips_v4_reachable);
>
> sync_addr_set(ovnsb_txn, ipv4_addrs_name,
> - &ipv4_addrs_sorted, &sb_address_sets);
> + &ipv4_addrs_sorted, &sb_address_sets, NULL);
> sorted_array_destroy(&ipv4_addrs_sorted);
> free(ipv4_addrs_name);
> }
> @@ -556,7 +574,7 @@ sync_addr_sets(struct ovsdb_idl_txn *ovnsb_txn,
> &lr_stateful_rec->lb_ips->ips_v6_reachable);
>
> sync_addr_set(ovnsb_txn, ipv6_addrs_name,
> - &ipv6_addrs_sorted, &sb_address_sets);
> + &ipv6_addrs_sorted, &sb_address_sets, NULL);
> sorted_array_destroy(&ipv6_addrs_sorted);
> free(ipv6_addrs_name);
> }
> @@ -570,7 +588,8 @@ sync_addr_sets(struct ovsdb_idl_txn *ovnsb_txn,
> struct sorted_array addrs =
> sorted_array_from_dbrec(nb_address_set, addresses);
> sync_addr_set(ovnsb_txn, nb_address_set->name,
> - &addrs, &sb_address_sets);
> + &addrs, &sb_address_sets,
> + &nb_address_set->options);
> sorted_array_destroy(&addrs);
> }
>
> diff --git a/ovn-ic-sb.ovsschema b/ovn-ic-sb.ovsschema
> index e0e0fef5e..dd1b2533c 100644
> --- a/ovn-ic-sb.ovsschema
> +++ b/ovn-ic-sb.ovsschema
> @@ -1,7 +1,7 @@
> {
> "name": "OVN_IC_Southbound",
> - "version": "2.5.0",
> - "cksum": "1892994110 9713",
> + "version": "2.6.0",
> + "cksum": "3333694935 10343",
> "tables": {
> "IC_SB_Global": {
> "columns": {
> @@ -192,6 +192,19 @@
> "external_ids": {
> "type": {"key": "string", "value": "string",
> "min": 0, "max": "unlimited"}}},
> + "isRoot": true},
> + "Address_Set": {
> + "columns": {
> + "name": {"type": "string"},
> + "addresses": {"type": {"key": "string",
> + "min": 0,
> + "max": "unlimited"}},
> + "availability_zone": {"type": {"key": {"type": "uuid",
> + "refTable": "Availability_Zone"}}},
> + "external_ids": {
> + "type": {"key": "string", "value": "string",
> + "min": 0, "max": "unlimited"}}},
> + "indexes": [["name"]],
> "isRoot": true}
> }
> }
> diff --git a/ovn-ic-sb.xml b/ovn-ic-sb.xml
> index f30760100..4fe491fac 100644
> --- a/ovn-ic-sb.xml
> +++ b/ovn-ic-sb.xml
> @@ -775,4 +775,30 @@
> Copy from source SBDB record.
> </column>
> </table>
> +
> + <table name="Address_Set" title="Address Sets">
> + <p>
> + Each row in this table represents an Address_Set advertised.
> + </p>
> +
> + <column name="name">
> + A name for the address set. Names are ASCII and must match
> + <code>[a-zA-Z_.][a-zA-Z_.0-9]*</code>.
> + </column>
> +
> + <column name="addresses">
> + The set of addresses in string form.
> + </column>
> +
> + <column name="availability_zone">
> + The availability zone that has advertised the Address_Set.
> + </column>
> +
> + <group title="Common Columns">
> + <column name="external_ids">
> + See <em>External IDs</em> at the beginning of this document.
> + </column>
> + </group>
> + </table>
> +
> </database>
> diff --git a/ovn-nb.ovsschema b/ovn-nb.ovsschema
> index e5945b831..00909fff4 100644
> --- a/ovn-nb.ovsschema
> +++ b/ovn-nb.ovsschema
> @@ -1,7 +1,7 @@
> {
> "name": "OVN_Northbound",
> - "version": "7.18.0",
> - "cksum": "1537030958 45190",
> + "version": "7.18.1",
> + "cksum": "2442107800 45407",
> "tables": {
> "NB_Global": {
> "columns": {
> @@ -292,6 +292,11 @@
> "addresses": {"type": {"key": "string",
> "min": 0,
> "max": "unlimited"}},
> + "options": {
> + "type": {"key": "string",
> + "value": "string",
> + "min": 0,
> + "max": "unlimited"}},
> "external_ids": {
> "type": {"key": "string", "value": "string",
> "min": 0, "max": "unlimited"}}},
> diff --git a/ovn-nb.xml b/ovn-nb.xml
> index 2e8a6a6f1..d98f3244d 100644
> --- a/ovn-nb.xml
> +++ b/ovn-nb.xml
> @@ -583,6 +583,25 @@
> </column>
> </group>
>
> + <group title="Options for interconnection address_set
> advertisement">
> + <p>
> + These options control whether address_sets propagate from one
> OVN
> + deployment to another via the global <ref
> db="OVN_IC_Southbound"/>.
> + </p>
> +
> + <column name="options" key="ic-as-adv">
> + A boolean value that enables address_set advertisement to the
> global
> + <ref db="OVN_IC_Southbound"/> database. Default is
> + <code>false</code>.
> + </column>
> +
> + <column name="options" key="ic-as-learn">
> + A boolean value that enables address_set learning from the
> global
> + <ref db="OVN_IC_Southbound"/> database. Default is
> + <code>false</code>.
> + </column>
> + </group>
> +
> </group>
>
> <group title="Connection Options">
> @@ -2409,6 +2428,18 @@
> See <em>External IDs</em> at the beginning of this document.
> </column>
> </group>
> +
> + <group title="Common options">
> + <column name="options">
> + This column provides general key/value settings. The supported
> + options are described individually below.
> + </column>
> +
> + <column name="options" key="ic-adv">
> + A boolean value that enables advertisement of this Address_Set to
> + the global OVN_IC_Southbound database. Default is false.
> + </column>
> + </group>
> </table>
>
> <table name="Port_Group" title="Port Groups">
> diff --git a/ovn-sb.ovsschema b/ovn-sb.ovsschema
> index d9a91739c..7057178ce 100644
> --- a/ovn-sb.ovsschema
> +++ b/ovn-sb.ovsschema
> @@ -1,7 +1,7 @@
> {
> "name": "OVN_Southbound",
> "version": "21.8.0",
> - "cksum": "614397313 36713",
> + "cksum": "2227484712 36930",
> "tables": {
> "SB_Global": {
> "columns": {
> @@ -77,7 +77,12 @@
> "name": {"type": "string"},
> "addresses": {"type": {"key": "string",
> "min": 0,
> - "max": "unlimited"}}},
> + "max": "unlimited"}},
> + "options": {
> + "type": {"key": "string",
> + "value": "string",
> + "min": 0,
> + "max": "unlimited"}}},
> "indexes": [["name"]],
> "isRoot": true},
> "Port_Group": {
> diff --git a/ovn-sb.xml b/ovn-sb.xml
> index 46568da62..51c16600d 100644
> --- a/ovn-sb.xml
> +++ b/ovn-sb.xml
> @@ -588,6 +588,10 @@
>
> <column name="name"/>
> <column name="addresses"/>
> + <group title="Common Columns">
> + <column name="options">
> + </column>
> + </group>
> </table>
>
> <table name="Port_Group" title="Port Groups">
> diff --git a/tests/ovn-ic.at b/tests/ovn-ic.at
> index 0fa7c4f29..45f64e551 100644
> --- a/tests/ovn-ic.at
> +++ b/tests/ovn-ic.at
> @@ -487,6 +487,178 @@ OVN_CLEANUP_IC([az1], [az2])
> AT_CLEANUP
> ])
>
> +OVN_FOR_EACH_NORTHD([
> +AT_SETUP([ovn-ic -- address-set sync])
> +
> +ovn_init_ic_db
> +ovn-ic-nbctl ts-add ts1
> +
> +for i in 1 2; do
> + ovn_start az$i
> + ovn_as az$i
> + check ovn-ic-nbctl --wait=sb sync
> + # Enable address-set learning at AZ level
> + check ovn-nbctl set nb_global . options:ic-as-learn=true
> + # Enable address-set advertising at AZ level
> + check ovn-nbctl set nb_global . options:ic-as-adv=true
> +
> + # Create address-set to advertise
> + check_uuid ovn-nbctl create Address_Set name=az$i-as1
> addresses=\"$i.1.1.1\",\"$i.1.1.2\" options:ic-adv=true
> +done
> +
> +wait_row_count ic-sb:Address_Set 2
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as1], [0], [dnl
> +addresses : [["1.1.1.1", "1.1.1.2"]]
> +])
> +
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az2-as1], [0], [dnl
> +addresses : [["2.1.1.1", "2.1.1.2"]]
> +])
> +
> +for i in 1 2; do
> + OVS_WAIT_UNTIL([ovn_as az$i ovn-nbctl list Address_Set | grep learnt])
> +done
> +
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az2-as1], [0], [dnl
> +addresses : [["2.1.1.1", "2.1.1.2"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az1-as1], [0], [dnl
> +addresses : [["1.1.1.1", "1.1.1.2"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +# Update AZ1 address_set (add 2 new IPs and delete one) and check it is
> synced to IC and AZ2.
> +ovn_as az1 ovn-nbctl set Address_Set az1-as1
> addresses=\"1.1.1.1\",\"1.1.1.3\",\"1.1.1.4\"
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as1], [0], [dnl
> +addresses : [["1.1.1.1", "1.1.1.3", "1.1.1.4"]]
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az1-as1], [0], [dnl
> +addresses : [["1.1.1.1", "1.1.1.3", "1.1.1.4"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +# Add 2 new address-sets in AZ1 - one to advertise, another not to.
> +ovn_as az1 ovn-nbctl create Address_Set name=az1-as2
> addresses=\"1.1.2.1\",\"1.1.2.2\" options:ic-adv=true
> +ovn_as az1 ovn-nbctl create Address_Set name=az1-as3
> addresses=\"1.1.3.1\",\"1.1.3.2\"
> +wait_row_count ic-sb:Address_Set 3
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as2], [0], [dnl
> +addresses : [["1.1.2.1", "1.1.2.2"]]
> +])
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as3], [0], [dnl
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az1-as2], [0], [dnl
> +addresses : [["1.1.2.1", "1.1.2.2"]]
> +external_ids : {ic-learnt="true"}
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids find
> Address_Set name=az1-as3], [0], [dnl
> +])
> +
> +# Now set the option field of the above address-set to advertise it.
> +ovn_as az1 ovn-nbctl set Address_Set az1-as3 options={ic-adv=true}
> +wait_row_count ic-sb:Address_Set 4
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as3], [0], [dnl
> +addresses : [["1.1.3.1", "1.1.3.2"]]
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az1-as3], [0], [dnl
> +addresses : [["1.1.3.1", "1.1.3.2"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +# Change the options of one of the address-sets to not advertise.
> +ovn_as az1 ovn-nbctl set Address_Set az1-as2 options={ic-adv=false}
> +wait_row_count ic-sb:Address_Set 3
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as2], [0], [dnl
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids find
> Address_Set name=az1-as2], [0], [dnl
> +])
> +
> +# Delete one of the advertised address-sets in AZ1.
> +ovn_as az1 ovn-nbctl destroy Address_Set az1-as3
> +wait_row_count ic-sb:Address_Set 2
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az1-as3], [0], [dnl
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids find
> Address_Set name=az1-as3], [0], [dnl
> +])
> +
> +# Add an IPv6 address-set in AZ2.
> +ovn_as az2 ovn-nbctl create Address_Set name=az2-as2
> addresses=\"2001:0db8:85a3:0000:0000:8a2e:0370:7334\",\"2001:0db8:85a3:0000:0000:8a2e:0370:7335\"
> options:ic-adv=true
> +wait_row_count ic-sb:Address_Set 3
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az2-as2], [0], [dnl
> +addresses : [["2001:0db8:85a3:0000:0000:8a2e:0370:7334",
> "2001:0db8:85a3:0000:0000:8a2e:0370:7335"]]
> +])
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az2-as2], [0], [dnl
> +addresses : [["2001:0db8:85a3:0000:0000:8a2e:0370:7334",
> "2001:0db8:85a3:0000:0000:8a2e:0370:7335"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +# Set the global option in AZ1 to not learn.
> +ovn_as az1 ovn-nbctl set nb_global . options:ic-as-learn=false
> +OVS_WAIT_WHILE([ovn_as az1 ovn-nbctl list Address_Set | grep learnt])
> +ovn_as az1 wait_row_count nb:Address_Set 2
> +
> +# Set the global option in AZ1 back to learn.
> +ovn_as az1 ovn-nbctl set nb_global . options:ic-as-learn=true
> +OVS_WAIT_UNTIL([ovn_as az1 ovn-nbctl list Address_Set | grep learnt])
> +ovn_as az1 wait_row_count nb:Address_Set 4
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az2-as2], [0], [dnl
> +addresses : [["2001:0db8:85a3:0000:0000:8a2e:0370:7334",
> "2001:0db8:85a3:0000:0000:8a2e:0370:7335"]]
> +external_ids : {ic-learnt="true"}
> +])
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az2-as1], [0], [dnl
> +addresses : [["2.1.1.1", "2.1.1.2"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +# Set the global option in AZ2 to not advertise.
> +ovn_as az2 ovn-nbctl set nb_global . options:ic-as-adv=false
> +wait_row_count ic-sb:Address_Set 1
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az2-as1], [0], [dnl])
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az2-as2], [0], [dnl])
> +OVS_WAIT_WHILE([ovn_as az1 ovn-nbctl list Address_Set | grep learnt])
> +
> +# Set the global option in AZ2 back to advertise.
> +ovn_as az2 ovn-nbctl set nb_global . options:ic-as-adv=true
> +wait_row_count ic-sb:Address_Set 3
> +ovn_as az1 wait_row_count nb:Address_Set 4
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az2-as1], [0], [dnl
> +addresses : [["2.1.1.1", "2.1.1.2"]]
> +])
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set
> name=az2-as2], [0], [dnl
> +addresses : [["2001:0db8:85a3:0000:0000:8a2e:0370:7334",
> "2001:0db8:85a3:0000:0000:8a2e:0370:7335"]]
> +])
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az2-as1], [0], [dnl
> +addresses : [["2.1.1.1", "2.1.1.2"]]
> +external_ids : {ic-learnt="true"}
> +])
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set az2-as2], [0], [dnl
> +addresses : [["2001:0db8:85a3:0000:0000:8a2e:0370:7334",
> "2001:0db8:85a3:0000:0000:8a2e:0370:7335"]]
> +external_ids : {ic-learnt="true"}
> +])
> +
> +# Create an address-set in AZ1 and then try to advertise an address-set
> with same name from AZ2.
> +# It will sync to IC-SB but will not propagate to AZ1.
> +ovn_as az1 ovn-nbctl create Address_Set name=dup-as addresses=\"5.5.5.1\"
> +ovn_as az2 ovn-nbctl create Address_Set name=dup-as addresses=\"5.5.5.2\"
> options:ic-adv=true
> +wait_row_count ic-sb:Address_Set 4
> +AT_CHECK([ovn-ic-sbctl --columns=addresses find Address_Set name=dup-as],
> [0], [dnl
> +addresses : [["5.5.5.2"]]
> +])
> +AT_CHECK([ovn_as az1 ovn-nbctl --columns=addresses,external_ids list
> Address_Set dup-as], [0], [dnl
> +addresses : [["5.5.5.1"]]
> +external_ids : {}
> +])
> +AT_CHECK([ovn_as az2 ovn-nbctl --columns=addresses,external_ids list
> Address_Set dup-as], [0], [dnl
> +addresses : [["5.5.5.2"]]
> +external_ids : {}
> +])
> +
> +OVN_CLEANUP_IC([az1], [az2])
> +
> +AT_CLEANUP
> +])
> +
> OVN_FOR_EACH_NORTHD([
> AT_SETUP([ovn-ic -- route sync])
>
> --
> 2.39.3
>
> _______________________________________________
> dev mailing list
> [email protected]
> https://mail.openvswitch.org/mailman/listinfo/ovs-dev
>
_______________________________________________
dev mailing list
[email protected]
https://mail.openvswitch.org/mailman/listinfo/ovs-dev