On Fri, Jun 6, 2025 at 8:14 PM Mark Michelson via dev <
ovs-dev@openvswitch.org> wrote:

> Prior to this commit, if you wanted to find the corresponding northbound
> UUID for a southbound Logical Datapath, you could find it in one of two
> places:
>
> For logical switches, it was in external-ids:logical-switch.
> For logical routers, it was in external-ids:logical-router.
>
> With this commit, we are separating the type and UUID into separate
> fields. This way, no matter the type of the datapath, you can find the
> UUID. And you can find the type of the datapath without having to
> potentially check multiple external-ids keys to do so.
>
> These fields are going to be used pretty heavily by northd in upcoming
> patches, so instead of making them external-ids, these are now
> full-fledged columns on the southbound Datapath_Binding. The UUID of the
> northbound logical datapath is in a column called "nb_uuid" and the type
> of the logical datapath is stored in an enumerated column called "type".
>
> Note that there is a seemingly-unrelated change in the check packet
> length test in tests/ovn.at. This test started failing because its use
> of `grep` was picking up the new "nb_uuid" external-id accidentally. The
> change to the test uses `fetch_column` since it is more precise.
>
> Note 2: I attempted to make the new "nb_uuid" column type "UUID" but
> this caused an assertion in ovs's db-ctl-base.c file when running the
> ovn-sbctl "invalid 0x flow" and "count-flows wrongDatapath" tests. This
> is because db-ctl-base.c is incapable of handling records other than of
> type "int" or "string". The downside is that we have to use
> uuid_to_string() in order to ensure the UUID is in proper form.
>
> Signed-off-by: Mark Michelson <mmich...@redhat.com>
> ---
>

Hi Mark,

thank you for v7. I have a few more comments down below.
Also there is a lot of CI failures, could you please take a look
at it?


