• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

kubeovn / kube-ovn / 21656192014

04 Feb 2026 02:35AM UTC coverage: 23.031% (-0.02%) from 23.05%
21656192014

push

github

web-flow
refactor: profiling, Retry, assignAddress, runIfLinkExists (#6258)

- Add pkg/util/profiling with DumpProfile(); use from cmd/cmdmain.go and cmd/controller/cmdmain.go
- cmd/cmdmain.go: remove local dumpProfile, use profiling.DumpProfile()
- cmd/controller/cmdmain.go: remove duplicated dumpProfile, use profiling.DumpProfile()
- cmd/daemon/cniserver.go: Retry param ctrl->cfg, return nil on success
- cmd/daemon/init.go: add runIfLinkExists helper for initForOS and setVxlanNicTxOff
- cmd/cni/cni.go: assignAddress(ipAddress, gateway, protocol, mask), merge IPv4/IPv6 case in generateCNIResult

Signed-off-by: Mengxin Liu <liumengxinfly@gmail.com>

0 of 50 new or added lines in 2 files covered. (0.0%)

2 existing lines in 1 file now uncovered.

12416 of 53909 relevant lines covered (23.03%)

0.27 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

0.0
/pkg/daemon/handler.go
1
package daemon
2

3
import (
4
        "context"
5
        "encoding/json"
6
        "fmt"
7
        "net"
8
        "net/http"
9
        "strconv"
10
        "strings"
11
        "time"
12

13
        "github.com/emicklei/go-restful/v3"
14
        v1 "k8s.io/api/core/v1"
15
        k8serrors "k8s.io/apimachinery/pkg/api/errors"
16
        metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
17
        "k8s.io/apimachinery/pkg/labels"
18
        "k8s.io/client-go/kubernetes"
19
        "k8s.io/klog/v2"
20
        kubevirtv1 "kubevirt.io/api/core/v1"
21

22
        kubeovnv1 "github.com/kubeovn/kube-ovn/pkg/apis/kubeovn/v1"
23
        clientset "github.com/kubeovn/kube-ovn/pkg/client/clientset/versioned"
24
        "github.com/kubeovn/kube-ovn/pkg/ovs"
25
        "github.com/kubeovn/kube-ovn/pkg/request"
26
        "github.com/kubeovn/kube-ovn/pkg/util"
27
)
28

29
const (
30
        gatewayCheckModeDisabled = iota
31
        gatewayCheckModePing
32
        gatewayCheckModeArping
33
        gatewayCheckModePingNotConcerned
34
        gatewayCheckModeArpingNotConcerned
35
)
36

37
type cniServerHandler struct {
38
        Config        *Configuration
39
        KubeClient    kubernetes.Interface
40
        KubeOvnClient clientset.Interface
41
        Controller    *Controller
42
}
43

44
func createCniServerHandler(config *Configuration, controller *Controller) *cniServerHandler {
×
45
        csh := &cniServerHandler{KubeClient: config.KubeClient, KubeOvnClient: config.KubeOvnClient, Config: config, Controller: controller}
×
46
        return csh
×
47
}
×
48

49
func (csh cniServerHandler) providerExists(provider string) (*kubeovnv1.Subnet, bool) {
×
50
        if util.IsOvnProvider(provider) {
×
51
                return nil, true
×
52
        }
×
53
        subnets, _ := csh.Controller.subnetsLister.List(labels.Everything())
×
54
        for _, subnet := range subnets {
×
55
                if subnet.Spec.Provider == provider {
×
56
                        return subnet.DeepCopy(), true
×
57
                }
×
58
        }
59
        return nil, false
×
60
}
61

62
func (csh cniServerHandler) handleAdd(req *restful.Request, resp *restful.Response) {
×
63
        podRequest := request.CniRequest{}
×
64
        if err := req.ReadEntity(&podRequest); err != nil {
×
65
                errMsg := fmt.Errorf("parse add request failed %w", err)
×
66
                klog.Error(errMsg)
×
67
                if err := resp.WriteHeaderAndEntity(http.StatusBadRequest, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
68
                        klog.Errorf("failed to write response, %v", err)
×
69
                }
×
70
                return
×
71
        }
72
        klog.V(5).Infof("request body is %v", podRequest)
×
73
        podSubnet, exist := csh.providerExists(podRequest.Provider)
×
74
        if !exist {
×
75
                errMsg := fmt.Errorf("provider %s not bind to any subnet", podRequest.Provider)
×
76
                klog.Error(errMsg)
×
77
                if err := resp.WriteHeaderAndEntity(http.StatusBadRequest, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
78
                        klog.Errorf("failed to write response, %v", err)
×
79
                }
×
80
                return
×
81
        }
82

83
        klog.Infof("add port request: %v", podRequest)
×
84
        if err := csh.validatePodRequest(&podRequest); err != nil {
×
85
                klog.Error(err)
×
86
                if err := resp.WriteHeaderAndEntity(http.StatusBadRequest, request.CniResponse{Err: err.Error()}); err != nil {
×
87
                        klog.Errorf("failed to write response, %v", err)
×
88
                }
×
89
                return
×
90
        }
91

92
        var gatewayCheckMode int
×
93
        var macAddr, ip, ipAddr, cidr, gw, subnet, ingress, egress, providerNetwork, ifName, nicType, podNicName, vmName, latency, limit, loss, jitter, u2oInterconnectionIP, oldPodName string
×
94
        var routes []request.Route
×
95
        var isDefaultRoute, noIPAM bool
×
96
        var pod *v1.Pod
×
97
        var err error
×
98
        for range 20 {
×
99
                if pod, err = csh.Controller.podsLister.Pods(podRequest.PodNamespace).Get(podRequest.PodName); err != nil {
×
100
                        errMsg := fmt.Errorf("get pod %s/%s failed %w", podRequest.PodNamespace, podRequest.PodName, err)
×
101
                        klog.Error(errMsg)
×
102
                        if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
103
                                klog.Errorf("failed to write response, %v", err)
×
104
                        }
×
105
                        return
×
106
                }
107
                if pod.Annotations[fmt.Sprintf(util.AllocatedAnnotationTemplate, podRequest.Provider)] != "true" {
×
108
                        klog.Infof("wait address for pod %s/%s provider %s", podRequest.PodNamespace, podRequest.PodName, podRequest.Provider)
×
109
                        // wait controller assign an address
×
110
                        cniWaitAddressResult.WithLabelValues(nodeName).Inc()
×
111
                        time.Sleep(1 * time.Second)
×
112
                        continue
×
113
                }
114

115
                if err := util.ValidatePodNetwork(pod.Annotations); err != nil {
×
116
                        klog.Errorf("validate pod %s/%s failed, %v", podRequest.PodNamespace, podRequest.PodName, err)
×
117
                        // wait controller assign an address
×
118
                        cniWaitAddressResult.WithLabelValues(nodeName).Inc()
×
119
                        time.Sleep(1 * time.Second)
×
120
                        continue
×
121
                }
122
                ip = pod.Annotations[fmt.Sprintf(util.IPAddressAnnotationTemplate, podRequest.Provider)]
×
123
                cidr = pod.Annotations[fmt.Sprintf(util.CidrAnnotationTemplate, podRequest.Provider)]
×
124
                gw = pod.Annotations[fmt.Sprintf(util.GatewayAnnotationTemplate, podRequest.Provider)]
×
125
                subnet = pod.Annotations[fmt.Sprintf(util.LogicalSwitchAnnotationTemplate, podRequest.Provider)]
×
126
                ingress = pod.Annotations[fmt.Sprintf(util.IngressRateAnnotationTemplate, podRequest.Provider)]
×
127
                egress = pod.Annotations[fmt.Sprintf(util.EgressRateAnnotationTemplate, podRequest.Provider)]
×
128
                latency = pod.Annotations[fmt.Sprintf(util.NetemQosLatencyAnnotationTemplate, podRequest.Provider)]
×
129
                limit = pod.Annotations[fmt.Sprintf(util.NetemQosLimitAnnotationTemplate, podRequest.Provider)]
×
130
                loss = pod.Annotations[fmt.Sprintf(util.NetemQosLossAnnotationTemplate, podRequest.Provider)]
×
131
                jitter = pod.Annotations[fmt.Sprintf(util.NetemQosJitterAnnotationTemplate, podRequest.Provider)]
×
132
                providerNetwork = pod.Annotations[fmt.Sprintf(util.ProviderNetworkTemplate, podRequest.Provider)]
×
133
                vmName = pod.Annotations[fmt.Sprintf(util.VMAnnotationTemplate, podRequest.Provider)]
×
134
                ipAddr, noIPAM, err = util.GetIPAddrWithMaskForCNI(ip, cidr)
×
135
                if err != nil {
×
136
                        errMsg := fmt.Errorf("failed to get ip address with mask, %w", err)
×
137
                        klog.Error(errMsg)
×
138
                        if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
139
                                klog.Errorf("failed to write response, %v", err)
×
140
                        }
×
141
                        return
×
142
                }
143
                oldPodName = podRequest.PodName
×
144
                if s := pod.Annotations[fmt.Sprintf(util.RoutesAnnotationTemplate, podRequest.Provider)]; s != "" {
×
145
                        if err = json.Unmarshal([]byte(s), &routes); err != nil {
×
146
                                errMsg := fmt.Errorf("invalid routes for pod %s/%s: %w", pod.Namespace, pod.Name, err)
×
147
                                klog.Error(errMsg)
×
148
                                if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
149
                                        klog.Errorf("failed to write response: %v", err)
×
150
                                }
×
151
                                return
×
152
                        }
153
                }
