Skip to content

Commit c11a46d

Browse files
committed
feature: add support for specifying pod's routes by subnet's .spec.routes
Signed-off-by: zhangzujian <[email protected]>
1 parent 57b3733 commit c11a46d

File tree

5 files changed

+123
-12
lines changed

5 files changed

+123
-12
lines changed

charts/kube-ovn/templates/kube-ovn-crd.yaml

+18
Original file line numberDiff line numberDiff line change
@@ -2461,6 +2461,24 @@ spec:
24612461
type: array
24622462
items:
24632463
type: string
2464+
routes:
2465+
description: |
2466+
Routes is a list of route rules for the pods which are in the subnet.
2467+
If specified, the routes will be added/replaced to the pod's network namespace.
2468+
type: array
2469+
items:
2470+
type: object
2471+
properties:
2472+
dst:
2473+
type: string
2474+
format: cidr
2475+
gw:
2476+
type: string
2477+
anyOf:
2478+
- format: ipv4
2479+
- format: ipv6
2480+
required:
2481+
- gw
24642482
gatewayType:
24652483
type: string
24662484
allowSubnets:

dist/images/install.sh

+18
Original file line numberDiff line numberDiff line change
@@ -2712,6 +2712,24 @@ spec:
27122712
type: array
27132713
items:
27142714
type: string
2715+
routes:
2716+
description: |
2717+
Routes is a list of route rules for the pods which are in the subnet.
2718+
If specified, the routes will be added/replaced to the pod's network namespace.
2719+
type: array
2720+
items:
2721+
type: object
2722+
properties:
2723+
dst:
2724+
type: string
2725+
format: cidr
2726+
gw:
2727+
type: string
2728+
anyOf:
2729+
- format: ipv4
2730+
- format: ipv6
2731+
required:
2732+
- gw
27152733
gatewayType:
27162734
type: string
27172735
allowSubnets:

pkg/apis/kubeovn/v1/subnet.go

+7
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,8 @@ import (
77
corev1 "k8s.io/api/core/v1"
88
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
99
"k8s.io/klog/v2"
10+
11+
"github.com/kubeovn/kube-ovn/pkg/request"
1012
)
1113

