
The grace-period is automatically set to 1 unless --force is provided, and the client waits until the object is deleted. This preserves backwards compatibility with 1.4 and earlier. It does not handle scenarios where the object is deleted and a new object is created with the same name.
585 lines
22 KiB
Go
585 lines
22 KiB
Go
/*
|
|
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 cmd
|
|
|
|
import (
|
|
"bytes"
|
|
"net/http"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"k8s.io/kubernetes/pkg/api"
|
|
"k8s.io/kubernetes/pkg/api/errors"
|
|
"k8s.io/kubernetes/pkg/api/meta"
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
|
"k8s.io/kubernetes/pkg/apimachinery/registered"
|
|
"k8s.io/kubernetes/pkg/client/restclient"
|
|
"k8s.io/kubernetes/pkg/client/restclient/fake"
|
|
"k8s.io/kubernetes/pkg/client/typed/dynamic"
|
|
"k8s.io/kubernetes/pkg/kubectl"
|
|
cmdtesting "k8s.io/kubernetes/pkg/kubectl/cmd/testing"
|
|
cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
|
|
"k8s.io/kubernetes/pkg/kubectl/resource"
|
|
)
|
|
|
|
var unstructuredSerializer = dynamic.ContentConfig().NegotiatedSerializer
|
|
|
|
func TestDeleteObjectByTuple(t *testing.T) {
|
|
_, _, rc := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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-controller" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: 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
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
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())
|
|
}
|
|
}
|
|
|
|
func TestDeleteNamedObject(t *testing.T) {
|
|
_, _, rc := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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-controller" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: 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
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
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())
|
|
}
|
|
}
|
|
|
|
func TestDeleteObject(t *testing.T) {
|
|
_, _, rc := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, &rc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
cmd.Flags().Set("filename", "../../../examples/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())
|
|
}
|
|
}
|
|
|
|
type fakeReaper struct {
|
|
namespace, name string
|
|
timeout time.Duration
|
|
deleteOptions *api.DeleteOptions
|
|
err error
|
|
}
|
|
|
|
func (r *fakeReaper) Stop(namespace, name string, timeout time.Duration, gracePeriod *api.DeleteOptions) error {
|
|
r.namespace, r.name = namespace, name
|
|
r.timeout = timeout
|
|
r.deleteOptions = gracePeriod
|
|
return r.err
|
|
}
|
|
|
|
type fakeReaperFactory struct {
|
|
cmdutil.Factory
|
|
reaper kubectl.Reaper
|
|
}
|
|
|
|
func (f *fakeReaperFactory) Reaper(mapping *meta.RESTMapping) (kubectl.Reaper, error) {
|
|
return f.reaper, nil
|
|
}
|
|
|
|
func TestDeleteObjectGraceZero(t *testing.T) {
|
|
pods, _, _ := testData()
|
|
|
|
objectDeletionWaitInterval = time.Millisecond
|
|
count := 0
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
|
|
default:
|
|
return &http.Response{StatusCode: 404, Header: defaultHeader(), Body: objBody(codec, &unversioned.Status{})}, nil
|
|
}
|
|
case p == "/api/v1/namespaces/test" && m == "GET":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &api.Namespace{})}, nil
|
|
case p == "/namespaces/test/pods/nginx" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
reaper := &fakeReaper{}
|
|
fake := &fakeReaperFactory{Factory: f, reaper: reaper}
|
|
cmd := NewCmdDelete(fake, buf, errBuf)
|
|
cmd.Flags().Set("output", "name")
|
|
cmd.Flags().Set("grace-period", "0")
|
|
cmd.Run(cmd, []string{"pod/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 reaper.deleteOptions == nil || reaper.deleteOptions.GracePeriodSeconds == nil || *reaper.deleteOptions.GracePeriodSeconds != 1 {
|
|
t.Errorf("unexpected reaper options: %#v", reaper)
|
|
}
|
|
if count != 4 {
|
|
t.Errorf("unexpected calls to GET: %d", count)
|
|
}
|
|
}
|
|
|
|
func TestDeleteObjectNotFound(t *testing.T) {
|
|
f, tf, _, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: stringBody("")}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
options := &resource.FilenameOptions{}
|
|
options.Filenames = []string{"../../../examples/guestbook/legacy/redis-master-controller.yaml"}
|
|
cmd.Flags().Set("cascade", "false")
|
|
cmd.Flags().Set("output", "name")
|
|
err := RunDelete(f, buf, errBuf, cmd, []string{}, options)
|
|
if err == nil || !errors.IsNotFound(err) {
|
|
t.Errorf("unexpected error: expected NotFound, got %v", err)
|
|
}
|
|
}
|
|
|
|
func TestDeleteObjectIgnoreNotFound(t *testing.T) {
|
|
f, tf, _, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: stringBody("")}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
cmd.Flags().Set("filename", "../../../examples/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) {
|
|
_, svc, _ := testData()
|
|
// Add an item to the list which will result in a 404 on delete
|
|
svc.Items = append(svc.Items, api.Service{ObjectMeta: api.ObjectMeta{Name: "foo"}})
|
|
notFoundError := &errors.NewNotFound(api.Resource("services"), "foo").ErrStatus
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, svc)}, nil
|
|
case p == "/namespaces/test/services/foo" && m == "DELETE":
|
|
return &http.Response{StatusCode: 404, Header: defaultHeader(), Body: objBody(codec, notFoundError)}, nil
|
|
case p == "/namespaces/test/services/baz" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &svc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
cmd.Flags().Set("all", "true")
|
|
cmd.Flags().Set("cascade", "false")
|
|
// Make sure we can explicitly choose to fail on NotFound errors, even with --all
|
|
cmd.Flags().Set("ignore-not-found", "false")
|
|
cmd.Flags().Set("output", "name")
|
|
|
|
err := RunDelete(f, buf, errBuf, cmd, []string{"services"}, &resource.FilenameOptions{})
|
|
if err == nil || !errors.IsNotFound(err) {
|
|
t.Errorf("unexpected error: expected NotFound, got %v", err)
|
|
}
|
|
}
|
|
|
|
func TestDeleteAllIgnoreNotFound(t *testing.T) {
|
|
_, svc, _ := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
|
|
// Add an item to the list which will result in a 404 on delete
|
|
svc.Items = append(svc.Items, api.Service{ObjectMeta: api.ObjectMeta{Name: "foo"}})
|
|
notFoundError := &errors.NewNotFound(api.Resource("services"), "foo").ErrStatus
|
|
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, svc)}, nil
|
|
case p == "/namespaces/test/services/foo" && m == "DELETE":
|
|
return &http.Response{StatusCode: 404, Header: defaultHeader(), Body: objBody(codec, notFoundError)}, nil
|
|
case p == "/namespaces/test/services/baz" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &svc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
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) {
|
|
_, svc, rc := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, &rc.Items[0])}, nil
|
|
case p == "/namespaces/test/services/frontend" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &svc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
cmd.Flags().Set("filename", "../../../examples/guestbook/legacy/redis-master-controller.yaml")
|
|
cmd.Flags().Set("filename", "../../../examples/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) {
|
|
_, svc, _ := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: stringBody("")}, nil
|
|
case p == "/namespaces/test/services/frontend" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &svc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
options := &resource.FilenameOptions{}
|
|
options.Filenames = []string{"../../../examples/guestbook/legacy/redis-master-controller.yaml", "../../../examples/guestbook/frontend-service.yaml"}
|
|
cmd.Flags().Set("cascade", "false")
|
|
cmd.Flags().Set("output", "name")
|
|
err := RunDelete(f, buf, errBuf, cmd, []string{}, options)
|
|
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) {
|
|
_, svc, rc := testData()
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, &rc.Items[0])}, nil
|
|
case p == "/namespaces/test/replicationcontrollers/foo" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &rc.Items[0])}, nil
|
|
case p == "/namespaces/test/services/baz" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &svc.Items[0])}, nil
|
|
case p == "/namespaces/test/services/foo" && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: 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
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
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) {
|
|
_, _, rc := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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: defaultHeader(), Body: objBody(codec, &rc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
cmd.Flags().Set("filename", "../../../examples/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) {
|
|
pods, svc, _ := testData()
|
|
|
|
f, tf, codec, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Client = &fake.RESTClient{
|
|
NegotiatedSerializer: unstructuredSerializer,
|
|
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(unversioned.LabelSelectorQueryParam(registered.GroupOrDie(api.GroupName).GroupVersion.String())) != "a=b" {
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
}
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, pods)}, nil
|
|
case p == "/namespaces/test/services" && m == "GET":
|
|
if req.URL.Query().Get(unversioned.LabelSelectorQueryParam(registered.GroupOrDie(api.GroupName).GroupVersion.String())) != "a=b" {
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
}
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, svc)}, nil
|
|
case strings.HasPrefix(p, "/namespaces/test/pods/") && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &pods.Items[0])}, nil
|
|
case strings.HasPrefix(p, "/namespaces/test/services/") && m == "DELETE":
|
|
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: objBody(codec, &svc.Items[0])}, nil
|
|
default:
|
|
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
|
|
return nil, nil
|
|
}
|
|
}),
|
|
}
|
|
tf.Namespace = "test"
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
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) {
|
|
testCases := map[string]struct {
|
|
args []string
|
|
flags map[string]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 {
|
|
f, tf, _, _ := cmdtesting.NewAPIFactory()
|
|
tf.Printer = &testPrinter{}
|
|
tf.Namespace = "test"
|
|
tf.ClientConfig = &restclient.Config{ContentConfig: restclient.ContentConfig{GroupVersion: ®istered.GroupOrDie(api.GroupName).GroupVersion}}
|
|
|
|
buf, errBuf := bytes.NewBuffer([]byte{}), bytes.NewBuffer([]byte{})
|
|
|
|
cmd := NewCmdDelete(f, buf, errBuf)
|
|
cmd.SetOutput(buf)
|
|
|
|
for k, v := range testCase.flags {
|
|
cmd.Flags().Set(k, v)
|
|
}
|
|
err := RunDelete(f, buf, errBuf, cmd, testCase.args, &resource.FilenameOptions{})
|
|
if !testCase.errFn(err) {
|
|
t.Errorf("%s: unexpected error: %v", k, err)
|
|
continue
|
|
}
|
|
if tf.Printer.(*testPrinter).Objects != nil {
|
|
t.Errorf("unexpected print to default printer")
|
|
}
|
|
if buf.Len() > 0 {
|
|
t.Errorf("buffer should be empty: %s", string(buf.Bytes()))
|
|
}
|
|
}
|
|
}
|