154
                if ifName = podRequest.IfName; ifName == "" {
×
155
                        ifName = "eth0"
×
156
                }
×
157

158
                // To support KubeVirt hotplug dpdk nic, forbidden set the volume name
159
                if podRequest.VhostUserSocketConsumption == util.ConsumptionKubevirt {
×
160
                        podRequest.VhostUserSocketVolumeName = util.VhostUserSocketVolumeName
×
161
                }
×
162

163
                switch {
×
164
                case podRequest.DeviceID != "":
×
165
                        nicType = util.OffloadType
×
166
                case podRequest.VhostUserSocketVolumeName != "":
×
167
                        nicType = util.DpdkType
×
168
                        if err = createShortSharedDir(pod, podRequest.VhostUserSocketVolumeName, podRequest.VhostUserSocketConsumption, csh.Config.KubeletDir); err != nil {
×
169
                                klog.Error(err.Error())
×
170
                                if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: err.Error()}); err != nil {
×
171
                                        klog.Errorf("failed to write response: %v", err)
×
172
                                }
×
173
                                return
×
174
                        }
175
                default:
×
176
                        nicType = pod.Annotations[fmt.Sprintf(util.PodNicAnnotationTemplate, podRequest.Provider)]
×
177
                }
178

179
                switch pod.Annotations[fmt.Sprintf(util.DefaultRouteAnnotationTemplate, podRequest.Provider)] {
×
180
                case "true":
×
181
                        isDefaultRoute = true
×
182
                case "false":
×
183
                        isDefaultRoute = false
×
184
                default:
×
185
                        isDefaultRoute = ifName == "eth0"
×
186
                }