> * v7
>   * Rebased.
>   * Changed from using external-ids for the nb_uuid and type to using
>     new columns in the Datapath_Binding table.
>
> * v6
>   * This is the first version of the series to contain this patch.
>  ic/ovn-ic.c             |  5 +++--
>  northd/northd.c         | 31 ++++++++++++++++++-------------
>  northd/northd.h         |  5 +++--
>  ovn-sb.ovsschema        |  7 +++++--
>  ovn-sb.xml              | 21 +++++++++++----------
>  tests/ovn-controller.at |  4 ++++
>  tests/ovn-northd.at     |  4 ++--
>  tests/ovn.at            |  6 ++----
>  utilities/ovn-sbctl.c   |  4 ++--
>  utilities/ovn-trace.c   |  3 +--
>  10 files changed, 51 insertions(+), 39 deletions(-)
>
> diff --git a/ic/ovn-ic.c b/ic/ovn-ic.c
> index 0dd079c4e..de5a91b23 100644
> --- a/ic/ovn-ic.c
> +++ b/ic/ovn-ic.c
> @@ -613,8 +613,7 @@ get_router_uuid_by_sb_pb(struct ic_context *ctx,
>          return NULL;
>      }
>
> -    return smap_get_uuid(&router_pb->datapath->external_ids,
> "logical-router",
> -                         router_uuid);
> +    return uuid_from_string(router_uuid, router_pb->datapath->nb_uuid);
>  }
>
>  static void
> @@ -2391,6 +2390,8 @@ main(int argc, char *argv[])
>      ovsdb_idl_add_table(ovnsb_idl_loop.idl,
> &sbrec_table_datapath_binding);
>      ovsdb_idl_add_column(ovnsb_idl_loop.idl,
>                           &sbrec_datapath_binding_col_external_ids);
> +    ovsdb_idl_add_column(ovnsb_idl_loop.idl,
> +                         &sbrec_datapath_binding_col_nb_uuid);
>
>      ovsdb_idl_add_table(ovnsb_idl_loop.idl, &sbrec_table_port_binding);
>      ovsdb_idl_add_column(ovnsb_idl_loop.idl,
> diff --git a/northd/northd.c b/northd/northd.c
> index 2da94e7e1..c3fa5d3eb 100644
> --- a/northd/northd.c
> +++ b/northd/northd.c
> @@ -580,14 +580,15 @@ ovn_datapath_from_sbrec(const struct hmap
> *ls_datapaths,
>      struct uuid key;
>      const struct hmap *dps;
>
> -    if (smap_get_uuid(&sb->external_ids, "logical-switch", &key)) {
> +    if (!strcmp(sb->type, "logical-switch")) {
>          dps = ls_datapaths;
> -    } else if (smap_get_uuid(&sb->external_ids, "logical-router", &key)) {
> +    } else if (!strcmp(sb->type, "logical-router")) {
>          dps = lr_datapaths;
>      } else {
>          return NULL;
>      }
> -    if (!dps) {
> +
> +    if (!uuid_from_string(&key, sb->nb_uuid)) {
>          return NULL;
>      }
>      struct ovn_datapath *od = ovn_datapath_find_(dps, &key);
> @@ -749,11 +750,9 @@ store_mcast_info_for_switch_datapath(const struct
> sbrec_ip_multicast *sb,
>  static void
>  ovn_datapath_update_external_ids(struct ovn_datapath *od)
>  {
> -    /* Get the logical-switch or logical-router UUID to set in
> -     * external-ids. */
> +    /* Get the NB  UUID to set in external-ids. */
>      char uuid_s[UUID_LEN + 1];
>      sprintf(uuid_s, UUID_FMT, UUID_ARGS(&od->key));
>

The uuid_s isn't actually used for anything.


> -    const char *key = od->nbs ? "logical-switch" : "logical-router";
>
>      /* Get names to set in external-ids. */
>      const char *name = od->nbs ? od->nbs->name : od->nbr->name;
> @@ -765,7 +764,6 @@ ovn_datapath_update_external_ids(struct ovn_datapath
> *od)
>
>      /* Set external-ids. */
>      struct smap ids = SMAP_INITIALIZER(&ids);
> -    smap_add(&ids, key, uuid_s);
>      smap_add(&ids, "name", name);
>      if (name2 && name2[0]) {
>          smap_add(&ids, "name2", name2);
> @@ -893,13 +891,10 @@ join_datapaths(const struct
> nbrec_logical_switch_table *nbrec_ls_table,
>      const struct sbrec_datapath_binding *sb;
>      SBREC_DATAPATH_BINDING_TABLE_FOR_EACH_SAFE (sb, sbrec_dp_table) {
>          struct uuid key;
> -        if (!smap_get_uuid(&sb->external_ids, "logical-switch", &key) &&
> -            !smap_get_uuid(&sb->external_ids, "logical-router", &key)) {
> +        if (!uuid_from_string(&key, sb->nb_uuid)) {
>              ovsdb_idl_txn_add_comment(
>                  ovnsb_txn,
> -                "deleting Datapath_Binding "UUID_FMT" that lacks "
> -                "external-ids:logical-switch and "
> -                "external-ids:logical-router",
> +                "deleting Datapath_Binding "UUID_FMT" that lacks nb_uuid",
>                  UUID_ARGS(&sb->header_.uuid));
>              sbrec_datapath_binding_delete(sb);
>              continue;
> @@ -909,7 +904,7 @@ join_datapaths(const struct nbrec_logical_switch_table
> *nbrec_ls_table,
>              static struct vlog_rate_limit rl = VLOG_RATE_LIMIT_INIT(5, 1);
>              VLOG_INFO_RL(
>                  &rl, "deleting Datapath_Binding "UUID_FMT" with "
> -                "duplicate external-ids:logical-switch/router "UUID_FMT,
> +                "duplicate nb_uuid "UUID_FMT,
>                  UUID_ARGS(&sb->header_.uuid), UUID_ARGS(&key));
>              sbrec_datapath_binding_delete(sb);
>              continue;
> @@ -1117,11 +1112,21 @@ build_datapaths(struct ovsdb_idl_txn *ovnsb_txn,
>              sbrec_datapath_binding_set_tunnel_key(od->sb, od->tunnel_key);
>          }
>          ovn_datapath_update_external_ids(od);
> +        char uuid_s[UUID_LEN + 1];
> +        sprintf(uuid_s, UUID_FMT, UUID_ARGS(&od->key));
> +        sbrec_datapath_binding_set_nb_uuid(od->sb, uuid_s);
> +        sbrec_datapath_binding_set_type(od->sb, od->nbs ?
> "logical-switch" :
> +                                        "logical-router");
>      }
>      LIST_FOR_EACH (od, list, &nb_only) {
>          od->sb = sbrec_datapath_binding_insert(ovnsb_txn);
>          ovn_datapath_update_external_ids(od);
>          sbrec_datapath_binding_set_tunnel_key(od->sb, od->tunnel_key);
> +        char uuid_s[UUID_LEN + 1];
> +        sprintf(uuid_s, UUID_FMT, UUID_ARGS(&od->key));
> +        sbrec_datapath_binding_set_nb_uuid(od->sb, uuid_s);
> +        sbrec_datapath_binding_set_type(od->sb, od->nbs ?
> "logical-switch" :
> +                                        "logical-router");
>      }
>      ovn_destroy_tnlids(&dp_tnlids);
>
> diff --git a/northd/northd.h b/northd/northd.h
> index fc138aab5..998ee8090 100644
> --- a/northd/northd.h
> +++ b/northd/northd.h
> @@ -346,7 +346,7 @@ DRR_MODES
>  #undef DRR_MODE
>
>  /* The 'key' comes from nbs->header_.uuid or nbr->header_.uuid or
> - * sb->external_ids:logical-switch. */
> + * sb->nb_uuid. */
>  struct ovn_datapath {
>      struct hmap_node key_node;  /* Index on 'key'. */
>      struct uuid key;            /* (nbs/nbr)->header_.uuid. */
> @@ -448,7 +448,8 @@ ovn_datapath_is_stale(const struct ovn_datapath *od)
>  /* The two purposes for which ovn-northd uses OVN logical datapaths. */
>  enum ovn_datapath_type {
>      DP_SWITCH,                  /* OVN logical switch. */
> -    DP_ROUTER                   /* OVN logical router. */
> +    DP_ROUTER,                  /* OVN logical router. */
> +    DP_MAX,
>  };
>
>  /* Returns an "enum ovn_stage" built from the arguments.
> diff --git a/ovn-sb.ovsschema b/ovn-sb.ovsschema
> index 4c24f5b51..a020a5c1c 100644
> --- a/ovn-sb.ovsschema
> +++ b/ovn-sb.ovsschema
> @@ -1,7 +1,7 @@
>  {
>      "name": "OVN_Southbound",
> -    "version": "21.2.0",
> -    "cksum": "29145795 34859",
> +    "version": "21.3.0",
> +    "cksum": "3112866017 35066",
>      "tables": {
>          "SB_Global": {
>              "columns": {
> @@ -196,6 +196,9 @@
>                  "load_balancers": {"type": {"key": {"type": "uuid"},
>                                              "min": 0,
>                                              "max": "unlimited"}},
> +                "type": {"type": {"key": {"type": "string",
> +                                          "enum": ["set",
> ["logical-switch", "logical-router"]]}}},
> +                "nb_uuid": {"type": "string"},
>

Why not use the type "uuid" directly? We could avoid the extra parsing.

                 "external_ids": {
>                      "type": {"key": "string", "value": "string",
>                               "min": 0, "max": "unlimited"}}},
> diff --git a/ovn-sb.xml b/ovn-sb.xml
> index db5faac66..ad09a472e 100644
> --- a/ovn-sb.xml
> +++ b/ovn-sb.xml
> @@ -3236,18 +3236,19 @@ tcp.flags = RST;
>          db="OVN_Northbound"/> database.
>        </p>
>
> -      <column name="external_ids" key="logical-switch" type='{"type":
> "uuid"}'>
> -        For a logical datapath that represents a logical switch,
> -        <code>ovn-northd</code> stores in this key the UUID of the
> -        corresponding <ref table="Logical_Switch" db="OVN_Northbound"/>
> row in
> -        the <ref db="OVN_Northbound"/> database.
> +      <column name="type" type='{"type": "string"}'>
> +        This represents the type of the northbound logical datapath that
> is
> +        represented by this southbound <code>Datapath_Binding</code>. The
> +        possible values for this are:
> +        <ul>
> +          <li><code>logical-switch</code></li>
> +          <li><code>logical-router</code></li>
> +        </ul>
>        </column>
>
> -      <column name="external_ids" key="logical-router" type='{"type":
> "uuid"}'>
> -        For a logical datapath that represents a logical router,
> -        <code>ovn-northd</code> stores in this key the UUID of the
> -        corresponding <ref table="Logical_Router" db="OVN_Northbound"/>
> row in
> -        the <ref db="OVN_Northbound"/> database.
> +      <column name="nb_uuid" type='{"type": "string"}'>
> +        This is the UUID of the corresponding northbound logical datapath
> +        represented by this southbound <code>Datapath_Binding</code>.
>        </column>
>
>        <column name="external_ids" key="interconn-ts" type='{"type":
> "string"}'>
> diff --git a/tests/ovn-controller.at b/tests/ovn-controller.at
> index ae7eb6f31..137442262 100644
> --- a/tests/ovn-controller.at
> +++ b/tests/ovn-controller.at
> @@ -123,6 +123,7 @@ check_patches
>  # another logical port bound to this chassis.
>  check_uuid ovn-sbctl \
>      -- --id=@dp101 create Datapath_Binding tunnel_key=101
> external_ids:name=dp101 \
> +       type=logical-switch \
>      -- create Port_Binding datapath=@dp101 logical_port=localnet1
> tunnel_key=1 \
>          type=localnet options:network_name=physnet1
>  check_patches
> @@ -131,6 +132,7 @@ check_patches
>  # Now we should get some patch ports created.
>  check_uuid ovn-sbctl \
>      -- --id=@dp102 create Datapath_Binding tunnel_key=102
> external_ids:name=dp102 \
> +       type=logical-switch \
>      -- create Port_Binding datapath=@dp102 logical_port=localnet2
> tunnel_key=1 \
>          type=localnet options:network_name=physnet1 \
>      -- create Port_Binding datapath=@dp102 logical_port=localvif2
> tunnel_key=2
> @@ -145,7 +147,9 @@ check_patches \
>  # the set of OVS patch ports doesn't change.
>  AT_CHECK([ovn-sbctl \
>      -- --id=@dp1 create Datapath_Binding tunnel_key=1
> external_ids:name=dp1 \
> +       type=logical-switch \
>      -- --id=@dp2 create Datapath_Binding tunnel_key=2
> external_ids:name=dp2 \
> +       type=logical-switch \
>      -- create Port_Binding datapath=@dp1 logical_port=foo tunnel_key=1
> type=patch options:peer=bar \
>      -- create Port_Binding datapath=@dp2 logical_port=bar tunnel_key=2
> type=patch options:peer=foo \
>      -- create Port_Binding datapath=@dp1 logical_port=dp1vif tunnel_key=3
> \
> diff --git a/tests/ovn-northd.at b/tests/ovn-northd.at
> index 97eaa6a40..f334698dd 100644
> --- a/tests/ovn-northd.at
> +++ b/tests/ovn-northd.at
> @@ -1887,7 +1887,7 @@ check ovn-nbctl --wait=sb \
>
>  check_row_count Datapath_Binding 1
>
> -nb_uuid=$(ovn-sbctl get Datapath_Binding . external_ids:logical-router)
> +nb_uuid=$(ovn-sbctl get Datapath_Binding . nb_uuid)
>  lr_uuid=\"$(ovn-nbctl get Logical_Router . _uuid)\"
>  echo nb_uuid="$nb_uuid" lr_uuid="$lr_uuid"
>  AT_CHECK([test "${nb_uuid}" = "${lr_uuid}"])
> @@ -7574,7 +7574,7 @@ ls1_uuid=$(fetch_column nb:Logical_Switch _uuid)
>
>  # create a duplicated sb datapath (and an IP_Mulicast record that
> references
>  # it) on purpose.
> -AT_CHECK([ovn-sbctl --id=@dp create Datapath_Binding
> external_ids:logical-switch=$ls1_uuid external_ids:name=ls1 tunnel_key=123
> -- create IP_Multicast datapath=@dp], [0], [ignore])
> +AT_CHECK([ovn-sbctl --id=@dp create Datapath_Binding nb_uuid=$ls1_uuid
> type=logical-switch external_ids:name=ls1 tunnel_key=123 -- create
> IP_Multicast datapath=@dp], [0], [ignore])
>
>  # northd should delete one of the datapaths in the end
>  wait_row_count Datapath_Binding 1
> diff --git a/tests/ovn.at b/tests/ovn.at
> index b561c69aa..f11dcd1f8 100644
> --- a/tests/ovn.at
> +++ b/tests/ovn.at
> @@ -22193,8 +22193,7 @@ AT_CAPTURE_FILE([sbflows])
>  AT_CHECK([as hv1 ovs-ofctl dump-flows br-int  \
>  | grep "check_pkt_larger" | wc -l], [0], [[0
>  ]])
> -dp_uuid=$(ovn-sbctl find datapath_binding | grep lr0 -B2 | grep _uuid | \
> -awk '{print $3}')
> +dp_uuid=$(fetch_column Datapath_Binding _uuid external-ids:name=lr0)
>  check_uuid ovn-sbctl create MAC_Binding ip=172.168.0.3 datapath=$dp_uuid \
>  logical_port=lr0-public mac="00\:00\:00\:12\:af\:11"
>
> @@ -22264,8 +22263,7 @@ check ovn-nbctl lr-nat-add lr1 snat 172.168.0.100
> 10.0.0.0/24
>  check ovn-nbctl lr-nat-add lr1 snat 2000::1 1000::/64
>  check ovn-nbctl --wait=sb sync
>
> -dp_uuid=$(ovn-sbctl find datapath_binding | grep lr1 -B2 | grep _uuid | \
> -awk '{print $3}')
> +dp_uuid=$(fetch_column Datapath_Binding _uuid external-ids:name=lr1)
>  check_uuid ovn-sbctl create MAC_Binding ip=172.168.0.3 datapath=$dp_uuid \
>  logical_port=lr1-public mac="00\:00\:00\:12\:af\:11"
>
> diff --git a/utilities/ovn-sbctl.c b/utilities/ovn-sbctl.c
> index 60cc39149..9c7be5e57 100644
> --- a/utilities/ovn-sbctl.c
> +++ b/utilities/ovn-sbctl.c
> @@ -1493,8 +1493,8 @@ static const struct ctl_table_class
> tables[SBREC_N_TABLES] = {
>      [SBREC_TABLE_DATAPATH_BINDING].row_ids
>       = {{&sbrec_datapath_binding_col_external_ids, "name", NULL},
>          {&sbrec_datapath_binding_col_external_ids, "name2", NULL},
> -        {&sbrec_datapath_binding_col_external_ids, "logical-switch",
> NULL},
> -        {&sbrec_datapath_binding_col_external_ids, "logical-router",
> NULL}},
> +        {&sbrec_datapath_binding_col_nb_uuid, NULL, NULL},
> +        {&sbrec_datapath_binding_col_type, NULL, NULL}},
>
>      [SBREC_TABLE_PORT_BINDING].row_ids
>       = {{&sbrec_port_binding_col_logical_port, NULL, NULL},
> diff --git a/utilities/ovn-trace.c b/utilities/ovn-trace.c
> index 0331eac21..648a1d43e 100644
> --- a/utilities/ovn-trace.c
> +++ b/utilities/ovn-trace.c
> @@ -703,8 +703,7 @@ read_datapaths(void)
>          const struct smap *ids = &sbdb->external_ids;
>
>          dp->sb_uuid = sbdb->header_.uuid;
> -        if (!smap_get_uuid(ids, "logical-switch", &dp->nb_uuid) &&
> -            !smap_get_uuid(ids, "logical-router", &dp->nb_uuid)) {
> +        if (!uuid_from_string(&dp->nb_uuid, sbdb->nb_uuid)) {
>              dp->nb_uuid = dp->sb_uuid;
>          }
>
> --
> 2.47.0
>
> _______________________________________________
> dev mailing list
> d...@openvswitch.org
> https://mail.openvswitch.org/mailman/listinfo/ovs-dev
>
>
Thanks,
Ales
_______________________________________________
dev mailing list
d...@openvswitch.org
https://mail.openvswitch.org/mailman/listinfo/ovs-dev

Reply via email to