k8s.io/kubernetes@v1.29.3/test/integration/apiserver/cve_2021_29923_test.go (about)

     1  /*
     2  Copyright 2022 The Kubernetes Authors.
     3  
     4  Licensed under the Apache License, Version 2.0 (the "License");
     5  you may not use this file except in compliance with the License.
     6  You may obtain a copy of the License at
     7  
     8      http://www.apache.org/licenses/LICENSE-2.0
     9  
    10  Unless required by applicable law or agreed to in writing, software
    11  distributed under the License is distributed on an "AS IS" BASIS,
    12  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    13  See the License for the specific language governing permissions and
    14  limitations under the License.
    15  */
    16  
    17  package apiserver
    18  
    19  import (
    20  	"context"
    21  	"encoding/json"
    22  	"testing"
    23  
    24  	metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
    25  	"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
    26  	"k8s.io/apimachinery/pkg/runtime/schema"
    27  	"k8s.io/client-go/dynamic"
    28  	clientset "k8s.io/client-go/kubernetes"
    29  	kubeapiservertesting "k8s.io/kubernetes/cmd/kube-apiserver/app/testing"
    30  	"k8s.io/kubernetes/test/integration/framework"
    31  )
    32  
    33  func gvr(g, v, r string) schema.GroupVersionResource {
    34  	return schema.GroupVersionResource{Group: g, Version: v, Resource: r}
    35  }
    36  
    37  // TestCanaryCVE_2021_29923 tests to make sure that objects that use the golang IP parsers allow IPv4 addresses with leading zeros.
    38  // Is it possible that exist more fields that can contain IPs, the test consider the most significative.
    39  // xref: https://issues.k8s.io/100895
    40  func TestCanaryCVE_2021_29923(t *testing.T) {
    41  	// Disable ServiceAccount admission plugin as we don't have serviceaccount controller running.
    42  	server := kubeapiservertesting.StartTestServerOrDie(t, nil, []string{"--disable-admission-plugins=ServiceAccount"}, framework.SharedEtcd())
    43  	defer server.TearDownFn()
    44  
    45  	client, err := clientset.NewForConfig(server.ClientConfig)
    46  	if err != nil {
    47  		t.Fatalf("unexpected error creating client: %v", err)
    48  	}
    49  
    50  	ns := framework.CreateNamespaceOrDie(client, "test-cve-2021-29923", t)
    51  	defer framework.DeleteNamespaceOrDie(client, ns, t)
    52  
    53  	dynamicClient, err := dynamic.NewForConfig(server.ClientConfig)
    54  	if err != nil {
    55  		t.Fatalf("unexpected error creating dynamic client: %v", err)
    56  	}
    57  
    58  	objects := map[schema.GroupVersionResource]string{
    59  		// k8s.io/kubernetes/pkg/api/v1
    60  		gvr("", "v1", "nodes"):     `{"kind": "Node", "apiVersion": "v1", "metadata": {"name": "node1"}, "spec": {"unschedulable": true}, "status": {"addresses":[{"address":"172.18.0.012","type":"InternalIP"}]}}`,
    61  		gvr("", "v1", "pods"):      `{"kind": "Pod", "apiVersion": "v1", "metadata": {"name": "pod1", "namespace": "test-cve-2021-29923"}, "spec": {"containers": [{"image": "` + "image" + `", "name": "container7", "resources": {"limits": {"cpu": "1M"}, "requests": {"cpu": "1M"}}}]}, "status": {"podIP":"10.244.0.05","podIPs":[{"ip":"10.244.0.05"}]}}`,
    62  		gvr("", "v1", "services"):  `{"kind": "Service", "apiVersion": "v1", "metadata": {"name": "service1", "namespace": "test-cve-2021-29923"}, "spec": {"clusterIP": "10.0.0.011", "externalIP": "192.168.0.012", "externalName": "service1name", "ports": [{"port": 10000, "targetPort": 11000}], "selector": {"test": "data"}}}`,
    63  		gvr("", "v1", "endpoints"): `{"kind": "Endpoints", "apiVersion": "v1", "metadata": {"name": "ep1name", "namespace": "test-cve-2021-29923"}, "subsets": [{"addresses": [{"hostname": "bar-001", "ip": "192.168.3.011"}], "ports": [{"port": 8000}]}]}`,
    64  		// k8s.io/kubernetes/pkg/apis/discovery/v1
    65  		gvr("discovery.k8s.io", "v1", "endpointslices"): `{"kind": "EndpointSlice", "apiVersion": "discovery.k8s.io/v1", "metadata": {"name": "slicev1", "namespace": "test-cve-2021-29923"}, "addressType": "IPv4", "protocol": "TCP", "ports": [], "endpoints": [{"addresses": ["10.244.0.011"], "conditions": {"ready": true, "serving": true, "terminating": false}, "nodeName": "control-plane"}]}`,
    66  		// k8s.io/kubernetes/pkg/apis/networking/v1
    67  		gvr("networking.k8s.io", "v1", "ingresses"):       `{"kind": "Ingress", "apiVersion": "networking.k8s.io/v1", "metadata": {"name": "ingress3", "namespace": "test-cve-2021-29923"}, "spec": {"defaultBackend": {"service":{"name":"service", "port":{"number": 5000}}}}, "status":{"loadBalancer":{"ingress": [{"ip":"10.0.0.013"}]}}}`,
    68  		gvr("networking.k8s.io", "v1", "networkpolicies"): `{"kind": "NetworkPolicy", "apiVersion": "networking.k8s.io/v1", "metadata": {"name": "np2", "namespace": "test-cve-2021-29923"}, "spec": {"egress":[{"ports":[{"port":5978,"protocol":"TCP"}],"to":[{"ipBlock":{"cidr":"10.0.012.0/24"}}]}],"ingress":[{"from":[{"ipBlock":{"cidr":"172.017.0.0/16","except":["172.17.001.0/24"]}},{"podSelector":{"matchLabels":{"role":"frontend"}}}],"ports":[{"port":6379,"protocol":"TCP"}]}],"podSelector":{"matchLabels":{"role":"db"}},"policyTypes":["Ingress","Egress"]}}`,
    69  	}
    70  
    71  	for gvr, data := range objects {
    72  		t.Run(gvr.String(), func(t *testing.T) {
    73  			obj := map[string]interface{}{}
    74  			if err := json.Unmarshal([]byte(data), &obj); err != nil {
    75  				t.Fatal(err)
    76  			}
    77  
    78  			cr := &unstructured.Unstructured{Object: obj}
    79  
    80  			_, err := dynamicClient.Resource(gvr).Namespace(cr.GetNamespace()).Create(context.TODO(), cr, metav1.CreateOptions{})
    81  			if err != nil {
    82  				t.Errorf("error creating resource %s with IPs with leading zeros %v", gvr.String(), err)
    83  			}
    84  
    85  		})
    86  	}
    87  
    88  }