187

188
                if isDefaultRoute && pod.Annotations[fmt.Sprintf(util.RoutedAnnotationTemplate, podRequest.Provider)] != "true" && util.IsOvnProvider(podRequest.Provider) {
×
189
                        klog.Infof("wait route ready for pod %s/%s provider %s", podRequest.PodNamespace, podRequest.PodName, podRequest.Provider)
×
190
                        cniWaitRouteResult.WithLabelValues(nodeName).Inc()
×
191
                        time.Sleep(1 * time.Second)
×
192
                        continue
×
193
                }
194

195
                if vmName != "" {
×
196
                        podRequest.PodName = vmName
×
197
                }
×
198

199
                break
×
200
        }
201

202
        if pod.Annotations[fmt.Sprintf(util.AllocatedAnnotationTemplate, podRequest.Provider)] != "true" {
×
203
                err := fmt.Errorf("no address allocated to pod %s/%s provider %s, please see kube-ovn-controller logs to find errors", pod.Namespace, pod.Name, podRequest.Provider)
×
204
                klog.Error(err)
×
205
                if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: err.Error()}); err != nil {
×
206
                        klog.Errorf("failed to write response, %v", err)
×
207
                }
×
208
                return
×
209
        }
210

211
        if subnet == "" && podSubnet != nil {
×
212
                subnet = podSubnet.Name
×
213
        }
