123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714 |
- /*
- 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 delete
- import (
- "encoding/json"
- "io"
- "io/ioutil"
- "net/http"
- "strings"
- "testing"
- "github.com/spf13/cobra"
- corev1 "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/api/errors"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/cli-runtime/pkg/genericclioptions"
- "k8s.io/cli-runtime/pkg/resource"
- "k8s.io/client-go/rest/fake"
- cmdtesting "k8s.io/kubernetes/pkg/kubectl/cmd/testing"
- "k8s.io/kubernetes/pkg/kubectl/scheme"
- )
- func fakecmd() *cobra.Command {
- cmd := &cobra.Command{
- Use: "delete ([-f FILENAME] | TYPE [(NAME | -l label | --all)])",
- DisableFlagsInUseLine: true,
- Run: func(cmd *cobra.Command, args []string) {},
- }
- return cmd
- }
- func TestDeleteObjectByTuple(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, _, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- // replication controller with cascade off
- case p == "/namespaces/test/replicationcontrollers/redis-master-controller" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- // secret with cascade on, but no client-side reaper
- case p == "/namespaces/test/secrets/mysecret" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- default:
- // Ensures no GET is performed when deleting by name
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"replicationcontrollers/redis-master-controller"})
- if buf.String() != "replicationcontroller/redis-master-controller\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- // Test cascading delete of object without client-side reaper doesn't make GET requests
- streams, _, buf, _ = genericclioptions.NewTestIOStreams()
- cmd = NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"secrets/mysecret"})
- if buf.String() != "secret/mysecret\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func hasExpectedPropagationPolicy(body io.ReadCloser, policy *metav1.DeletionPropagation) bool {
- if body == nil || policy == nil {
- return body == nil && policy == nil
- }
- var parsedBody metav1.DeleteOptions
- rawBody, _ := ioutil.ReadAll(body)
- json.Unmarshal(rawBody, &parsedBody)
- if parsedBody.PropagationPolicy == nil {
- return false
- }
- return *policy == *parsedBody.PropagationPolicy
- }
- // Tests that DeleteOptions.OrphanDependents is appropriately set while deleting objects.
- func TestOrphanDependentsInDeleteObject(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, _, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- var policy *metav1.DeletionPropagation
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m, b := req.URL.Path, req.Method, req.Body; {
- case p == "/namespaces/test/secrets/mysecret" && m == "DELETE" && hasExpectedPropagationPolicy(b, policy):
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- default:
- return nil, nil
- }
- }),
- }
- // DeleteOptions.PropagationPolicy should be Background, when cascade is true (default).
- backgroundPolicy := metav1.DeletePropagationBackground
- policy = &backgroundPolicy
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"secrets/mysecret"})
- if buf.String() != "secret/mysecret\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- // Test that delete options should be set to orphan when cascade is false.
- orphanPolicy := metav1.DeletePropagationOrphan
- policy = &orphanPolicy
- streams, _, buf, _ = genericclioptions.NewTestIOStreams()
- cmd = NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"secrets/mysecret"})
- if buf.String() != "secret/mysecret\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteNamedObject(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- cmdtesting.InitTestErrorHandler(t)
- _, _, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- // replication controller with cascade off
- case p == "/namespaces/test/replicationcontrollers/redis-master-controller" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- // secret with cascade on, but no client-side reaper
- case p == "/namespaces/test/secrets/mysecret" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- default:
- // Ensures no GET is performed when deleting by name
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"replicationcontrollers", "redis-master-controller"})
- if buf.String() != "replicationcontroller/redis-master-controller\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- // Test cascading delete of object without client-side reaper doesn't make GET requests
- streams, _, buf, _ = genericclioptions.NewTestIOStreams()
- cmd = NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"secrets", "mysecret"})
- if buf.String() != "secret/mysecret\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteObject(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, _, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/replicationcontrollers/redis-master" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("filename", "../../../../test/e2e/testing-manifests/guestbook/legacy/redis-master-controller.yaml")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{})
- // uses the name from the file, not the response
- if buf.String() != "replicationcontroller/redis-master\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteObjectGraceZero(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- pods, _, _ := cmdtesting.TestData()
- count := 0
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- t.Logf("got request %s %s", req.Method, req.URL.Path)
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/pods/nginx" && m == "GET":
- count++
- switch count {
- case 1, 2, 3:
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &pods.Items[0])}, nil
- default:
- return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &metav1.Status{})}, nil
- }
- case p == "/api/v1/namespaces/test" && m == "GET":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &corev1.Namespace{})}, nil
- case p == "/namespaces/test/pods/nginx" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &pods.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, errBuf := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("output", "name")
- cmd.Flags().Set("grace-period", "0")
- cmd.Run(cmd, []string{"pods/nginx"})
- // uses the name from the file, not the response
- if buf.String() != "pod/nginx\n" {
- t.Errorf("unexpected output: %s\n---\n%s", buf.String(), errBuf.String())
- }
- if count != 0 {
- t.Errorf("unexpected calls to GET: %d", count)
- }
- }
- func TestDeleteObjectNotFound(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/replicationcontrollers/redis-master" && m == "DELETE":
- return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.StringBody("")}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- options := &DeleteOptions{
- FilenameOptions: resource.FilenameOptions{
- Filenames: []string{"../../../../test/e2e/testing-manifests/guestbook/legacy/redis-master-controller.yaml"},
- },
- GracePeriod: -1,
- Cascade: false,
- Output: "name",
- IOStreams: genericclioptions.NewTestIOStreamsDiscard(),
- }
- err := options.Complete(tf, []string{}, fakecmd())
- if err != nil {
- t.Errorf("unexpected error: %v", err)
- }
- err = options.RunDelete()
- if err == nil || !errors.IsNotFound(err) {
- t.Errorf("unexpected error: expected NotFound, got %v", err)
- }
- }
- func TestDeleteObjectIgnoreNotFound(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/replicationcontrollers/redis-master" && m == "DELETE":
- return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.StringBody("")}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("filename", "../../../../test/e2e/testing-manifests/guestbook/legacy/redis-master-controller.yaml")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("ignore-not-found", "true")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{})
- if buf.String() != "" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteAllNotFound(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, svc, _ := cmdtesting.TestData()
- // Add an item to the list which will result in a 404 on delete
- svc.Items = append(svc.Items, corev1.Service{ObjectMeta: metav1.ObjectMeta{Name: "foo"}})
- notFoundError := &errors.NewNotFound(corev1.Resource("services"), "foo").ErrStatus
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/services" && m == "GET":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, svc)}, nil
- case p == "/namespaces/test/services/foo" && m == "DELETE":
- return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, notFoundError)}, nil
- case p == "/namespaces/test/services/baz" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- // Make sure we can explicitly choose to fail on NotFound errors, even with --all
- options := &DeleteOptions{
- FilenameOptions: resource.FilenameOptions{},
- GracePeriod: -1,
- Cascade: false,
- DeleteAll: true,
- IgnoreNotFound: false,
- Output: "name",
- IOStreams: genericclioptions.NewTestIOStreamsDiscard(),
- }
- err := options.Complete(tf, []string{"services"}, fakecmd())
- if err != nil {
- t.Errorf("unexpected error: %v", err)
- }
- err = options.RunDelete()
- if err == nil || !errors.IsNotFound(err) {
- t.Errorf("unexpected error: expected NotFound, got %v", err)
- }
- }
- func TestDeleteAllIgnoreNotFound(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, svc, _ := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- // Add an item to the list which will result in a 404 on delete
- svc.Items = append(svc.Items, corev1.Service{ObjectMeta: metav1.ObjectMeta{Name: "foo"}})
- notFoundError := &errors.NewNotFound(corev1.Resource("services"), "foo").ErrStatus
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/services" && m == "GET":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, svc)}, nil
- case p == "/namespaces/test/services/foo" && m == "DELETE":
- return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, notFoundError)}, nil
- case p == "/namespaces/test/services/baz" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("all", "true")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"services"})
- if buf.String() != "service/baz\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteMultipleObject(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, svc, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/replicationcontrollers/redis-master" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- case p == "/namespaces/test/services/frontend" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("filename", "../../../../test/e2e/testing-manifests/guestbook/legacy/redis-master-controller.yaml")
- cmd.Flags().Set("filename", "../../../../test/e2e/testing-manifests/guestbook/frontend-service.yaml")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{})
- if buf.String() != "replicationcontroller/redis-master\nservice/frontend\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteMultipleObjectContinueOnMissing(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, svc, _ := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/replicationcontrollers/redis-master" && m == "DELETE":
- return &http.Response{StatusCode: 404, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.StringBody("")}, nil
- case p == "/namespaces/test/services/frontend" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- options := &DeleteOptions{
- FilenameOptions: resource.FilenameOptions{
- Filenames: []string{"../../../../test/e2e/testing-manifests/guestbook/legacy/redis-master-controller.yaml", "../../../../test/e2e/testing-manifests/guestbook/frontend-service.yaml"},
- },
- GracePeriod: -1,
- Cascade: false,
- Output: "name",
- IOStreams: streams,
- }
- err := options.Complete(tf, []string{}, fakecmd())
- if err != nil {
- t.Errorf("unexpected error: %v", err)
- }
- err = options.RunDelete()
- if err == nil || !errors.IsNotFound(err) {
- t.Errorf("unexpected error: expected NotFound, got %v", err)
- }
- if buf.String() != "service/frontend\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteMultipleResourcesWithTheSameName(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, svc, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/replicationcontrollers/baz" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- case p == "/namespaces/test/replicationcontrollers/foo" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- case p == "/namespaces/test/services/baz" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- case p == "/namespaces/test/services/foo" && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- default:
- // Ensures no GET is performed when deleting by name
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("namespace", "test")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"replicationcontrollers,services", "baz", "foo"})
- if buf.String() != "replicationcontroller/baz\nreplicationcontroller/foo\nservice/baz\nservice/foo\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteDirectory(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- _, _, rc := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case strings.HasPrefix(p, "/namespaces/test/replicationcontrollers/") && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &rc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("filename", "../../../../test/e2e/testing-manifests/guestbook/legacy")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{})
- if buf.String() != "replicationcontroller/frontend\nreplicationcontroller/redis-master\nreplicationcontroller/redis-slave\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestDeleteMultipleSelector(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- pods, svc, _ := cmdtesting.TestData()
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- codec := scheme.Codecs.LegacyCodec(scheme.Scheme.PrioritizedVersionsAllGroups()...)
- tf.UnstructuredClient = &fake.RESTClient{
- NegotiatedSerializer: resource.UnstructuredPlusDefaultContentConfig().NegotiatedSerializer,
- Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
- switch p, m := req.URL.Path, req.Method; {
- case p == "/namespaces/test/pods" && m == "GET":
- if req.URL.Query().Get(metav1.LabelSelectorQueryParam("v1")) != "a=b" {
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- }
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, pods)}, nil
- case p == "/namespaces/test/services" && m == "GET":
- if req.URL.Query().Get(metav1.LabelSelectorQueryParam("v1")) != "a=b" {
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- }
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, svc)}, nil
- case strings.HasPrefix(p, "/namespaces/test/pods/") && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &pods.Items[0])}, nil
- case strings.HasPrefix(p, "/namespaces/test/services/") && m == "DELETE":
- return &http.Response{StatusCode: 200, Header: cmdtesting.DefaultHeader(), Body: cmdtesting.ObjBody(codec, &svc.Items[0])}, nil
- default:
- t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
- return nil, nil
- }
- }),
- }
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- cmd := NewCmdDelete(tf, streams)
- cmd.Flags().Set("selector", "a=b")
- cmd.Flags().Set("cascade", "false")
- cmd.Flags().Set("output", "name")
- cmd.Run(cmd, []string{"pods,services"})
- if buf.String() != "pod/foo\npod/bar\nservice/baz\n" {
- t.Errorf("unexpected output: %s", buf.String())
- }
- }
- func TestResourceErrors(t *testing.T) {
- cmdtesting.InitTestErrorHandler(t)
- testCases := map[string]struct {
- args []string
- errFn func(error) bool
- }{
- "no args": {
- args: []string{},
- errFn: func(err error) bool { return strings.Contains(err.Error(), "You must provide one or more resources") },
- },
- "resources but no selectors": {
- args: []string{"pods"},
- errFn: func(err error) bool {
- return strings.Contains(err.Error(), "resource(s) were provided, but no name, label selector, or --all flag specified")
- },
- },
- "multiple resources but no selectors": {
- args: []string{"pods,deployments"},
- errFn: func(err error) bool {
- return strings.Contains(err.Error(), "resource(s) were provided, but no name, label selector, or --all flag specified")
- },
- },
- }
- for k, testCase := range testCases {
- t.Run(k, func(t *testing.T) {
- tf := cmdtesting.NewTestFactory().WithNamespace("test")
- defer tf.Cleanup()
- tf.ClientConfigVal = cmdtesting.DefaultClientConfig()
- streams, _, buf, _ := genericclioptions.NewTestIOStreams()
- options := &DeleteOptions{
- FilenameOptions: resource.FilenameOptions{},
- GracePeriod: -1,
- Cascade: false,
- Output: "name",
- IOStreams: streams,
- }
- err := options.Complete(tf, testCase.args, fakecmd())
- if !testCase.errFn(err) {
- t.Errorf("%s: unexpected error: %v", k, err)
- return
- }
- if buf.Len() > 0 {
- t.Errorf("buffer should be empty: %s", string(buf.Bytes()))
- }
- })
- }
- }
|