diff --git a/cmd/cmd.go b/cmd/cmd.go index 991d4bf..34d8bff 100644 --- a/cmd/cmd.go +++ b/cmd/cmd.go @@ -9,6 +9,7 @@ import ( "os" "github.com/opiproject/godpu/cmd/common" + "github.com/opiproject/godpu/cmd/evpnipsec" "github.com/opiproject/godpu/cmd/inventory" "github.com/opiproject/godpu/cmd/ipsec" "github.com/opiproject/godpu/cmd/network" @@ -37,7 +38,7 @@ func NewCommand() *cobra.Command { c.AddCommand(ipsec.NewIPSecCommand()) c.AddCommand(storage.NewStorageCommand()) c.AddCommand(network.NewEvpnCommand()) - + c.AddCommand(evpnipsec.NewEvpnIPSecCommand()) flags := c.PersistentFlags() flags.String(common.AddrCmdLineArg, "localhost:50151", "address of OPI gRPC server") flags.String(common.TLSFiles, "", "TLS files in client_cert:client_key:ca_cert format.") diff --git a/cmd/evpnipsec/ipsec-evpn.go b/cmd/evpnipsec/ipsec-evpn.go new file mode 100644 index 0000000..b31f587 --- /dev/null +++ b/cmd/evpnipsec/ipsec-evpn.go @@ -0,0 +1,218 @@ +// SPDX-License-Identifier: Apache-2.0 +// Copyright (c) 2022-2023 Dell Inc, or its subsidiaries. + +// Package evpnipsec implements the ipsec related CLI commands +package evpnipsec + +import ( + "context" + "fmt" + "log" + "time" + + "github.com/opiproject/godpu/cmd/common" + "github.com/opiproject/godpu/evpnipsec" + pb "github.com/opiproject/opi-evpn-bridge/pkg/ipsec/gen/go" + "github.com/spf13/cobra" +) + +// AddSaCommand Add Sa Command +func AddSaCommand() *cobra.Command { + var ( + src string + dst string + spi uint32 + proto int32 + ifID uint32 + reqid uint32 + mode int32 + intrface string + encAlg string + encKey string + intAlg string + intKey string + replayWindow uint32 + tfc uint32 + encap int32 + esn int32 + copyDf int32 + copyEcn int32 + copyDscp int32 + initiator int32 + inbound int32 + update int32 + ) + // Create the map of string to CryptoAlgorithm + var EncAlgorithms = map[string]pb.CryptoAlgorithm{ + "rsvd": pb.CryptoAlgorithm_ENCR_RSVD, + "null": pb.CryptoAlgorithm_ENCR_NULL, + "aes_cbc": pb.CryptoAlgorithm_ENCR_AES_CBC, + "aes_ctr": pb.CryptoAlgorithm_ENCR_AES_CTR, + "aes_ccm_icv_8": pb.CryptoAlgorithm_ENCR_AES_CCM_8, + "aes_ccm_icv_12": pb.CryptoAlgorithm_ENCR_AES_CCM_12, + "aes_ccm_icv_16": pb.CryptoAlgorithm_ENCR_AES_CCM_16, + "aes_gcm_icv_8": pb.CryptoAlgorithm_ENCR_AES_GCM_8, + "aes_gcm_icv_12": pb.CryptoAlgorithm_ENCR_AES_GCM_12, + "aes_gcm_icv_16": pb.CryptoAlgorithm_ENCR_AES_GCM_16, + "aes_gmac": pb.CryptoAlgorithm_ENCR_NULL_AUTH_AES_GMAC, + "chacha_poly": pb.CryptoAlgorithm_ENCR_CHACHA20_POLY1305, + } + var IntAlgorithms = map[string]pb.IntegAlgorithm{ + "sha1_96": pb.IntegAlgorithm_AUTH_HMAC_SHA1_96, + "xcbc_96": pb.IntegAlgorithm_AUTH_AES_XCBC_96, + "cmac_96": pb.IntegAlgorithm_AUTH_AES_CMAC_96, + "gmac_128": pb.IntegAlgorithm_AUTH_AES_128_GMAC, + "gmac_192": pb.IntegAlgorithm_AUTH_AES_192_GMAC, + "gmac_256": pb.IntegAlgorithm_AUTH_AES_256_GMAC, + "sha2_128": pb.IntegAlgorithm_AUTH_HMAC_SHA2_256_128, + "sha2_192": pb.IntegAlgorithm_AUTH_HMAC_SHA2_384_192, + "sha2_256": pb.IntegAlgorithm_AUTH_HMAC_SHA2_512_256, + "none": pb.IntegAlgorithm_NONE, + } + + var cmd = &cobra.Command{ + Use: "add-sa", + Aliases: []string{"c"}, + Short: "add-sa functionality", + Args: cobra.NoArgs, + Run: func(c *cobra.Command, _ []string) { + tlsFiles, err := c.Flags().GetString(common.TLSFiles) + cobra.CheckErr(err) + + addr, err := c.Flags().GetString(common.AddrCmdLineArg) + cobra.CheckErr(err) + + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + + IPSecEvpnClient, err := evpnipsec.NewIPSecClient(addr, tlsFiles) + if err != nil { + log.Printf("error Adding SA: %s\n", err) + log.Println("ONE") + } + + data, err := IPSecEvpnClient.AddSA(ctx, + src, dst, spi, proto, ifID, reqid, mode, intrface, int32(EncAlgorithms[encAlg]), encKey, int32(IntAlgorithms[intAlg]), intKey, + replayWindow, tfc, encap, esn, copyDf, copyEcn, copyDscp, initiator, inbound, update, + ) + if err != nil { + log.Printf("error error Adding SA: %s\n", err) + log.Println("TWO") + } + fmt.Println("Add SA Req marshaled successfully:", data) + }, + } + + cmd.Flags().StringVar(&src, "src", "", "Source address or hostname") + cmd.Flags().StringVar(&dst, "dst", "", "Destination address or hostname") + cmd.Flags().Uint32Var(&spi, "spi", 0, "SPI") + cmd.Flags().Int32Var(&proto, "proto", 0, "Protocol (ESP/AH)") + cmd.Flags().Uint32Var(&ifID, "if_id", 0, "Interface ID") + cmd.Flags().Uint32Var(&reqid, "reqid", 0, "Reqid") + cmd.Flags().Int32Var(&mode, "mode", 0, "Mode (tunnel, transport...)") + cmd.Flags().StringVar(&intrface, "interface", "", "Network interface restricting policy") + cmd.Flags().StringVar(&encAlg, "enc_alg", "aes_cbc", "rsvd, null, aes_cbc, aes_ctr, aes_ccm_icv_8, aes_ccm_icv_12, aes_ccm_icv_16, aes_gcm_icv_8, aes_gcm_icv_12, aes_gcm_icv_16, aes_gmac, chacha_poly") + cmd.Flags().StringVar(&encKey, "enc_key", "", "Encryption key") + cmd.Flags().StringVar(&intAlg, "int_alg", "none", "Integrity protection algorithm: sha1_96, xcbc_96, cmac_96, gmac_128, gmac_192, gmac_256, sha2_128, sha2_192, sha2_256, none") + cmd.Flags().StringVar(&intKey, "int_key", "", "Integrity protection key") + cmd.Flags().Uint32Var(&replayWindow, "replay_window", 0, "Anti-replay window size") + cmd.Flags().Uint32Var(&tfc, "tfc", 0, "Traffic Flow Confidentiality padding") + cmd.Flags().Int32Var(&encap, "encap", 0, "Enable UDP encapsulation for NAT traversal") + cmd.Flags().Int32Var(&esn, "esn", 0, "Mark the SA should apply to packets after processing") + cmd.Flags().Int32Var(©Df, "copy_df", 0, "Copy the DF bit to the outer IPv4 header in tunnel mode") + cmd.Flags().Int32Var(©Ecn, "copy_ecn", 0, "Copy the ECN header field to/from the outer header") + cmd.Flags().Int32Var(©Dscp, "copy_dscp", 0, "Copy the DSCP header field to/from the outer header") + cmd.Flags().Int32Var(&initiator, "initiator", 0, "TRUE if initiator of the exchange creating the SA") + cmd.Flags().Int32Var(&inbound, "inbound", 0, "TRUE if this is an inbound SA") + cmd.Flags().Int32Var(&update, "update", 0, "TRUE if an SPI has already been allocated for this SA") + + if err := cmd.MarkFlagRequired("src"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + if err := cmd.MarkFlagRequired("dst"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + if err := cmd.MarkFlagRequired("spi"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + if err := cmd.MarkFlagRequired("if_id"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + + return cmd +} + +// DelSaCommand tests the del SA +func DelSaCommand() *cobra.Command { + var ( + src string + dst string + spi uint32 + proto int32 + ifID uint32 + ) + + var cmd = &cobra.Command{ + Use: "Del-sa", + Aliases: []string{"c"}, + Short: "add-sa functionality", + Args: cobra.NoArgs, + Run: func(c *cobra.Command, _ []string) { + tlsFiles, err := c.Flags().GetString(common.TLSFiles) + cobra.CheckErr(err) + + addr, err := c.Flags().GetString(common.AddrCmdLineArg) + cobra.CheckErr(err) + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + IPSecEvpnClient, err := evpnipsec.NewIPSecClient(addr, tlsFiles) + if err != nil { + log.Printf("error Deleting SA %s\n", err) + } + data, err := IPSecEvpnClient.DelSA(ctx, src, dst, spi, proto, ifID) + if err != nil { + log.Printf("error Deleting SA %s\n", err) + } + fmt.Println("Deleting SA successfully:", data) + }, + } + + cmd.Flags().StringVar(&src, "src", "", "Source address or hostname") + cmd.Flags().StringVar(&dst, "dst", "", "Destination address or hostname") + cmd.Flags().Uint32Var(&spi, "spi", 0, "SPI") + cmd.Flags().Int32Var(&proto, "proto", 0, "Protocol (ESP/AH)") + cmd.Flags().Uint32Var(&ifID, "if_id", 0, "Interface ID") + if err := cmd.MarkFlagRequired("src"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + if err := cmd.MarkFlagRequired("dst"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + if err := cmd.MarkFlagRequired("spi"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + if err := cmd.MarkFlagRequired("if_id"); err != nil { + log.Fatalf("Error marking flag as required: %v", err) + } + return cmd +} + +// NewEvpnIPSecCommand tests the inventory +func NewEvpnIPSecCommand() *cobra.Command { + cmd := &cobra.Command{ + Use: "evpnipsec", + Aliases: []string{"g"}, + Short: "Tests ipsec functionality", + Args: cobra.NoArgs, + Run: func(cmd *cobra.Command, _ []string) { + err := cmd.Help() + if err != nil { + log.Fatalf("[ERROR] %s", err.Error()) + } + }, + } + + cmd.AddCommand(AddSaCommand()) + cmd.AddCommand(DelSaCommand()) + return cmd +} diff --git a/evpnipsec/ipsec-evpn-test.go b/evpnipsec/ipsec-evpn-test.go new file mode 100644 index 0000000..001686d --- /dev/null +++ b/evpnipsec/ipsec-evpn-test.go @@ -0,0 +1,131 @@ +// SPDX-License-Identifier: Apache-2.0 +// Copyright (c) 2022-2023 Intel Corporation, or its subsidiaries. +// Copyright (c) 2022-2023 Dell Inc, or its subsidiaries. + +// Package evpnipsec implements the go library for OPI to be used to establish networking +package evpnipsec + +import ( + "context" + "log" + + //pb "github.com/opiproject/godpu/evpnipsec/gen/go" + pb "github.com/opiproject/opi-evpn-bridge/pkg/ipsec/gen/go" +) + +// AddSA adds a new SA +func (c IPSecEvpnClientImpl) AddSA(ctx context.Context, src string, dst string, spi uint32, proto int32, ifID uint32, reqid uint32, mode int32, intrface string, encAlg int32, encKey string, + intAlg int32, intKey string, replayWindow uint32, tfc uint32, encap int32, esn int32, copyDf int32, copyEcn int32, copyDscp int32, initiator int32, inbound int32, + update int32) (*pb.AddSAResp, error) { + conn, closer, err := c.NewConn() + if err != nil { + log.Printf("error creating connection: %s\n", err) + log.Println("THREE") + return nil, err + } + defer closer() + + client := c.getIPSecClient(conn) + data, err := client.AddSA(ctx, &pb.AddSAReq{ + SaId: &pb.SAIdentifier{ + Src: src, + Dst: dst, + Spi: spi, + Proto: pb.IPSecProtocol(proto), + IfId: ifID, + }, + + SaData: &pb.AddSAReqData{ + Reqid: reqid, + Mode: pb.IPSecMode(mode), + Interface: intrface, + EncAlg: pb.CryptoAlgorithm(encAlg), + EncKey: []byte(encKey), + IntAlg: pb.IntegAlgorithm(intAlg), + IntKey: []byte(intKey), + ReplayWindow: replayWindow, + Tfc: tfc, + Encap: pb.Bool(encap), + Esn: pb.Bool(esn), + CopyDf: pb.Bool(copyDf), + CopyEcn: pb.Bool(copyEcn), + CopyDscp: pb.DSCPCopy(copyDscp), + Initiator: pb.Bool(initiator), + Inbound: pb.Bool(inbound), + Update: pb.Bool(update), + }}) + if err != nil { + log.Printf("error creating logical bridge: %s\n", err) + log.Println("FOUR") + return nil, err + } + + return data, nil +} + +// CreateLogicalBridge creates an Logical Bridge an OPI server +/*func (c IPSecEvpnClientImpl) AddSA(ctx context.Context, sareq *pb.AddSAReq) (*pb.AddSAResp, error) { + + conn, closer, err := c.NewConn() + if err != nil { + log.Printf("error creating connection: %s\n", err) + return nil, err + } + defer closer() + + client := c.getIPSecClient(conn) + data, err := client.AddSA(ctx, sareq) + if err != nil { + log.Printf("error creating logical bridge: %s\n", err) + return nil, err + } + + return data, nil +}*/ + +// DelSA deletes an SA an OPI server +func (c IPSecEvpnClientImpl) DelSA(ctx context.Context, src string, dst string, spi uint32, proto int32, ifID uint32) (*pb.DeleteSAResp, error) { + conn, closer, err := c.NewConn() + if err != nil { + log.Printf("error creating connection: %s\n", err) + return nil, err + } + defer closer() + + client := c.getIPSecClient(conn) + data, err := client.DeleteSA(ctx, &pb.DeleteSAReq{ + SaId: &pb.SAIdentifier{ + Src: src, + Dst: dst, + Spi: spi, + Proto: pb.IPSecProtocol(proto), + IfId: ifID, + }}) + + if err != nil { + log.Printf("error creating logical bridge: %s\n", err) + return nil, err + } + + return data, nil +} + +// CreateLogicalBridge creates an Logical Bridge an OPI server +/*func (c IPSecEvpnClientImpl) DelSA(ctx context.Context, sareq *pb.DeleteSAReq) (*pb.DeleteSAResp, error) { + + conn, closer, err := c.NewConn() + if err != nil { + log.Printf("error creating connection: %s\n", err) + return nil, err + } + defer closer() + + client := c.getIPSecClient(conn) + data, err := client.DeleteSA(ctx, sareq) + if err != nil { + log.Printf("error creating logical bridge: %s\n", err) + return nil, err + } + + return data, nil +}*/ diff --git a/evpnipsec/ipsec-evpn.go b/evpnipsec/ipsec-evpn.go new file mode 100644 index 0000000..35293bf --- /dev/null +++ b/evpnipsec/ipsec-evpn.go @@ -0,0 +1,65 @@ +// SPDX-License-Identifier: Apache-2.0 +// Copyright (c) 2022-2023 Intel Corporation, or its subsidiaries. +// Copyright (c) 2022-2023 Dell Inc, or its subsidiaries. + +// Package evpnipsec implements the go library for OPI to be used to establish networking +package evpnipsec + +import ( + "context" + "errors" + + pb "github.com/opiproject/opi-evpn-bridge/pkg/ipsec/gen/go" + grpcOpi "github.com/opiproject/godpu/grpc" + "google.golang.org/grpc" +) + +// PbEvpnIPSecClientGetter defines the function type used to retrieve an evpn protobuf LogicalBridgeServiceClient +type PbEvpnIPSecClientGetter func(c grpc.ClientConnInterface) pb.IPUIPSecClient + +// IPSecEvpnClientImpl defines the function type used to retrieve an evpn protobuf LogicalBridgeServiceClient +type IPSecEvpnClientImpl struct { + // getIPSecClient PbEvpnIPSecClientGetter + getIPSecClient PbEvpnIPSecClientGetter + + grpcOpi.Connector +} + +// IPSecClient is an interface for querying evpn data from an OPI server +type IPSecClient interface { + + // AddSA interfaces + AddSA(ctx context.Context, src string, dst string, spi uint32, proto int32, ifID uint32, reqid uint32, mode int32, intrface string, encAlg int32, encKey string, + intAlg int32, intKey string, replayWindow uint32, tfc uint32, encap int32, esn int32, copyDf int32, copyEcn int32, copyDscp int32, initiator int32, inbound int32, + update int32) (*pb.AddSAResp, error) + DelSA(ctx context.Context, src string, dst string, spi uint32, proto int32, ifID uint32) (*pb.DeleteSAResp, error) +} + +// NewIPSecClient creates an evpn Logical Bridge client for use with OPI server at the given address +func NewIPSecClient(addr string, tls string) (IPSecClient, error) { + c, err := grpcOpi.New(addr, tls) + if err != nil { + return nil, err + } + + // Default is to use the OPI grpc client and the pb client generated from the protobuf spec + + return NewIPSecWithArgs(c, pb.NewIPUIPSecClient) +} + +// NewIPSecWithArgs creates an evpn Logical Bridge client for use with OPI server using the given gRPC client and the given function for +// retrieving an evpn protobuf client +func NewIPSecWithArgs(c grpcOpi.Connector, getter PbEvpnIPSecClientGetter) (IPSecClient, error) { + if c == nil { + return nil, errors.New("grpc connector is nil") + } + + if getter == nil { + return nil, errors.New("protobuf client getter is nil") + } + + return IPSecEvpnClientImpl{ + getIPSecClient: getter, + Connector: c, + }, nil +} diff --git a/go.mod b/go.mod index 3de127c..be4b263 100644 --- a/go.mod +++ b/go.mod @@ -11,34 +11,37 @@ require ( github.com/onsi/ginkgo/v2 v2.14.0 github.com/onsi/gomega v1.30.0 github.com/opiproject/opi-api v0.0.0-20240304222410-5dba226aaa9e + github.com/opiproject/opi-evpn-bridge v0.2.0 github.com/spf13/cobra v1.8.0 github.com/stretchr/testify v1.8.4 go.einride.tech/aip v0.66.0 - golang.org/x/net v0.20.0 + golang.org/x/net v0.21.0 golang.org/x/text v0.14.0 - google.golang.org/grpc v1.60.1 + google.golang.org/grpc v1.61.0 google.golang.org/protobuf v1.32.0 ) +replace github.com/opiproject/opi-evpn-bridge => github.com/mardim91/opi-evpn-bridge v0.0.0-20241209100717-35ff8ff45934 + require ( github.com/davecgh/go-spew v1.1.1 // indirect - github.com/go-logr/logr v1.3.0 // indirect + github.com/go-logr/logr v1.4.1 // indirect github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect github.com/golang/protobuf v1.5.3 // indirect github.com/google/go-cmp v0.6.0 // indirect github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38 // indirect - github.com/grpc-ecosystem/grpc-gateway/v2 v2.18.0 // indirect + github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 // indirect github.com/inconshreveable/mousetrap v1.1.0 // indirect github.com/kr/text v0.2.0 // indirect github.com/pmezard/go-difflib v1.0.0 // indirect github.com/rogpeppe/go-internal v1.9.0 // indirect github.com/spf13/pflag v1.0.5 // indirect - github.com/stretchr/objx v0.5.0 // indirect - golang.org/x/sync v0.5.0 // indirect - golang.org/x/sys v0.16.0 // indirect - golang.org/x/tools v0.16.1 // indirect - google.golang.org/genproto v0.0.0-20231012201019-e917dd12ba7a // indirect - google.golang.org/genproto/googleapis/api v0.0.0-20231002182017-d307bd883b97 // indirect - google.golang.org/genproto/googleapis/rpc v0.0.0-20231016165738-49dd2c1f3d0b // indirect + github.com/stretchr/objx v0.5.1 // indirect + golang.org/x/sync v0.6.0 // indirect + golang.org/x/sys v0.17.0 // indirect + golang.org/x/tools v0.17.0 // indirect + google.golang.org/genproto v0.0.0-20240108191215-35c7eff3a6b1 // indirect + google.golang.org/genproto/googleapis/api v0.0.0-20240108191215-35c7eff3a6b1 // indirect + google.golang.org/genproto/googleapis/rpc v0.0.0-20240108191215-35c7eff3a6b1 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index ebaa5fd..bde07a3 100644 --- a/go.sum +++ b/go.sum @@ -10,13 +10,12 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/go-chi/chi v1.5.5 h1:vOB/HbEMt9QqBqErz07QehcOKHaWFtuj87tTDVz2qXE= github.com/go-chi/chi v1.5.5/go.mod h1:C9JqLr3tIYjDOZpzn+BCuxY8z8vmca43EeMgyZt7irw= -github.com/go-logr/logr v1.3.0 h1:2y3SDp0ZXuc6/cjLSZ+Q3ir+QB9T/iG5yYRXqsagWSY= -github.com/go-logr/logr v1.3.0/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= +github.com/go-logr/logr v1.4.1 h1:pKouT5E8xu9zeFC39JXRDukb6JFQPXM5p5I91188VAQ= +github.com/go-logr/logr v1.4.1/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= github.com/go-ping/ping v1.1.0 h1:3MCGhVX4fyEUuhsfwPrsEdQw6xspHkv5zHsiSoDFZYw= github.com/go-ping/ping v1.1.0/go.mod h1:xIFjORFzTxqIV/tDVGO4eDy/bLuSyawEeojSm3GfRGk= github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 h1:tfuBGBXKqDEevZMzYi5KSi8KkcZtzBcTgAUUtapy0OI= github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572/go.mod h1:9Pwr4B2jHnOSGXyyzV8ROjYa2ojvAY6HCGYYfMoC3Ls= -github.com/golang/glog v1.1.2 h1:DVjP2PbBOzHyzA+dn3WhHIq4NdVu3Q+pvivFICf/7fo= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= @@ -28,8 +27,8 @@ github.com/google/pprof v0.0.0-20210407192527-94a9f03dee38/go.mod h1:kpwsk12EmLe github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.5.0 h1:1p67kYwdtXjb0gL0BPiP1Av9wiZPo5A8z2cWkTZ+eyU= github.com/google/uuid v1.5.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/grpc-ecosystem/grpc-gateway/v2 v2.18.0 h1:RtRsiaGvWxcwd8y3BiRZxsylPT8hLWZ5SPcfI+3IDNk= -github.com/grpc-ecosystem/grpc-gateway/v2 v2.18.0/go.mod h1:TzP6duP4Py2pHLVPPQp42aoYI92+PCrVotyR5e8Vqlk= +github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 h1:Wqo399gCIufwto+VfwCSvsnfGpF/w5E9CNxSwbpD6No= +github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0/go.mod h1:qmOFXW2epJhM0qSnUUYpldc7gVz2KMQwJ/QYCDIa7XU= github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= @@ -38,12 +37,12 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= github.com/lithammer/fuzzysearch v1.1.8/go.mod h1:IdqeyBClc3FFqSzYq/MXESsS4S0FsZ5ajtkr5xPLts4= +github.com/mardim91/opi-evpn-bridge v0.0.0-20241209100717-35ff8ff45934 h1:NfpPZshW29DcrW44DCgmShmK7WuOhV/7Bdd2mMA/xpI= +github.com/mardim91/opi-evpn-bridge v0.0.0-20241209100717-35ff8ff45934/go.mod h1:Lnt6BnGJEKm/poq+/blpIoyF8Fq8blkKc4kAcNCKlZ8= github.com/onsi/ginkgo/v2 v2.14.0 h1:vSmGj2Z5YPb9JwCWT6z6ihcUvDhuXLc3sJiqd3jMKAY= github.com/onsi/ginkgo/v2 v2.14.0/go.mod h1:JkUdW7JkN0V6rFvsHcJ478egV3XH9NxpD27Hal/PhZw= github.com/onsi/gomega v1.30.0 h1:hvMK7xYz4D3HapigLTeGdId/NcfQx1VHMJc60ew99+8= github.com/onsi/gomega v1.30.0/go.mod h1:9sxs+SwGrKI0+PWe4Fxa9tFQQBG5xSsSbMXOI8PPpoQ= -github.com/opiproject/opi-api v0.0.0-20240118183513-e44db269fba4 h1:YBjvYWQQAbNIGsAXvB6FwL9Encr1nzo3/w+bB/tXltM= -github.com/opiproject/opi-api v0.0.0-20240118183513-e44db269fba4/go.mod h1:92pv4ulvvPMuxCJ9ND3aYbmBfEMLx0VCjpkiR7ZTqPY= github.com/opiproject/opi-api v0.0.0-20240304222410-5dba226aaa9e h1:jUa7DmVLjzLKg051y7rYyCD0NAbEHZQMimM1451D74I= github.com/opiproject/opi-api v0.0.0-20240304222410-5dba226aaa9e/go.mod h1:92pv4ulvvPMuxCJ9ND3aYbmBfEMLx0VCjpkiR7ZTqPY= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= @@ -57,11 +56,13 @@ github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= -github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c= github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/objx v0.5.1 h1:4VhoImhV/Bm0ToFkXFi8hXNXwpDRZ/ynw3amt82mzq0= +github.com/stretchr/objx v0.5.1/go.mod h1:/iHQpkQwBD6DLUmQ4pE+s1TXdob1mORJ4/UFdrifcy0= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= github.com/stretchr/testify v1.8.4 h1:CcVxjf3Q8PM0mHUKJCdn+eZZtm5yQwehR5yeSVQQcUk= github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= @@ -76,14 +77,14 @@ golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= -golang.org/x/net v0.20.0 h1:aCL9BSgETF1k+blQaYUBx9hJ9LOGP3gAVemcZlf1Kpo= -golang.org/x/net v0.20.0/go.mod h1:z8BVo6PvndSri0LbOE3hAn0apkU+1YvI6E70E9jsnvY= +golang.org/x/net v0.21.0 h1:AQyQV4dYCvJ7vGmJyKki9+PBdyvhkSd8EIx/qb0AYv4= +golang.org/x/net v0.21.0/go.mod h1:bIjVDfnllIU7BJ2DNgfnXvpSvtn8VRwhlsaeUTyUS44= golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.5.0 h1:60k92dhOjHxJkrqnwsfl8KuaHbn/5dl0lUPUklKo3qE= -golang.org/x/sync v0.5.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= +golang.org/x/sync v0.6.0 h1:5BMeUDZ7vkXGfEr1x9B4bRcTH4lpkTkpdh0T/J+qjbQ= +golang.org/x/sync v0.6.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -92,8 +93,8 @@ golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBc golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.16.0 h1:xWw16ngr6ZMtmxDyKyIgsE93KNKz5HKmMa3b8ALHidU= -golang.org/x/sys v0.16.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/sys v0.17.0 h1:25cE3gD+tdBA7lp7QfhuV+rJiE9YXTcS3VG1SqssI/Y= +golang.org/x/sys v0.17.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= @@ -108,18 +109,18 @@ golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGm golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= -golang.org/x/tools v0.16.1 h1:TLyB3WofjdOEepBHAU20JdNC1Zbg87elYofWYAY5oZA= -golang.org/x/tools v0.16.1/go.mod h1:kYVVN6I1mBNoB1OX+noeBjbRk4IUEPa7JJ+TJMEooJ0= +golang.org/x/tools v0.17.0 h1:FvmRgNOcs3kOa+T20R1uhfP9F6HgG2mfxDv1vrx1Htc= +golang.org/x/tools v0.17.0/go.mod h1:xsh6VxdV005rRVaS6SSAf9oiAqljS7UZUacMZ8Bnsps= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= -google.golang.org/genproto v0.0.0-20231012201019-e917dd12ba7a h1:fwgW9j3vHirt4ObdHoYNwuO24BEZjSzbh+zPaNWoiY8= -google.golang.org/genproto v0.0.0-20231012201019-e917dd12ba7a/go.mod h1:EMfReVxb80Dq1hhioy0sOsY9jCE46YDgHlJ7fWVUWRE= -google.golang.org/genproto/googleapis/api v0.0.0-20231002182017-d307bd883b97 h1:W18sezcAYs+3tDZX4F80yctqa12jcP1PUS2gQu1zTPU= -google.golang.org/genproto/googleapis/api v0.0.0-20231002182017-d307bd883b97/go.mod h1:iargEX0SFPm3xcfMI0d1domjg0ZF4Aa0p2awqyxhvF0= -google.golang.org/genproto/googleapis/rpc v0.0.0-20231016165738-49dd2c1f3d0b h1:ZlWIi1wSK56/8hn4QcBp/j9M7Gt3U/3hZw3mC7vDICo= -google.golang.org/genproto/googleapis/rpc v0.0.0-20231016165738-49dd2c1f3d0b/go.mod h1:swOH3j0KzcDDgGUWr+SNpyTen5YrXjS3eyPzFYKc6lc= -google.golang.org/grpc v1.60.1 h1:26+wFr+cNqSGFcOXcabYC0lUVJVRa2Sb2ortSK7VrEU= -google.golang.org/grpc v1.60.1/go.mod h1:OlCHIeLYqSSsLi6i49B5QGdzaMZK9+M7LXN2FKz4eGM= +google.golang.org/genproto v0.0.0-20240108191215-35c7eff3a6b1 h1:/IWabOtPziuXTEtI1KYCpM6Ss7vaAkeMxk+uXV/xvZs= +google.golang.org/genproto v0.0.0-20240108191215-35c7eff3a6b1/go.mod h1:+Rvu7ElI+aLzyDQhpHMFMMltsD6m7nqpuWDd2CwJw3k= +google.golang.org/genproto/googleapis/api v0.0.0-20240108191215-35c7eff3a6b1 h1:OPXtXn7fNMaXwO3JvOmF1QyTc00jsSFFz1vXXBOdCDo= +google.golang.org/genproto/googleapis/api v0.0.0-20240108191215-35c7eff3a6b1/go.mod h1:B5xPO//w8qmBDjGReYLpR6UJPnkldGkCSMoH/2vxJeg= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240108191215-35c7eff3a6b1 h1:gphdwh0npgs8elJ4T6J+DQJHPVF7RsuJHCfwztUb4J4= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240108191215-35c7eff3a6b1/go.mod h1:daQN87bsDqDoe316QbbvX60nMoJQa4r6Ds0ZuoAe5yA= +google.golang.org/grpc v1.61.0 h1:TOvOcuXn30kRao+gfcvsebNEa5iZIiLkisYEkf7R7o0= +google.golang.org/grpc v1.61.0/go.mod h1:VUbo7IFqmF1QtCAstipjG0GIoq49KvMe9+h1jFLBNJs= google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.32.0 h1:pPC6BG5ex8PDFnkbrGU3EixyhKcQ2aDuBS36lqK/C7I=