×
214
        if !noIPAM {
×
215
                if err := csh.UpdateIPCR(podRequest, subnet, ip); err != nil {
×
216
                        if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: err.Error()}); err != nil {
×
217
                                klog.Errorf("failed to write response, %v", err)
×
218
                        }
×
219
                        return
×
220
                }
221
        }
222

223
        if isDefaultRoute && pod.Annotations[fmt.Sprintf(util.RoutedAnnotationTemplate, podRequest.Provider)] != "true" && util.IsOvnProvider(podRequest.Provider) {
×
224
                err := fmt.Errorf("route is not ready for pod %s/%s provider %s, please see kube-ovn-controller logs to find errors", pod.Namespace, pod.Name, podRequest.Provider)
×
225
                klog.Error(err)
×
226
                if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: err.Error()}); err != nil {
×
227
                        klog.Errorf("failed to write response, %v", err)
×
228
                }
×
229
                return
×
230
        }
231

232
        var mtu int
×
233
        routes = append(podRequest.Routes, routes...)
×
234
        if strings.HasSuffix(podRequest.Provider, util.OvnProvider) && subnet != "" {
×
235
                podSubnet, err := csh.Controller.subnetsLister.Get(subnet)
×
236
                if err != nil {
×
237
                        errMsg := fmt.Errorf("failed to get subnet %s: %w", subnet, err)
×
238
                        klog.Error(errMsg)
×
239
                        if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
240
                                klog.Errorf("failed to write response: %v", err)
×
241
                        }
×
242
                        return
×
243
                }
244

245
                if podSubnet.Status.U2OInterconnectionIP == "" && podSubnet.Spec.U2OInterconnection {
×
246
                        errMsg := fmt.Errorf("failed to generate u2o ip on subnet %s", podSubnet.Name)
×
247
                        klog.Error(errMsg)
×
248
                        return
×
249
                }
×
250

251
                if podSubnet.Status.U2OInterconnectionIP != "" && podSubnet.Spec.U2OInterconnection {
×
252
                        u2oInterconnectionIP = podSubnet.Status.U2OInterconnectionIP
×
253
                }
×
254

255
                var vmMigration bool
×
256
                subnetHasVlan := podSubnet.Spec.Vlan != ""
×
257
                // skip ping check gateway for pods during live migration
×
258
                if pod.Annotations[kubevirtv1.MigrationJobNameAnnotation] == "" {
×
259
                        if subnetHasVlan && !podSubnet.Spec.LogicalGateway {
×
260
                                if podSubnet.Spec.DisableGatewayCheck {
×
261
                                        gatewayCheckMode = gatewayCheckModeArpingNotConcerned
×
262
                                } else {
×
263
                                        gatewayCheckMode = gatewayCheckModeArping
×
264
                                }
×
265
                        } else {
×
266
                                if podSubnet.Spec.DisableGatewayCheck {
×
267
                                        gatewayCheckMode = gatewayCheckModePingNotConcerned
×
268
                                } else {
×
269
                                        gatewayCheckMode = gatewayCheckModePing
×
270
                                }
×
271
                        }
272
                } else {
×
273
                        vmMigration = true
×
274
                }
