123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699 |
- /*
- Copyright 2014 The Kubernetes Authors.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package pod
- import (
- "context"
- "fmt"
- "net/http"
- "net/url"
- "reflect"
- "testing"
- "k8s.io/apimachinery/pkg/api/errors"
- "k8s.io/apimachinery/pkg/api/resource"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/fields"
- "k8s.io/apimachinery/pkg/labels"
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/types"
- genericapirequest "k8s.io/apiserver/pkg/endpoints/request"
- utilfeature "k8s.io/apiserver/pkg/util/feature"
- "k8s.io/client-go/tools/cache"
- featuregatetesting "k8s.io/component-base/featuregate/testing"
- apitesting "k8s.io/kubernetes/pkg/api/testing"
- api "k8s.io/kubernetes/pkg/apis/core"
- "k8s.io/kubernetes/pkg/features"
- "k8s.io/kubernetes/pkg/kubelet/client"
- // ensure types are installed
- _ "k8s.io/kubernetes/pkg/apis/core/install"
- )
- func TestMatchPod(t *testing.T) {
- testCases := []struct {
- in *api.Pod
- fieldSelector fields.Selector
- expectMatch bool
- }{
- {
- in: &api.Pod{
- Spec: api.PodSpec{NodeName: "nodeA"},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.nodeName=nodeA"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{NodeName: "nodeB"},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.nodeName=nodeA"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{RestartPolicy: api.RestartPolicyAlways},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.restartPolicy=Always"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{RestartPolicy: api.RestartPolicyAlways},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.restartPolicy=Never"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{SchedulerName: "scheduler1"},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.schedulerName=scheduler1"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{SchedulerName: "scheduler1"},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.schedulerName=scheduler2"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{ServiceAccountName: "serviceAccount1"},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.serviceAccountName=serviceAccount1"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{SchedulerName: "serviceAccount1"},
- },
- fieldSelector: fields.ParseSelectorOrDie("spec.serviceAccountName=serviceAccount2"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{Phase: api.PodRunning},
- },
- fieldSelector: fields.ParseSelectorOrDie("status.phase=Running"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{Phase: api.PodRunning},
- },
- fieldSelector: fields.ParseSelectorOrDie("status.phase=Pending"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{
- PodIPs: []api.PodIP{
- {IP: "1.2.3.4"},
- },
- },
- },
- fieldSelector: fields.ParseSelectorOrDie("status.podIP=1.2.3.4"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{
- PodIPs: []api.PodIP{
- {IP: "1.2.3.4"},
- },
- },
- },
- fieldSelector: fields.ParseSelectorOrDie("status.podIP=4.3.2.1"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{NominatedNodeName: "node1"},
- },
- fieldSelector: fields.ParseSelectorOrDie("status.nominatedNodeName=node1"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{NominatedNodeName: "node1"},
- },
- fieldSelector: fields.ParseSelectorOrDie("status.nominatedNodeName=node2"),
- expectMatch: false,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{
- PodIPs: []api.PodIP{
- {IP: "2001:db8::"},
- },
- },
- },
- fieldSelector: fields.ParseSelectorOrDie("status.podIP=2001:db8::"),
- expectMatch: true,
- },
- {
- in: &api.Pod{
- Status: api.PodStatus{
- PodIPs: []api.PodIP{
- {IP: "2001:db8::"},
- },
- },
- },
- fieldSelector: fields.ParseSelectorOrDie("status.podIP=2001:db7::"),
- expectMatch: false,
- },
- }
- for _, testCase := range testCases {
- m := MatchPod(labels.Everything(), testCase.fieldSelector)
- result, err := m.Matches(testCase.in)
- if err != nil {
- t.Errorf("Unexpected error %v", err)
- }
- if result != testCase.expectMatch {
- t.Errorf("Result %v, Expected %v, Selector: %v, Pod: %v", result, testCase.expectMatch, testCase.fieldSelector.String(), testCase.in)
- }
- }
- }
- func getResourceList(cpu, memory string) api.ResourceList {
- res := api.ResourceList{}
- if cpu != "" {
- res[api.ResourceCPU] = resource.MustParse(cpu)
- }
- if memory != "" {
- res[api.ResourceMemory] = resource.MustParse(memory)
- }
- return res
- }
- func getResourceRequirements(requests, limits api.ResourceList) api.ResourceRequirements {
- res := api.ResourceRequirements{}
- res.Requests = requests
- res.Limits = limits
- return res
- }
- func newContainer(name string, requests api.ResourceList, limits api.ResourceList) api.Container {
- return api.Container{
- Name: name,
- Resources: getResourceRequirements(requests, limits),
- }
- }
- func newPod(name string, containers []api.Container) *api.Pod {
- return &api.Pod{
- ObjectMeta: metav1.ObjectMeta{
- Name: name,
- },
- Spec: api.PodSpec{
- Containers: containers,
- },
- }
- }
- func TestGetPodQOS(t *testing.T) {
- testCases := []struct {
- pod *api.Pod
- expected api.PodQOSClass
- }{
- {
- pod: newPod("guaranteed", []api.Container{
- newContainer("guaranteed", getResourceList("100m", "100Mi"), getResourceList("100m", "100Mi")),
- }),
- expected: api.PodQOSGuaranteed,
- },
- {
- pod: newPod("best-effort", []api.Container{
- newContainer("best-effort", getResourceList("", ""), getResourceList("", "")),
- }),
- expected: api.PodQOSBestEffort,
- },
- {
- pod: newPod("burstable", []api.Container{
- newContainer("burstable", getResourceList("100m", "100Mi"), getResourceList("", "")),
- }),
- expected: api.PodQOSBurstable,
- },
- }
- for id, testCase := range testCases {
- Strategy.PrepareForCreate(genericapirequest.NewContext(), testCase.pod)
- actual := testCase.pod.Status.QOSClass
- if actual != testCase.expected {
- t.Errorf("[%d]: invalid qos pod %s, expected: %s, actual: %s", id, testCase.pod.Name, testCase.expected, actual)
- }
- }
- }
- func TestCheckGracefulDelete(t *testing.T) {
- defaultGracePeriod := int64(30)
- tcs := []struct {
- in *api.Pod
- gracePeriod int64
- }{
- {
- in: &api.Pod{
- Spec: api.PodSpec{NodeName: "something"},
- Status: api.PodStatus{Phase: api.PodPending},
- },
- gracePeriod: defaultGracePeriod,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{NodeName: "something"},
- Status: api.PodStatus{Phase: api.PodFailed},
- },
- gracePeriod: 0,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{},
- Status: api.PodStatus{Phase: api.PodPending},
- },
- gracePeriod: 0,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{},
- Status: api.PodStatus{Phase: api.PodSucceeded},
- },
- gracePeriod: 0,
- },
- {
- in: &api.Pod{
- Spec: api.PodSpec{},
- Status: api.PodStatus{},
- },
- gracePeriod: 0,
- },
- }
- for _, tc := range tcs {
- out := &metav1.DeleteOptions{GracePeriodSeconds: &defaultGracePeriod}
- Strategy.CheckGracefulDelete(genericapirequest.NewContext(), tc.in, out)
- if out.GracePeriodSeconds == nil {
- t.Errorf("out grace period was nil but supposed to be %v", tc.gracePeriod)
- }
- if *(out.GracePeriodSeconds) != tc.gracePeriod {
- t.Errorf("out grace period was %v but was expected to be %v", *out, tc.gracePeriod)
- }
- }
- }
- type mockPodGetter struct {
- pod *api.Pod
- }
- func (g mockPodGetter) Get(context.Context, string, *metav1.GetOptions) (runtime.Object, error) {
- return g.pod, nil
- }
- func TestCheckLogLocation(t *testing.T) {
- defer featuregatetesting.SetFeatureGateDuringTest(t, utilfeature.DefaultFeatureGate, features.EphemeralContainers, true)()
- ctx := genericapirequest.NewDefaultContext()
- fakePodName := "test"
- tcs := []struct {
- name string
- in *api.Pod
- opts *api.PodLogOptions
- expectedErr error
- expectedTransport http.RoundTripper
- }{
- {
- name: "simple",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "mycontainer"},
- },
- NodeName: "foo",
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{},
- expectedErr: nil,
- expectedTransport: fakeSecureRoundTripper,
- },
- {
- name: "insecure",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "mycontainer"},
- },
- NodeName: "foo",
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{
- InsecureSkipTLSVerifyBackend: true,
- },
- expectedErr: nil,
- expectedTransport: fakeInsecureRoundTripper,
- },
- {
- name: "missing container",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{},
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{},
- expectedErr: errors.NewBadRequest("a container name must be specified for pod test"),
- expectedTransport: nil,
- },
- {
- name: "choice of two containers",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "container1"},
- {Name: "container2"},
- },
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{},
- expectedErr: errors.NewBadRequest("a container name must be specified for pod test, choose one of: [container1 container2]"),
- expectedTransport: nil,
- },
- {
- name: "initcontainers",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "container1"},
- {Name: "container2"},
- },
- InitContainers: []api.Container{
- {Name: "initcontainer1"},
- },
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{},
- expectedErr: errors.NewBadRequest("a container name must be specified for pod test, choose one of: [initcontainer1 container1 container2]"),
- expectedTransport: nil,
- },
- {
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "container1"},
- {Name: "container2"},
- },
- InitContainers: []api.Container{
- {Name: "initcontainer1"},
- },
- EphemeralContainers: []api.EphemeralContainer{
- {EphemeralContainerCommon: api.EphemeralContainerCommon{Name: "debugger"}},
- },
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{},
- expectedErr: errors.NewBadRequest("a container name must be specified for pod test, choose one of: [initcontainer1 container1 container2 debugger]"),
- expectedTransport: nil,
- },
- {
- name: "bad container",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "container1"},
- {Name: "container2"},
- },
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{
- Container: "unknown",
- },
- expectedErr: errors.NewBadRequest("container unknown is not valid for pod test"),
- expectedTransport: nil,
- },
- {
- name: "good with two containers",
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Name: fakePodName},
- Spec: api.PodSpec{
- Containers: []api.Container{
- {Name: "container1"},
- {Name: "container2"},
- },
- NodeName: "foo",
- },
- Status: api.PodStatus{},
- },
- opts: &api.PodLogOptions{
- Container: "container2",
- },
- expectedErr: nil,
- expectedTransport: fakeSecureRoundTripper,
- },
- }
- for _, tc := range tcs {
- t.Run(tc.name, func(t *testing.T) {
- getter := &mockPodGetter{tc.in}
- connectionGetter := &mockConnectionInfoGetter{&client.ConnectionInfo{
- Transport: fakeSecureRoundTripper,
- InsecureSkipTLSVerifyTransport: fakeInsecureRoundTripper,
- }}
- _, actualTransport, err := LogLocation(ctx, getter, connectionGetter, fakePodName, tc.opts)
- if !reflect.DeepEqual(err, tc.expectedErr) {
- t.Errorf("expected %q, got %q", tc.expectedErr, err)
- }
- if actualTransport != tc.expectedTransport {
- t.Errorf("expected %q, got %q", tc.expectedTransport, actualTransport)
- }
- })
- }
- }
- func TestSelectableFieldLabelConversions(t *testing.T) {
- apitesting.TestSelectableFieldLabelConversionsOfKind(t,
- "v1",
- "Pod",
- ToSelectableFields(&api.Pod{}),
- nil,
- )
- }
- type mockConnectionInfoGetter struct {
- info *client.ConnectionInfo
- }
- func (g mockConnectionInfoGetter) GetConnectionInfo(ctx context.Context, nodeName types.NodeName) (*client.ConnectionInfo, error) {
- return g.info, nil
- }
- func TestPortForwardLocation(t *testing.T) {
- ctx := genericapirequest.NewDefaultContext()
- tcs := []struct {
- in *api.Pod
- info *client.ConnectionInfo
- opts *api.PodPortForwardOptions
- expectedErr error
- expectedURL *url.URL
- }{
- {
- in: &api.Pod{
- Spec: api.PodSpec{},
- },
- opts: &api.PodPortForwardOptions{},
- expectedErr: errors.NewBadRequest("pod test does not have a host assigned"),
- },
- {
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: "ns",
- Name: "pod1",
- },
- Spec: api.PodSpec{
- NodeName: "node1",
- },
- },
- info: &client.ConnectionInfo{},
- opts: &api.PodPortForwardOptions{},
- expectedURL: &url.URL{Host: ":", Path: "/portForward/ns/pod1"},
- },
- {
- in: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: "ns",
- Name: "pod1",
- },
- Spec: api.PodSpec{
- NodeName: "node1",
- },
- },
- info: &client.ConnectionInfo{},
- opts: &api.PodPortForwardOptions{Ports: []int32{80}},
- expectedURL: &url.URL{Host: ":", Path: "/portForward/ns/pod1", RawQuery: "port=80"},
- },
- }
- for _, tc := range tcs {
- getter := &mockPodGetter{tc.in}
- connectionGetter := &mockConnectionInfoGetter{tc.info}
- loc, _, err := PortForwardLocation(ctx, getter, connectionGetter, "test", tc.opts)
- if !reflect.DeepEqual(err, tc.expectedErr) {
- t.Errorf("expected %v, got %v", tc.expectedErr, err)
- }
- if !reflect.DeepEqual(loc, tc.expectedURL) {
- t.Errorf("expected %v, got %v", tc.expectedURL, loc)
- }
- }
- }
- func TestGetPodIP(t *testing.T) {
- testCases := []struct {
- name string
- pod *api.Pod
- expectedIP string
- }{
- {
- name: "nil pod",
- pod: nil,
- expectedIP: "",
- },
- {
- name: "no status object",
- pod: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Namespace: "ns", Name: "pod1"},
- Spec: api.PodSpec{},
- },
- expectedIP: "",
- },
- {
- name: "no pod ips",
- pod: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Namespace: "ns", Name: "pod1"},
- Spec: api.PodSpec{},
- Status: api.PodStatus{},
- },
- expectedIP: "",
- },
- {
- name: "empty list",
- pod: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Namespace: "ns", Name: "pod1"},
- Spec: api.PodSpec{},
- Status: api.PodStatus{
- PodIPs: []api.PodIP{},
- },
- },
- expectedIP: "",
- },
- {
- name: "1 ip",
- pod: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Namespace: "ns", Name: "pod1"},
- Spec: api.PodSpec{},
- Status: api.PodStatus{
- PodIPs: []api.PodIP{
- {IP: "10.0.0.10"},
- },
- },
- },
- expectedIP: "10.0.0.10",
- },
- {
- name: "multiple ips",
- pod: &api.Pod{
- ObjectMeta: metav1.ObjectMeta{Namespace: "ns", Name: "pod1"},
- Spec: api.PodSpec{},
- Status: api.PodStatus{
- PodIPs: []api.PodIP{
- {IP: "10.0.0.10"},
- {IP: "10.0.0.20"},
- },
- },
- },
- expectedIP: "10.0.0.10",
- },
- }
- for _, tc := range testCases {
- t.Run(tc.name, func(t *testing.T) {
- podIP := getPodIP(tc.pod)
- if podIP != tc.expectedIP {
- t.Errorf("expected pod ip:%v does not match actual %v", tc.expectedIP, podIP)
- }
- })
- }
- }
- type fakeTransport struct {
- val string
- }
- func (f fakeTransport) RoundTrip(*http.Request) (*http.Response, error) {
- return nil, nil
- }
- var (
- fakeSecureRoundTripper = fakeTransport{val: "secure"}
- fakeInsecureRoundTripper = fakeTransport{val: "insecure"}
- )
- func TestPodIndexFunc(t *testing.T) {
- tcs := []struct {
- name string
- indexFunc cache.IndexFunc
- pod interface{}
- expectedValue string
- expectedErr error
- }{
- {
- name: "node name index",
- indexFunc: NodeNameIndexFunc,
- pod: &api.Pod{
- Spec: api.PodSpec{
- NodeName: "test-pod",
- },
- },
- expectedValue: "test-pod",
- expectedErr: nil,
- },
- {
- name: "not a pod failed",
- indexFunc: NodeNameIndexFunc,
- pod: "not a pod object",
- expectedValue: "test-pod",
- expectedErr: fmt.Errorf("not a pod"),
- },
- }
- for _, tc := range tcs {
- indexValues, err := tc.indexFunc(tc.pod)
- if !reflect.DeepEqual(err, tc.expectedErr) {
- t.Errorf("name %v, expected %v, got %v", tc.name, tc.expectedErr, err)
- }
- if err == nil && len(indexValues) != 1 && indexValues[0] != tc.expectedValue {
- t.Errorf("name %v, expected %v, got %v", tc.name, tc.expectedValue, indexValues)
- }
- }
- }
|