diff --git a/.golangci.yml b/.golangci.yml index 253c4ff23..fa00a9c57 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -74,7 +74,7 @@ linters: disable-all: true enable: - bodyclose - - depguard +# - depguard - dogsled # TODO fix issues- dupl # TODO fix issues- errcheck diff --git a/api/v1/zz_generated.deepcopy.go b/api/v1/zz_generated.deepcopy.go index 9ffb982b8..fc9477593 100644 --- a/api/v1/zz_generated.deepcopy.go +++ b/api/v1/zz_generated.deepcopy.go @@ -1,5 +1,4 @@ //go:build !ignore_autogenerated -// +build !ignore_autogenerated /* Copyright 2021. diff --git a/config/crd/bases/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml b/config/crd/bases/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml index 2cb2ece31..da3465127 100644 --- a/config/crd/bases/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml +++ b/config/crd/bases/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml @@ -83,11 +83,13 @@ spec: items: type: string type: array + x-kubernetes-list-type: atomic required: - key - operator type: object type: array + x-kubernetes-list-type: atomic matchLabels: additionalProperties: type: string diff --git a/controllers/sriovnetworknodepolicy_controller.go b/controllers/sriovnetworknodepolicy_controller.go index a275beb5d..1115b1bd3 100644 --- a/controllers/sriovnetworknodepolicy_controller.go +++ b/controllers/sriovnetworknodepolicy_controller.go @@ -196,7 +196,7 @@ func (r *SriovNetworkNodePolicyReconciler) SetupWithManager(mgr ctrl.Manager) er For(&sriovnetworkv1.SriovNetworkNodePolicy{}). Watches(&corev1.Node{}, nodeEvenHandler). Watches(&sriovnetworkv1.SriovNetworkNodePolicy{}, delayedEventHandler). - WatchesRawSource(&source.Channel{Source: eventChan}, delayedEventHandler). + WatchesRawSource(source.Channel(eventChan, &handler.EnqueueRequestForObject{})). Complete(r) } diff --git a/controllers/suite_test.go b/controllers/suite_test.go index b830e2fc2..83326a71f 100644 --- a/controllers/suite_test.go +++ b/controllers/suite_test.go @@ -37,6 +37,7 @@ import ( logf "sigs.k8s.io/controller-runtime/pkg/log" "sigs.k8s.io/controller-runtime/pkg/log/zap" "sigs.k8s.io/controller-runtime/pkg/manager" + "sigs.k8s.io/controller-runtime/pkg/metrics/server" netattdefv1 "github.com/k8snetworkplumbingwg/network-attachment-definition-client/pkg/apis/k8s.cni.cncf.io/v1" openshiftconfigv1 "github.com/openshift/api/config/v1" @@ -62,7 +63,8 @@ const testNamespace = "openshift-sriov-network-operator" func setupK8sManagerForTest() (manager.Manager, error) { k8sManager, err := ctrl.NewManager(cfg, ctrl.Options{ - Scheme: scheme.Scheme, + Scheme: scheme.Scheme, + Metrics: server.Options{BindAddress: "0"}, // we don't need metrics server for tests }) if err != nil { diff --git a/deployment/sriov-network-operator-chart/crds/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml b/deployment/sriov-network-operator-chart/crds/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml index 2cb2ece31..da3465127 100644 --- a/deployment/sriov-network-operator-chart/crds/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml +++ b/deployment/sriov-network-operator-chart/crds/sriovnetwork.openshift.io_sriovnetworkpoolconfigs.yaml @@ -83,11 +83,13 @@ spec: items: type: string type: array + x-kubernetes-list-type: atomic required: - key - operator type: object type: array + x-kubernetes-list-type: atomic matchLabels: additionalProperties: type: string diff --git a/go.mod b/go.mod index 2dc49afe9..0c1e6b615 100644 --- a/go.mod +++ b/go.mod @@ -1,6 +1,8 @@ module github.com/k8snetworkplumbingwg/sriov-network-operator -go 1.22 +go 1.22.0 + +toolchain go1.22.4 require ( github.com/Masterminds/sprig/v3 v3.2.2 @@ -8,39 +10,39 @@ require ( github.com/cenkalti/backoff v2.2.1+incompatible github.com/coreos/go-systemd/v22 v22.5.0 github.com/fsnotify/fsnotify v1.7.0 - github.com/go-logr/logr v1.2.4 + github.com/go-logr/logr v1.4.1 github.com/golang/mock v1.4.4 github.com/google/go-cmp v0.6.0 github.com/hashicorp/go-retryablehttp v0.7.7 - github.com/jaypipes/ghw v0.9.0 + github.com/jaypipes/ghw v0.12.0 github.com/jaypipes/pcidb v1.0.0 - github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.4.0 + github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.6.0 github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3 - github.com/k8snetworkplumbingwg/sriovnet v1.2.0 - github.com/onsi/ginkgo/v2 v2.11.0 - github.com/onsi/gomega v1.27.10 + github.com/k8snetworkplumbingwg/sriovnet v1.2.1-0.20240128120937-3ca5e43034e6 + github.com/onsi/ginkgo/v2 v2.17.1 + github.com/onsi/gomega v1.32.0 github.com/openshift-kni/k8sreporter v1.0.4 - github.com/openshift/api v0.0.0-20230807132801-600991d550ac + github.com/openshift/api v0.0.0-20240508125607-95e22923d553 github.com/openshift/client-go v0.0.0-20230607134213-3cd0021bbee3 github.com/openshift/machine-config-operator v0.0.1-0.20231024085435-7e1fb719c1ba github.com/pkg/errors v0.9.1 github.com/safchain/ethtool v0.3.0 github.com/spf13/cobra v1.7.0 - github.com/stretchr/testify v1.8.4 + github.com/stretchr/testify v1.9.0 github.com/vishvananda/netlink v1.2.1-beta.2.0.20240221172127-ec7bcb248e94 github.com/vishvananda/netns v0.0.4 - go.uber.org/zap v1.25.0 + go.uber.org/zap v1.26.0 golang.org/x/time v0.3.0 gopkg.in/yaml.v3 v3.0.1 - k8s.io/api v0.28.3 - k8s.io/apiextensions-apiserver v0.28.3 - k8s.io/apimachinery v0.28.3 - k8s.io/client-go v0.28.3 - k8s.io/code-generator v0.28.3 - k8s.io/klog/v2 v2.100.1 - k8s.io/kubectl v0.28.3 + k8s.io/api v0.30.2 + k8s.io/apiextensions-apiserver v0.30.2 + k8s.io/apimachinery v0.30.2 + k8s.io/client-go v0.30.2 + k8s.io/code-generator v0.30.2 + k8s.io/klog/v2 v2.120.1 + k8s.io/kubectl v0.30.2 k8s.io/utils v0.0.0-20230726121419-3b25d923346b - sigs.k8s.io/controller-runtime v0.16.3 + sigs.k8s.io/controller-runtime v0.18.4 ) require ( @@ -48,7 +50,7 @@ require ( github.com/MakeNowJust/heredoc v1.0.0 // indirect github.com/Masterminds/goutils v1.1.1 // indirect github.com/Masterminds/semver/v3 v3.1.1 // indirect - github.com/Mellanox/sriovnet v1.0.3 // indirect + github.com/Mellanox/rdmamap v1.1.0 // indirect github.com/StackExchange/wmi v1.2.1 // indirect github.com/ajeddeloh/go-json v0.0.0-20200220154158-5ae607161559 // indirect github.com/aws/aws-sdk-go v1.44.204 // indirect @@ -68,27 +70,28 @@ require ( github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect github.com/emicklei/go-restful/v3 v3.11.0 // indirect github.com/evanphx/json-patch v5.6.0+incompatible // indirect - github.com/evanphx/json-patch/v5 v5.7.0 // indirect + github.com/evanphx/json-patch/v5 v5.9.0 // indirect github.com/exponent-io/jsonpath v0.0.0-20151013193312-d6023ce2651d // indirect github.com/frankban/quicktest v1.14.4 // indirect github.com/ghodss/yaml v1.0.1-0.20190212211648-25d852aebe32 // indirect github.com/go-errors/errors v1.4.2 // indirect - github.com/go-logr/zapr v1.2.4 // indirect + github.com/go-logr/zapr v1.3.0 // indirect github.com/go-ole/go-ole v1.2.6 // indirect github.com/go-openapi/jsonpointer v0.20.0 // indirect github.com/go-openapi/jsonreference v0.20.2 // indirect github.com/go-openapi/swag v0.22.4 // indirect github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect github.com/gogo/protobuf v1.3.2 // indirect - github.com/golang/glog v1.1.0 // indirect + github.com/golang/glog v1.2.1 // indirect github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect - github.com/golang/protobuf v1.5.3 // indirect + github.com/golang/protobuf v1.5.4 // indirect github.com/google/btree v1.0.1 // indirect github.com/google/gnostic-models v0.6.8 // indirect github.com/google/gofuzz v1.2.0 // indirect github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1 // indirect github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect - github.com/google/uuid v1.3.1 // indirect + github.com/google/uuid v1.6.0 // indirect + github.com/gorilla/websocket v1.5.0 // indirect github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79 // indirect github.com/hashicorp/go-cleanhttp v0.5.2 // indirect github.com/huandu/xstrings v1.3.2 // indirect @@ -110,6 +113,7 @@ require ( github.com/modern-go/reflect2 v1.0.2 // indirect github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00 // indirect github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect + github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f // indirect github.com/openshift/library-go v0.0.0-20231020125025-211b32f1a1f2 // indirect github.com/peterbourgon/diskv v2.0.1+incompatible // indirect github.com/pmezard/go-difflib v1.0.0 // indirect @@ -120,10 +124,10 @@ require ( github.com/robfig/cron v1.2.0 // indirect github.com/russross/blackfriday/v2 v2.1.0 // indirect github.com/shopspring/decimal v1.2.0 // indirect - github.com/spf13/afero v1.9.4 // indirect + github.com/spf13/afero v1.9.5 // indirect github.com/spf13/cast v1.5.0 // indirect github.com/spf13/pflag v1.0.6-0.20210604193023-d5e0c0615ace // indirect - github.com/stretchr/objx v0.5.0 // indirect + github.com/stretchr/objx v0.5.2 // indirect github.com/vincent-petithory/dataurl v1.0.0 // indirect github.com/xlab/treeprint v1.2.0 // indirect go.starlark.net v0.0.0-20230525235612-a134d8f9ddca // indirect @@ -131,35 +135,37 @@ require ( go4.org v0.0.0-20200104003542-c7e774b10ea0 // indirect golang.org/x/crypto v0.21.0 // indirect golang.org/x/exp v0.0.0-20231006140011-7918f672742d // indirect - golang.org/x/mod v0.13.0 // indirect + golang.org/x/mod v0.15.0 // indirect golang.org/x/net v0.23.0 // indirect - golang.org/x/oauth2 v0.13.0 // indirect - golang.org/x/sync v0.4.0 // indirect + golang.org/x/oauth2 v0.17.0 // indirect + golang.org/x/sync v0.6.0 // indirect golang.org/x/sys v0.20.0 // indirect golang.org/x/term v0.18.0 // indirect golang.org/x/text v0.14.0 // indirect - golang.org/x/tools v0.14.0 // indirect + golang.org/x/tools v0.18.0 // indirect gomodules.xyz/jsonpatch/v2 v2.4.0 // indirect google.golang.org/appengine v1.6.8 // indirect - google.golang.org/genproto/googleapis/rpc v0.0.0-20230525234030-28d5490b6b19 // indirect - google.golang.org/grpc v1.56.3 // indirect + google.golang.org/genproto/googleapis/rpc v0.0.0-20240227224415-6ceb2ff114de // indirect + google.golang.org/grpc v1.63.0 // indirect google.golang.org/protobuf v1.33.0 // indirect gopkg.in/inf.v0 v0.9.1 // indirect gopkg.in/yaml.v2 v2.4.0 // indirect howett.net/plist v1.0.0 // indirect - k8s.io/apiserver v0.28.3 // indirect - k8s.io/cli-runtime v0.28.3 // indirect - k8s.io/component-base v0.28.3 // indirect - k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01 // indirect + k8s.io/apiserver v0.30.2 // indirect + k8s.io/cli-runtime v0.30.2 // indirect + k8s.io/component-base v0.30.2 // indirect + k8s.io/gengo/v2 v2.0.0-20240228010128-51d4e06bde70 // indirect k8s.io/kube-aggregator v0.27.4 // indirect - k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 // indirect - k8s.io/kubelet v0.27.7 // indirect + k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 // indirect + k8s.io/kubelet v0.29.3 // indirect sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect sigs.k8s.io/kube-storage-version-migrator v0.0.6-0.20230721195810-5c8923c5ff96 // indirect sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3 // indirect sigs.k8s.io/kustomize/kyaml v0.14.3-0.20230601165947-6ce0bf390ce3 // indirect - sigs.k8s.io/structured-merge-diff/v4 v4.3.0 // indirect + sigs.k8s.io/structured-merge-diff/v4 v4.4.1 // indirect sigs.k8s.io/yaml v1.4.0 // indirect ) replace github.com/emicklei/go-restful => github.com/emicklei/go-restful v2.16.0+incompatible + +replace github.com/k8snetworkplumbingwg/sriov-network-device-plugin => github.com/SchSeba/sriov-network-device-plugin v0.0.0-20240626111634-92cb855fbc29 diff --git a/go.sum b/go.sum index 32cba669d..22fe65892 100644 --- a/go.sum +++ b/go.sum @@ -48,8 +48,10 @@ github.com/Masterminds/semver/v3 v3.1.1 h1:hLg3sBzpNErnxhQtUy/mmLR2I9foDujNK030I github.com/Masterminds/semver/v3 v3.1.1/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs= github.com/Masterminds/sprig/v3 v3.2.2 h1:17jRggJu518dr3QaafizSXOjKYp94wKfABxUmyxvxX8= github.com/Masterminds/sprig/v3 v3.2.2/go.mod h1:UoaO7Yp8KlPnJIYWTFkMaqPUYKTfGFPhxNuwnnxkKlk= -github.com/Mellanox/sriovnet v1.0.3 h1:Nlmxr2mkp16aIP4CJcsnqCczxQQgOuzNDm/nu9qTBZ8= -github.com/Mellanox/sriovnet v1.0.3/go.mod h1:8TlYc3iOTEvUM+WAbC7MU6U6JXqIfhl2DcWIGVUsjIE= +github.com/Mellanox/rdmamap v1.1.0 h1:A/W1wAXw+6vm58f3VklrIylgV+eDJlPVIMaIKuxgUT4= +github.com/Mellanox/rdmamap v1.1.0/go.mod h1:fN+/V9lf10ABnDCwTaXRjeeWijLt2iVLETnK+sx/LY8= +github.com/SchSeba/sriov-network-device-plugin v0.0.0-20240626111634-92cb855fbc29 h1:2CPV3pCDxkPZQokS3HXEuQpY3gVutSZrJWszjm/31BY= +github.com/SchSeba/sriov-network-device-plugin v0.0.0-20240626111634-92cb855fbc29/go.mod h1:mvTdwqCgsw/xGQ49ojmu61tIBLqrvOT/R0+GOCXAvys= github.com/StackExchange/wmi v1.2.1 h1:VIkavFPXSjcnS+O8yTq7NI32k0R5Aj+v39y29VYDOSA= github.com/StackExchange/wmi v1.2.1/go.mod h1:rcmrprowKIVzvc+NUiLncP2uuArMWLCbu9SBzvHz7e8= github.com/ajeddeloh/go-json v0.0.0-20170920214419-6a2fe990e083/go.mod h1:otnto4/Icqn88WCcM4bhIJNSgsh9VLBuspyyCfvof9c= @@ -62,9 +64,6 @@ github.com/ashcrow/osrelease v0.0.0-20180626175927-9b292693c55c/go.mod h1:BRljTy github.com/aws/aws-sdk-go v1.19.11/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= github.com/aws/aws-sdk-go v1.44.204 h1:7/tPUXfNOHB390A63t6fJIwmlwVQAkAwcbzKsU2/6OQ= github.com/aws/aws-sdk-go v1.44.204/go.mod h1:aVsgQcEevwlmQ7qHE9I3h+dtQgpqhFB+i8Phjh7fkwI= -github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= -github.com/benbjohnson/clock v1.3.0 h1:ip6w0uFQkncKQ979AypyG0ER7mqUSBdKLOgAle/AT8A= -github.com/benbjohnson/clock v1.3.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= github.com/blang/semver v3.5.1+incompatible h1:cQNTCjp13qL8KC3Nbxr/y2Bqb63oX6wdnnjpJbkM4JQ= @@ -129,8 +128,8 @@ github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.m github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= github.com/evanphx/json-patch v5.6.0+incompatible h1:jBYDEEiFBPxA0v50tFdvOzQQTCvpL6mnFh5mB2/l16U= github.com/evanphx/json-patch v5.6.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk= -github.com/evanphx/json-patch/v5 v5.7.0 h1:nJqP7uwL84RJInrohHfW0Fx3awjbm8qZeFv0nW9SYGc= -github.com/evanphx/json-patch/v5 v5.7.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ= +github.com/evanphx/json-patch/v5 v5.9.0 h1:kcBlZQbplgElYIlo/n1hJbls2z/1awpXxpRi0/FOJfg= +github.com/evanphx/json-patch/v5 v5.9.0/go.mod h1:VNkHZ/282BpEyt/tObQO8s5CMPmYYq14uClGH4abBuQ= github.com/exponent-io/jsonpath v0.0.0-20151013193312-d6023ce2651d h1:105gxyaGwCFad8crR9dcMQWvV9Hvulu6hwUh4tWPJnM= github.com/exponent-io/jsonpath v0.0.0-20151013193312-d6023ce2651d/go.mod h1:ZZMPRZwes7CROmyNKgQzC3XPs6L/G2EJLHddWejkmf4= github.com/fatih/color v1.16.0 h1:zmkK9Ngbjj+K0yRhTVONQh1p/HknKYSlNT+vZCzyokM= @@ -139,7 +138,6 @@ github.com/frankban/quicktest v1.14.4 h1:g2rn0vABPOOXmZUj+vbmUp0lPoXEMuhTpIluN0X github.com/frankban/quicktest v1.14.4/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA= github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM= -github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= github.com/ghodss/yaml v1.0.1-0.20190212211648-25d852aebe32 h1:Mn26/9ZMNWSw9C9ERFA1PUxfmGpolnw2v0bKOREu5ew= github.com/ghodss/yaml v1.0.1-0.20190212211648-25d852aebe32/go.mod h1:GIjDIg/heH5DOkXY3YJ/wNhfHsQHoXGjl8G8amsYQ1I= github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA= @@ -147,12 +145,10 @@ github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3Bop github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= -github.com/go-logr/logr v0.2.0/go.mod h1:z6/tIYblkpsD+a4lm/fGIIU9mZ+XfAiaFtq7xTgseGU= -github.com/go-logr/logr v1.2.0/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= -github.com/go-logr/logr v1.2.4 h1:g01GSCwiDw2xSZfjJ2/T9M+S6pFdcNtFYsp+Y43HYDQ= -github.com/go-logr/logr v1.2.4/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= -github.com/go-logr/zapr v1.2.4 h1:QHVo+6stLbfJmYGkQ7uGHUCu5hnAFAj6mDe6Ea0SeOo= -github.com/go-logr/zapr v1.2.4/go.mod h1:FyHWQIzQORZ0QVE1BtVHv3cKtNLuXsbNLtpuhNapBOA= +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-logr/zapr v1.3.0 h1:XGdV8XW8zdwFiwOA2Dryh1gj2KRQyOOoNmBy4EplIcQ= +github.com/go-logr/zapr v1.3.0/go.mod h1:YKepepNBd1u/oyhd/yQmtjVXmm9uML4IXUgMOwR8/Gg= github.com/go-ole/go-ole v1.2.5/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= @@ -171,8 +167,8 @@ github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5x github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= -github.com/golang/glog v1.1.0 h1:/d3pCKDPWNnvIWe0vVUpNP32qc8U3PDVxySP/y360qE= -github.com/golang/glog v1.1.0/go.mod h1:pfYeQZ3JWZoXTV5sFc986z3HTpwQs9At6P4ImfuP3NQ= +github.com/golang/glog v1.2.1 h1:OptwRhECazUx5ix5TTWC3EZhsZEHWcYWY4FQHTIubm4= +github.com/golang/glog v1.2.1/go.mod h1:6AhwSGph0fcJtXVM/PEHPqZlFeoLxhs7/t5UDAwmO+w= github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= @@ -202,8 +198,8 @@ github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= -github.com/golang/protobuf v1.5.3 h1:KhyjKVUg7Usr/dYsdSqoFveMYd5ko72D+zANwlG1mmg= -github.com/golang/protobuf v1.5.3/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= +github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek= +github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps= github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/btree v1.0.1 h1:gK4Kx5IaGY9CD5sPJ36FHiBJ6ZXl0kilRiiCj+jdYp4= @@ -224,7 +220,6 @@ github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeN github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= -github.com/google/gofuzz v1.1.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0= github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= @@ -247,13 +242,15 @@ github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 h1:El6M4kTTCOh6aBiKaU github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510/go.mod h1:pupxD2MaaD3pAXIBCelhxNneeOaAeabZDe5s4K6zSpQ= github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= -github.com/google/uuid v1.3.1 h1:KjJaJ9iWZ3jOFZIf1Lqf4laDRCasjl0BCmnEGxkdLb4= -github.com/google/uuid v1.3.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= +github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g= github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= +github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc= +github.com/gorilla/websocket v1.5.0/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79 h1:+ngKgrYPPJrOjhax5N+uePQ0Fh1Z7PheYoUI/0nzkPA= github.com/gregjones/httpcache v0.0.0-20190611155906-901d90724c79/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA= github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ= @@ -272,11 +269,10 @@ github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1: github.com/imdario/mergo v0.3.11/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA= github.com/imdario/mergo v0.3.16 h1:wwQJbIsHYGMUyLSPrEq1CT16AhnhNJQ51+4fdHUnCl4= github.com/imdario/mergo v0.3.16/go.mod h1:WBLT9ZmE3lPoWsEzCh9LPo3TiwVN+ZKEjmz+hD27ysY= -github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= -github.com/jaypipes/ghw v0.9.0 h1:TWF4wNIGtZcgDJaiNcFgby5BR8s2ixcUe0ydxNO2McY= -github.com/jaypipes/ghw v0.9.0/go.mod h1:dXMo19735vXOjpIBDyDYSp31sB2u4hrtRCMxInqQ64k= +github.com/jaypipes/ghw v0.12.0 h1:xU2/MDJfWmBhJnujHY9qwXQLs3DBsf0/Xa9vECY0Tho= +github.com/jaypipes/ghw v0.12.0/go.mod h1:jeJGbkRB2lL3/gxYzNYzEDETV1ZJ56OKr+CSeSEym+g= github.com/jaypipes/pcidb v1.0.0 h1:vtZIfkiCUE42oYbJS0TAq9XSfSmcsgo9IdxSm9qzYU8= github.com/jaypipes/pcidb v1.0.0/go.mod h1:TnYUvqhPBzCKnH34KrIX22kAeEbDCSRJ9cqLRCuNDfk= github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI= @@ -292,17 +288,14 @@ github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/X github.com/jtolds/gls v4.2.1+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= github.com/k8snetworkplumbingwg/govdpa v0.1.4 h1:e6mM7JFZkLVJeMQw3px96EigHAhnb4VUlqhNub/2Psk= github.com/k8snetworkplumbingwg/govdpa v0.1.4/go.mod h1:UQR1xu7A+nnRK1dkLEi12OnNL0OiBPpIKOYDuaQQkck= -github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.4.0 h1:VzM3TYHDgqPkettiP6I6q2jOeQFL4nrJM+UcAc4f6Fs= -github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.4.0/go.mod h1:nqCI7aelBJU61wiBeeZWJ6oi4bJy5nrjkM6lWIMA4j0= -github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3 h1:hIHzF4vNTCFb9UMngrcJAMvdNslCekaSvNbfQt21CUw= -github.com/k8snetworkplumbingwg/sriov-network-device-plugin v0.0.0-20221127172732-a5a7395122e3/go.mod h1:b0YSmUuNOy6CkEmV27XfmZ3a7njs2pjxHoFAvfLbUII= -github.com/k8snetworkplumbingwg/sriovnet v1.2.0 h1:6ELfAxCB1dvosGUy3DVRmfH+HWTzmPD3W67HKQvMR1M= -github.com/k8snetworkplumbingwg/sriovnet v1.2.0/go.mod h1:jyWzGe6ZtYiPq6ih6aXCOy6mZ49Y9mNyBOLBBXnli+k= +github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.6.0 h1:BT3ghAY0q7lWib9rz+tVXDFkm27dJV6SLCn7TunZwo4= +github.com/k8snetworkplumbingwg/network-attachment-definition-client v1.6.0/go.mod h1:wxt2YWRVItDtaQmVSmaN5ubE2L1c9CiNoHQwSJnM8Ko= +github.com/k8snetworkplumbingwg/sriovnet v1.2.1-0.20240128120937-3ca5e43034e6 h1:Ho6fhRwqgow7ytMgQ1/55j1gG5AbkxIYRbAhJnXr/MM= +github.com/k8snetworkplumbingwg/sriovnet v1.2.1-0.20240128120937-3ca5e43034e6/go.mod h1:LuzcqxxXdSgopWe1yo2kQFSgFTz9Ec5qLu6bb0s5Ut4= github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= @@ -323,7 +316,6 @@ github.com/matttproud/golang_protobuf_extensions/v2 v2.0.0/go.mod h1:QUyp042oQth github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw= github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= -github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= github.com/mitchellh/go-homedir v1.1.0 h1:lukF9ziXFxDFPkA1vsr5zpc1XuPDn/wFntq5mG+4E0Y= github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= github.com/mitchellh/go-wordwrap v1.0.1 h1:TLuKupo69TCn6TQSyGxwI1EblZZEsQ0vMlAFQflz0v0= @@ -344,18 +336,20 @@ github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00 h1:n6/ github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00/go.mod h1:Pm3mSP3c5uWn86xMLZ5Sa7JB9GsEZySvHYXCTK4E9q4= github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA= github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ= +github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f h1:y5//uYreIhSUg3J1GEMiLbxo1LJaP8RfCpH6pymGZus= +github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw= github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= -github.com/onsi/ginkgo/v2 v2.11.0 h1:WgqUCUt/lT6yXoQ8Wef0fsNn5cAuMK7+KT9UFRz2tcU= -github.com/onsi/ginkgo/v2 v2.11.0/go.mod h1:ZhrRA5XmEE3x3rhlzamx/JJvujdZoJ2uvgI7kR0iZvM= -github.com/onsi/gomega v1.27.10 h1:naR28SdDFlqrG6kScpT8VWpu1xWY5nJRCF3XaYyBjhI= -github.com/onsi/gomega v1.27.10/go.mod h1:RsS8tutOdbdgzbPtzzATp12yT7kM5I5aElG3evPbQ0M= +github.com/onsi/ginkgo/v2 v2.17.1 h1:V++EzdbhI4ZV4ev0UTIj0PzhzOcReJFyJaLjtSF55M8= +github.com/onsi/ginkgo/v2 v2.17.1/go.mod h1:llBI3WDLL9Z6taip6f33H76YcWtJv+7R3HigUjbIBOs= +github.com/onsi/gomega v1.32.0 h1:JRYU78fJ1LPxlckP6Txi/EYqJvjtMrDC04/MM5XRHPk= +github.com/onsi/gomega v1.32.0/go.mod h1:a4x4gW6Pz2yK1MAmvluYme5lvYTn61afQ2ETw/8n4Lg= github.com/openshift-kni/k8sreporter v1.0.4 h1:jEwX6Pqei60kO1U0JLo+ePjQaP7DNn/M6d63KCS2tS0= github.com/openshift-kni/k8sreporter v1.0.4/go.mod h1:fg8HI9yxiKAi6UzR6NTtrmQmA2WKzUqmkRUHwQ1+Bj8= -github.com/openshift/api v0.0.0-20230807132801-600991d550ac h1:HqT8MmYGXiUGUW0BjygTGOOvqO2wIsTaG3q8nboJyPY= -github.com/openshift/api v0.0.0-20230807132801-600991d550ac/go.mod h1:yimSGmjsI+XF1mr+AKBs2//fSXIOhhetHGbMlBEfXbs= +github.com/openshift/api v0.0.0-20240508125607-95e22923d553 h1:xMTwK9/Co/HyzXkNQLDcHRBgrTdL52/bgNHyDS+YY94= +github.com/openshift/api v0.0.0-20240508125607-95e22923d553/go.mod h1:CxgbWAlvu2iQB0UmKTtRu1YfepRg1/vJ64n2DlIEVz4= github.com/openshift/client-go v0.0.0-20230607134213-3cd0021bbee3 h1:uVCq/Sx2y4UZh+qCsCL1BBUJpc3DULHkN4j7XHHgHtw= github.com/openshift/client-go v0.0.0-20230607134213-3cd0021bbee3/go.mod h1:M+VUIcqx5IvgzejcbgmQnxETPrXRYlcufHpw2bAgz9Y= github.com/openshift/library-go v0.0.0-20231020125025-211b32f1a1f2 h1:TWG/YVRhSvjYq8iIwJ2Wpoopgg0zuh+ZAl1RSm4J8Z0= @@ -366,10 +360,8 @@ github.com/peterbourgon/diskv v2.0.1+incompatible h1:UBdAOUP5p4RWqPBg048CAvpKN+v github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU= github.com/pin/tftp v2.1.0+incompatible/go.mod h1:xVpZOMCXTy+A5QMjEVN0Glwa1sUvaJhFXbr/aAxuxGY= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= -github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= -github.com/pkg/sftp v1.10.1/go.mod h1:lYOWFsE0bwd1+KfKJaKeuokY15vzFx25BLbzYYoAxZI= github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= @@ -398,23 +390,21 @@ github.com/shopspring/decimal v1.2.0 h1:abSATXmQEYyShuxI4/vyW3tV1MrKAJzCZ/0zLUXY github.com/shopspring/decimal v1.2.0/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o= github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= github.com/smartystreets/goconvey v0.0.0-20190222223459-a17d461953aa/go.mod h1:2RVY1rIf+2J2o/IM9+vPq9RzmHDSseB7FoXiSNIUsoU= -github.com/spf13/afero v1.4.1/go.mod h1:Ai8FlHk4v/PARR026UzYexafAt9roJ7LcLMAmO6Z93I= -github.com/spf13/afero v1.9.4 h1:Sd43wM1IWz/s1aVXdOBkjJvuP8UdyqioeE4AmM0QsBs= -github.com/spf13/afero v1.9.4/go.mod h1:iUV7ddyEEZPO5gA3zD4fJt6iStLlL+Lg4m2cihcDf8Y= +github.com/spf13/afero v1.9.5 h1:stMpOSZFs//0Lv29HduCmli3GUfpFoF3Y1Q/aXj/wVM= +github.com/spf13/afero v1.9.5/go.mod h1:UBogFpq8E9Hx+xc5CNTTEpTnuHVmXDwZcZcE1eb/UhQ= github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= github.com/spf13/cast v1.5.0 h1:rj3WzYc11XZaIZMPKmwP96zkFEnnAmV8s6XbB2aY32w= github.com/spf13/cast v1.5.0/go.mod h1:SpXXQ5YoyJw6s3/6cMTQuxvgRl3PCJiyaX9p6b155UU= -github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= github.com/spf13/cobra v1.7.0 h1:hyqWnYt1ZQShIddO5kBpj3vu05/++x6tJ6dg8EC572I= github.com/spf13/cobra v1.7.0/go.mod h1:uLxZILRyS/50WlhOIKD7W6V5bgeIt+4sICxh6uRMrb0= -github.com/spf13/pflag v1.0.2/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/pflag v1.0.6-0.20210604193023-d5e0c0615ace h1:9PNP1jnUjRhfmGMlkXHjYPishpcw4jpSt/V/xYY3FMA= github.com/spf13/pflag v1.0.6-0.20210604193023-d5e0c0615ace/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.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY= +github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= @@ -424,14 +414,15 @@ github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/ 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.1/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/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= +github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= github.com/vincent-petithory/dataurl v0.0.0-20160330182126-9a301d65acbb/go.mod h1:FHafX5vmDzyP+1CQATJn7WFKc9CvnvxyvZy6I1MrG/U= github.com/vincent-petithory/dataurl v1.0.0 h1:cXw+kPto8NLuJtlMsI152irrVw9fRDX8AbShPRpg2CI= github.com/vincent-petithory/dataurl v1.0.0/go.mod h1:FHafX5vmDzyP+1CQATJn7WFKc9CvnvxyvZy6I1MrG/U= -github.com/vishvananda/netlink v1.1.1-0.20210518155637-4cb3795f2ccb/go.mod h1:twkDnbuQxJYemMlGd4JFIcuhgX83tXhKS2B/PRMpOho= +github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= github.com/vishvananda/netlink v1.2.1-beta.2.0.20240221172127-ec7bcb248e94 h1:EO+D8nUD4bAhHHcJUSvH+8puZrDrcjc50tePSevGQrw= github.com/vishvananda/netlink v1.2.1-beta.2.0.20240221172127-ec7bcb248e94/go.mod h1:whJevzBpTrid75eZy99s3DqCmy05NfibNaF2Ol5Ox5A= +github.com/vishvananda/netns v0.0.0-20191106174202-0a2b9b5464df/go.mod h1:JP3t17pCcGlemwknint6hfoeCVQrEMVwxRLRjXpq+BU= github.com/vishvananda/netns v0.0.0-20200728191858-db3c7e526aae/go.mod h1:DD4vA1DwXk04H54A1oHXtwZmA0grkVMdPxx/VGLCah0= github.com/vishvananda/netns v0.0.4 h1:Oeaw1EM2JMxD51g9uhtC0D7erkIjgmj8+JZc26m1YX8= github.com/vishvananda/netns v0.0.4/go.mod h1:SpkAiCQRtJ6TvvxPnOSyH3BMl6unz3xZlaprSwhNNJM= @@ -443,7 +434,6 @@ github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9de github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= -github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= @@ -453,28 +443,23 @@ go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= go.starlark.net v0.0.0-20230525235612-a134d8f9ddca h1:VdD38733bfYv5tUZwEIskMM93VanwNIi5bIKnDrJdEY= go.starlark.net v0.0.0-20230525235612-a134d8f9ddca/go.mod h1:jxU+3+j+71eXOW14274+SmmuW82qJzl6iZSeqEtTGds= -go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= -go.uber.org/goleak v1.1.11/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= -go.uber.org/goleak v1.2.1 h1:NBol2c7O1ZokfZ0LEU9K6Whx/KnwvepVetCUhtKja4A= -go.uber.org/goleak v1.2.1/go.mod h1:qlT2yGI9QafXHhZZLxlSuNsMw3FFLxBr+tBRlmO1xH4= -go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= +go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= +go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= -go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg= -go.uber.org/zap v1.25.0 h1:4Hvk6GtkucQ790dqmj7l1eEnRdKm3k3ZUrUMS2d5+5c= -go.uber.org/zap v1.25.0/go.mod h1:JIAUzQIH94IC4fOJQm7gMmBJP5k7wQfdcnYdPoEXJYk= +go.uber.org/zap v1.26.0 h1:sI7k6L95XOKS281NhVKOFCUNIvv9e0w4BF8N3u+tCRo= +go.uber.org/zap v1.26.0/go.mod h1:dtElttAiwGvoJ/vj4IwHBS/gXsEu/pZ50mUIRWuG0so= go4.org v0.0.0-20200104003542-c7e774b10ea0 h1:M6XsnQeLwG+rHQ+/rrGh3puBI3WZEy9TBWmf2H+enQA= go4.org v0.0.0-20200104003542-c7e774b10ea0/go.mod h1:MkTOUMDaeVYJUOUsaDXIhWPZYa1yOyC1qaOBpL57BhE= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200414173820-0848c9571904/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= -golang.org/x/crypto v0.0.0-20211108221036-ceb1ce70b4fa/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/crypto v0.21.0 h1:X31++rzVUdKhX5sWmSOFZxx8UW/ldWx55cbf08iNAMA= golang.org/x/crypto v0.21.0/go.mod h1:0BP7YvVV9gBbVKyeTG0Gyn+gZm94bibOW5BjDEYAOMs= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= @@ -512,10 +497,9 @@ golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= -golang.org/x/mod v0.13.0 h1:I/DsJXRlw/8l/0c24sM9yb0T4z9liZTduXvdAWYiysY= -golang.org/x/mod v0.13.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= +golang.org/x/mod v0.15.0 h1:SernR4v+D55NyBH2QiEQrlBAnj1ECL6AGrA5+dPaMY8= +golang.org/x/mod v0.15.0/go.mod h1:hTbmBsO62+eylJbnUtE2MGJUyE7QWk4xUqPFrRgJ+7c= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= @@ -548,7 +532,7 @@ golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwY golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= golang.org/x/net v0.1.0/go.mod h1:Cx3nUiGt4eDBEyega/BKRp+/AlGL8hYe7U9odMt2Cco= golang.org/x/net v0.23.0 h1:7EYJ93RZ9vYSZAIb2x3lnuvqO5zneoD6IvWjuhfxjTs= @@ -562,8 +546,8 @@ golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.13.0 h1:jDDenyj+WgFtmV3zYVoi8aE2BwtXFLWOA67ZfNWftiY= -golang.org/x/oauth2 v0.13.0/go.mod h1:/JMhi4ZRXAf4HG9LiNmxvk+45+96RUlVThiH8FzNBn0= +golang.org/x/oauth2 v0.17.0 h1:6m3ZPmLEFdVxKKWnKq4VqZ60gutO35zm+zrAHVmHyDQ= +golang.org/x/oauth2 v0.17.0/go.mod h1:OzPDGQiuQMguemayvdylqddI7qcD9lnSDb+1FiwQ5HA= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -574,10 +558,9 @@ golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/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.4.0 h1:zxkM55ReGkDlKSM+Fu41A+zmbZuaPVbGMzvvdUPznYQ= -golang.org/x/sync v0.4.0/go.mod h1:FU7BRWz2tNW+3quACPkgCx/L+uEAv1htQ0V83Z9Rj+Y= +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-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -585,6 +568,7 @@ golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190606203320-7fc4e5ec1444/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -605,7 +589,6 @@ golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200728102440-3e129f6d46b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -614,12 +597,10 @@ golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210225134936-a50acf3fe073/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220319134239-a9b59b0215f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220715151400-c0bba94af5f8/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -640,6 +621,7 @@ golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3 golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= @@ -684,7 +666,6 @@ golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjs golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= -golang.org/x/tools v0.0.0-20200505023115-26f46d2f7ef8/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= @@ -700,10 +681,9 @@ golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4f golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210108195828-e2f9c7f1fc8e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= -golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= -golang.org/x/tools v0.14.0 h1:jvNa2pY0M4r62jkRQ6RwEZZyPcymeL9XZMLBbV7U2nc= -golang.org/x/tools v0.14.0/go.mod h1:uYBEerGOWcJyEORxN+Ek8+TT266gXkNlHdJBwexUsBg= +golang.org/x/tools v0.18.0 h1:k8NLag8AGHnn+PHbl7g43CtqZAwG60vZkLqgyZgIHgQ= +golang.org/x/tools v0.18.0/go.mod h1:GL7B4CwcLLeo59yx/9UWWuNOW1n3VZ4f5axWfML7Lcg= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -774,8 +754,8 @@ google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6D google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto/googleapis/rpc v0.0.0-20230525234030-28d5490b6b19 h1:0nDDozoAU19Qb2HwhXadU8OcsiO/09cnTqhUtq2MEOM= -google.golang.org/genproto/googleapis/rpc v0.0.0-20230525234030-28d5490b6b19/go.mod h1:66JfowdXAEgad5O9NnYcsNPLCPZJD++2L9X0PCMODrA= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240227224415-6ceb2ff114de h1:cZGRis4/ot9uVm639a+rHCUaG0JJHEsdyzSQTMX+suY= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240227224415-6ceb2ff114de/go.mod h1:H4O17MA/PE9BsGx3w+a+W2VOLLD1Qf7oJneAoU6WktY= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= @@ -792,8 +772,8 @@ google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.56.3 h1:8I4C0Yq1EjstUzUJzpcRVbuYA2mODtEmpWiQoN/b2nc= -google.golang.org/grpc v1.56.3/go.mod h1:I9bI3vqKfayGqPUAwGdOSu7kt6oIJLixfffKrpXqQ9s= +google.golang.org/grpc v1.63.0 h1:WjKe+dnvABXyPJMD7KDNLxtoGk5tgk+YFWN6cBWjZE8= +google.golang.org/grpc v1.63.0/go.mod h1:WAX/8DgncnokcFUldAxq7GeB5DXHDbMF+lLvDomNkRA= google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= @@ -838,42 +818,41 @@ honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9 honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k= howett.net/plist v1.0.0 h1:7CrbWYbPPO/PyNy38b2EB/+gYbjCe2DXBxgtOOZbSQM= howett.net/plist v1.0.0/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g= -k8s.io/api v0.28.3 h1:Gj1HtbSdB4P08C8rs9AR94MfSGpRhJgsS+GF9V26xMM= -k8s.io/api v0.28.3/go.mod h1:MRCV/jr1dW87/qJnZ57U5Pak65LGmQVkKTzf3AtKFHc= -k8s.io/apiextensions-apiserver v0.28.3 h1:Od7DEnhXHnHPZG+W9I97/fSQkVpVPQx2diy+2EtmY08= -k8s.io/apiextensions-apiserver v0.28.3/go.mod h1:NE1XJZ4On0hS11aWWJUTNkmVB03j9LM7gJSisbRt8Lc= -k8s.io/apimachinery v0.28.3 h1:B1wYx8txOaCQG0HmYF6nbpU8dg6HvA06x5tEffvOe7A= -k8s.io/apimachinery v0.28.3/go.mod h1:uQTKmIqs+rAYaq+DFaoD2X7pcjLOqbQX2AOiO0nIpb8= -k8s.io/apiserver v0.28.3 h1:8Ov47O1cMyeDzTXz0rwcfIIGAP/dP7L8rWbEljRcg5w= -k8s.io/apiserver v0.28.3/go.mod h1:YIpM+9wngNAv8Ctt0rHG4vQuX/I5rvkEMtZtsxW2rNM= -k8s.io/cli-runtime v0.28.3 h1:lvuJYVkwCqHEvpS6KuTZsUVwPePFjBfSGvuaLl2SxzA= -k8s.io/cli-runtime v0.28.3/go.mod h1:jeX37ZPjIcENVuXDDTskG3+FnVuZms5D9omDXS/2Jjc= -k8s.io/client-go v0.28.3 h1:2OqNb72ZuTZPKCl+4gTKvqao0AMOl9f3o2ijbAj3LI4= -k8s.io/client-go v0.28.3/go.mod h1:LTykbBp9gsA7SwqirlCXBWtK0guzfhpoW4qSm7i9dxo= -k8s.io/code-generator v0.28.3 h1:I847QvdpYx7xKiG2KVQeCSyNF/xU9TowaDAg601mvlw= -k8s.io/code-generator v0.28.3/go.mod h1:A2EAHTRYvCvBrb/MM2zZBNipeCk3f8NtpdNIKawC43M= -k8s.io/component-base v0.28.3 h1:rDy68eHKxq/80RiMb2Ld/tbH8uAE75JdCqJyi6lXMzI= -k8s.io/component-base v0.28.3/go.mod h1:fDJ6vpVNSk6cRo5wmDa6eKIG7UlIQkaFmZN2fYgIUD8= -k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01 h1:pWEwq4Asjm4vjW7vcsmijwBhOr1/shsbSYiWXmNGlks= -k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E= -k8s.io/klog/v2 v2.2.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y= -k8s.io/klog/v2 v2.100.1 h1:7WCHKK6K8fNhTqfBhISHQ97KrnJNFZMcQvKp7gP/tmg= -k8s.io/klog/v2 v2.100.1/go.mod h1:y1WjHnz7Dj687irZUWR/WLkLc5N1YHtjLdmgWjndZn0= +k8s.io/api v0.30.2 h1:+ZhRj+28QT4UOH+BKznu4CBgPWgkXO7XAvMcMl0qKvI= +k8s.io/api v0.30.2/go.mod h1:ULg5g9JvOev2dG0u2hig4Z7tQ2hHIuS+m8MNZ+X6EmI= +k8s.io/apiextensions-apiserver v0.30.2 h1:l7Eue2t6QiLHErfn2vwK4KgF4NeDgjQkCXtEbOocKIE= +k8s.io/apiextensions-apiserver v0.30.2/go.mod h1:lsJFLYyK40iguuinsb3nt+Sj6CmodSI4ACDLep1rgjw= +k8s.io/apimachinery v0.30.2 h1:fEMcnBj6qkzzPGSVsAZtQThU62SmQ4ZymlXRC5yFSCg= +k8s.io/apimachinery v0.30.2/go.mod h1:iexa2somDaxdnj7bha06bhb43Zpa6eWH8N8dbqVjTUc= +k8s.io/apiserver v0.30.2 h1:ACouHiYl1yFI2VFI3YGM+lvxgy6ir4yK2oLOsLI1/tw= +k8s.io/apiserver v0.30.2/go.mod h1:BOTdFBIch9Sv0ypSEcUR6ew/NUFGocRFNl72Ra7wTm8= +k8s.io/cli-runtime v0.30.2 h1:ooM40eEJusbgHNEqnHziN9ZpLN5U4WcQGsdLKVxpkKE= +k8s.io/cli-runtime v0.30.2/go.mod h1:Y4g/2XezFyTATQUbvV5WaChoUGhojv/jZAtdp5Zkm0A= +k8s.io/client-go v0.30.2 h1:sBIVJdojUNPDU/jObC+18tXWcTJVcwyqS9diGdWHk50= +k8s.io/client-go v0.30.2/go.mod h1:JglKSWULm9xlJLx4KCkfLLQ7XwtlbflV6uFFSHTMgVs= +k8s.io/code-generator v0.30.2 h1:ZY1+aGkqZVwKIyGsOzquaeZ5rSfE6wZHur8z3jQAaiw= +k8s.io/code-generator v0.30.2/go.mod h1:RQP5L67QxqgkVquk704CyvWFIq0e6RCMmLTXxjE8dVA= +k8s.io/component-base v0.30.2 h1:pqGBczYoW1sno8q9ObExUqrYSKhtE5rW3y6gX88GZII= +k8s.io/component-base v0.30.2/go.mod h1:yQLkQDrkK8J6NtP+MGJOws+/PPeEXNpwFixsUI7h/OE= +k8s.io/gengo/v2 v2.0.0-20240228010128-51d4e06bde70 h1:NGrVE502P0s0/1hudf8zjgwki1X/TByhmAoILTarmzo= +k8s.io/gengo/v2 v2.0.0-20240228010128-51d4e06bde70/go.mod h1:VH3AT8AaQOqiGjMF9p0/IM1Dj+82ZwjfxUP1IxaHE+8= +k8s.io/klog/v2 v2.120.1 h1:QXU6cPEOIslTGvZaXvFWiP9VKyeet3sawzTOvdXb4Vw= +k8s.io/klog/v2 v2.120.1/go.mod h1:3Jpz1GvMt720eyJH1ckRHK1EDfpxISzJ7I9OYgaDtPE= k8s.io/kube-aggregator v0.27.4 h1:WdK9iiBr32G8bWfpUEFVQl70RZO2dU19ZAktUXL5JFc= k8s.io/kube-aggregator v0.27.4/go.mod h1:+eG83gkAyh0uilQEAOgheeQW4hr+PkyV+5O1nLGsjlM= -k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00 h1:aVUu9fTY98ivBPKR9Y5w/AuzbMm96cd3YHRTU83I780= -k8s.io/kube-openapi v0.0.0-20231010175941-2dd684a91f00/go.mod h1:AsvuZPBlUDVuCdzJ87iajxtXuR9oktsTctW/R9wwouA= -k8s.io/kubectl v0.28.3 h1:H1Peu1O3EbN9zHkJCcvhiJ4NUj6lb88sGPO5wrWIM6k= -k8s.io/kubectl v0.28.3/go.mod h1:RDAudrth/2wQ3Sg46fbKKl4/g+XImzvbsSRZdP2RiyE= -k8s.io/kubelet v0.27.7 h1:DiptBLFbl6nyadTP9DUfhiReasBDV1qyE1r8h2o5mXc= -k8s.io/kubelet v0.27.7/go.mod h1:WKoEgiCa6/hzmgN4UgVioEwcpLC8wg+9Xzzc8fqOCYs= +k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340 h1:BZqlfIlq5YbRMFko6/PM7FjZpUb45WallggurYhKGag= +k8s.io/kube-openapi v0.0.0-20240228011516-70dd3763d340/go.mod h1:yD4MZYeKMBwQKVht279WycxKyM84kkAx2DPrTXaeb98= +k8s.io/kubectl v0.30.2 h1:cgKNIvsOiufgcs4yjvgkK0+aPCfa8pUwzXdJtkbhsH8= +k8s.io/kubectl v0.30.2/go.mod h1:rz7GHXaxwnigrqob0lJsiA07Df8RE3n1TSaC2CTeuB4= +k8s.io/kubelet v0.29.3 h1:X9h0ZHzc+eUeNTaksbN0ItHyvGhQ7Z0HPjnQD2oHdwU= +k8s.io/kubelet v0.29.3/go.mod h1:jDiGuTkFOUynyBKzOoC1xRSWlgAZ9UPcTYeFyjr6vas= k8s.io/utils v0.0.0-20230726121419-3b25d923346b h1:sgn3ZU783SCgtaSJjpcVVlRqd6GSnlTLKgpAAttJvpI= k8s.io/utils v0.0.0-20230726121419-3b25d923346b/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0= rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= -sigs.k8s.io/controller-runtime v0.16.3 h1:2TuvuokmfXvDUamSx1SuAOO3eTyye+47mJCigwG62c4= -sigs.k8s.io/controller-runtime v0.16.3/go.mod h1:j7bialYoSn142nv9sCOJmQgDXQXxnroFU4VnX/brVJ0= +sigs.k8s.io/controller-runtime v0.18.4 h1:87+guW1zhvuPLh1PHybKdYFLU0YJp4FhJRmiHvm5BZw= +sigs.k8s.io/controller-runtime v0.18.4/go.mod h1:TVoGrfdpbA9VRFaRnKgk9P5/atA0pMwq+f+msb9M8Sg= sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd h1:EDPBXCAspyGV4jQlpZSudPeMmr1bNJefnuqLsRAsHZo= sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd/go.mod h1:B8JuhiUyNFVKdsE8h686QcCxMaH6HrOAZj4vswFpcB0= sigs.k8s.io/kube-storage-version-migrator v0.0.6-0.20230721195810-5c8923c5ff96 h1:PFWFSkpArPNJxFX4ZKWAk9NSeRoZaXschn+ULa4xVek= @@ -882,8 +861,7 @@ sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3 h1:XX3Ajgzov2RKU sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3/go.mod h1:9n16EZKMhXBNSiUC5kSdFQJkdH3zbxS/JoO619G1VAY= sigs.k8s.io/kustomize/kyaml v0.14.3-0.20230601165947-6ce0bf390ce3 h1:W6cLQc5pnqM7vh3b7HvGNfXrJ/xL6BDMS0v1V/HHg5U= sigs.k8s.io/kustomize/kyaml v0.14.3-0.20230601165947-6ce0bf390ce3/go.mod h1:JWP1Fj0VWGHyw3YUPjXSQnRnrwezrZSrApfX5S0nIag= -sigs.k8s.io/structured-merge-diff/v4 v4.3.0 h1:UZbZAZfX0wV2zr7YZorDz6GXROfDFj6LvqCRm4VUVKk= -sigs.k8s.io/structured-merge-diff/v4 v4.3.0/go.mod h1:N8hJocpFajUSSeSJ9bOZ77VzejKZaXsTtZo4/u7Io08= -sigs.k8s.io/yaml v1.2.0/go.mod h1:yfXDCHCao9+ENCvLSE62v9VSji2MKu5jeNfTrofGhJc= +sigs.k8s.io/structured-merge-diff/v4 v4.4.1 h1:150L+0vs/8DA78h1u02ooW1/fFq/Lwr+sGiqlzvrtq4= +sigs.k8s.io/structured-merge-diff/v4 v4.4.1/go.mod h1:N8hJocpFajUSSeSJ9bOZ77VzejKZaXsTtZo4/u7Io08= sigs.k8s.io/yaml v1.4.0 h1:Mk1wCc2gy/F0THH0TAp1QYyJNzRm2KCLy3o5ASXVI5E= sigs.k8s.io/yaml v1.4.0/go.mod h1:Ejl7/uTz7PSA4eKMyQCUTnhZYNmLIl+5c2lQPGR2BPY= diff --git a/pkg/drain/drainer.go b/pkg/drain/drainer.go index a3500dc47..d05d253f0 100644 --- a/pkg/drain/drainer.go +++ b/pkg/drain/drainer.go @@ -77,7 +77,7 @@ func (d *Drainer) DrainNode(ctx context.Context, node *corev1.Node, fullNodeDrai var lastErr error reqLogger.Info("drainNode(): Start draining") - if err = wait.ExponentialBackoff(backoff, func() (bool, error) { + if err = wait.ExponentialBackoffWithContext(ctx, backoff, func(ctx context.Context) (bool, error) { err := drain.RunCordonOrUncordon(drainHelper, node, true) if err != nil { lastErr = err @@ -143,7 +143,7 @@ func createDrainHelper(kubeClient kubernetes.Interface, ctx context.Context, ful DeleteEmptyDirData: true, GracePeriodSeconds: -1, Timeout: 90 * time.Second, - OnPodDeletedOrEvicted: func(pod *corev1.Pod, usingEviction bool) { + OnPodDeletionOrEvictionFinished: func(pod *corev1.Pod, usingEviction bool, err error) { verbStr := constants.DrainDeleted if usingEviction { verbStr = constants.DrainEvicted diff --git a/pkg/helper/mock/mock_helper.go b/pkg/helper/mock/mock_helper.go index 2b2ff4e33..34eebe8b8 100644 --- a/pkg/helper/mock/mock_helper.go +++ b/pkg/helper/mock/mock_helper.go @@ -1127,6 +1127,20 @@ func (mr *MockHostHelpersInterfaceMockRecorder) TryEnableVhostNet() *gomock.Call return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "TryEnableVhostNet", reflect.TypeOf((*MockHostHelpersInterface)(nil).TryEnableVhostNet)) } +// TryGetInterfaceIndex mocks base method. +func (m *MockHostHelpersInterface) TryGetInterfaceIndex(pciAddr string) int { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "TryGetInterfaceIndex", pciAddr) + ret0, _ := ret[0].(int) + return ret0 +} + +// TryGetInterfaceIndex indicates an expected call of TryGetInterfaceIndex. +func (mr *MockHostHelpersInterfaceMockRecorder) TryGetInterfaceIndex(pciAddr interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "TryGetInterfaceIndex", reflect.TypeOf((*MockHostHelpersInterface)(nil).TryGetInterfaceIndex), pciAddr) +} + // TryGetInterfaceName mocks base method. func (m *MockHostHelpersInterface) TryGetInterfaceName(pciAddr string) string { m.ctrl.T.Helper() diff --git a/pkg/host/internal/lib/dputils/dputils.go b/pkg/host/internal/lib/dputils/dputils.go index ef503a851..c8e6071ff 100644 --- a/pkg/host/internal/lib/dputils/dputils.go +++ b/pkg/host/internal/lib/dputils/dputils.go @@ -12,6 +12,8 @@ func New() DPUtilsLib { type DPUtilsLib interface { // GetNetNames returns host net interface names as string for a PCI device from its pci address GetNetNames(pciAddr string) ([]string, error) + // GetNetIndex returns host net interface index as int for a PCI device from its pci address + GetNetIndex(pciAddr string) (int, error) // GetDriverName returns current driver attached to a pci device from its pci address GetDriverName(pciAddr string) (string, error) // GetVFID returns VF ID index (within specific PF) based on PCI address @@ -37,6 +39,10 @@ func (w *libWrapper) GetNetNames(pciAddr string) ([]string, error) { return dputils.GetNetNames(pciAddr) } +func (w *libWrapper) GetNetIndex(pciAddr string) (int, error) { + return dputils.GetNetIndex(pciAddr) +} + // GetDriverName returns current driver attached to a pci device from its pci address func (w *libWrapper) GetDriverName(pciAddr string) (string, error) { return dputils.GetDriverName(pciAddr) diff --git a/pkg/host/internal/lib/dputils/mock/mock_dputils.go b/pkg/host/internal/lib/dputils/mock/mock_dputils.go index de32180f2..de9cce907 100644 --- a/pkg/host/internal/lib/dputils/mock/mock_dputils.go +++ b/pkg/host/internal/lib/dputils/mock/mock_dputils.go @@ -48,6 +48,21 @@ func (mr *MockDPUtilsLibMockRecorder) GetDriverName(pciAddr interface{}) *gomock return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDriverName", reflect.TypeOf((*MockDPUtilsLib)(nil).GetDriverName), pciAddr) } +// GetNetIndex mocks base method. +func (m *MockDPUtilsLib) GetNetIndex(pciAddr string) (int, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetNetIndex", pciAddr) + ret0, _ := ret[0].(int) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetNetIndex indicates an expected call of GetNetIndex. +func (mr *MockDPUtilsLibMockRecorder) GetNetIndex(pciAddr interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetNetIndex", reflect.TypeOf((*MockDPUtilsLib)(nil).GetNetIndex), pciAddr) +} + // GetNetNames mocks base method. func (m *MockDPUtilsLib) GetNetNames(pciAddr string) ([]string, error) { m.ctrl.T.Helper() diff --git a/pkg/host/internal/lib/netlink/mock/mock_netlink.go b/pkg/host/internal/lib/netlink/mock/mock_netlink.go index e3e7d649f..367f57018 100644 --- a/pkg/host/internal/lib/netlink/mock/mock_netlink.go +++ b/pkg/host/internal/lib/netlink/mock/mock_netlink.go @@ -159,6 +159,21 @@ func (mr *MockNetlinkLibMockRecorder) IsLinkAdminStateUp(link interface{}) *gomo return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "IsLinkAdminStateUp", reflect.TypeOf((*MockNetlinkLib)(nil).IsLinkAdminStateUp), link) } +// LinkByIndex mocks base method. +func (m *MockNetlinkLib) LinkByIndex(index int) (netlink.Link, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "LinkByIndex", index) + ret0, _ := ret[0].(netlink.Link) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// LinkByIndex indicates an expected call of LinkByIndex. +func (mr *MockNetlinkLibMockRecorder) LinkByIndex(index interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "LinkByIndex", reflect.TypeOf((*MockNetlinkLib)(nil).LinkByIndex), index) +} + // LinkByName mocks base method. func (m *MockNetlinkLib) LinkByName(name string) (netlink.Link, error) { m.ctrl.T.Helper() diff --git a/pkg/host/internal/lib/netlink/netlink.go b/pkg/host/internal/lib/netlink/netlink.go index 95b620bba..e53ee85bf 100644 --- a/pkg/host/internal/lib/netlink/netlink.go +++ b/pkg/host/internal/lib/netlink/netlink.go @@ -24,6 +24,8 @@ type NetlinkLib interface { LinkSetVfPortGUID(link Link, vf int, portguid net.HardwareAddr) error // LinkByName finds a link by name and returns a pointer to the object. LinkByName(name string) (Link, error) + // LinkByName finds a link by index and returns a pointer to the object. + LinkByIndex(index int) (Link, error) // LinkSetVfHardwareAddr sets the hardware address of a vf for the link. // Equivalent to: `ip link set $link vf $vf mac $hwaddr` LinkSetVfHardwareAddr(link Link, vf int, hwaddr net.HardwareAddr) error @@ -79,11 +81,16 @@ func (w *libWrapper) LinkSetVfPortGUID(link Link, vf int, portguid net.HardwareA return netlink.LinkSetVfPortGUID(link, vf, portguid) } -// LinkByName finds a link by name and returns a pointer to the object.// LinkByName finds a link by name and returns a pointer to the object. +// LinkByName finds a link by name and returns a pointer to the object. func (w *libWrapper) LinkByName(name string) (Link, error) { return netlink.LinkByName(name) } +// LinkByIndex finds a link by index and returns a pointer to the object. +func (w *libWrapper) LinkByIndex(index int) (Link, error) { + return netlink.LinkByIndex(index) +} + // LinkSetVfHardwareAddr sets the hardware address of a vf for the link. // Equivalent to: `ip link set $link vf $vf mac $hwaddr` func (w *libWrapper) LinkSetVfHardwareAddr(link Link, vf int, hwaddr net.HardwareAddr) error { diff --git a/pkg/host/internal/network/network.go b/pkg/host/internal/network/network.go index 5c026f217..681fef3d8 100644 --- a/pkg/host/internal/network/network.go +++ b/pkg/host/internal/network/network.go @@ -75,6 +75,7 @@ func (n *network) TryToGetVirtualInterfaceName(pciAddr string) string { func (n *network) TryGetInterfaceName(pciAddr string) string { names, err := n.dputilsLib.GetNetNames(pciAddr) if err != nil || len(names) < 1 { + log.Log.Error(err, "TryGetInterfaceName(): failed to get interface name") return "" } netDevName := names[0] @@ -95,10 +96,20 @@ func (n *network) TryGetInterfaceName(pciAddr string) string { return name } - log.Log.V(2).Info("tryGetInterfaceName()", "name", netDevName) + log.Log.V(2).Info("TryGetInterfaceName()", "name", netDevName) return netDevName } +// TryGetInterfaceName tries to find the SR-IOV virtual interface index base on pci address +func (n *network) TryGetInterfaceIndex(pciAddr string) int { + ifIndex, err := n.dputilsLib.GetNetIndex(pciAddr) + if err != nil { + log.Log.Error(err, "TryGetInterfaceIndex(): failed to get interface index") + return -1 + } + return ifIndex +} + func (n *network) GetPhysSwitchID(name string) (string, error) { swIDFile := filepath.Join(vars.FilesystemRoot, consts.SysClassNet, name, "phys_switch_id") physSwitchID, err := os.ReadFile(swIDFile) diff --git a/pkg/host/internal/sriov/sriov.go b/pkg/host/internal/sriov/sriov.go index 1b8d99091..1a825077f 100644 --- a/pkg/host/internal/sriov/sriov.go +++ b/pkg/host/internal/sriov/sriov.go @@ -189,12 +189,13 @@ func (s *sriov) VFIsReady(pciAddr string) (netlink.Link, error) { var err error var vfLink netlink.Link err = wait.PollImmediate(time.Second, 10*time.Second, func() (bool, error) { - vfName := s.networkHelper.TryGetInterfaceName(pciAddr) - vfLink, err = s.netlinkLib.LinkByName(vfName) + vfIndex := s.networkHelper.TryGetInterfaceIndex(pciAddr) + vfLink, err = s.netlinkLib.LinkByIndex(vfIndex) if err != nil { log.Log.Error(err, "VFIsReady(): unable to get VF link", "device", pciAddr) + return false, nil } - return err == nil, nil + return true, nil }) if err != nil { return vfLink, err diff --git a/pkg/host/internal/sriov/sriov_test.go b/pkg/host/internal/sriov/sriov_test.go index 8c4e8ef6f..e1da764d5 100644 --- a/pkg/host/internal/sriov/sriov_test.go +++ b/pkg/host/internal/sriov/sriov_test.go @@ -235,11 +235,11 @@ var _ = Describe("SRIOV", func() { hostMock.EXPECT().UnbindDriverIfNeeded("0000:d8:00.2", true).Return(nil) hostMock.EXPECT().BindDefaultDriver("0000:d8:00.2").Return(nil) hostMock.EXPECT().SetNetdevMTU("0000:d8:00.2", 2000).Return(nil) - hostMock.EXPECT().TryGetInterfaceName("0000:d8:00.2").Return("enp216s0f0_0") + hostMock.EXPECT().TryGetInterfaceIndex("0000:d8:00.2").Return(42).Times(1) vf0LinkMock := netlinkMockPkg.NewMockLink(testCtrl) vf0Mac, _ := net.ParseMAC("02:42:19:51:2f:af") - vf0LinkMock.EXPECT().Attrs().Return(&netlink.LinkAttrs{HardwareAddr: vf0Mac}) - netlinkLibMock.EXPECT().LinkByName("enp216s0f0_0").Return(vf0LinkMock, nil) + vf0LinkMock.EXPECT().Attrs().Return(&netlink.LinkAttrs{Name: "enp216s0f0_0", HardwareAddr: vf0Mac}) + netlinkLibMock.EXPECT().LinkByIndex(42).Return(vf0LinkMock, nil) netlinkLibMock.EXPECT().LinkSetVfHardwareAddr(vf0LinkMock, 0, vf0Mac).Return(nil) dputilsLibMock.EXPECT().GetVFID("0000:d8:00.3").Return(1, nil) @@ -359,11 +359,11 @@ var _ = Describe("SRIOV", func() { hostMock.EXPECT().UnbindDriverIfNeeded("0000:d8:00.2", true).Return(nil) hostMock.EXPECT().BindDefaultDriver("0000:d8:00.2").Return(nil) hostMock.EXPECT().SetNetdevMTU("0000:d8:00.2", 2000).Return(nil) - hostMock.EXPECT().TryGetInterfaceName("0000:d8:00.2").Return("enp216s0f0_0") + hostMock.EXPECT().TryGetInterfaceIndex("0000:d8:00.2").Return(42) vf0LinkMock := netlinkMockPkg.NewMockLink(testCtrl) vf0Mac, _ := net.ParseMAC("02:42:19:51:2f:af") - vf0LinkMock.EXPECT().Attrs().Return(&netlink.LinkAttrs{HardwareAddr: vf0Mac}) - netlinkLibMock.EXPECT().LinkByName("enp216s0f0_0").Return(vf0LinkMock, nil) + vf0LinkMock.EXPECT().Attrs().Return(&netlink.LinkAttrs{Name: "enp216s0f0_0", HardwareAddr: vf0Mac}) + netlinkLibMock.EXPECT().LinkByIndex(42).Return(vf0LinkMock, nil).AnyTimes() netlinkLibMock.EXPECT().LinkSetVfHardwareAddr(vf0LinkMock, 0, vf0Mac).Return(nil) hostMock.EXPECT().GetPhysPortName("enp216s0f0np0").Return("p0", nil) hostMock.EXPECT().GetPhysSwitchID("enp216s0f0np0").Return("7cfe90ff2cc0", nil) @@ -537,6 +537,21 @@ var _ = Describe("SRIOV", func() { helpers.GinkgoAssertFileContentsEquals("/sys/bus/pci/devices/0000:d8:00.0/sriov_numvfs", "2") }) }) + + Context("VfIsReady", func() { + It("Should retry if interface index is -1", func() { + hostMock.EXPECT().TryGetInterfaceIndex("0000:d8:00.2").Return(-1).Times(1) + hostMock.EXPECT().TryGetInterfaceIndex("0000:d8:00.2").Return(42).Times(1) + vf0LinkMock := netlinkMockPkg.NewMockLink(testCtrl) + vf0Mac, _ := net.ParseMAC("02:42:19:51:2f:af") + vf0LinkMock.EXPECT().Attrs().Return(&netlink.LinkAttrs{Name: "enp216s0f0_0", HardwareAddr: vf0Mac}) + netlinkLibMock.EXPECT().LinkByIndex(-1).Return(vf0LinkMock, fmt.Errorf("interface not found")) + netlinkLibMock.EXPECT().LinkByIndex(42).Return(vf0LinkMock, nil).Times(1) + vfLink, err := s.VFIsReady("0000:d8:00.2") + Expect(err).ToNot(HaveOccurred()) + Expect(vfLink.Attrs().HardwareAddr).To(Equal(vf0Mac)) + }) + }) }) func getTestPCIDevices() []*ghw.PCIDevice { diff --git a/pkg/host/mock/mock_host.go b/pkg/host/mock/mock_host.go index e9ccdb9cd..5a09c1440 100644 --- a/pkg/host/mock/mock_host.go +++ b/pkg/host/mock/mock_host.go @@ -970,6 +970,20 @@ func (mr *MockHostManagerInterfaceMockRecorder) TryEnableVhostNet() *gomock.Call return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "TryEnableVhostNet", reflect.TypeOf((*MockHostManagerInterface)(nil).TryEnableVhostNet)) } +// TryGetInterfaceIndex mocks base method. +func (m *MockHostManagerInterface) TryGetInterfaceIndex(pciAddr string) int { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "TryGetInterfaceIndex", pciAddr) + ret0, _ := ret[0].(int) + return ret0 +} + +// TryGetInterfaceIndex indicates an expected call of TryGetInterfaceIndex. +func (mr *MockHostManagerInterfaceMockRecorder) TryGetInterfaceIndex(pciAddr interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "TryGetInterfaceIndex", reflect.TypeOf((*MockHostManagerInterface)(nil).TryGetInterfaceIndex), pciAddr) +} + // TryGetInterfaceName mocks base method. func (m *MockHostManagerInterface) TryGetInterfaceName(pciAddr string) string { m.ctrl.T.Helper() diff --git a/pkg/host/types/interfaces.go b/pkg/host/types/interfaces.go index 886776032..fd622afe6 100644 --- a/pkg/host/types/interfaces.go +++ b/pkg/host/types/interfaces.go @@ -81,6 +81,8 @@ type NetworkInterface interface { TryToGetVirtualInterfaceName(pciAddr string) string // TryGetInterfaceName tries to find the SR-IOV virtual interface name base on pci address TryGetInterfaceName(pciAddr string) string + // TryGetInterfaceName tries to find the SR-IOV virtual interface index base on pci address + TryGetInterfaceIndex(pciAddr string) int // GetPhysSwitchID returns the physical switch ID for a specific pci address GetPhysSwitchID(name string) (string, error) // GetPhysPortName returns the physical port name for a specific pci address diff --git a/vendor/github.com/Mellanox/sriovnet/.golangci.yml b/vendor/github.com/Mellanox/rdmamap/.golangci.yml similarity index 93% rename from vendor/github.com/Mellanox/sriovnet/.golangci.yml rename to vendor/github.com/Mellanox/rdmamap/.golangci.yml index 13903829c..09248db5f 100644 --- a/vendor/github.com/Mellanox/sriovnet/.golangci.yml +++ b/vendor/github.com/Mellanox/rdmamap/.golangci.yml @@ -24,7 +24,7 @@ linters-settings: gocognit: min-complexity: 30 goimports: - local-prefixes: github.com/Mellanox/sriovnet + local-prefixes: github.com/Mellanox/rdmamap golint: min-confidence: 0 gomnd: @@ -32,7 +32,6 @@ linters-settings: mnd: # don't include the "operation" and "assign" checks: argument,case,condition,return - ignored-numbers: 2 govet: check-shadowing: true settings: @@ -43,9 +42,6 @@ linters-settings: line-length: 120 misspell: locale: US - ignore-words: - - flavour - - flavours prealloc: # Report preallocation suggestions only on simple loops that have no returns/breaks/continues/gotos in them. # True by default. @@ -83,7 +79,7 @@ linters: - nakedret - prealloc - rowserrcheck - - exportloopref + - scopelint - staticcheck - structcheck - stylecheck diff --git a/vendor/github.com/Mellanox/rdmamap/.travis.yml b/vendor/github.com/Mellanox/rdmamap/.travis.yml new file mode 100644 index 000000000..97f41cea8 --- /dev/null +++ b/vendor/github.com/Mellanox/rdmamap/.travis.yml @@ -0,0 +1,12 @@ +language: go + +go: + - "1.13" + +before_script: + - go get -u github.com/mattn/goveralls + +script: + - make lint + - make test-coverage + - goveralls -coverprofile=rdmamap.cover -service=travis-ci diff --git a/vendor/github.com/Mellanox/sriovnet/LICENSE b/vendor/github.com/Mellanox/rdmamap/LICENSE similarity index 100% rename from vendor/github.com/Mellanox/sriovnet/LICENSE rename to vendor/github.com/Mellanox/rdmamap/LICENSE diff --git a/vendor/github.com/Mellanox/sriovnet/Makefile b/vendor/github.com/Mellanox/rdmamap/Makefile similarity index 97% rename from vendor/github.com/Mellanox/sriovnet/Makefile rename to vendor/github.com/Mellanox/rdmamap/Makefile index d4cb6fa42..dd7d7ad3c 100644 --- a/vendor/github.com/Mellanox/sriovnet/Makefile +++ b/vendor/github.com/Mellanox/rdmamap/Makefile @@ -1,5 +1,5 @@ # Package related -PACKAGE=sriovnet +PACKAGE=rdmamap ORG_PATH=github.com/Mellanox REPO_PATH=$(ORG_PATH)/$(PACKAGE) GOPATH=$(CURDIR)/.gopath @@ -17,7 +17,7 @@ GOLANGCI_LINT = $(GOBIN)/golangci-lint # golangci-lint version should be updated periodically # we keep it fixed to avoid it from unexpectedly failing on the project # in case of a version bump -GOLANGCI_LINT_VER = v1.39.0 +GOLANGCI_LINT_VER = v1.23.8 TIMEOUT = 15 Q = $(if $(filter 1,$V),,@) @@ -65,7 +65,7 @@ COVERAGE_MODE = count test-coverage-tools: $(GOVERALLS) test-coverage: COVERAGE_DIR := $(CURDIR)/test test-coverage: test-coverage-tools $(BASE) ; $(info running coverage tests...) @ ## Run coverage tests - $Q cd $(BASE); $(GO) test -covermode=$(COVERAGE_MODE) -coverprofile=sriovnet.cover ./... + $Q cd $(BASE); $(GO) test -covermode=$(COVERAGE_MODE) -coverprofile=rdmamap.cover ./... # Misc diff --git a/vendor/github.com/Mellanox/rdmamap/README.md b/vendor/github.com/Mellanox/rdmamap/README.md new file mode 100644 index 000000000..22aaa5c1e --- /dev/null +++ b/vendor/github.com/Mellanox/rdmamap/README.md @@ -0,0 +1,40 @@ +[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](http://www.apache.org/licenses/LICENSE-2.0) +[![Go Report Card](https://goreportcard.com/badge/github.com/Mellanox/rdmamap)](https://goreportcard.com/report/github.com/Mellanox/rdmamap) +[![Build Status](https://travis-ci.com/Mellanox/rdmamap.svg?branch=master)](https://travis-ci.com/Mellanox/rdmamap) +[![Coverage Status](https://coveralls.io/repos/github/Mellanox/rdmamap/badge.svg)](https://coveralls.io/github/Mellanox/rdmamap) + +# rdmamap + +This is golang package that provides map of rdma device with its character and network devices. + +It uses sysfs and netlink interfaces provided by kernel to perform this mapping. + +Local build and test + +You can use go get command: +``` +go get github.com/Mellanox/rdmamap +``` + +Example: + +``` +package main + +import ( + "fmt" + "github.com/Mellanox/rdmamap" +) + +func main() { + rdmaDevices := rdmamap.GetRdmaDeviceList() + fmt.Println("Devices: ", rdmaDevices) + + for _, dev := range rdmaDevices { + charDevices := rdmamap.GetRdmaCharDevices(dev) + fmt.Printf("Rdma device: = %s", dev) + fmt.Println(" Char devices: = ", charDevices) + } +} + +``` diff --git a/vendor/github.com/Mellanox/rdmamap/rdma_map.go b/vendor/github.com/Mellanox/rdmamap/rdma_map.go new file mode 100644 index 000000000..44e17e63b --- /dev/null +++ b/vendor/github.com/Mellanox/rdmamap/rdma_map.go @@ -0,0 +1,396 @@ +package rdmamap + +import ( + "bytes" + "fmt" + "io" + "io/ioutil" + "os" + "path/filepath" + "strconv" + "strings" + + "github.com/vishvananda/netlink" +) + +const ( + RdmaClassName = "infiniband" + RdmaClassDir = "/sys/class/infiniband" + RdmaIbUcmDir = "/sys/class/infiniband_cm" + RdmaUcmFilePrefix = "ucm" + + RdmaUmadDir = "/sys/class/infiniband_mad" + RdmaIssmFilePrefix = "issm" + RdmaUmadFilxPrefix = "umad" + + RdmaUverbsDir = "/sys/class/infiniband_verbs" + RdmaUverbsFilxPrefix = "uverbs" + + RdmaGidAttrDir = "gid_attrs" //nolint:stylecheck,golint + RdmaGidAttrNdevDir = "ndevs" //nolint:stylecheck,golint + RdmaPortsdir = "ports" + + RdmaNodeGuidFile = "node_guid" //nolint:stylecheck,golint + RdmaUcmDevice = "/dev/infiniband/rdma_cm" + RdmaDeviceDir = "/dev/infiniband" + + RdmaCountersDir = "counters" + RdmaHwCountersDir = "hw_counters" + + PciDevDir = "/sys/bus/pci/devices" + AuxDevDir = "/sys/bus/auxiliary/devices" + + // For local usage + prevDir = ".." + nibbleBitSize = 4 + loopBackIfName = "lo" +) + +// Returns a list of rdma device names +//nolint:prealloc +func GetRdmaDeviceList() []string { + var rdmaDevices []string + fd, err := os.Open(RdmaClassDir) + if err != nil { + return nil + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return nil + } + + for i := range fileInfos { + if fileInfos[i].IsDir() { + continue + } + rdmaDevices = append(rdmaDevices, fileInfos[i].Name()) + } + return rdmaDevices +} + +func isDirForRdmaDevice(rdmaDeviceName, dirName string) bool { + fileName := filepath.Join(dirName, "ibdev") + + fd, err := os.OpenFile(fileName, os.O_RDONLY, 0444) + if err != nil { + return false + } + defer fd.Close() + + if _, err = fd.Seek(0, io.SeekStart); err != nil { + return false + } + + data, err := ioutil.ReadAll(fd) + if err != nil { + return false + } + return (strings.Compare(strings.Trim(string(data), "\n"), rdmaDeviceName) == 0) +} + +func getCharDevice(rdmaDeviceName, classDir, charDevPrefix string) (string, error) { + fd, err := os.Open(classDir) + if err != nil { + return "", err + } + defer fd.Close() + fileInfos, err := fd.Readdir(-1) + if err != nil { + return "", nil + } + + for i := range fileInfos { + if fileInfos[i].Name() == "." || fileInfos[i].Name() == prevDir { + continue + } + if !strings.Contains(fileInfos[i].Name(), charDevPrefix) { + continue + } + dirName := filepath.Join(classDir, fileInfos[i].Name()) + if !isDirForRdmaDevice(rdmaDeviceName, dirName) { + continue + } + deviceFile := filepath.Join("/dev/infiniband", fileInfos[i].Name()) + return deviceFile, nil + } + return "", fmt.Errorf("no ucm device found") +} + +func getUcmDevice(rdmaDeviceName string) (string, error) { + return getCharDevice(rdmaDeviceName, + RdmaIbUcmDir, + RdmaUcmFilePrefix) +} + +func getIssmDevice(rdmaDeviceName string) (string, error) { + return getCharDevice(rdmaDeviceName, + RdmaUmadDir, + RdmaIssmFilePrefix) +} + +func getUmadDevice(rdmaDeviceName string) (string, error) { + return getCharDevice(rdmaDeviceName, + RdmaUmadDir, + RdmaUmadFilxPrefix) +} + +func getUverbDevice(rdmaDeviceName string) (string, error) { + return getCharDevice(rdmaDeviceName, + RdmaUverbsDir, + RdmaUverbsFilxPrefix) +} + +func getRdmaUcmDevice() (string, error) { + info, err := os.Stat(RdmaUcmDevice) + if err != nil { + return "", err + } + if info.Name() == "rdma_cm" { + return RdmaUcmDevice, nil + } + + return "", fmt.Errorf("invalid file name rdma_cm") +} + +// Returns a list of character device absolute path for a requested +// rdmaDeviceName. +// Returns nil if no character devices are found. +func GetRdmaCharDevices(rdmaDeviceName string) []string { + var rdmaCharDevices []string + + ucm, err := getUcmDevice(rdmaDeviceName) + if err == nil { + rdmaCharDevices = append(rdmaCharDevices, ucm) + } + issm, err := getIssmDevice(rdmaDeviceName) + if err == nil { + rdmaCharDevices = append(rdmaCharDevices, issm) + } + umad, err := getUmadDevice(rdmaDeviceName) + if err == nil { + rdmaCharDevices = append(rdmaCharDevices, umad) + } + uverb, err := getUverbDevice(rdmaDeviceName) + if err == nil { + rdmaCharDevices = append(rdmaCharDevices, uverb) + } + rdmaCm, err := getRdmaUcmDevice() + if err == nil { + rdmaCharDevices = append(rdmaCharDevices, rdmaCm) + } + + return rdmaCharDevices +} + +// Gets a list of ports for a specified device +//nolint:prealloc +func GetPorts(rdmaDeviceName string) []string { + var ports []string + + portsDir := filepath.Join(RdmaClassDir, rdmaDeviceName, RdmaPortsdir) + fd, err := os.Open(portsDir) + if err != nil { + return nil + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return nil + } + + for i := range fileInfos { + if fileInfos[i].Name() == "." || fileInfos[i].Name() == prevDir { + continue + } + ports = append(ports, fileInfos[i].Name()) + } + return ports +} + +//nolint:prealloc +func getNetdeviceIds(rdmaDeviceName, port string) []string { + var indices []string + + dir := filepath.Join(RdmaClassDir, rdmaDeviceName, RdmaPortsdir, port, + RdmaGidAttrDir, RdmaGidAttrNdevDir) + + fd, err := os.Open(dir) + if err != nil { + return nil + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return nil + } + + for i := range fileInfos { + if fileInfos[i].Name() == "." || fileInfos[i].Name() == prevDir { + continue + } + indices = append(indices, fileInfos[i].Name()) + } + return indices +} + +func isNetdevForRdma(rdmaDeviceName, port, index, netdevName string) bool { + fileName := filepath.Join(RdmaClassDir, rdmaDeviceName, RdmaPortsdir, port, + RdmaGidAttrDir, RdmaGidAttrNdevDir, index) + + fd, err := os.OpenFile(fileName, os.O_RDONLY, 0444) + if err != nil { + return false + } + defer fd.Close() + + if _, err = fd.Seek(0, io.SeekStart); err != nil { + return false + } + + data, err := ioutil.ReadAll(fd) + if err != nil { + return false + } + return (strings.TrimSuffix(string(data), "\n") == netdevName) +} + +func getRdmaDeviceForEth(netdevName string) (string, error) { + // Iterate over the list of rdma devices, + // read the gid table attribute netdev + // if the netdev matches, found the matching rdma device + + devices := GetRdmaDeviceList() + for _, dev := range devices { + ports := GetPorts(dev) + for _, port := range ports { + indices := getNetdeviceIds(dev, port) + for _, index := range indices { + found := isNetdevForRdma(dev, port, index, netdevName) + if found { + return dev, nil + } + } + } + } + return "", fmt.Errorf("rdma device not found for netdev %v", netdevName) +} + +func getNodeGUID(rdmaDeviceName string) ([]byte, error) { + var nodeGUID []byte + + fileName := filepath.Join(RdmaClassDir, rdmaDeviceName, RdmaNodeGuidFile) + + fd, err := os.OpenFile(fileName, os.O_RDONLY, 0444) + if err != nil { + return nil, err + } + defer fd.Close() + + if _, err = fd.Seek(0, io.SeekStart); err != nil { + return nil, err + } + data, err := ioutil.ReadAll(fd) + if err != nil { + return nil, err + } + data = data[:len(data)-1] + var j int + for _, b := range data { + if b == ':' { + continue + } + c, err := strconv.ParseUint(string(b), 16, 8) + if err != nil { + return nil, err + } + if (j % 2) == 0 { + nodeGUID = append(nodeGUID, byte(c)<//hw_counters */ + Stats []RdmaStatEntry /* /sys/class/infiniband///counters */ + Port int +} + +type RdmaStats struct { + PortStats []RdmaPortStats +} + +func readCounter(name string) uint64 { + fd, err := os.OpenFile(name, os.O_RDONLY, 0444) + if err != nil { + return 0 + } + defer fd.Close() + + if _, err = fd.Seek(0, io.SeekStart); err != nil { + return 0 + } + + data, err := ioutil.ReadAll(fd) + if err != nil { + return 0 + } + dataStr := string(data) + dataStr = strings.Trim(dataStr, "\n") + value, _ := strconv.ParseUint(dataStr, 10, 64) + return value +} + +//nolint:prealloc +func getCountersFromDir(path string) ([]RdmaStatEntry, error) { + var stats []RdmaStatEntry + + fd, err := os.Open(path) + if err != nil { + return stats, err + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return stats, err + } + + for _, file := range fileInfos { + if file.IsDir() { + continue + } + value := readCounter(filepath.Join(path, file.Name())) + entry := RdmaStatEntry{file.Name(), value} + stats = append(stats, entry) + } + return stats, nil +} + +// Get RDMA Sysfs stats from counters directory of a port of a rdma device +// Port number starts from 1. +func GetRdmaSysfsStats(rdmaDevice string, port int) ([]RdmaStatEntry, error) { + path := filepath.Join(RdmaClassDir, rdmaDevice, + RdmaPortsdir, strconv.Itoa(port), RdmaCountersDir) + + rdmastats, err := getCountersFromDir(path) + return rdmastats, err +} + +// Get RDMA Sysfs stats from hw_counters directory of a port of a rdma device +// Port number starts from 1. +func GetRdmaSysfsHwStats(rdmaDevice string, port int) ([]RdmaStatEntry, error) { + path := filepath.Join(RdmaClassDir, rdmaDevice, + RdmaPortsdir, strconv.Itoa(port), RdmaHwCountersDir) + + rdmastats, err := getCountersFromDir(path) + return rdmastats, err +} + +// Get RDMA sysfs starts from counter and hw_counters directory for a requested +// port of a device. +func GetRdmaSysfsAllStats(rdmaDevice string, port int) (RdmaPortStats, error) { + var portstats RdmaPortStats + + hwstats, err := GetRdmaSysfsHwStats(rdmaDevice, port) + if err != nil { + return portstats, nil + } + portstats.HwStats = hwstats + + stats, err := GetRdmaSysfsStats(rdmaDevice, port) + if err != nil { + return portstats, nil + } + portstats.Stats = stats + portstats.Port = port + return portstats, nil +} + +// Get RDMA sysfs starts from counter and hw_counters directory for a +// rdma device. +func GetRdmaSysfsAllPortsStats(rdmaDevice string) (RdmaStats, error) { + var allstats RdmaStats + + path := filepath.Join(RdmaClassDir, rdmaDevice, RdmaPortsdir) + fd, err := os.Open(path) + if err != nil { + return allstats, err + } + defer fd.Close() + + fileInfos, err := fd.Readdir(-1) + if err != nil { + return allstats, err + } + + for i, file := range fileInfos { + if fileInfos[i].Name() == "." || fileInfos[i].Name() == ".." { + continue + } + if !file.IsDir() { + continue + } + port, _ := strconv.Atoi(file.Name()) + portstats, err := GetRdmaSysfsAllStats(rdmaDevice, port) + if err != nil { + return allstats, err + } + allstats.PortStats = append(allstats.PortStats, portstats) + } + return allstats, nil +} + +func printRdmaStats(device string, stats *RdmaStats) { + for _, portstats := range stats.PortStats { + fmt.Printf("device: %s, port: %d\n", device, portstats.Port) + fmt.Println("Hw stats:") + for _, entry := range portstats.HwStats { + fmt.Printf("%s: %d\n", entry.Name, entry.Value) + } + fmt.Println("Stats:") + for _, entry := range portstats.Stats { + fmt.Printf("%s: %d\n", entry.Name, entry.Value) + } + } +} + +// Get RDMA statistics of a docker container. +// containerID is prefixed matched against the running docker containers, +// so a non ambiguous short identifier can be supplied as well. +func GetDockerContainerRdmaStats(containerID string) { + // Lock the OS Thread so we don't accidentally switch namespaces + runtime.LockOSThread() + defer runtime.UnlockOSThread() + + originalHandle, err := netns.Get() + if err != nil { + log.Println("Fail to get handle of current net ns", err) + return + } + + nsHandle, err := netns.GetFromDocker(containerID) + if err != nil { + log.Println("Invalid docker id: ", containerID) + return + } + if netns.Set(nsHandle) != nil { + return + } + + ifaces, err := net.Interfaces() + if err != nil { + _ = netns.Set(originalHandle) + return + } + log.Printf("Net Interfaces: %v\n", ifaces) + for _, iface := range ifaces { + if iface.Name == loopBackIfName { + continue + } + rdmadev, err := GetRdmaDeviceForNetdevice(iface.Name) + if err != nil { + continue + } + rdmastats, err := GetRdmaSysfsAllPortsStats(rdmadev) + if err != nil { + log.Println("Fail to query device stats: ", err) + continue + } + printRdmaStats(rdmadev, &rdmastats) + } + _ = netns.Set(originalHandle) +} diff --git a/vendor/github.com/Mellanox/sriovnet/.travis.yml b/vendor/github.com/Mellanox/sriovnet/.travis.yml deleted file mode 100644 index 64e272201..000000000 --- a/vendor/github.com/Mellanox/sriovnet/.travis.yml +++ /dev/null @@ -1,16 +0,0 @@ -# see https://docs.travis-ci.com/user/reference/overview/#Virtualization-environments -language: go - -# use the latest ubuntu environment (18.04) available on travis -dist: xenial - -go: - - 1.13.x - -before_script: - - go get -u github.com/mattn/goveralls - -script: - - make lint - - make test-coverage - - goveralls -coverprofile=sriovnet.cover -service=travis-ci diff --git a/vendor/github.com/Mellanox/sriovnet/README.md b/vendor/github.com/Mellanox/sriovnet/README.md deleted file mode 100644 index 138dfae16..000000000 --- a/vendor/github.com/Mellanox/sriovnet/README.md +++ /dev/null @@ -1,59 +0,0 @@ -[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](http://www.apache.org/licenses/LICENSE-2.0) -[![Go Report Card](https://goreportcard.com/badge/github.com/Mellanox/sriovnet)](https://goreportcard.com/report/github.com/Mellanox/sriovnet) -[![Build Status](https://travis-ci.com/Mellanox/sriovnet.svg?branch=master)](https://travis-ci.com/Mellanox/sriovnet) -[![Coverage Status](https://coveralls.io/repos/github/Mellanox/sriovnet/badge.svg)](https://coveralls.io/github/Mellanox/sriovnet) - -# sriovnet -Go library to configure SRIOV networking devices - -Local build and test - -You can use go get command: -``` -go get github.com/Mellanox/sriovnet -``` - -Example: - -```go -package main - -import ( - "fmt" - - "github.com/Mellanox/sriovnet" -) - -func main() { - var vfList[10] *sriovnet.VfObj - - err1 := sriovnet.EnableSriov("ib0") - if err1 != nil { - return - } - - handle, err2 := sriovnet.GetPfNetdevHandle("ib0") - if err2 != nil { - return - } - err3 := sriovnet.ConfigVfs(handle, false) - if err3 != nil { - return - } - for i := 0; i < 10; i++ { - vfList[i], _ = sriovnet.AllocateVf(handle) - } - for _, vf := range handle.List { - fmt.Printf("after allocation vf = %v\n", vf) - } - for i := 0; i < 10; i++ { - if vfList[i] == nil { - continue - } - sriovnet.FreeVf(handle, vfList[i]) - } - for _, vf := range handle.List { - fmt.Printf("after free vf = %v\n", vf) - } -} -``` diff --git a/vendor/github.com/Mellanox/sriovnet/file_access.go b/vendor/github.com/Mellanox/sriovnet/file_access.go deleted file mode 100644 index a22b4911e..000000000 --- a/vendor/github.com/Mellanox/sriovnet/file_access.go +++ /dev/null @@ -1,139 +0,0 @@ -package sriovnet - -import ( - "io" - "io/ioutil" - "os" - "strconv" - "strings" - "syscall" -) - -type fileObject struct { - Path string - File *os.File -} - -func (attrib *fileObject) Exists() bool { - return fileExists(attrib.Path) -} - -func (attrib *fileObject) Open() (err error) { - attrib.File, err = os.OpenFile(attrib.Path, os.O_RDWR|syscall.O_NONBLOCK, 0660) - return err -} - -func (attrib *fileObject) OpenRO() (err error) { - attrib.File, err = os.OpenFile(attrib.Path, os.O_RDONLY, 0444) - return err -} - -func (attrib *fileObject) OpenWO() (err error) { - attrib.File, err = os.OpenFile(attrib.Path, os.O_WRONLY, 0444) - return err -} - -func (attrib *fileObject) Close() (err error) { - err = attrib.File.Close() - attrib.File = nil - return err -} - -func (attrib *fileObject) Read() (str string, err error) { - if attrib.File == nil { - err = attrib.OpenRO() - if err != nil { - return - } - defer func() { - e := attrib.Close() - if err == nil { - err = e - } - }() - } - _, err = attrib.File.Seek(0, io.SeekStart) - if err != nil { - return "", err - } - data, err := ioutil.ReadAll(attrib.File) - if err != nil { - return "", err - } - return string(data), nil -} - -func (attrib *fileObject) Write(value string) (err error) { - if attrib.File == nil { - err = attrib.OpenWO() - if err != nil { - return - } - defer func() { - e := attrib.Close() - if err == nil { - err = e - } - }() - } - _, err = attrib.File.Seek(0, io.SeekStart) - if err != nil { - return err - } - _, err = attrib.File.WriteString(value) - return err -} - -func (attrib *fileObject) ReadInt() (value int, err error) { - s, err := attrib.Read() - if err != nil { - return 0, err - } - s = strings.Trim(s, "\n") - value, err = strconv.Atoi(s) - if err != nil { - return 0, err - } - - return value, err -} - -func (attrib *fileObject) WriteInt(value int) (err error) { - return attrib.Write(strconv.Itoa(value)) -} - -func lsFilesWithPrefix(dir, filePrefix string, ignoreDir bool) ([]string, error) { - var desiredFiles []string - - f, err := os.Open(dir) - if err != nil { - return nil, err - } - defer f.Close() - fileInfos, err := f.Readdir(-1) - if err != nil { - return nil, err - } - - for i := range fileInfos { - if ignoreDir && fileInfos[i].IsDir() { - continue - } - - if filePrefix == "" || - strings.Contains(fileInfos[i].Name(), filePrefix) { - desiredFiles = append(desiredFiles, fileInfos[i].Name()) - } - } - return desiredFiles, nil -} - -func dirExists(dirname string) bool { - info, err := os.Stat(dirname) - return err == nil && info.IsDir() -} - -func fileExists(dirname string) bool { - info, err := os.Stat(dirname) - return err == nil && !info.IsDir() -} diff --git a/vendor/github.com/Mellanox/sriovnet/mofed_ib_helper.go b/vendor/github.com/Mellanox/sriovnet/mofed_ib_helper.go deleted file mode 100644 index 0e99e4191..000000000 --- a/vendor/github.com/Mellanox/sriovnet/mofed_ib_helper.go +++ /dev/null @@ -1,57 +0,0 @@ -package sriovnet - -import ( - "net" - "path/filepath" - "strconv" -) - -const ( - ibSriovCfgDir = "sriov" - ibSriovNodeFile = "node" - ibSriovPortFile = "port" - ibSriovPortAdminFile = "policy" - ibSriovPortAdminStateFollow = "Follow" -) - -func ibGetPortAdminState(pfNetdevName string, vfIndex int) (string, error) { - path := filepath.Join( - NetSysDir, pfNetdevName, pcidevPrefix, ibSriovCfgDir, strconv.Itoa(vfIndex), ibSriovPortAdminFile) - adminStateFile := fileObject{ - Path: path, - } - - state, err := adminStateFile.Read() - if err != nil { - return "", err - } - return state, nil -} - -func ibSetPortAdminState(pfNetdevName string, vfIndex int, newState string) error { - path := filepath.Join( - NetSysDir, pfNetdevName, pcidevPrefix, ibSriovCfgDir, strconv.Itoa(vfIndex), ibSriovPortAdminFile) - adminStateFile := fileObject{ - Path: path, - } - - return adminStateFile.Write(newState) -} - -func ibSetNodeGUID(pfNetdevName string, vfIndex int, guid net.HardwareAddr) error { - path := filepath.Join(NetSysDir, pfNetdevName, pcidevPrefix, ibSriovCfgDir, strconv.Itoa(vfIndex), ibSriovNodeFile) - nodeGUIDFile := fileObject{ - Path: path, - } - kernelGUIDFormat := guid.String() - return nodeGUIDFile.Write(kernelGUIDFormat) -} - -func ibSetPortGUID(pfNetdevName string, vfIndex int, guid net.HardwareAddr) error { - path := filepath.Join(NetSysDir, pfNetdevName, pcidevPrefix, ibSriovCfgDir, strconv.Itoa(vfIndex), ibSriovPortFile) - portGUIDFile := fileObject{ - Path: path, - } - kernelGUIDFormat := guid.String() - return portGUIDFile.Write(kernelGUIDFormat) -} diff --git a/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/defaultfs.go b/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/defaultfs.go deleted file mode 100644 index c795caf51..000000000 --- a/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/defaultfs.go +++ /dev/null @@ -1,119 +0,0 @@ -package filesystem - -import ( - "io/ioutil" - "os" - "path/filepath" - "time" -) - -// DefaultFs implements Filesystem using same-named functions from "os" and "io/ioutil" -type DefaultFs struct{} - -// Stat via os.Stat -func (DefaultFs) Stat(name string) (os.FileInfo, error) { - return os.Stat(name) -} - -// Create via os.Create -func (DefaultFs) Create(name string) (File, error) { - file, err := os.Create(name) - if err != nil { - return nil, err - } - return &defaultFile{file}, nil -} - -// Rename via os.Rename -func (DefaultFs) Rename(oldpath, newpath string) error { - return os.Rename(oldpath, newpath) -} - -// MkdirAll via os.MkdirAll -func (DefaultFs) MkdirAll(path string, perm os.FileMode) error { - return os.MkdirAll(path, perm) -} - -// Chtimes via os.Chtimes -func (DefaultFs) Chtimes(name string, atime, mtime time.Time) error { - return os.Chtimes(name, atime, mtime) -} - -// RemoveAll via os.RemoveAll -func (DefaultFs) RemoveAll(path string) error { - return os.RemoveAll(path) -} - -// Remove via os.RemoveAll -func (DefaultFs) Remove(name string) error { - return os.Remove(name) -} - -// Readlink via os.Readlink -func (DefaultFs) Readlink(name string) (string, error) { - return os.Readlink(name) -} - -// Symlink via os.Symlink -func (DefaultFs) Symlink(oldname, newname string) error { - return os.Symlink(oldname, newname) -} - -// ReadFile via ioutil.ReadFile -func (DefaultFs) ReadFile(filename string) ([]byte, error) { - return ioutil.ReadFile(filename) -} - -// TempDir via ioutil.TempDir -func (DefaultFs) TempDir(dir, prefix string) (string, error) { - return ioutil.TempDir(dir, prefix) -} - -// TempFile via ioutil.TempFile -func (DefaultFs) TempFile(dir, prefix string) (File, error) { - file, err := ioutil.TempFile(dir, prefix) - if err != nil { - return nil, err - } - return &defaultFile{file}, nil -} - -// ReadDir via ioutil.ReadDir -func (DefaultFs) ReadDir(dirname string) ([]os.FileInfo, error) { - return ioutil.ReadDir(dirname) -} - -// Walk via filepath.Walk -func (DefaultFs) Walk(root string, walkFn filepath.WalkFunc) error { - return filepath.Walk(root, walkFn) -} - -// WriteFile via ioutil.Writefile -func (DefaultFs) WriteFile(filename string, data []byte, perm os.FileMode) error { - return ioutil.WriteFile(filename, data, perm) -} - -// defaultFile implements File using same-named functions from "os" -type defaultFile struct { - file *os.File -} - -// Name via os.File.Name -func (file *defaultFile) Name() string { - return file.file.Name() -} - -// Write via os.File.Write -func (file *defaultFile) Write(b []byte) (n int, err error) { - return file.file.Write(b) -} - -// Sync via os.File.Sync -func (file *defaultFile) Sync() error { - return file.file.Sync() -} - -// Close via os.File.Close -func (file *defaultFile) Close() error { - return file.file.Close() -} diff --git a/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/fakefs.go b/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/fakefs.go deleted file mode 100644 index 05e6a4ca9..000000000 --- a/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/fakefs.go +++ /dev/null @@ -1,151 +0,0 @@ -//nolint:gomnd -package filesystem - -import ( - "fmt" - "os" - "path/filepath" - "time" - - "github.com/spf13/afero" -) - -// FakeFs is implemented in terms of afero -type FakeFs struct { - a afero.Afero -} - -// NewFakeFs returns a fake Filesystem that exists at fakeFsRoot as its base path, useful for unit tests. -// Returns: Filesystem interface, teardown method (cleanup of provided root path) and error. -// teardown method should be called at the end of each test to ensure environment is left clean. -func NewFakeFs(fakeFsRoot string) (Filesystem, func(), error) { - _, err := os.Stat(fakeFsRoot) - // if fakeFsRoot dir exists remove it. - if err == nil { - err = os.RemoveAll(fakeFsRoot) - if err != nil { - return nil, nil, fmt.Errorf("failed to cleanup fake root dir %s. %s", fakeFsRoot, err) - } - } else if !os.IsNotExist(err) { - return nil, nil, fmt.Errorf("failed to lstat fake root dir %s. %s", fakeFsRoot, err) - } - - // create fakeFsRoot dir - if err = os.MkdirAll(fakeFsRoot, os.FileMode(0755)); err != nil { - return nil, nil, fmt.Errorf("failed to create fake root dir: %s. %s", fakeFsRoot, err) - } - - return &FakeFs{a: afero.Afero{Fs: afero.NewBasePathFs(afero.NewOsFs(), fakeFsRoot)}}, - func() { - os.RemoveAll(fakeFsRoot) - }, - nil -} - -// Stat via afero.Fs.Stat -func (fs *FakeFs) Stat(name string) (os.FileInfo, error) { - return fs.a.Fs.Stat(name) -} - -// Create via afero.Fs.Create -func (fs *FakeFs) Create(name string) (File, error) { - file, err := fs.a.Fs.Create(name) - if err != nil { - return nil, err - } - return &fakeFile{file}, nil -} - -// Rename via afero.Fs.Rename -func (fs *FakeFs) Rename(oldpath, newpath string) error { - return fs.a.Fs.Rename(oldpath, newpath) -} - -// MkdirAll via afero.Fs.MkdirAll -func (fs *FakeFs) MkdirAll(path string, perm os.FileMode) error { - return fs.a.Fs.MkdirAll(path, perm) -} - -// Chtimes via afero.Fs.Chtimes -func (fs *FakeFs) Chtimes(name string, atime, mtime time.Time) error { - return fs.a.Fs.Chtimes(name, atime, mtime) -} - -// ReadFile via afero.ReadFile -func (fs *FakeFs) ReadFile(filename string) ([]byte, error) { - return fs.a.ReadFile(filename) -} - -// WriteFile via afero.WriteFile -func (fs *FakeFs) WriteFile(filename string, data []byte, perm os.FileMode) error { - return fs.a.WriteFile(filename, data, perm) -} - -// TempDir via afero.TempDir -func (fs *FakeFs) TempDir(dir, prefix string) (string, error) { - return fs.a.TempDir(dir, prefix) -} - -// TempFile via afero.TempFile -func (fs *FakeFs) TempFile(dir, prefix string) (File, error) { - file, err := fs.a.TempFile(dir, prefix) - if err != nil { - return nil, err - } - return &fakeFile{file}, nil -} - -// ReadDir via afero.ReadDir -func (fs *FakeFs) ReadDir(dirname string) ([]os.FileInfo, error) { - return fs.a.ReadDir(dirname) -} - -// Walk via afero.Walk -func (fs *FakeFs) Walk(root string, walkFn filepath.WalkFunc) error { - return fs.a.Walk(root, walkFn) -} - -// RemoveAll via afero.RemoveAll -func (fs *FakeFs) RemoveAll(path string) error { - return fs.a.RemoveAll(path) -} - -// Remove via afero.Remove -func (fs *FakeFs) Remove(name string) error { - return fs.a.Remove(name) -} - -// Readlink via afero.ReadlinkIfPossible -func (fs *FakeFs) Readlink(name string) (string, error) { - return fs.a.Fs.(afero.Symlinker).ReadlinkIfPossible(name) -} - -// Symlink via afero.FS.(Symlinker).SymlinkIfPossible -func (fs *FakeFs) Symlink(oldname, newname string) error { - return fs.a.Fs.(afero.Symlinker).SymlinkIfPossible(oldname, newname) -} - -// fakeFile implements File; for use with FakeFs -type fakeFile struct { - file afero.File -} - -// Name via afero.File.Name -func (file *fakeFile) Name() string { - return file.file.Name() -} - -// Write via afero.File.Write -func (file *fakeFile) Write(b []byte) (n int, err error) { - return file.file.Write(b) -} - -// Sync via afero.File.Sync -func (file *fakeFile) Sync() error { - return file.file.Sync() -} - -// Close via afero.File.Close -func (file *fakeFile) Close() error { - return file.file.Close() -} diff --git a/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/filesystem.go b/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/filesystem.go deleted file mode 100644 index 99073b3cc..000000000 --- a/vendor/github.com/Mellanox/sriovnet/pkg/utils/filesystem/filesystem.go +++ /dev/null @@ -1,41 +0,0 @@ -package filesystem - -import ( - "os" - "path/filepath" - "time" -) - -var Fs Filesystem = DefaultFs{} - -// Filesystem is an interface that we can use to mock various filesystem operations -type Filesystem interface { - // from "os" - Stat(name string) (os.FileInfo, error) - Create(name string) (File, error) - Rename(oldpath, newpath string) error - MkdirAll(path string, perm os.FileMode) error - Chtimes(name string, atime time.Time, mtime time.Time) error - RemoveAll(path string) error - Remove(name string) error - Readlink(name string) (string, error) - Symlink(oldname, newname string) error - - // from "io/ioutil" - ReadFile(filename string) ([]byte, error) - WriteFile(filename string, data []byte, perm os.FileMode) error - TempDir(dir, prefix string) (string, error) - TempFile(dir, prefix string) (File, error) - ReadDir(dirname string) ([]os.FileInfo, error) - Walk(root string, walkFn filepath.WalkFunc) error -} - -// File is an interface that we can use to mock various filesystem operations typically -// accessed through the File object from the "os" package -type File interface { - // for now, the only os.File methods used are those below, add more as necessary - Name() string - Write(b []byte) (n int, err error) - Sync() error - Close() error -} diff --git a/vendor/github.com/Mellanox/sriovnet/pkg/utils/netlinkops/netlinkops.go b/vendor/github.com/Mellanox/sriovnet/pkg/utils/netlinkops/netlinkops.go deleted file mode 100644 index ce458a315..000000000 --- a/vendor/github.com/Mellanox/sriovnet/pkg/utils/netlinkops/netlinkops.go +++ /dev/null @@ -1,114 +0,0 @@ -package netlinkops - -import ( - "fmt" - "net" - - "github.com/vishvananda/netlink" -) - -var nlOpsImpl NetlinkOps - -// NetlinkOps is an interface wrapping netlink to be used by sriovnet -type NetlinkOps interface { - // LinkByName gets link by netdev name - LinkByName(name string) (netlink.Link, error) - // LinkSetUp sets Link state to up - LinkSetUp(link netlink.Link) error - // LinkSetVfHardwareAddr sets VF hardware address - LinkSetVfHardwareAddr(link netlink.Link, vf int, hwaddr net.HardwareAddr) error - // LinkSetVfVlan sets VF vlan - LinkSetVfVlan(link netlink.Link, vf, vlan int) error - // LinkSetVfNodeGUID sets VF Node GUID - LinkSetVfNodeGUID(link netlink.Link, vf int, nodeguid net.HardwareAddr) error - // LinkSetVfPortGUID sets VF Port GUID - LinkSetVfPortGUID(link netlink.Link, vf int, portguid net.HardwareAddr) error - // LinkSetVfTrust sets VF trust for the given VF - LinkSetVfTrust(link netlink.Link, vf int, state bool) error - // LinkSetVfSpoofchk sets VF spoofchk for the given VF - LinkSetVfSpoofchk(link netlink.Link, vf int, check bool) error - // DevLinkGetAllPortList gets all devlink ports - DevLinkGetAllPortList() ([]*netlink.DevlinkPort, error) - // DevLinkGetPortByNetdevName gets devlink port by netdev name - DevLinkGetPortByNetdevName(netdev string) (*netlink.DevlinkPort, error) -} - -// GetNetlinkOps returns NetlinkOps interface -func GetNetlinkOps() NetlinkOps { - if nlOpsImpl == nil { - nlOpsImpl = &netlinkOps{} - } - return nlOpsImpl -} - -// SetNetlinkOps sets NetlinkOps interface (to be used by unit tests) -func SetNetlinkOps(nlops NetlinkOps) { - nlOpsImpl = nlops -} - -// ResetNetlinkOps resets nlOpsImpl to nil -func ResetNetlinkOps() { - nlOpsImpl = nil -} - -type netlinkOps struct{} - -// LinkByName gets link by netdev name -func (nlo *netlinkOps) LinkByName(name string) (netlink.Link, error) { - return netlink.LinkByName(name) -} - -// LinkSetUp sets Link state to up -func (nlo *netlinkOps) LinkSetUp(link netlink.Link) error { - return netlink.LinkSetUp(link) -} - -// LinkSetVfHardwareAddr sets VF hardware address -func (nlo *netlinkOps) LinkSetVfHardwareAddr(link netlink.Link, vf int, hwaddr net.HardwareAddr) error { - return netlink.LinkSetVfHardwareAddr(link, vf, hwaddr) -} - -// LinkSetVfVlan sets VF vlan -func (nlo *netlinkOps) LinkSetVfVlan(link netlink.Link, vf, vlan int) error { - return netlink.LinkSetVfVlan(link, vf, vlan) -} - -// LinkSetVfNodeGUID sets VF Node GUID -func (nlo *netlinkOps) LinkSetVfNodeGUID(link netlink.Link, vf int, nodeguid net.HardwareAddr) error { - return netlink.LinkSetVfNodeGUID(link, vf, nodeguid) -} - -// LinkSetVfPortGUID sets VF Port GUID -func (nlo *netlinkOps) LinkSetVfPortGUID(link netlink.Link, vf int, portguid net.HardwareAddr) error { - return netlink.LinkSetVfPortGUID(link, vf, portguid) -} - -// LinkSetVfTrust sets VF trust for the given VF -func (nlo *netlinkOps) LinkSetVfTrust(link netlink.Link, vf int, state bool) error { - return netlink.LinkSetVfTrust(link, vf, state) -} - -// LinkSetVfSpoofchk sets VF spoofchk for the given VF -func (nlo *netlinkOps) LinkSetVfSpoofchk(link netlink.Link, vf int, check bool) error { - return netlink.LinkSetVfSpoofchk(link, vf, check) -} - -// DevLinkGetAllPortList gets all devlink ports -func (nlo *netlinkOps) DevLinkGetAllPortList() ([]*netlink.DevlinkPort, error) { - return netlink.DevLinkGetAllPortList() -} - -// DevLinkGetPortByNetdevName gets devlink port by netdev name -func (nlo *netlinkOps) DevLinkGetPortByNetdevName(netdev string) (*netlink.DevlinkPort, error) { - ports, err := netlink.DevLinkGetAllPortList() - if err != nil { - return nil, err - } - - for _, port := range ports { - if netdev == port.NetdeviceName { - return port, nil - } - } - return nil, fmt.Errorf("failed to get devlink port for netdev %s", netdev) -} diff --git a/vendor/github.com/Mellanox/sriovnet/sriovnet.go b/vendor/github.com/Mellanox/sriovnet/sriovnet.go deleted file mode 100644 index 579783321..000000000 --- a/vendor/github.com/Mellanox/sriovnet/sriovnet.go +++ /dev/null @@ -1,466 +0,0 @@ -package sriovnet - -import ( - "fmt" - "log" - "net" - "os" - "path" - "path/filepath" - "regexp" - "strconv" - "strings" - - "github.com/google/uuid" - "github.com/vishvananda/netlink" - - utilfs "github.com/Mellanox/sriovnet/pkg/utils/filesystem" - "github.com/Mellanox/sriovnet/pkg/utils/netlinkops" -) - -const ( - // Used locally - etherEncapType = "ether" - ibEncapType = "infiniband" -) - -var virtFnRe = regexp.MustCompile(`virtfn(\d+)`) - -type VfObj struct { - Index int - PciAddress string - Bound bool - Allocated bool -} - -type PfNetdevHandle struct { - PfNetdevName string - pfLinkHandle netlink.Link - - List []*VfObj -} - -func SetPFLinkUp(pfNetdevName string) error { - handle, err := netlinkops.GetNetlinkOps().LinkByName(pfNetdevName) - if err != nil { - return err - } - - return netlinkops.GetNetlinkOps().LinkSetUp(handle) -} - -func IsVfPciVfioBound(pciAddr string) bool { - driverLink := filepath.Join(PciSysDir, pciAddr, "driver") - driverPath, err := utilfs.Fs.Readlink(driverLink) - if err != nil { - return false - } - driverName := filepath.Base(driverPath) - return driverName == "vfio-pci" -} - -func IsSriovSupported(netdevName string) bool { - maxvfs, err := getMaxVfCount(netdevName) - if maxvfs == 0 || err != nil { - return false - } - return true -} - -func IsSriovEnabled(netdevName string) bool { - curvfs, err := getCurrentVfCount(netdevName) - if curvfs == 0 || err != nil { - return false - } - return true -} - -func EnableSriov(pfNetdevName string) error { - var maxVfCount int - var err error - - devDirName := netDevDeviceDir(pfNetdevName) - - devExist := dirExists(devDirName) - if !devExist { - return fmt.Errorf("device %s not found", pfNetdevName) - } - - maxVfCount, err = getMaxVfCount(pfNetdevName) - if err != nil { - log.Println("Fail to read max vf count of PF", pfNetdevName) - return err - } - - if maxVfCount == 0 { - return fmt.Errorf("sriov unsupported for device: %s", pfNetdevName) - } - - curVfCount, err2 := getCurrentVfCount(pfNetdevName) - if err2 != nil { - log.Println("Fail to read current vf count of PF", pfNetdevName) - return err - } - if curVfCount == 0 { - return setMaxVfCount(pfNetdevName, maxVfCount) - } - return nil -} - -func DisableSriov(pfNetdevName string) error { - devDirName := netDevDeviceDir(pfNetdevName) - - devExist := dirExists(devDirName) - if !devExist { - return fmt.Errorf("device %s not found", pfNetdevName) - } - - return setMaxVfCount(pfNetdevName, 0) -} - -func GetPfNetdevHandle(pfNetdevName string) (*PfNetdevHandle, error) { - pfLinkHandle, err := netlinkops.GetNetlinkOps().LinkByName(pfNetdevName) - if err != nil { - return nil, err - } - - handle := PfNetdevHandle{ - PfNetdevName: pfNetdevName, - pfLinkHandle: pfLinkHandle, - } - - list, err := GetVfPciDevList(pfNetdevName) - if err != nil { - return nil, err - } - - for _, vfDir := range list { - vfIndexStr := strings.TrimPrefix(vfDir, netDevVfDevicePrefix) - vfIndex, _ := strconv.Atoi(vfIndexStr) - vfNetdevName := vfNetdevNameFromParent(pfNetdevName, vfIndex) - pciAddress, err := vfPCIDevNameFromVfIndex(pfNetdevName, vfIndex) - if err != nil { - log.Printf("Failed to read PCI Address for VF %v from PF %v: %v\n", - vfNetdevName, pfNetdevName, err) - continue - } - vfObj := VfObj{ - Index: vfIndex, - PciAddress: pciAddress, - } - if vfNetdevName != "" { - vfObj.Bound = true - } else { - vfObj.Bound = false - } - vfObj.Allocated = false - handle.List = append(handle.List, &vfObj) - } - return &handle, nil -} - -func UnbindVf(handle *PfNetdevHandle, vf *VfObj) error { - cmdFile := filepath.Join(NetSysDir, handle.PfNetdevName, netdevDriverDir, netdevUnbindFile) - cmdFileObj := fileObject{ - Path: cmdFile, - } - err := cmdFileObj.Write(vf.PciAddress) - if err != nil { - vf.Bound = false - } - return err -} - -func BindVf(handle *PfNetdevHandle, vf *VfObj) error { - cmdFile := filepath.Join(NetSysDir, handle.PfNetdevName, netdevDriverDir, netdevBindFile) - cmdFileObj := fileObject{ - Path: cmdFile, - } - err := cmdFileObj.Write(vf.PciAddress) - if err != nil { - vf.Bound = true - } - return err -} - -func GetVfDefaultMacAddr(vfNetdevName string) (string, error) { - ethHandle, err1 := netlinkops.GetNetlinkOps().LinkByName(vfNetdevName) - if err1 != nil { - return "", err1 - } - - ethAttr := ethHandle.Attrs() - return ethAttr.HardwareAddr.String(), nil -} - -func SetVfDefaultMacAddress(handle *PfNetdevHandle, vf *VfObj) error { - netdevName := vfNetdevNameFromParent(handle.PfNetdevName, vf.Index) - ethHandle, err1 := netlinkops.GetNetlinkOps().LinkByName(netdevName) - if err1 != nil { - return err1 - } - ethAttr := ethHandle.Attrs() - return netlinkops.GetNetlinkOps().LinkSetVfHardwareAddr(handle.pfLinkHandle, vf.Index, ethAttr.HardwareAddr) -} - -func SetVfVlan(handle *PfNetdevHandle, vf *VfObj, vlan int) error { - return netlinkops.GetNetlinkOps().LinkSetVfVlan(handle.pfLinkHandle, vf.Index, vlan) -} - -func setVfNodeGUID(handle *PfNetdevHandle, vf *VfObj, guid []byte) error { - var err error - - nodeGUIDHwAddr := net.HardwareAddr(guid) - - err = ibSetNodeGUID(handle.PfNetdevName, vf.Index, nodeGUIDHwAddr) - if err == nil { - return nil - } - err = netlinkops.GetNetlinkOps().LinkSetVfNodeGUID(handle.pfLinkHandle, vf.Index, guid) - return err -} - -func setVfPortGUID(handle *PfNetdevHandle, vf *VfObj, guid []byte) error { - var err error - - portGUIDHwAddr := net.HardwareAddr(guid) - - err = ibSetPortGUID(handle.PfNetdevName, vf.Index, portGUIDHwAddr) - if err == nil { - return nil - } - err = netlinkops.GetNetlinkOps().LinkSetVfPortGUID(handle.pfLinkHandle, vf.Index, guid) - return err -} - -func SetVfDefaultGUID(handle *PfNetdevHandle, vf *VfObj) error { - randUUID, err := uuid.NewRandom() - if err != nil { - return err - } - guid := randUUID[0:8] - guid[7] = byte(vf.Index) - - err = setVfNodeGUID(handle, vf, guid) - if err != nil { - return err - } - - err = setVfPortGUID(handle, vf, guid) - return err -} - -func SetVfPrivileged(handle *PfNetdevHandle, vf *VfObj, privileged bool) error { - var spoofChk bool - var trusted bool - - ethAttr := handle.pfLinkHandle.Attrs() - if ethAttr.EncapType != etherEncapType { - return nil - } - // Only ether type is supported - if privileged { - spoofChk = false - trusted = true - } else { - spoofChk = true - trusted = false - } - - /* do not check for error status as older kernels doesn't - * have support for it. - * golangci-lint complains on missing error check. ignore it - * with nolint comment until we update the code to ignore ENOTSUP error - */ - netlinkops.GetNetlinkOps().LinkSetVfTrust(handle.pfLinkHandle, vf.Index, trusted) //nolint - netlinkops.GetNetlinkOps().LinkSetVfSpoofchk(handle.pfLinkHandle, vf.Index, spoofChk) //nolint - return nil -} - -func setDefaultHwAddr(handle *PfNetdevHandle, vf *VfObj) error { - var err error - - ethAttr := handle.pfLinkHandle.Attrs() - if ethAttr.EncapType == etherEncapType { - err = SetVfDefaultMacAddress(handle, vf) - } else if ethAttr.EncapType == ibEncapType { - err = SetVfDefaultGUID(handle, vf) - } - return err -} - -func setPortAdminState(handle *PfNetdevHandle, vf *VfObj) error { - ethAttr := handle.pfLinkHandle.Attrs() - if ethAttr.EncapType == ibEncapType { - state, err2 := ibGetPortAdminState(handle.PfNetdevName, vf.Index) - // Ignore the error where this file is not available - if err2 != nil { - return nil - } - log.Printf("Admin state = %v", state) - err2 = ibSetPortAdminState(handle.PfNetdevName, vf.Index, ibSriovPortAdminStateFollow) - if err2 != nil { - // If file exist, we must be able to write - log.Printf("Admin state setting error = %v", err2) - return err2 - } - } - return nil -} - -func ConfigVfs(handle *PfNetdevHandle, privileged bool) error { - var err error - - for _, vf := range handle.List { - log.Printf("vf = %v\n", vf) - err = setPortAdminState(handle, vf) - if err != nil { - break - } - // skip VFs in another namespace - netdevName := vfNetdevNameFromParent(handle.PfNetdevName, vf.Index) - if _, err = netlinkops.GetNetlinkOps().LinkByName(netdevName); err != nil { - continue - } - err = setDefaultHwAddr(handle, vf) - if err != nil { - break - } - _ = SetVfPrivileged(handle, vf, privileged) - } - if err != nil { - return err - } - for _, vf := range handle.List { - if !vf.Bound { - continue - } - - err = UnbindVf(handle, vf) - if err != nil { - log.Printf("Fail to unbind err=%v\n", err) - break - } - - err = BindVf(handle, vf) - if err != nil { - log.Printf("Fail to bind err=%v\n", err) - break - } - log.Printf("vf = %v unbind/bind completed", vf) - } - return nil -} - -func AllocateVf(handle *PfNetdevHandle) (*VfObj, error) { - for _, vf := range handle.List { - if vf.Allocated { - continue - } - vf.Allocated = true - log.Printf("Allocated vf = %v\n", *vf) - return vf, nil - } - return nil, fmt.Errorf("all Vfs for %v are allocated", handle.PfNetdevName) -} - -func AllocateVfByMacAddress(handle *PfNetdevHandle, vfMacAddress string) (*VfObj, error) { - for _, vf := range handle.List { - if vf.Allocated { - continue - } - - netdevName := vfNetdevNameFromParent(handle.PfNetdevName, vf.Index) - macAddr, _ := GetVfDefaultMacAddr(netdevName) - if macAddr != vfMacAddress { - continue - } - vf.Allocated = true - log.Printf("Allocated vf by mac = %v\n", *vf) - return vf, nil - } - return nil, fmt.Errorf("all Vfs for %v are allocated for mac address %v", - handle.PfNetdevName, vfMacAddress) -} - -func FreeVf(handle *PfNetdevHandle, vf *VfObj) { - vf.Allocated = false - log.Printf("Free vf = %v\n", *vf) -} - -func FreeVfByNetdevName(handle *PfNetdevHandle, vfIndex int) error { - vfNetdevName := fmt.Sprintf("%s%v", netDevVfDevicePrefix, vfIndex) - for _, vf := range handle.List { - netdevName := vfNetdevNameFromParent(handle.PfNetdevName, vf.Index) - if vf.Allocated && netdevName == vfNetdevName { - vf.Allocated = true - return nil - } - } - return fmt.Errorf("vf netdev %v not found", vfNetdevName) -} - -func GetVfNetdevName(handle *PfNetdevHandle, vf *VfObj) string { - return vfNetdevNameFromParent(handle.PfNetdevName, vf.Index) -} - -// GetVfIndexByPciAddress gets a VF PCI address (e.g '0000:03:00.4') and -// returns the correlate VF index. -func GetVfIndexByPciAddress(vfPciAddress string) (int, error) { - vfPath := filepath.Join(PciSysDir, vfPciAddress, "physfn", "virtfn*") - matches, err := filepath.Glob(vfPath) - if err != nil { - return -1, err - } - for _, match := range matches { - tmp, err := os.Readlink(match) - if err != nil { - continue - } - if strings.Contains(tmp, vfPciAddress) { - result := virtFnRe.FindStringSubmatch(match) - vfIndex, err := strconv.Atoi(result[1]) - if err != nil { - continue - } - return vfIndex, nil - } - } - return -1, fmt.Errorf("vf index for %s not found", vfPciAddress) -} - -// GetNetDevicesFromPci gets a PCI address (e.g '0000:03:00.1') and -// returns the correlate list of netdevices -func GetNetDevicesFromPci(pciAddress string) ([]string, error) { - pciDir := filepath.Join(PciSysDir, pciAddress, "net") - _, err := utilfs.Fs.Stat(pciDir) - if err != nil { - return nil, fmt.Errorf("cannot get a network device with pci address %v %v", pciAddress, err) - } - - netDevicesFiles, err := utilfs.Fs.ReadDir(pciDir) - if err != nil { - return nil, fmt.Errorf("failed to get network device name in %v %v", pciDir, err) - } - - netDevices := make([]string, 0, len(netDevicesFiles)) - for _, netDeviceFile := range netDevicesFiles { - netDevices = append(netDevices, strings.TrimSpace(netDeviceFile.Name())) - } - return netDevices, nil -} - -// GetPfPciFromVfPci retrieves the parent PF PCI address of the provided VF PCI address in D:B:D.f format -func GetPfPciFromVfPci(vfPciAddress string) (string, error) { - pfPath := filepath.Join(PciSysDir, vfPciAddress, "physfn") - pciDevDir, err := utilfs.Fs.Readlink(pfPath) - if err != nil { - return "", fmt.Errorf("failed to read physfn link, provided address may not be a VF. %v", err) - } - - pf := path.Base(pciDevDir) - if pf == "" { - return pf, fmt.Errorf("could not find PF PCI Address") - } - return pf, err -} diff --git a/vendor/github.com/Mellanox/sriovnet/sriovnet_helper.go b/vendor/github.com/Mellanox/sriovnet/sriovnet_helper.go deleted file mode 100644 index bee37c6c9..000000000 --- a/vendor/github.com/Mellanox/sriovnet/sriovnet_helper.go +++ /dev/null @@ -1,129 +0,0 @@ -package sriovnet - -import ( - "fmt" - "log" - "os" - "path/filepath" -) - -const ( - NetSysDir = "/sys/class/net" - PciSysDir = "/sys/bus/pci/devices" - pcidevPrefix = "device" - netdevDriverDir = "device/driver" - netdevUnbindFile = "unbind" - netdevBindFile = "bind" - - netDevMaxVfCountFile = "sriov_totalvfs" - netDevCurrentVfCountFile = "sriov_numvfs" - netDevVfDevicePrefix = "virtfn" -) - -type VfObject struct { - NetdevName string - PCIDevName string -} - -func netDevDeviceDir(netDevName string) string { - devDirName := filepath.Join(NetSysDir, netDevName, pcidevPrefix) - return devDirName -} - -func getMaxVfCount(pfNetdevName string) (int, error) { - devDirName := netDevDeviceDir(pfNetdevName) - - maxDevFile := fileObject{ - Path: filepath.Join(devDirName, netDevMaxVfCountFile), - } - - maxVfs, err := maxDevFile.ReadInt() - if err != nil { - return 0, err - } - log.Println("max_vfs = ", maxVfs) - return maxVfs, nil -} - -func setMaxVfCount(pfNetdevName string, maxVfs int) error { - devDirName := netDevDeviceDir(pfNetdevName) - - maxDevFile := fileObject{ - Path: filepath.Join(devDirName, netDevCurrentVfCountFile), - } - - return maxDevFile.WriteInt(maxVfs) -} - -func getCurrentVfCount(pfNetdevName string) (int, error) { - devDirName := netDevDeviceDir(pfNetdevName) - - maxDevFile := fileObject{ - Path: filepath.Join(devDirName, netDevCurrentVfCountFile), - } - - curVfs, err := maxDevFile.ReadInt() - if err != nil { - return 0, err - } - log.Println("cur_vfs = ", curVfs) - return curVfs, nil -} - -func vfNetdevNameFromParent(pfNetdevName string, vfIndex int) string { - devDirName := netDevDeviceDir(pfNetdevName) - vfNetdev, _ := lsFilesWithPrefix(fmt.Sprintf("%s/%s%v/net", devDirName, - netDevVfDevicePrefix, vfIndex), "", false) - if len(vfNetdev) == 0 { - return "" - } - return vfNetdev[0] -} - -func readPCIsymbolicLink(symbolicLink string) (string, error) { - pciDevDir, err := os.Readlink(symbolicLink) - //nolint:gomnd - if len(pciDevDir) <= 3 { - return "", fmt.Errorf("could not find PCI Address") - } - - return pciDevDir[3:], err -} -func vfPCIDevNameFromVfIndex(pfNetdevName string, vfIndex int) (string, error) { - symbolicLink := filepath.Join(NetSysDir, pfNetdevName, pcidevPrefix, fmt.Sprintf("%s%v", - netDevVfDevicePrefix, vfIndex)) - pciAddress, err := readPCIsymbolicLink(symbolicLink) - if err != nil { - err = fmt.Errorf("%v for VF %s%v of PF %s", err, - netDevVfDevicePrefix, vfIndex, pfNetdevName) - } - return pciAddress, err -} - -func getPCIFromDeviceName(netdevName string) (string, error) { - symbolicLink := filepath.Join(NetSysDir, netdevName, pcidevPrefix) - pciAddress, err := readPCIsymbolicLink(symbolicLink) - if err != nil { - err = fmt.Errorf("%v for netdevice %s", err, netdevName) - } - return pciAddress, err -} - -func GetVfPciDevList(pfNetdevName string) ([]string, error) { - var i int - devDirName := netDevDeviceDir(pfNetdevName) - - virtFnDirs, err := lsFilesWithPrefix(devDirName, netDevVfDevicePrefix, true) - - if err != nil { - return nil, err - } - - i = 0 - vfDirList := make([]string, 0, len(virtFnDirs)) - for _, vfDir := range virtFnDirs { - vfDirList = append(vfDirList, vfDir) - i++ - } - return vfDirList, nil -} diff --git a/vendor/github.com/Mellanox/sriovnet/sriovnet_switchdev.go b/vendor/github.com/Mellanox/sriovnet/sriovnet_switchdev.go deleted file mode 100644 index 18904aeb7..000000000 --- a/vendor/github.com/Mellanox/sriovnet/sriovnet_switchdev.go +++ /dev/null @@ -1,391 +0,0 @@ -package sriovnet - -import ( - "errors" - "fmt" - "net" - "os" - "path/filepath" - "regexp" - "strconv" - "strings" - - utilfs "github.com/Mellanox/sriovnet/pkg/utils/filesystem" - "github.com/Mellanox/sriovnet/pkg/utils/netlinkops" -) - -const ( - netdevPhysSwitchID = "phys_switch_id" - netdevPhysPortName = "phys_port_name" -) - -type PortFlavour uint16 - -// Keep things consistent with netlink lib constants -// nolint:golint,stylecheck -const ( - PORT_FLAVOUR_PHYSICAL = iota - PORT_FLAVOUR_CPU - PORT_FLAVOUR_DSA - PORT_FLAVOUR_PCI_PF - PORT_FLAVOUR_PCI_VF - PORT_FLAVOUR_VIRTUAL - PORT_FLAVOUR_UNUSED - PORT_FLAVOUR_PCI_SF - PORT_FLAVOUR_UNKNOWN = 0xffff -) - -// Regex that matches on the physical/upling port name -var physPortRepRegex = regexp.MustCompile(`^p(\d+)$`) - -// Regex that matches on PF representor port name. These ports exists on DPUs. -var pfPortRepRegex = regexp.MustCompile(`^(?:c\d+)?pf(\d+)$`) - -// Regex that matches on VF representor port name -var vfPortRepRegex = regexp.MustCompile(`^(?:c\d+)?pf(\d+)vf(\d+)$`) - -func parsePortName(physPortName string) (pfRepIndex, vfRepIndex int, err error) { - pfRepIndex = -1 - vfRepIndex = -1 - - // old kernel syntax of phys_port_name is vf index - physPortName = strings.TrimSpace(physPortName) - physPortNameInt, err := strconv.Atoi(physPortName) - if err == nil { - vfRepIndex = physPortNameInt - } else { - // new kernel syntax of phys_port_name [cZ]pfXVfY - matches := vfPortRepRegex.FindStringSubmatch(physPortName) - //nolint:gomnd - if len(matches) != 3 { - err = fmt.Errorf("failed to parse physPortName %s", physPortName) - } else { - pfRepIndex, err = strconv.Atoi(matches[1]) - if err == nil { - vfRepIndex, err = strconv.Atoi(matches[2]) - } - } - } - return pfRepIndex, vfRepIndex, err -} - -func isSwitchdev(netdevice string) bool { - swIDFile := filepath.Join(NetSysDir, netdevice, netdevPhysSwitchID) - physSwitchID, err := utilfs.Fs.ReadFile(swIDFile) - if err != nil { - return false - } - if physSwitchID != nil && string(physSwitchID) != "" { - return true - } - return false -} - -// GetUplinkRepresentor gets a VF or PF PCI address (e.g '0000:03:00.4') and -// returns the uplink represntor netdev name for that VF or PF. -func GetUplinkRepresentor(pciAddress string) (string, error) { - devicePath := filepath.Join(PciSysDir, pciAddress, "physfn", "net") - if _, err := utilfs.Fs.Stat(devicePath); errors.Is(err, os.ErrNotExist) { - // If physfn symlink to the parent PF doesn't exist, use the current device's dir - devicePath = filepath.Join(PciSysDir, pciAddress, "net") - } - - devices, err := utilfs.Fs.ReadDir(devicePath) - if err != nil { - return "", fmt.Errorf("failed to lookup %s: %v", pciAddress, err) - } - for _, device := range devices { - if isSwitchdev(device.Name()) { - // Try to get the phys port name, if not exists then fallback to check without it - // phys_port_name should be in formant p e.g p0,p1,p2 ...etc. - if devicePhysPortName, err := getNetDevPhysPortName(device.Name()); err == nil { - if !physPortRepRegex.MatchString(devicePhysPortName) { - continue - } - } - - return device.Name(), nil - } - } - return "", fmt.Errorf("uplink for %s not found", pciAddress) -} - -func GetVfRepresentor(uplink string, vfIndex int) (string, error) { - swIDFile := filepath.Join(NetSysDir, uplink, netdevPhysSwitchID) - physSwitchID, err := utilfs.Fs.ReadFile(swIDFile) - if err != nil || string(physSwitchID) == "" { - return "", fmt.Errorf("cant get uplink %s switch id", uplink) - } - - pfSubsystemPath := filepath.Join(NetSysDir, uplink, "subsystem") - devices, err := utilfs.Fs.ReadDir(pfSubsystemPath) - if err != nil { - return "", err - } - for _, device := range devices { - devicePath := filepath.Join(NetSysDir, device.Name()) - deviceSwIDFile := filepath.Join(devicePath, netdevPhysSwitchID) - deviceSwID, err := utilfs.Fs.ReadFile(deviceSwIDFile) - if err != nil || string(deviceSwID) != string(physSwitchID) { - continue - } - physPortNameStr, err := getNetDevPhysPortName(device.Name()) - if err != nil { - continue - } - pfRepIndex, vfRepIndex, _ := parsePortName(physPortNameStr) - if pfRepIndex != -1 { - pfPCIAddress, err := getPCIFromDeviceName(uplink) - if err != nil { - continue - } - PCIFuncAddress, err := strconv.Atoi(string((pfPCIAddress[len(pfPCIAddress)-1]))) - if pfRepIndex != PCIFuncAddress || err != nil { - continue - } - } - // At this point we're confident we have a representor. - if vfRepIndex == vfIndex { - return device.Name(), nil - } - } - return "", fmt.Errorf("failed to find VF representor for uplink %s", uplink) -} - -func getNetDevPhysPortName(netDev string) (string, error) { - devicePortNameFile := filepath.Join(NetSysDir, netDev, netdevPhysPortName) - physPortName, err := utilfs.Fs.ReadFile(devicePortNameFile) - if err != nil { - return "", err - } - return strings.TrimSpace(string(physPortName)), nil -} - -// findNetdevWithPortNameCriteria returns representor netdev that matches a criteria function on the -// physical port name -func findNetdevWithPortNameCriteria(criteria func(string) bool) (string, error) { - netdevs, err := utilfs.Fs.ReadDir(NetSysDir) - if err != nil { - return "", err - } - - for _, netdev := range netdevs { - // find matching VF representor - netdevName := netdev.Name() - - // skip non switchdev netdevs - if !isSwitchdev(netdevName) { - continue - } - - portName, err := getNetDevPhysPortName(netdevName) - if err != nil { - continue - } - - if criteria(portName) { - return netdevName, nil - } - } - return "", fmt.Errorf("no representor matched criteria") -} - -// GetVfRepresentorDPU returns VF representor on DPU for a host VF identified by pfID and vfIndex -func GetVfRepresentorDPU(pfID, vfIndex string) (string, error) { - // TODO(Adrianc): This method should change to get switchID and vfIndex as input, then common logic can - // be shared with GetVfRepresentor, backward compatibility should be preserved when this happens. - - // pfID should be 0 or 1 - if pfID != "0" && pfID != "1" { - return "", fmt.Errorf("unexpected pfID(%s). It should be 0 or 1", pfID) - } - - // vfIndex should be an unsinged integer provided as a decimal number - if _, err := strconv.ParseUint(vfIndex, 10, 32); err != nil { - return "", fmt.Errorf("unexpected vfIndex(%s). It should be an unsigned decimal number", vfIndex) - } - - // map for easy search of expected VF rep port name. - // Note: no supoport for Multi-Chassis DPUs - expectedPhysPortNames := map[string]interface{}{ - fmt.Sprintf("pf%svf%s", pfID, vfIndex): nil, - fmt.Sprintf("c1pf%svf%s", pfID, vfIndex): nil, - } - - netdev, err := findNetdevWithPortNameCriteria(func(portName string) bool { - // if phys port name == pfvf or c1pfvf we have a match - if _, ok := expectedPhysPortNames[portName]; ok { - return true - } - return false - }) - - if err != nil { - return "", fmt.Errorf("vf representor for pfID:%s, vfIndex:%s not found", pfID, vfIndex) - } - return netdev, nil -} - -// GetRepresentorPortFlavour returns the representor port flavour -// Note: this method does not support old representor names used by old kernels -// e.g and will return PORT_FLAVOUR_UNKNOWN for such cases. -func GetRepresentorPortFlavour(netdev string) (PortFlavour, error) { - if !isSwitchdev(netdev) { - return PORT_FLAVOUR_UNKNOWN, fmt.Errorf("net device %s is does not represent an eswitch port", netdev) - } - - // Attempt to get information via devlink (Kernel >= 5.9.0) - port, err := netlinkops.GetNetlinkOps().DevLinkGetPortByNetdevName(netdev) - if err == nil { - return PortFlavour(port.PortFlavour), nil - } - - // Fallback to Get PortFlavour by phys_port_name - // read phy_port_name - portName, err := getNetDevPhysPortName(netdev) - if err != nil { - return PORT_FLAVOUR_UNKNOWN, err - } - - typeToRegex := map[PortFlavour]*regexp.Regexp{ - PORT_FLAVOUR_PHYSICAL: physPortRepRegex, - PORT_FLAVOUR_PCI_PF: pfPortRepRegex, - PORT_FLAVOUR_PCI_VF: vfPortRepRegex, - } - for flavour, regex := range typeToRegex { - if regex.MatchString(portName) { - return flavour, nil - } - } - return PORT_FLAVOUR_UNKNOWN, nil -} - -// parseDPUConfigFileOutput parses the config file content of a DPU -// representor port. The format of the file is a set of : pairs as follows: -// -// ``` -// MAC : 0c:42:a1:c6:cf:7c -// MaxTxRate : 0 -// State : Follow -// ``` -func parseDPUConfigFileOutput(out string) map[string]string { - configMap := make(map[string]string) - for _, line := range strings.Split(strings.TrimSuffix(out, "\n"), "\n") { - entry := strings.SplitN(line, ":", 2) - if len(entry) != 2 { - // unexpected line format - continue - } - configMap[strings.Trim(entry[0], " \t\n")] = strings.Trim(entry[1], " \t\n") - } - return configMap -} - -// GetRepresentorPeerMacAddress returns the MAC address of the peer netdev associated with the given -// representor netdev -// Note: -// This method functionality is currently supported only on DPUs. -// Currently only netdev representors with PORT_FLAVOUR_PCI_PF are supported -func GetRepresentorPeerMacAddress(netdev string) (net.HardwareAddr, error) { - flavor, err := GetRepresentorPortFlavour(netdev) - if err != nil { - return nil, fmt.Errorf("unknown port flavour for netdev %s. %v", netdev, err) - } - if flavor == PORT_FLAVOUR_UNKNOWN { - return nil, fmt.Errorf("unknown port flavour for netdev %s", netdev) - } - if flavor != PORT_FLAVOUR_PCI_PF { - return nil, fmt.Errorf("unsupported port flavour for netdev %s", netdev) - } - - // Attempt to get information via devlink (Kernel >= 5.9.0) - port, err := netlinkops.GetNetlinkOps().DevLinkGetPortByNetdevName(netdev) - if err == nil { - if port.Fn != nil { - return port.Fn.HwAddr, nil - } - } - - // Get information via sysfs - // read phy_port_name - portName, err := getNetDevPhysPortName(netdev) - if err != nil { - return nil, err - } - // Extract port num - portNum := pfPortRepRegex.FindStringSubmatch(portName) - if len(portNum) < 2 { - return nil, fmt.Errorf("failed to extract physical port number from port name %s of netdev %s", - portName, netdev) - } - uplinkPhysPortName := "p" + portNum[1] - // Find uplink netdev for that port - // Note(adrianc): As we support only DPUs ATM we do not need to deal with netdevs from different - // eswitch (i.e different switch IDs). - uplinkNetdev, err := findNetdevWithPortNameCriteria(func(pname string) bool { return pname == uplinkPhysPortName }) - if err != nil { - return nil, fmt.Errorf("failed to find uplink port for netdev %s. %v", netdev, err) - } - // get MAC address for netdev - configPath := filepath.Join(NetSysDir, uplinkNetdev, "smart_nic", "pf", "config") - out, err := utilfs.Fs.ReadFile(configPath) - if err != nil { - return nil, fmt.Errorf("failed to read DPU config via uplink %s for %s. %v", - uplinkNetdev, netdev, err) - } - config := parseDPUConfigFileOutput(string(out)) - macStr, ok := config["MAC"] - if !ok { - return nil, fmt.Errorf("MAC address not found for %s", netdev) - } - mac, err := net.ParseMAC(macStr) - if err != nil { - return nil, fmt.Errorf("failed to parse MAC address \"%s\" for %s. %v", macStr, netdev, err) - } - return mac, nil -} - -// SetRepresentorPeerMacAddress sets the given MAC addresss of the peer netdev associated with the given -// representor netdev. -// Note: This method functionality is currently supported only for DPUs. -// Currently only netdev representors with PORT_FLAVOUR_PCI_VF are supported -func SetRepresentorPeerMacAddress(netdev string, mac net.HardwareAddr) error { - flavor, err := GetRepresentorPortFlavour(netdev) - if err != nil { - return fmt.Errorf("unknown port flavour for netdev %s. %v", netdev, err) - } - if flavor == PORT_FLAVOUR_UNKNOWN { - return fmt.Errorf("unknown port flavour for netdev %s", netdev) - } - if flavor != PORT_FLAVOUR_PCI_VF { - return fmt.Errorf("unsupported port flavour for netdev %s", netdev) - } - - physPortNameStr, err := getNetDevPhysPortName(netdev) - if err != nil { - return fmt.Errorf("failed to get phys_port_name for netdev %s: %v", netdev, err) - } - pfID, vfIndex, err := parsePortName(physPortNameStr) - if err != nil { - return fmt.Errorf("failed to get the pf and vf index for netdev %s "+ - "with phys_port_name %s: %v", netdev, physPortNameStr, err) - } - - uplinkPhysPortName := fmt.Sprintf("p%d", pfID) - uplinkNetdev, err := findNetdevWithPortNameCriteria(func(pname string) bool { return pname == uplinkPhysPortName }) - if err != nil { - return fmt.Errorf("failed to find netdev for physical port name %s. %v", uplinkPhysPortName, err) - } - vfRepName := fmt.Sprintf("vf%d", vfIndex) - sysfsVfRepMacFile := filepath.Join(NetSysDir, uplinkNetdev, "smart_nic", vfRepName, "mac") - _, err = utilfs.Fs.Stat(sysfsVfRepMacFile) - if err != nil { - return fmt.Errorf("couldn't stat VF representor's sysfs file %s: %v", sysfsVfRepMacFile, err) - } - err = utilfs.Fs.WriteFile(sysfsVfRepMacFile, []byte(mac.String()), 0) - if err != nil { - return fmt.Errorf("failed to write the MAC address %s to VF reprentor %s", - mac.String(), sysfsVfRepMacFile) - } - return nil -} diff --git a/vendor/github.com/evanphx/json-patch/v5/internal/json/decode.go b/vendor/github.com/evanphx/json-patch/v5/internal/json/decode.go new file mode 100644 index 000000000..e9bb0efe7 --- /dev/null +++ b/vendor/github.com/evanphx/json-patch/v5/internal/json/decode.go @@ -0,0 +1,1385 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Represents JSON data structure using native Go types: booleans, floats, +// strings, arrays, and maps. + +package json + +import ( + "encoding" + "encoding/base64" + "fmt" + "reflect" + "strconv" + "strings" + "sync" + "unicode" + "unicode/utf16" + "unicode/utf8" +) + +// Unmarshal parses the JSON-encoded data and stores the result +// in the value pointed to by v. If v is nil or not a pointer, +// Unmarshal returns an InvalidUnmarshalError. +// +// Unmarshal uses the inverse of the encodings that +// Marshal uses, allocating maps, slices, and pointers as necessary, +// with the following additional rules: +// +// To unmarshal JSON into a pointer, Unmarshal first handles the case of +// the JSON being the JSON literal null. In that case, Unmarshal sets +// the pointer to nil. Otherwise, Unmarshal unmarshals the JSON into +// the value pointed at by the pointer. If the pointer is nil, Unmarshal +// allocates a new value for it to point to. +// +// To unmarshal JSON into a value implementing the Unmarshaler interface, +// Unmarshal calls that value's UnmarshalJSON method, including +// when the input is a JSON null. +// Otherwise, if the value implements encoding.TextUnmarshaler +// and the input is a JSON quoted string, Unmarshal calls that value's +// UnmarshalText method with the unquoted form of the string. +// +// To unmarshal JSON into a struct, Unmarshal matches incoming object +// keys to the keys used by Marshal (either the struct field name or its tag), +// preferring an exact match but also accepting a case-insensitive match. By +// default, object keys which don't have a corresponding struct field are +// ignored (see Decoder.DisallowUnknownFields for an alternative). +// +// To unmarshal JSON into an interface value, +// Unmarshal stores one of these in the interface value: +// +// bool, for JSON booleans +// float64, for JSON numbers +// string, for JSON strings +// []interface{}, for JSON arrays +// map[string]interface{}, for JSON objects +// nil for JSON null +// +// To unmarshal a JSON array into a slice, Unmarshal resets the slice length +// to zero and then appends each element to the slice. +// As a special case, to unmarshal an empty JSON array into a slice, +// Unmarshal replaces the slice with a new empty slice. +// +// To unmarshal a JSON array into a Go array, Unmarshal decodes +// JSON array elements into corresponding Go array elements. +// If the Go array is smaller than the JSON array, +// the additional JSON array elements are discarded. +// If the JSON array is smaller than the Go array, +// the additional Go array elements are set to zero values. +// +// To unmarshal a JSON object into a map, Unmarshal first establishes a map to +// use. If the map is nil, Unmarshal allocates a new map. Otherwise Unmarshal +// reuses the existing map, keeping existing entries. Unmarshal then stores +// key-value pairs from the JSON object into the map. The map's key type must +// either be any string type, an integer, implement json.Unmarshaler, or +// implement encoding.TextUnmarshaler. +// +// If the JSON-encoded data contain a syntax error, Unmarshal returns a SyntaxError. +// +// If a JSON value is not appropriate for a given target type, +// or if a JSON number overflows the target type, Unmarshal +// skips that field and completes the unmarshaling as best it can. +// If no more serious errors are encountered, Unmarshal returns +// an UnmarshalTypeError describing the earliest such error. In any +// case, it's not guaranteed that all the remaining fields following +// the problematic one will be unmarshaled into the target object. +// +// The JSON null value unmarshals into an interface, map, pointer, or slice +// by setting that Go value to nil. Because null is often used in JSON to mean +// “not present,” unmarshaling a JSON null into any other Go type has no effect +// on the value and produces no error. +// +// When unmarshaling quoted strings, invalid UTF-8 or +// invalid UTF-16 surrogate pairs are not treated as an error. +// Instead, they are replaced by the Unicode replacement +// character U+FFFD. +func Unmarshal(data []byte, v any) error { + // Check for well-formedness. + // Avoids filling out half a data structure + // before discovering a JSON syntax error. + d := ds.Get().(*decodeState) + defer ds.Put(d) + //var d decodeState + d.useNumber = true + err := checkValid(data, &d.scan) + if err != nil { + return err + } + + d.init(data) + return d.unmarshal(v) +} + +var ds = sync.Pool{ + New: func() any { + return new(decodeState) + }, +} + +func UnmarshalWithKeys(data []byte, v any) ([]string, error) { + // Check for well-formedness. + // Avoids filling out half a data structure + // before discovering a JSON syntax error. + + d := ds.Get().(*decodeState) + defer ds.Put(d) + //var d decodeState + d.useNumber = true + err := checkValid(data, &d.scan) + if err != nil { + return nil, err + } + + d.init(data) + err = d.unmarshal(v) + if err != nil { + return nil, err + } + + return d.lastKeys, nil +} + +func UnmarshalValid(data []byte, v any) error { + // Check for well-formedness. + // Avoids filling out half a data structure + // before discovering a JSON syntax error. + d := ds.Get().(*decodeState) + defer ds.Put(d) + //var d decodeState + d.useNumber = true + + d.init(data) + return d.unmarshal(v) +} + +func UnmarshalValidWithKeys(data []byte, v any) ([]string, error) { + // Check for well-formedness. + // Avoids filling out half a data structure + // before discovering a JSON syntax error. + + d := ds.Get().(*decodeState) + defer ds.Put(d) + //var d decodeState + d.useNumber = true + + d.init(data) + err := d.unmarshal(v) + if err != nil { + return nil, err + } + + return d.lastKeys, nil +} + +// Unmarshaler is the interface implemented by types +// that can unmarshal a JSON description of themselves. +// The input can be assumed to be a valid encoding of +// a JSON value. UnmarshalJSON must copy the JSON data +// if it wishes to retain the data after returning. +// +// By convention, to approximate the behavior of Unmarshal itself, +// Unmarshalers implement UnmarshalJSON([]byte("null")) as a no-op. +type Unmarshaler interface { + UnmarshalJSON([]byte) error +} + +// An UnmarshalTypeError describes a JSON value that was +// not appropriate for a value of a specific Go type. +type UnmarshalTypeError struct { + Value string // description of JSON value - "bool", "array", "number -5" + Type reflect.Type // type of Go value it could not be assigned to + Offset int64 // error occurred after reading Offset bytes + Struct string // name of the struct type containing the field + Field string // the full path from root node to the field +} + +func (e *UnmarshalTypeError) Error() string { + if e.Struct != "" || e.Field != "" { + return "json: cannot unmarshal " + e.Value + " into Go struct field " + e.Struct + "." + e.Field + " of type " + e.Type.String() + } + return "json: cannot unmarshal " + e.Value + " into Go value of type " + e.Type.String() +} + +// An UnmarshalFieldError describes a JSON object key that +// led to an unexported (and therefore unwritable) struct field. +// +// Deprecated: No longer used; kept for compatibility. +type UnmarshalFieldError struct { + Key string + Type reflect.Type + Field reflect.StructField +} + +func (e *UnmarshalFieldError) Error() string { + return "json: cannot unmarshal object key " + strconv.Quote(e.Key) + " into unexported field " + e.Field.Name + " of type " + e.Type.String() +} + +// An InvalidUnmarshalError describes an invalid argument passed to Unmarshal. +// (The argument to Unmarshal must be a non-nil pointer.) +type InvalidUnmarshalError struct { + Type reflect.Type +} + +func (e *InvalidUnmarshalError) Error() string { + if e.Type == nil { + return "json: Unmarshal(nil)" + } + + if e.Type.Kind() != reflect.Pointer { + return "json: Unmarshal(non-pointer " + e.Type.String() + ")" + } + return "json: Unmarshal(nil " + e.Type.String() + ")" +} + +func (d *decodeState) unmarshal(v any) error { + rv := reflect.ValueOf(v) + if rv.Kind() != reflect.Pointer || rv.IsNil() { + return &InvalidUnmarshalError{reflect.TypeOf(v)} + } + + d.scan.reset() + d.scanWhile(scanSkipSpace) + // We decode rv not rv.Elem because the Unmarshaler interface + // test must be applied at the top level of the value. + err := d.value(rv) + if err != nil { + return d.addErrorContext(err) + } + return d.savedError +} + +// A Number represents a JSON number literal. +type Number string + +// String returns the literal text of the number. +func (n Number) String() string { return string(n) } + +// Float64 returns the number as a float64. +func (n Number) Float64() (float64, error) { + return strconv.ParseFloat(string(n), 64) +} + +// Int64 returns the number as an int64. +func (n Number) Int64() (int64, error) { + return strconv.ParseInt(string(n), 10, 64) +} + +// An errorContext provides context for type errors during decoding. +type errorContext struct { + Struct reflect.Type + FieldStack []string +} + +// decodeState represents the state while decoding a JSON value. +type decodeState struct { + data []byte + off int // next read offset in data + opcode int // last read result + scan scanner + errorContext *errorContext + savedError error + useNumber bool + disallowUnknownFields bool + lastKeys []string +} + +// readIndex returns the position of the last byte read. +func (d *decodeState) readIndex() int { + return d.off - 1 +} + +// phasePanicMsg is used as a panic message when we end up with something that +// shouldn't happen. It can indicate a bug in the JSON decoder, or that +// something is editing the data slice while the decoder executes. +const phasePanicMsg = "JSON decoder out of sync - data changing underfoot?" + +func (d *decodeState) init(data []byte) *decodeState { + d.data = data + d.off = 0 + d.savedError = nil + if d.errorContext != nil { + d.errorContext.Struct = nil + // Reuse the allocated space for the FieldStack slice. + d.errorContext.FieldStack = d.errorContext.FieldStack[:0] + } + return d +} + +// saveError saves the first err it is called with, +// for reporting at the end of the unmarshal. +func (d *decodeState) saveError(err error) { + if d.savedError == nil { + d.savedError = d.addErrorContext(err) + } +} + +// addErrorContext returns a new error enhanced with information from d.errorContext +func (d *decodeState) addErrorContext(err error) error { + if d.errorContext != nil && (d.errorContext.Struct != nil || len(d.errorContext.FieldStack) > 0) { + switch err := err.(type) { + case *UnmarshalTypeError: + err.Struct = d.errorContext.Struct.Name() + err.Field = strings.Join(d.errorContext.FieldStack, ".") + } + } + return err +} + +// skip scans to the end of what was started. +func (d *decodeState) skip() { + s, data, i := &d.scan, d.data, d.off + depth := len(s.parseState) + for { + op := s.step(s, data[i]) + i++ + if len(s.parseState) < depth { + d.off = i + d.opcode = op + return + } + } +} + +// scanNext processes the byte at d.data[d.off]. +func (d *decodeState) scanNext() { + if d.off < len(d.data) { + d.opcode = d.scan.step(&d.scan, d.data[d.off]) + d.off++ + } else { + d.opcode = d.scan.eof() + d.off = len(d.data) + 1 // mark processed EOF with len+1 + } +} + +// scanWhile processes bytes in d.data[d.off:] until it +// receives a scan code not equal to op. +func (d *decodeState) scanWhile(op int) { + s, data, i := &d.scan, d.data, d.off + for i < len(data) { + newOp := s.step(s, data[i]) + i++ + if newOp != op { + d.opcode = newOp + d.off = i + return + } + } + + d.off = len(data) + 1 // mark processed EOF with len+1 + d.opcode = d.scan.eof() +} + +// rescanLiteral is similar to scanWhile(scanContinue), but it specialises the +// common case where we're decoding a literal. The decoder scans the input +// twice, once for syntax errors and to check the length of the value, and the +// second to perform the decoding. +// +// Only in the second step do we use decodeState to tokenize literals, so we +// know there aren't any syntax errors. We can take advantage of that knowledge, +// and scan a literal's bytes much more quickly. +func (d *decodeState) rescanLiteral() { + data, i := d.data, d.off +Switch: + switch data[i-1] { + case '"': // string + for ; i < len(data); i++ { + switch data[i] { + case '\\': + i++ // escaped char + case '"': + i++ // tokenize the closing quote too + break Switch + } + } + case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-': // number + for ; i < len(data); i++ { + switch data[i] { + case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', + '.', 'e', 'E', '+', '-': + default: + break Switch + } + } + case 't': // true + i += len("rue") + case 'f': // false + i += len("alse") + case 'n': // null + i += len("ull") + } + if i < len(data) { + d.opcode = stateEndValue(&d.scan, data[i]) + } else { + d.opcode = scanEnd + } + d.off = i + 1 +} + +// value consumes a JSON value from d.data[d.off-1:], decoding into v, and +// reads the following byte ahead. If v is invalid, the value is discarded. +// The first byte of the value has been read already. +func (d *decodeState) value(v reflect.Value) error { + switch d.opcode { + default: + panic(phasePanicMsg) + + case scanBeginArray: + if v.IsValid() { + if err := d.array(v); err != nil { + return err + } + } else { + d.skip() + } + d.scanNext() + + case scanBeginObject: + if v.IsValid() { + if err := d.object(v); err != nil { + return err + } + } else { + d.skip() + } + d.scanNext() + + case scanBeginLiteral: + // All bytes inside literal return scanContinue op code. + start := d.readIndex() + d.rescanLiteral() + + if v.IsValid() { + if err := d.literalStore(d.data[start:d.readIndex()], v, false); err != nil { + return err + } + } + } + return nil +} + +type unquotedValue struct{} + +// valueQuoted is like value but decodes a +// quoted string literal or literal null into an interface value. +// If it finds anything other than a quoted string literal or null, +// valueQuoted returns unquotedValue{}. +func (d *decodeState) valueQuoted() any { + switch d.opcode { + default: + panic(phasePanicMsg) + + case scanBeginArray, scanBeginObject: + d.skip() + d.scanNext() + + case scanBeginLiteral: + v := d.literalInterface() + switch v.(type) { + case nil, string: + return v + } + } + return unquotedValue{} +} + +// indirect walks down v allocating pointers as needed, +// until it gets to a non-pointer. +// If it encounters an Unmarshaler, indirect stops and returns that. +// If decodingNull is true, indirect stops at the first settable pointer so it +// can be set to nil. +func indirect(v reflect.Value, decodingNull bool) (Unmarshaler, encoding.TextUnmarshaler, reflect.Value) { + // Issue #24153 indicates that it is generally not a guaranteed property + // that you may round-trip a reflect.Value by calling Value.Addr().Elem() + // and expect the value to still be settable for values derived from + // unexported embedded struct fields. + // + // The logic below effectively does this when it first addresses the value + // (to satisfy possible pointer methods) and continues to dereference + // subsequent pointers as necessary. + // + // After the first round-trip, we set v back to the original value to + // preserve the original RW flags contained in reflect.Value. + v0 := v + haveAddr := false + + // If v is a named type and is addressable, + // start with its address, so that if the type has pointer methods, + // we find them. + if v.Kind() != reflect.Pointer && v.Type().Name() != "" && v.CanAddr() { + haveAddr = true + v = v.Addr() + } + for { + // Load value from interface, but only if the result will be + // usefully addressable. + if v.Kind() == reflect.Interface && !v.IsNil() { + e := v.Elem() + if e.Kind() == reflect.Pointer && !e.IsNil() && (!decodingNull || e.Elem().Kind() == reflect.Pointer) { + haveAddr = false + v = e + continue + } + } + + if v.Kind() != reflect.Pointer { + break + } + + if decodingNull && v.CanSet() { + break + } + + // Prevent infinite loop if v is an interface pointing to its own address: + // var v interface{} + // v = &v + if v.Elem().Kind() == reflect.Interface && v.Elem().Elem() == v { + v = v.Elem() + break + } + if v.IsNil() { + v.Set(reflect.New(v.Type().Elem())) + } + if v.Type().NumMethod() > 0 && v.CanInterface() { + if u, ok := v.Interface().(Unmarshaler); ok { + return u, nil, reflect.Value{} + } + if !decodingNull { + if u, ok := v.Interface().(encoding.TextUnmarshaler); ok { + return nil, u, reflect.Value{} + } + } + } + + if haveAddr { + v = v0 // restore original value after round-trip Value.Addr().Elem() + haveAddr = false + } else { + v = v.Elem() + } + } + return nil, nil, v +} + +// array consumes an array from d.data[d.off-1:], decoding into v. +// The first byte of the array ('[') has been read already. +func (d *decodeState) array(v reflect.Value) error { + // Check for unmarshaler. + u, ut, pv := indirect(v, false) + if u != nil { + start := d.readIndex() + d.skip() + return u.UnmarshalJSON(d.data[start:d.off]) + } + if ut != nil { + d.saveError(&UnmarshalTypeError{Value: "array", Type: v.Type(), Offset: int64(d.off)}) + d.skip() + return nil + } + v = pv + + // Check type of target. + switch v.Kind() { + case reflect.Interface: + if v.NumMethod() == 0 { + // Decoding into nil interface? Switch to non-reflect code. + ai := d.arrayInterface() + v.Set(reflect.ValueOf(ai)) + return nil + } + // Otherwise it's invalid. + fallthrough + default: + d.saveError(&UnmarshalTypeError{Value: "array", Type: v.Type(), Offset: int64(d.off)}) + d.skip() + return nil + case reflect.Array, reflect.Slice: + break + } + + i := 0 + for { + // Look ahead for ] - can only happen on first iteration. + d.scanWhile(scanSkipSpace) + if d.opcode == scanEndArray { + break + } + + // Get element of array, growing if necessary. + if v.Kind() == reflect.Slice { + // Grow slice if necessary + if i >= v.Cap() { + newcap := v.Cap() + v.Cap()/2 + if newcap < 4 { + newcap = 4 + } + newv := reflect.MakeSlice(v.Type(), v.Len(), newcap) + reflect.Copy(newv, v) + v.Set(newv) + } + if i >= v.Len() { + v.SetLen(i + 1) + } + } + + if i < v.Len() { + // Decode into element. + if err := d.value(v.Index(i)); err != nil { + return err + } + } else { + // Ran out of fixed array: skip. + if err := d.value(reflect.Value{}); err != nil { + return err + } + } + i++ + + // Next token must be , or ]. + if d.opcode == scanSkipSpace { + d.scanWhile(scanSkipSpace) + } + if d.opcode == scanEndArray { + break + } + if d.opcode != scanArrayValue { + panic(phasePanicMsg) + } + } + + if i < v.Len() { + if v.Kind() == reflect.Array { + // Array. Zero the rest. + z := reflect.Zero(v.Type().Elem()) + for ; i < v.Len(); i++ { + v.Index(i).Set(z) + } + } else { + v.SetLen(i) + } + } + if i == 0 && v.Kind() == reflect.Slice { + v.Set(reflect.MakeSlice(v.Type(), 0, 0)) + } + return nil +} + +var nullLiteral = []byte("null") +var textUnmarshalerType = reflect.TypeOf((*encoding.TextUnmarshaler)(nil)).Elem() + +// object consumes an object from d.data[d.off-1:], decoding into v. +// The first byte ('{') of the object has been read already. +func (d *decodeState) object(v reflect.Value) error { + // Check for unmarshaler. + u, ut, pv := indirect(v, false) + if u != nil { + start := d.readIndex() + d.skip() + return u.UnmarshalJSON(d.data[start:d.off]) + } + if ut != nil { + d.saveError(&UnmarshalTypeError{Value: "object", Type: v.Type(), Offset: int64(d.off)}) + d.skip() + return nil + } + v = pv + t := v.Type() + + // Decoding into nil interface? Switch to non-reflect code. + if v.Kind() == reflect.Interface && v.NumMethod() == 0 { + oi := d.objectInterface() + v.Set(reflect.ValueOf(oi)) + return nil + } + + var fields structFields + + // Check type of target: + // struct or + // map[T1]T2 where T1 is string, an integer type, + // or an encoding.TextUnmarshaler + switch v.Kind() { + case reflect.Map: + // Map key must either have string kind, have an integer kind, + // or be an encoding.TextUnmarshaler. + switch t.Key().Kind() { + case reflect.String, + reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, + reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + default: + if !reflect.PointerTo(t.Key()).Implements(textUnmarshalerType) { + d.saveError(&UnmarshalTypeError{Value: "object", Type: t, Offset: int64(d.off)}) + d.skip() + return nil + } + } + if v.IsNil() { + v.Set(reflect.MakeMap(t)) + } + case reflect.Struct: + fields = cachedTypeFields(t) + // ok + default: + d.saveError(&UnmarshalTypeError{Value: "object", Type: t, Offset: int64(d.off)}) + d.skip() + return nil + } + + var mapElem reflect.Value + var origErrorContext errorContext + if d.errorContext != nil { + origErrorContext = *d.errorContext + } + + var keys []string + + for { + // Read opening " of string key or closing }. + d.scanWhile(scanSkipSpace) + if d.opcode == scanEndObject { + // closing } - can only happen on first iteration. + break + } + if d.opcode != scanBeginLiteral { + panic(phasePanicMsg) + } + + // Read key. + start := d.readIndex() + d.rescanLiteral() + item := d.data[start:d.readIndex()] + key, ok := unquoteBytes(item) + if !ok { + panic(phasePanicMsg) + } + + keys = append(keys, string(key)) + + // Figure out field corresponding to key. + var subv reflect.Value + destring := false // whether the value is wrapped in a string to be decoded first + + if v.Kind() == reflect.Map { + elemType := t.Elem() + if !mapElem.IsValid() { + mapElem = reflect.New(elemType).Elem() + } else { + mapElem.Set(reflect.Zero(elemType)) + } + subv = mapElem + } else { + var f *field + if i, ok := fields.nameIndex[string(key)]; ok { + // Found an exact name match. + f = &fields.list[i] + } else { + // Fall back to the expensive case-insensitive + // linear search. + for i := range fields.list { + ff := &fields.list[i] + if ff.equalFold(ff.nameBytes, key) { + f = ff + break + } + } + } + if f != nil { + subv = v + destring = f.quoted + for _, i := range f.index { + if subv.Kind() == reflect.Pointer { + if subv.IsNil() { + // If a struct embeds a pointer to an unexported type, + // it is not possible to set a newly allocated value + // since the field is unexported. + // + // See https://golang.org/issue/21357 + if !subv.CanSet() { + d.saveError(fmt.Errorf("json: cannot set embedded pointer to unexported struct: %v", subv.Type().Elem())) + // Invalidate subv to ensure d.value(subv) skips over + // the JSON value without assigning it to subv. + subv = reflect.Value{} + destring = false + break + } + subv.Set(reflect.New(subv.Type().Elem())) + } + subv = subv.Elem() + } + subv = subv.Field(i) + } + if d.errorContext == nil { + d.errorContext = new(errorContext) + } + d.errorContext.FieldStack = append(d.errorContext.FieldStack, f.name) + d.errorContext.Struct = t + } else if d.disallowUnknownFields { + d.saveError(fmt.Errorf("json: unknown field %q", key)) + } + } + + // Read : before value. + if d.opcode == scanSkipSpace { + d.scanWhile(scanSkipSpace) + } + if d.opcode != scanObjectKey { + panic(phasePanicMsg) + } + d.scanWhile(scanSkipSpace) + + if destring { + switch qv := d.valueQuoted().(type) { + case nil: + if err := d.literalStore(nullLiteral, subv, false); err != nil { + return err + } + case string: + if err := d.literalStore([]byte(qv), subv, true); err != nil { + return err + } + default: + d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal unquoted value into %v", subv.Type())) + } + } else { + if err := d.value(subv); err != nil { + return err + } + } + + // Write value back to map; + // if using struct, subv points into struct already. + if v.Kind() == reflect.Map { + kt := t.Key() + var kv reflect.Value + switch { + case reflect.PointerTo(kt).Implements(textUnmarshalerType): + kv = reflect.New(kt) + if err := d.literalStore(item, kv, true); err != nil { + return err + } + kv = kv.Elem() + case kt.Kind() == reflect.String: + kv = reflect.ValueOf(key).Convert(kt) + default: + switch kt.Kind() { + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + s := string(key) + n, err := strconv.ParseInt(s, 10, 64) + if err != nil || reflect.Zero(kt).OverflowInt(n) { + d.saveError(&UnmarshalTypeError{Value: "number " + s, Type: kt, Offset: int64(start + 1)}) + break + } + kv = reflect.ValueOf(n).Convert(kt) + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + s := string(key) + n, err := strconv.ParseUint(s, 10, 64) + if err != nil || reflect.Zero(kt).OverflowUint(n) { + d.saveError(&UnmarshalTypeError{Value: "number " + s, Type: kt, Offset: int64(start + 1)}) + break + } + kv = reflect.ValueOf(n).Convert(kt) + default: + panic("json: Unexpected key type") // should never occur + } + } + if kv.IsValid() { + v.SetMapIndex(kv, subv) + } + } + + // Next token must be , or }. + if d.opcode == scanSkipSpace { + d.scanWhile(scanSkipSpace) + } + if d.errorContext != nil { + // Reset errorContext to its original state. + // Keep the same underlying array for FieldStack, to reuse the + // space and avoid unnecessary allocs. + d.errorContext.FieldStack = d.errorContext.FieldStack[:len(origErrorContext.FieldStack)] + d.errorContext.Struct = origErrorContext.Struct + } + if d.opcode == scanEndObject { + break + } + if d.opcode != scanObjectValue { + panic(phasePanicMsg) + } + } + + if v.Kind() == reflect.Map { + d.lastKeys = keys + } + return nil +} + +// convertNumber converts the number literal s to a float64 or a Number +// depending on the setting of d.useNumber. +func (d *decodeState) convertNumber(s string) (any, error) { + if d.useNumber { + return Number(s), nil + } + f, err := strconv.ParseFloat(s, 64) + if err != nil { + return nil, &UnmarshalTypeError{Value: "number " + s, Type: reflect.TypeOf(0.0), Offset: int64(d.off)} + } + return f, nil +} + +var numberType = reflect.TypeOf(Number("")) + +// literalStore decodes a literal stored in item into v. +// +// fromQuoted indicates whether this literal came from unwrapping a +// string from the ",string" struct tag option. this is used only to +// produce more helpful error messages. +func (d *decodeState) literalStore(item []byte, v reflect.Value, fromQuoted bool) error { + // Check for unmarshaler. + if len(item) == 0 { + //Empty string given + d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) + return nil + } + isNull := item[0] == 'n' // null + u, ut, pv := indirect(v, isNull) + if u != nil { + return u.UnmarshalJSON(item) + } + if ut != nil { + if item[0] != '"' { + if fromQuoted { + d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) + return nil + } + val := "number" + switch item[0] { + case 'n': + val = "null" + case 't', 'f': + val = "bool" + } + d.saveError(&UnmarshalTypeError{Value: val, Type: v.Type(), Offset: int64(d.readIndex())}) + return nil + } + s, ok := unquoteBytes(item) + if !ok { + if fromQuoted { + return fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()) + } + panic(phasePanicMsg) + } + return ut.UnmarshalText(s) + } + + v = pv + + switch c := item[0]; c { + case 'n': // null + // The main parser checks that only true and false can reach here, + // but if this was a quoted string input, it could be anything. + if fromQuoted && string(item) != "null" { + d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) + break + } + switch v.Kind() { + case reflect.Interface, reflect.Pointer, reflect.Map, reflect.Slice: + v.Set(reflect.Zero(v.Type())) + // otherwise, ignore null for primitives/string + } + case 't', 'f': // true, false + value := item[0] == 't' + // The main parser checks that only true and false can reach here, + // but if this was a quoted string input, it could be anything. + if fromQuoted && string(item) != "true" && string(item) != "false" { + d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) + break + } + switch v.Kind() { + default: + if fromQuoted { + d.saveError(fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type())) + } else { + d.saveError(&UnmarshalTypeError{Value: "bool", Type: v.Type(), Offset: int64(d.readIndex())}) + } + case reflect.Bool: + v.SetBool(value) + case reflect.Interface: + if v.NumMethod() == 0 { + v.Set(reflect.ValueOf(value)) + } else { + d.saveError(&UnmarshalTypeError{Value: "bool", Type: v.Type(), Offset: int64(d.readIndex())}) + } + } + + case '"': // string + s, ok := unquoteBytes(item) + if !ok { + if fromQuoted { + return fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()) + } + panic(phasePanicMsg) + } + switch v.Kind() { + default: + d.saveError(&UnmarshalTypeError{Value: "string", Type: v.Type(), Offset: int64(d.readIndex())}) + case reflect.Slice: + if v.Type().Elem().Kind() != reflect.Uint8 { + d.saveError(&UnmarshalTypeError{Value: "string", Type: v.Type(), Offset: int64(d.readIndex())}) + break + } + b := make([]byte, base64.StdEncoding.DecodedLen(len(s))) + n, err := base64.StdEncoding.Decode(b, s) + if err != nil { + d.saveError(err) + break + } + v.SetBytes(b[:n]) + case reflect.String: + if v.Type() == numberType && !isValidNumber(string(s)) { + return fmt.Errorf("json: invalid number literal, trying to unmarshal %q into Number", item) + } + v.SetString(string(s)) + case reflect.Interface: + if v.NumMethod() == 0 { + v.Set(reflect.ValueOf(string(s))) + } else { + d.saveError(&UnmarshalTypeError{Value: "string", Type: v.Type(), Offset: int64(d.readIndex())}) + } + } + + default: // number + if c != '-' && (c < '0' || c > '9') { + if fromQuoted { + return fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()) + } + panic(phasePanicMsg) + } + s := string(item) + switch v.Kind() { + default: + if v.Kind() == reflect.String && v.Type() == numberType { + // s must be a valid number, because it's + // already been tokenized. + v.SetString(s) + break + } + if fromQuoted { + return fmt.Errorf("json: invalid use of ,string struct tag, trying to unmarshal %q into %v", item, v.Type()) + } + d.saveError(&UnmarshalTypeError{Value: "number", Type: v.Type(), Offset: int64(d.readIndex())}) + case reflect.Interface: + n, err := d.convertNumber(s) + if err != nil { + d.saveError(err) + break + } + if v.NumMethod() != 0 { + d.saveError(&UnmarshalTypeError{Value: "number", Type: v.Type(), Offset: int64(d.readIndex())}) + break + } + v.Set(reflect.ValueOf(n)) + + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + n, err := strconv.ParseInt(s, 10, 64) + if err != nil || v.OverflowInt(n) { + d.saveError(&UnmarshalTypeError{Value: "number " + s, Type: v.Type(), Offset: int64(d.readIndex())}) + break + } + v.SetInt(n) + + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + n, err := strconv.ParseUint(s, 10, 64) + if err != nil || v.OverflowUint(n) { + d.saveError(&UnmarshalTypeError{Value: "number " + s, Type: v.Type(), Offset: int64(d.readIndex())}) + break + } + v.SetUint(n) + + case reflect.Float32, reflect.Float64: + n, err := strconv.ParseFloat(s, v.Type().Bits()) + if err != nil || v.OverflowFloat(n) { + d.saveError(&UnmarshalTypeError{Value: "number " + s, Type: v.Type(), Offset: int64(d.readIndex())}) + break + } + v.SetFloat(n) + } + } + return nil +} + +// The xxxInterface routines build up a value to be stored +// in an empty interface. They are not strictly necessary, +// but they avoid the weight of reflection in this common case. + +// valueInterface is like value but returns interface{} +func (d *decodeState) valueInterface() (val any) { + switch d.opcode { + default: + panic(phasePanicMsg) + case scanBeginArray: + val = d.arrayInterface() + d.scanNext() + case scanBeginObject: + val = d.objectInterface() + d.scanNext() + case scanBeginLiteral: + val = d.literalInterface() + } + return +} + +// arrayInterface is like array but returns []interface{}. +func (d *decodeState) arrayInterface() []any { + var v = make([]any, 0) + for { + // Look ahead for ] - can only happen on first iteration. + d.scanWhile(scanSkipSpace) + if d.opcode == scanEndArray { + break + } + + v = append(v, d.valueInterface()) + + // Next token must be , or ]. + if d.opcode == scanSkipSpace { + d.scanWhile(scanSkipSpace) + } + if d.opcode == scanEndArray { + break + } + if d.opcode != scanArrayValue { + panic(phasePanicMsg) + } + } + return v +} + +// objectInterface is like object but returns map[string]interface{}. +func (d *decodeState) objectInterface() map[string]any { + m := make(map[string]any) + for { + // Read opening " of string key or closing }. + d.scanWhile(scanSkipSpace) + if d.opcode == scanEndObject { + // closing } - can only happen on first iteration. + break + } + if d.opcode != scanBeginLiteral { + panic(phasePanicMsg) + } + + // Read string key. + start := d.readIndex() + d.rescanLiteral() + item := d.data[start:d.readIndex()] + key, ok := unquote(item) + if !ok { + panic(phasePanicMsg) + } + + // Read : before value. + if d.opcode == scanSkipSpace { + d.scanWhile(scanSkipSpace) + } + if d.opcode != scanObjectKey { + panic(phasePanicMsg) + } + d.scanWhile(scanSkipSpace) + + // Read value. + m[key] = d.valueInterface() + + // Next token must be , or }. + if d.opcode == scanSkipSpace { + d.scanWhile(scanSkipSpace) + } + if d.opcode == scanEndObject { + break + } + if d.opcode != scanObjectValue { + panic(phasePanicMsg) + } + } + return m +} + +// literalInterface consumes and returns a literal from d.data[d.off-1:] and +// it reads the following byte ahead. The first byte of the literal has been +// read already (that's how the caller knows it's a literal). +func (d *decodeState) literalInterface() any { + // All bytes inside literal return scanContinue op code. + start := d.readIndex() + d.rescanLiteral() + + item := d.data[start:d.readIndex()] + + switch c := item[0]; c { + case 'n': // null + return nil + + case 't', 'f': // true, false + return c == 't' + + case '"': // string + s, ok := unquote(item) + if !ok { + panic(phasePanicMsg) + } + return s + + default: // number + if c != '-' && (c < '0' || c > '9') { + panic(phasePanicMsg) + } + n, err := d.convertNumber(string(item)) + if err != nil { + d.saveError(err) + } + return n + } +} + +// getu4 decodes \uXXXX from the beginning of s, returning the hex value, +// or it returns -1. +func getu4(s []byte) rune { + if len(s) < 6 || s[0] != '\\' || s[1] != 'u' { + return -1 + } + var r rune + for _, c := range s[2:6] { + switch { + case '0' <= c && c <= '9': + c = c - '0' + case 'a' <= c && c <= 'f': + c = c - 'a' + 10 + case 'A' <= c && c <= 'F': + c = c - 'A' + 10 + default: + return -1 + } + r = r*16 + rune(c) + } + return r +} + +// unquote converts a quoted JSON string literal s into an actual string t. +// The rules are different than for Go, so cannot use strconv.Unquote. +func unquote(s []byte) (t string, ok bool) { + s, ok = unquoteBytes(s) + t = string(s) + return +} + +func unquoteBytes(s []byte) (t []byte, ok bool) { + if len(s) < 2 || s[0] != '"' || s[len(s)-1] != '"' { + return + } + s = s[1 : len(s)-1] + + // Check for unusual characters. If there are none, + // then no unquoting is needed, so return a slice of the + // original bytes. + r := 0 + for r < len(s) { + c := s[r] + if c == '\\' || c == '"' || c < ' ' { + break + } + if c < utf8.RuneSelf { + r++ + continue + } + rr, size := utf8.DecodeRune(s[r:]) + if rr == utf8.RuneError && size == 1 { + break + } + r += size + } + if r == len(s) { + return s, true + } + + b := make([]byte, len(s)+2*utf8.UTFMax) + w := copy(b, s[0:r]) + for r < len(s) { + // Out of room? Can only happen if s is full of + // malformed UTF-8 and we're replacing each + // byte with RuneError. + if w >= len(b)-2*utf8.UTFMax { + nb := make([]byte, (len(b)+utf8.UTFMax)*2) + copy(nb, b[0:w]) + b = nb + } + switch c := s[r]; { + case c == '\\': + r++ + if r >= len(s) { + return + } + switch s[r] { + default: + return + case '"', '\\', '/', '\'': + b[w] = s[r] + r++ + w++ + case 'b': + b[w] = '\b' + r++ + w++ + case 'f': + b[w] = '\f' + r++ + w++ + case 'n': + b[w] = '\n' + r++ + w++ + case 'r': + b[w] = '\r' + r++ + w++ + case 't': + b[w] = '\t' + r++ + w++ + case 'u': + r-- + rr := getu4(s[r:]) + if rr < 0 { + return + } + r += 6 + if utf16.IsSurrogate(rr) { + rr1 := getu4(s[r:]) + if dec := utf16.DecodeRune(rr, rr1); dec != unicode.ReplacementChar { + // A valid pair; consume. + r += 6 + w += utf8.EncodeRune(b[w:], dec) + break + } + // Invalid surrogate; fall back to replacement rune. + rr = unicode.ReplacementChar + } + w += utf8.EncodeRune(b[w:], rr) + } + + // Quote, control characters are invalid. + case c == '"', c < ' ': + return + + // ASCII + case c < utf8.RuneSelf: + b[w] = c + r++ + w++ + + // Coerce to well-formed UTF-8. + default: + rr, size := utf8.DecodeRune(s[r:]) + r += size + w += utf8.EncodeRune(b[w:], rr) + } + } + return b[0:w], true +} diff --git a/vendor/github.com/evanphx/json-patch/v5/internal/json/encode.go b/vendor/github.com/evanphx/json-patch/v5/internal/json/encode.go new file mode 100644 index 000000000..2e6eca448 --- /dev/null +++ b/vendor/github.com/evanphx/json-patch/v5/internal/json/encode.go @@ -0,0 +1,1486 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package json implements encoding and decoding of JSON as defined in +// RFC 7159. The mapping between JSON and Go values is described +// in the documentation for the Marshal and Unmarshal functions. +// +// See "JSON and Go" for an introduction to this package: +// https://golang.org/doc/articles/json_and_go.html +package json + +import ( + "bytes" + "encoding" + "encoding/base64" + "fmt" + "math" + "reflect" + "sort" + "strconv" + "strings" + "sync" + "unicode" + "unicode/utf8" +) + +// Marshal returns the JSON encoding of v. +// +// Marshal traverses the value v recursively. +// If an encountered value implements the Marshaler interface +// and is not a nil pointer, Marshal calls its MarshalJSON method +// to produce JSON. If no MarshalJSON method is present but the +// value implements encoding.TextMarshaler instead, Marshal calls +// its MarshalText method and encodes the result as a JSON string. +// The nil pointer exception is not strictly necessary +// but mimics a similar, necessary exception in the behavior of +// UnmarshalJSON. +// +// Otherwise, Marshal uses the following type-dependent default encodings: +// +// Boolean values encode as JSON booleans. +// +// Floating point, integer, and Number values encode as JSON numbers. +// +// String values encode as JSON strings coerced to valid UTF-8, +// replacing invalid bytes with the Unicode replacement rune. +// So that the JSON will be safe to embed inside HTML