×
275
                if pod.Annotations[fmt.Sprintf(util.ActivationStrategyTemplate, podRequest.Provider)] != "" {
×
276
                        gatewayCheckMode = gatewayCheckModeDisabled
×
277
                }
×
278

279
                if podSubnet.Spec.Mtu > 0 {
×
280
                        mtu = int(podSubnet.Spec.Mtu)
×
281
                } else {
×
282
                        if providerNetwork != "" && !podSubnet.Spec.LogicalGateway && !podSubnet.Spec.U2OInterconnection {
×
283
                                node, err := csh.Controller.nodesLister.Get(csh.Config.NodeName)
×
284
                                if err != nil {
×
285
                                        errMsg := fmt.Errorf("failed to get node %s: %w", csh.Config.NodeName, err)
×
286
                                        klog.Error(errMsg)
×
287
                                        if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
288
                                                klog.Errorf("failed to write response: %v", err)
×
289
                                        }
×
290
                                        return
×
291
                                }
292
                                mtuStr := node.Labels[fmt.Sprintf(util.ProviderNetworkMtuTemplate, providerNetwork)]
×
293
                                if mtuStr != "" {
×
294
                                        if mtu, err = strconv.Atoi(mtuStr); err != nil || mtu <= 0 {
×
295
                                                errMsg := fmt.Errorf("failed to parse provider network MTU %s: %w", mtuStr, err)
×
296
                                                klog.Error(errMsg)
×
297
                                                if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
298
                                                        klog.Errorf("failed to write response: %v", err)
×
299
                                                }
×
300
                                                return
×
301
                                        }
302
                                }
303
                        } else {
×
304
                                mtu = csh.Config.MTU
×
305
                        }
×
306
                }
307

308
                macAddr = pod.Annotations[fmt.Sprintf(util.MacAddressAnnotationTemplate, podRequest.Provider)]
×
309
                klog.Infof("create container interface %s mac %s, ip %s, cidr %s, gw %s, custom routes %v", ifName, macAddr, ipAddr, cidr, gw, routes)
×
310
                podNicName = ifName
×
311

×
312
                var encapIP string
×
313
                if podSubnet.Spec.NodeNetwork != "" {
×
314
                        encapIP, err = csh.Config.GetEncapIPByNetwork(podSubnet.Spec.NodeNetwork)
×
315
                        if err != nil {
×
316
                                errMsg := fmt.Errorf("failed to get encap IP for node network %s: %w", podSubnet.Spec.NodeNetwork, err)
×
317
                                klog.Error(errMsg)
×
318
                                if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
319
                                        klog.Errorf("failed to write response: %v", err)
×
320
                                }
×
321
                                return
×
322
                        }
323
                }
324

325
                switch nicType {
×
326
                case util.DpdkType:
×
327
                        err = csh.configureDpdkNic(podRequest.PodName, podRequest.PodNamespace, podRequest.Provider, podRequest.NetNs, podRequest.ContainerID, ifName, macAddr, mtu, ipAddr, gw, ingress, egress, getShortSharedDir(pod.UID, podRequest.VhostUserSocketVolumeName), podRequest.VhostUserSocketName, podRequest.VhostUserSocketConsumption)
×
328
                        routes = nil
×
329
                default:
×
330
                        routes, err = csh.configureNic(podRequest.PodName, podRequest.PodNamespace, podRequest.Provider, podRequest.NetNs, podRequest.ContainerID, podRequest.VfDriver, ifName, macAddr, mtu, ipAddr, gw, isDefaultRoute, vmMigration, routes, podRequest.DNS.Nameservers, podRequest.DNS.Search, ingress, egress, podRequest.DeviceID, latency, limit, loss, jitter, gatewayCheckMode, u2oInterconnectionIP, oldPodName, encapIP)
×
331
                }
332
                if err != nil {
×
333
                        errMsg := fmt.Errorf("configure nic %s for pod %s/%s failed: %w", ifName, podRequest.PodName, podRequest.PodNamespace, err)
×
334
                        klog.Error(errMsg)
×
335
                        if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
336
                                klog.Errorf("failed to write response, %v", err)
×
337
                        }