1214
const (
@@ -49,6 +51,11 @@ type SubnetSpec struct {
4951
ExcludeIps []string `json:"excludeIps,omitempty"`
5052
Provider string `json:"provider,omitempty"`
5153

54+
// Routes is a list of route rules for the pods which are in the subnet.
55+
// If specified, the routes will be added/replaced to the pod's network namespace.
56+
// +optional
57+
Routes []request.Route `json:"routes,omitempty"`
58+
5259
GatewayType string `json:"gatewayType,omitempty"`
5360
GatewayNode string `json:"gatewayNode"`
5461
NatOutgoing bool `json:"natOutgoing"`

pkg/daemon/handler.go

+11-12
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ import (
66
"fmt"
77
"net"
88
"net/http"
9+
"slices"
910
"strconv"
10-
"strings"
1111
"time"
1212

1313
"github.com/emicklei/go-restful/v3"
@@ -226,19 +226,18 @@ func (csh cniServerHandler) handleAdd(req *restful.Request, resp *restful.Respon
226226
return
227227
}
228228

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

238+
var mtu int
239+
routes = slices.Concat(podRequest.Routes, podSubnet.Spec.Routes, routes)
240+
if util.IsOvnProvider(podRequest.Provider) {
242241
if podSubnet.Status.U2OInterconnectionIP == "" && podSubnet.Spec.U2OInterconnection {
243242
errMsg := fmt.Errorf("failed to generate u2o ip on subnet %s", podSubnet.Name)
244243
klog.Error(errMsg)

test/e2e/kube-ovn/subnet/subnet.go

+69
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import (
1313
"time"
1414

1515
corev1 "k8s.io/api/core/v1"
16+
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
1617
clientset "k8s.io/client-go/kubernetes"
1718
e2enode "k8s.io/kubernetes/test/e2e/framework/node"
1819
e2epodoutput "k8s.io/kubernetes/test/e2e/framework/pod/output"
@@ -21,6 +22,7 @@ import (
2122
"github.com/onsi/gomega"
2223

2324
apiv1 "github.com/kubeovn/kube-ovn/pkg/apis/kubeovn/v1"
25+
"github.com/kubeovn/kube-ovn/pkg/request"
2426
"github.com/kubeovn/kube-ovn/pkg/util"
2527
"github.com/kubeovn/kube-ovn/test/e2e/framework"
2628
"github.com/kubeovn/kube-ovn/test/e2e/framework/docker"
@@ -289,6 +291,73 @@ var _ = framework.Describe("[group:subnet]", func() {
289291
}
290292
})
291293

294+
framework.ConformanceIt(`should be able to configure pod routes by subnet's ".spec.routes" field`, func() {
295+
subnet = framework.MakeSubnet(subnetName, "", cidr, "", "", "", nil, nil, nil)
296+
ginkgo.By("Creating subnet " + subnetName + " with cidr " + subnet.Spec.CIDRBlock)
297+
subnet = subnetClient.CreateSync(subnet)
298+
299+
ginkgo.By(`Constructing specified routes by subnet's ".spec.routes" field`)
300+
var routeDst string
301+
for i := 0; i < 3; i++ {
302+
routeDst = framework.RandomCIDR(f.ClusterIPFamily)
303+
if routeDst != subnet.Spec.CIDRBlock {
304+
break
305+
}
306+
}
307+
framework.ExpectNotEqual(routeDst, subnet.Spec.CIDRBlock)
308+
routeGw := framework.RandomIPs(subnet.Spec.CIDRBlock, "", 1)
309+
ipv4Gateway, ipv6Gateway := util.SplitStringIP(subnet.Spec.Gateway)
310+
ipv4RouteDst, ipv6RouteDst := util.SplitStringIP(routeDst)
311+
ipv4RouteGw, ipv6RouteGw := util.SplitStringIP(routeGw)
312+
routes := make([]request.Route, 0, 4)
313+
if f.HasIPv4() {
314+
routes = append(routes, request.Route{Gateway: ipv4RouteGw})
315+
routes = append(routes, request.Route{Destination: ipv4RouteDst, Gateway: ipv4Gateway})
316+
317+
}
318+
if f.HasIPv6() {
319+
routes = append(routes, request.Route{Gateway: ipv6RouteGw})
320+
routes = append(routes, request.Route{Destination: ipv6RouteDst, Gateway: ipv6Gateway})
321+
}
322+
323+
ginkgo.By("Updating subnet " + subnetName + " with routes " + fmt.Sprintf("%v", routes))
324+
subnet.Spec.Routes = routes
325+
subnet = subnetClient.Update(subnet, metav1.UpdateOptions{}, 2*time.Second)
326+
327+
ginkgo.By("Creating pod " + podName)
328+
cmd := []string{"sleep", "infinity"}
329+
annotations := map[string]string{util.LogicalSwitchAnnotation: subnetName}
330+
pod := framework.MakePrivilegedPod(namespaceName, podName, nil, annotations, f.KubeOVNImage, cmd, nil)
331+
_ = podClient.CreateSync(pod)
332+
333+
ginkgo.By("Retrieving pod routes")
334+
podRoutes, err := iproute.RouteShow("", "", func(cmd ...string) ([]byte, []byte, error) {
335+
return framework.KubectlExec(namespaceName, podName, cmd...)
336+
})
337+
framework.ExpectNoError(err)
338+
339+
ginkgo.By("Validating pod routes")
340+
actualRoutes := make([]request.Route, 0, len(podRoutes))
341+
for _, r := range podRoutes {
342+
if r.Gateway != "" || r.Dst != "" {
343+
actualRoutes = append(actualRoutes, request.Route{Destination: r.Dst, Gateway: r.Gateway})
344+
}
345+
}
346+
ipv4CIDR, ipv6CIDR := util.SplitStringIP(subnet.Spec.CIDRBlock)
347+
if f.HasIPv4() {
348+
framework.ExpectContainElement(actualRoutes, request.Route{Destination: ipv4CIDR})
349+
framework.ExpectNotContainElement(actualRoutes, request.Route{Destination: "default", Gateway: ipv4Gateway})
350+
framework.ExpectContainElement(actualRoutes, request.Route{Destination: "default", Gateway: ipv4RouteGw})
351+
framework.ExpectContainElement(actualRoutes, request.Route{Destination: ipv4RouteDst, Gateway: ipv4Gateway})
352+
}
353+
if f.HasIPv6() {
354+
framework.ExpectContainElement(actualRoutes, request.Route{Destination: ipv6CIDR})
355+
framework.ExpectNotContainElement(actualRoutes, request.Route{Destination: "default", Gateway: ipv6Gateway})
356+
framework.ExpectContainElement(actualRoutes, request.Route{Destination: "default", Gateway: ipv6RouteGw})
357+
framework.ExpectContainElement(actualRoutes, request.Route{Destination: ipv6RouteDst, Gateway: ipv6Gateway})
358+
}
359+
})
360+
292361
framework.ConformanceIt("should create subnet with centralized gateway", func() {
293362
ginkgo.By("Getting nodes")
294363
nodes, err := e2enode.GetReadySchedulableNodes(context.Background(), cs)

0 commit comments

Comments
 (0)