×
338
                        return
×
339
                }
340

341
                ifaceID := ovs.PodNameToPortName(podRequest.PodName, podRequest.PodNamespace, podRequest.Provider)
×
342
                if err = ovs.ConfigInterfaceMirror(csh.Config.EnableMirror, pod.Annotations[fmt.Sprintf(util.MirrorControlAnnotationTemplate, podRequest.Provider)], ifaceID); err != nil {
×
343
                        klog.Errorf("failed mirror to mirror0, %v", err)
×
344
                        return
×
345
                }
×
346

347
                if err = csh.Controller.addEgressConfig(podSubnet, ip); err != nil {
×
348
                        errMsg := fmt.Errorf("failed to add egress configuration: %w", err)
×
349
                        klog.Error(errMsg)
×
350
                        if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
351
                                klog.Errorf("failed to write response, %v", err)
×
352
                        }
×
353
                        return
×
354
                }
355
        } else if len(routes) != 0 {
×
356
                hasDefaultRoute := make(map[string]bool, 2)
×
357
                for _, r := range routes {
×
358
                        if r.Destination == "" {
×
359
                                hasDefaultRoute[util.CheckProtocol(r.Gateway)] = true
×
360
                                continue
×
361
                        }
362
                        if _, cidr, err := net.ParseCIDR(r.Destination); err == nil {
×
363
                                if ones, _ := cidr.Mask.Size(); ones == 0 {
×
364
                                        hasDefaultRoute[util.CheckProtocol(r.Gateway)] = true
×
365
                                }
×
366
                        }
367
                }
368
                if len(hasDefaultRoute) != 0 {
×
369
                        // remove existing default route so other CNI plugins, such as macvlan, can add the new default route correctly
×
370
                        if err = csh.removeDefaultRoute(podRequest.NetNs, hasDefaultRoute[kubeovnv1.ProtocolIPv4], hasDefaultRoute[kubeovnv1.ProtocolIPv6]); err != nil {
×
371
                                errMsg := fmt.Errorf("failed to remove existing default route for interface %s of pod %s/%s: %w", podRequest.IfName, podRequest.PodNamespace, podRequest.PodName, err)
×
372
                                klog.Error(errMsg)
×
373
                                if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
374
                                        klog.Errorf("failed to write response: %v", err)
×
375
                                }
×
376
                                return
×
377
                        }
378
                }
379
        }
380

NEW
381
        v4IP, v6IP := util.SplitStringIP(ip)
×
NEW
382
        v4CIDR, v6CIDR := util.SplitStringIP(cidr)
×
NEW
383
        v4GW, v6GW := util.SplitStringIP(gw)
×
NEW
384

×
NEW
385
        var ips []request.IPConfig
×
NEW
386
        if v4IP != "" {
×
NEW
387
                cfg := request.IPConfig{Protocol: kubeovnv1.ProtocolIPv4, IP: v4IP, CIDR: v4CIDR}
×
NEW
388
                if isDefaultRoute {
×
NEW
389
                        cfg.Gateway = v4GW
×
NEW
390
                }
×
NEW
391
                ips = append(ips, cfg)
×
392
        }
NEW
393
        if v6IP != "" {
×
NEW
394
                cfg := request.IPConfig{Protocol: kubeovnv1.ProtocolIPv6, IP: v6IP, CIDR: v6CIDR}
×
NEW
395
                if isDefaultRoute {
×
NEW
396
                        cfg.Gateway = v6GW
×
NEW
397
                }
×
NEW
398
                ips = append(ips, cfg)
×
399
        }
400

401
        response := &request.CniResponse{
×
NEW
402
                IPs:        ips,
×
403
                MacAddress: macAddr,
×
404
                PodNicName: podNicName,
×
405
                Routes:     routes,
×
406
                Mtu:        mtu,
×
407
        }
×
408
        if err := resp.WriteHeaderAndEntity(http.StatusOK, response); err != nil {
×
409
                klog.Errorf("failed to write response, %v", err)
×
410
        }
×
411
}
412

413
func (csh cniServerHandler) UpdateIPCR(podRequest request.CniRequest, subnet, ip string) error {
×
414
        ipCRName := ovs.PodNameToPortName(podRequest.PodName, podRequest.PodNamespace, podRequest.Provider)
×
415
        for range 20 {
×
416
                ipCR, err := csh.KubeOvnClient.KubeovnV1().IPs().Get(context.Background(), ipCRName, metav1.GetOptions{})
×
417
                if err != nil {
×
418
                        err = fmt.Errorf("failed to get ip crd for %s, %w", ip, err)
×
419
                        // maybe create a backup pod with previous annotations
×
420
                        klog.Error(err)
×
421
                } else if ipCR.Spec.NodeName != csh.Config.NodeName {
×
422
                        ipCR := ipCR.DeepCopy()
×
423
                        if ipCR.Labels == nil {
×
424
                                ipCR.Labels = map[string]string{}
×
425
                        }
×
426
                        ipCR.Spec.NodeName = csh.Config.NodeName
×
427
                        ipCR.Spec.AttachIPs = []string{}
×
428
                        ipCR.Labels[subnet] = ""
×
429
                        ipCR.Labels[util.NodeNameLabel] = csh.Config.NodeName
×
430
                        ipCR.Spec.AttachSubnets = []string{}
×
431
                        ipCR.Spec.AttachMacs = []string{}
×
432
                        if _, err := csh.KubeOvnClient.KubeovnV1().IPs().Update(context.Background(), ipCR, metav1.UpdateOptions{}); err != nil {
×
433
                                err = fmt.Errorf("failed to update ip crd for %s, %w", ip, err)
×
434
                                klog.Error(err)
×
435
                        } else {
×
436
                                return nil
×
437
                        }
×
438
                }
439
                if err != nil {
×
440
                        klog.Warningf("wait pod ip %s to be ready", ipCRName)
×
441
                        time.Sleep(1 * time.Second)
×
442
                } else {
×
443
                        return nil
×
444
                }
×
445
        }
446
        // update ip spec node is not that necessary, so we just log the error
447
        return nil
×
448
}
449

450
func (csh cniServerHandler) handleDel(req *restful.Request, resp *restful.Response) {
×
451
        var podRequest request.CniRequest
×
452
        if err := req.ReadEntity(&podRequest); err != nil {
×
453
                errMsg := fmt.Errorf("parse del request failed %w", err)
×
454
                klog.Error(errMsg)
×
455
                if err := resp.WriteHeaderAndEntity(http.StatusBadRequest, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
456
                        klog.Errorf("failed to write response, %v", err)
×
457
                }
×
458
                return
×
459
        }
460

461
        // Try to get the Pod, but if it fails due to not being found, log a warning and continue.
462
        pod, err := csh.Controller.podsLister.Pods(podRequest.PodNamespace).Get(podRequest.PodName)
×
463
        if err != nil && !k8serrors.IsNotFound(err) {
×
464
                errMsg := fmt.Errorf("failed to retrieve Pod %s/%s: %w", podRequest.PodNamespace, podRequest.PodName, err)
×
465
                klog.Error(errMsg)
×
466
                if err := resp.WriteHeaderAndEntity(http.StatusBadRequest, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
467
                        klog.Errorf("failed to write response, %v", err)
×
468
                }
×
469
                return
×
470
        }
471

472
        if podRequest.NetNs == "" {
×
473
                klog.Infof("skip del port request: %v", podRequest)
×
474
                resp.WriteHeader(http.StatusNoContent)
×
475
                return
×
476
        }
×
477

478
        klog.Infof("del port request: %v", podRequest)
×
479
        if err := csh.validatePodRequest(&podRequest); err != nil {
×
480
                klog.Error(err)
×
481
                if err := resp.WriteHeaderAndEntity(http.StatusBadRequest, request.CniResponse{Err: err.Error()}); err != nil {
×
482
                        klog.Errorf("failed to write response, %v", err)
×
483
                }
×
484
                return
×
485
        }
486

487
        var nicType string
×
488
        var vmName string
×
489

×
490
        // If the Pod was found, process its annotations and labels.
×
491
        if pod != nil {
×
492
                if pod.Annotations != nil && (util.IsOvnProvider(podRequest.Provider) || podRequest.CniType == util.CniTypeName) {
×
493
                        subnet := pod.Annotations[fmt.Sprintf(util.LogicalSwitchAnnotationTemplate, podRequest.Provider)]
×
494
                        if subnet != "" {
×
495
                                ip := pod.Annotations[fmt.Sprintf(util.IPAddressAnnotationTemplate, podRequest.Provider)]
×
496
                                if err = csh.Controller.removeEgressConfig(subnet, ip); err != nil {
×
497
                                        errMsg := fmt.Errorf("failed to remove egress configuration: %w", err)
×
498
                                        klog.Error(errMsg)
×
499
                                        if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
500
                                                klog.Errorf("failed to write response, %v", err)
×
501
                                        }
×
502
                                        return
×
503
                                }
504
                        }
505

506
                        switch {
×
507
                        case podRequest.DeviceID != "":
×
508
                                nicType = util.OffloadType
×
509
                        case podRequest.VhostUserSocketVolumeName != "":
×
510
                                nicType = util.DpdkType
×
511
                                if err = removeShortSharedDir(pod, podRequest.VhostUserSocketVolumeName, podRequest.VhostUserSocketConsumption); err != nil {
×
512
                                        klog.Error(err.Error())
×
513
                                        if err = resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: err.Error()}); err != nil {
×
514
                                                klog.Errorf("failed to write response: %v", err)
×
515
                                        }
×
516
                                        return
×
517
                                }
518
                        default:
×
519
                                nicType = pod.Annotations[fmt.Sprintf(util.PodNicAnnotationTemplate, podRequest.Provider)]
×
520
                        }
521

522
                        vmName = pod.Annotations[fmt.Sprintf(util.VMAnnotationTemplate, podRequest.Provider)]
×
523
                        if vmName != "" {
×
524
                                podRequest.PodName = vmName
×
525
                        }
×
526
                }
527
        } else {
×
528
                // If the Pod is not found, assign a default value.
×
529
                klog.Warningf("Pod %s not found, proceeding with NIC deletion using ContainerID and NetNs", podRequest.PodName)
×
530
                switch {
×
531
                case podRequest.DeviceID != "":
×
532
                        nicType = util.OffloadType
×
533
                case podRequest.VhostUserSocketVolumeName != "":
×
534
                        nicType = util.DpdkType
×
535
                default:
×
536
                        nicType = util.VethType
×
537
                }
538
        }
539

540
        // To support KubeVirt hotplug dpdk nic, forbidden set the volume name
541
        if podRequest.VhostUserSocketConsumption == util.ConsumptionKubevirt {
×
542
                podRequest.VhostUserSocketVolumeName = util.VhostUserSocketVolumeName
×
543
        }
×
544

545
        // Proceed to delete the NIC regardless of whether the Pod was found or not.
546
        err = csh.deleteNic(podRequest.PodName, podRequest.PodNamespace, podRequest.ContainerID, podRequest.NetNs, podRequest.DeviceID, podRequest.IfName, nicType)
×
547
        if err != nil {
×
548
                errMsg := fmt.Errorf("del nic failed %w", err)
×
549
                klog.Error(errMsg)
×
550
                if err := resp.WriteHeaderAndEntity(http.StatusInternalServerError, request.CniResponse{Err: errMsg.Error()}); err != nil {
×
551
                        klog.Errorf("failed to write response, %v", err)
×
552
                }
×
553
                return
×
554
        }
555
        resp.WriteHeader(http.StatusNoContent)
×
556
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc