Merge pull request #4381 from sdminonne/bug_fix

kubectl delete resource : remove all resources
This commit is contained in:
Clayton Coleman 2015-02-17 12:25:56 -05:00
commit 0e07e61a53
7 changed files with 55 additions and 49 deletions

View File

@ -315,11 +315,15 @@ Examples:
// Delete a pod with ID 1234-56-7890-234234-456456. // Delete a pod with ID 1234-56-7890-234234-456456.
$ kubectl delete pod 1234-56-7890-234234-456456 $ kubectl delete pod 1234-56-7890-234234-456456
// Delete all pods
$ kubectl delete pods --all
Usage: Usage:
``` ```
kubectl delete ([-f filename] | (<resource> [(<id> | -l <label>)] [flags] kubectl delete ([-f filename] | (<resource> [(<id> | -l <label> | --all)] [flags]
Flags: Flags:
--all=false: [-all] to select all the specified resources
-f, --filename=[]: Filename, directory, or URL to a file containing the resource to delete -f, --filename=[]: Filename, directory, or URL to a file containing the resource to delete
-l, --selector="": Selector (label query) to filter on -l, --selector="": Selector (label query) to filter on

View File

@ -145,6 +145,16 @@ for version in "${kube_api_versions[@]}"; do
kubectl get pods -o yaml --output-version=v1beta1 "${kube_flags[@]}" | grep -q "id: redis-master" kubectl get pods -o yaml --output-version=v1beta1 "${kube_flags[@]}" | grep -q "id: redis-master"
kubectl describe pod redis-master "${kube_flags[@]}" | grep -q 'Name:.*redis-master' kubectl describe pod redis-master "${kube_flags[@]}" | grep -q 'Name:.*redis-master'
kubectl delete -f examples/guestbook/redis-master.json "${kube_flags[@]}" kubectl delete -f examples/guestbook/redis-master.json "${kube_flags[@]}"
kubectl delete pods -l name=redis-master "${kube_flags[@]}"
[ ! $(kubectl get pods "${kube_flags[@]}" -lname=redis-master | grep -q 'redis-master') ]
kubectl create -f examples/guestbook/redis-master.json "${kube_flags[@]}"
kubectl get pods "${kube_flags[@]}" -lname=redis-master | grep -q 'redis-master'
[ ! $(kubectl delete pods "${kube_flags[@]}" ) ]
kubectl get pods "${kube_flags[@]}" -lname=redis-master | grep -q 'redis-master'
[ ! $(delete pods --all pods -l name=redis-master) ] # not --all and label selector together
kubectl delete --all pods "${kube_flags[@]}" # --all remove all the pods
howmanypods="$(kubectl get pods -o template -t "{{ len .items }}" "${kube_flags[@]}")"
[ "$howmanypods" -eq 0 ]
kube::log::status "Testing kubectl(${version}:services)" kube::log::status "Testing kubectl(${version}:services)"
kubectl get services "${kube_flags[@]}" kubectl get services "${kube_flags[@]}"

View File

@ -33,7 +33,7 @@ func (f *Factory) NewCmdDelete(out io.Writer) *cobra.Command {
Filenames util.StringList Filenames util.StringList
}{} }{}
cmd := &cobra.Command{ cmd := &cobra.Command{
Use: "delete ([-f filename] | (<resource> [(<id> | -l <label>)]", Use: "delete ([-f filename] | (<resource> [(<id> | -l <label> | --all)]",
Short: "Delete a resource by filename, stdin, or resource and ID.", Short: "Delete a resource by filename, stdin, or resource and ID.",
Long: `Delete a resource by filename, stdin, resource and ID, or by resources and label selector. Long: `Delete a resource by filename, stdin, resource and ID, or by resources and label selector.
@ -58,18 +58,22 @@ Examples:
$ kubectl delete pods,services -l name=myLabel $ kubectl delete pods,services -l name=myLabel
// Delete a pod with ID 1234-56-7890-234234-456456. // Delete a pod with ID 1234-56-7890-234234-456456.
$ kubectl delete pod 1234-56-7890-234234-456456`, $ kubectl delete pod 1234-56-7890-234234-456456
// Delete all pods
$ kubectl delete pods --all`,
Run: func(cmd *cobra.Command, args []string) { Run: func(cmd *cobra.Command, args []string) {
cmdNamespace, err := f.DefaultNamespace(cmd) cmdNamespace, err := f.DefaultNamespace(cmd)
checkErr(err) checkErr(err)
mapper, typer := f.Object(cmd) mapper, typer := f.Object(cmd)
r := resource.NewBuilder(mapper, typer, f.ClientMapperForCommand(cmd)). r := resource.NewBuilder(mapper, typer, f.ClientMapperForCommand(cmd)).
ContinueOnError(). ContinueOnError().
NamespaceParam(cmdNamespace).DefaultNamespace(). NamespaceParam(cmdNamespace).DefaultNamespace().
FilenameParam(flags.Filenames...). FilenameParam(flags.Filenames...).
SelectorParam(cmdutil.GetFlagString(cmd, "selector")). SelectorParam(cmdutil.GetFlagString(cmd, "selector")).
ResourceTypeOrNameArgs(args...). SelectAllParam(cmdutil.GetFlagBool(cmd, "all")).
ResourceTypeOrNameArgs(false, args...).
Flatten(). Flatten().
Do() Do()
checkErr(r.Err()) checkErr(r.Err())
@ -91,5 +95,6 @@ Examples:
} }
cmd.Flags().VarP(&flags.Filenames, "filename", "f", "Filename, directory, or URL to a file containing the resource to delete") cmd.Flags().VarP(&flags.Filenames, "filename", "f", "Filename, directory, or URL to a file containing the resource to delete")
cmd.Flags().StringP("selector", "l", "", "Selector (label query) to filter on") cmd.Flags().StringP("selector", "l", "", "Selector (label query) to filter on")
cmd.Flags().Bool("all", false, "[-all] to select all the specified resources")
return cmd return cmd
} }

View File

@ -22,7 +22,6 @@ import (
"strings" "strings"
"testing" "testing"
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
"github.com/GoogleCloudPlatform/kubernetes/pkg/client" "github.com/GoogleCloudPlatform/kubernetes/pkg/client"
) )
@ -83,37 +82,6 @@ func TestDeleteObjectIgnoreNotFound(t *testing.T) {
} }
} }
func TestDeleteNoObjects(t *testing.T) {
f, tf, codec := NewAPIFactory()
tf.Printer = &testPrinter{}
tf.Client = &client.FakeRESTClient{
Codec: codec,
Client: client.HTTPClientFunc(func(req *http.Request) (*http.Response, error) {
switch p, m := req.URL.Path, req.Method; {
case p == "/namespaces/test/pods" && m == "GET":
return &http.Response{StatusCode: 200, Body: objBody(codec, &api.PodList{})}, nil
default:
t.Fatalf("unexpected request: %#v\n%#v", req.URL, req)
return nil, nil
}
}),
}
tf.Namespace = "test"
buf := bytes.NewBuffer([]byte{})
stderr := bytes.NewBuffer([]byte{})
cmd := f.NewCmdDelete(buf)
cmd.SetOutput(stderr)
cmd.Run(cmd, []string{"pods"})
if buf.String() != "" {
t.Errorf("unexpected output: %s", buf.String())
}
if stderr.String() != "No resources found\n" {
t.Errorf("unexpected output: %s", stderr.String())
}
}
func TestDeleteMultipleObject(t *testing.T) { func TestDeleteMultipleObject(t *testing.T) {
pods, svc := testData() pods, svc := testData()

View File

@ -87,7 +87,7 @@ func RunGet(f *Factory, out io.Writer, cmd *cobra.Command, args []string) {
r := resource.NewBuilder(mapper, typer, f.ClientMapperForCommand(cmd)). r := resource.NewBuilder(mapper, typer, f.ClientMapperForCommand(cmd)).
NamespaceParam(cmdNamespace).DefaultNamespace(). NamespaceParam(cmdNamespace).DefaultNamespace().
SelectorParam(selector). SelectorParam(selector).
ResourceTypeOrNameArgs(args...). ResourceTypeOrNameArgs(true, args...).
SingleResourceType(). SingleResourceType().
Do() Do()
checkErr(r.Err()) checkErr(r.Err())
@ -124,7 +124,7 @@ func RunGet(f *Factory, out io.Writer, cmd *cobra.Command, args []string) {
b := resource.NewBuilder(mapper, typer, f.ClientMapperForCommand(cmd)). b := resource.NewBuilder(mapper, typer, f.ClientMapperForCommand(cmd)).
NamespaceParam(cmdNamespace).DefaultNamespace(). NamespaceParam(cmdNamespace).DefaultNamespace().
SelectorParam(selector). SelectorParam(selector).
ResourceTypeOrNameArgs(args...). ResourceTypeOrNameArgs(true, args...).
Latest() Latest()
printer, generic, err := util.PrinterForCommand(cmd) printer, generic, err := util.PrinterForCommand(cmd)
checkErr(err) checkErr(err)

View File

@ -43,6 +43,7 @@ type Builder struct {
dir bool dir bool
selector labels.Selector selector labels.Selector
selectAll bool
resources []string resources []string
@ -171,6 +172,10 @@ func (b *Builder) SelectorParam(s string) *Builder {
if selector.Empty() { if selector.Empty() {
return b return b
} }
if b.selectAll {
b.errs = append(b.errs, fmt.Errorf("found non empty selector %q with previously set 'all' parameter. ", s))
return b
}
return b.Selector(selector) return b.Selector(selector)
} }
@ -202,19 +207,29 @@ func (b *Builder) RequireNamespace() *Builder {
return b return b
} }
// SelectEverythingParam
func (b *Builder) SelectAllParam(selectAll bool) *Builder {
if selectAll && b.selector != nil {
b.errs = append(b.errs, fmt.Errorf("setting 'all' parameter but found a non empty selector. "))
return b
}
b.selectAll = selectAll
return b
}
// ResourceTypeOrNameArgs indicates that the builder should accept one or two arguments // ResourceTypeOrNameArgs indicates that the builder should accept one or two arguments
// of the form `(<type1>[,<type2>,...]|<type> <name>)`. When one argument is received, the types // of the form `(<type1>[,<type2>,...]|<type> <name>)`. When one argument is received, the types
// provided will be retrieved from the server (and be comma delimited). When two arguments are // provided will be retrieved from the server (and be comma delimited). When two arguments are
// received, they must be a single type and name. If more than two arguments are provided an // received, they must be a single type and name. If more than two arguments are provided an
// error is set. // error is set. The allowEmptySelector permits to select all the resources (via Everything func).
func (b *Builder) ResourceTypeOrNameArgs(args ...string) *Builder { func (b *Builder) ResourceTypeOrNameArgs(allowEmptySelector bool, args ...string) *Builder {
switch len(args) { switch len(args) {
case 2: case 2:
b.name = args[1] b.name = args[1]
b.ResourceTypes(SplitResourceArgument(args[0])...) b.ResourceTypes(SplitResourceArgument(args[0])...)
case 1: case 1:
b.ResourceTypes(SplitResourceArgument(args[0])...) b.ResourceTypes(SplitResourceArgument(args[0])...)
if b.selector == nil { if b.selector == nil && allowEmptySelector {
b.selector = labels.Everything() b.selector = labels.Everything()
} }
case 0: case 0:
@ -291,6 +306,10 @@ func (b *Builder) visitorResult() *Result {
return &Result{err: errors.NewAggregate(b.errs)} return &Result{err: errors.NewAggregate(b.errs)}
} }
if b.selectAll {
b.selector = labels.Everything()
}
// visit selectors // visit selectors
if b.selector != nil { if b.selector != nil {
if len(b.name) != 0 { if len(b.name) != 0 {

View File

@ -315,7 +315,7 @@ func TestResourceByName(t *testing.T) {
t.Errorf("unexpected non-error") t.Errorf("unexpected non-error")
} }
b.ResourceTypeOrNameArgs("pods", "foo") b.ResourceTypeOrNameArgs(true, "pods", "foo")
err := b.Do().IntoSingular(&singular).Visit(test.Handle) err := b.Do().IntoSingular(&singular).Visit(test.Handle)
if err != nil || !singular || len(test.Infos) != 1 { if err != nil || !singular || len(test.Infos) != 1 {
@ -341,7 +341,7 @@ func TestResourceByNameAndEmptySelector(t *testing.T) {
})). })).
NamespaceParam("test"). NamespaceParam("test").
SelectorParam(""). SelectorParam("").
ResourceTypeOrNameArgs("pods", "foo") ResourceTypeOrNameArgs(true, "pods", "foo")
singular := false singular := false
infos, err := b.Do().IntoSingular(&singular).Infos() infos, err := b.Do().IntoSingular(&singular).Infos()
@ -378,7 +378,7 @@ func TestSelector(t *testing.T) {
t.Errorf("unexpected non-error") t.Errorf("unexpected non-error")
} }
b.ResourceTypeOrNameArgs("pods,service") b.ResourceTypeOrNameArgs(true, "pods,service")
err := b.Do().IntoSingular(&singular).Visit(test.Handle) err := b.Do().IntoSingular(&singular).Visit(test.Handle)
if err != nil || singular || len(test.Infos) != 3 { if err != nil || singular || len(test.Infos) != 3 {
@ -408,7 +408,7 @@ func TestSingleResourceType(t *testing.T) {
b := NewBuilder(latest.RESTMapper, api.Scheme, fakeClient()). b := NewBuilder(latest.RESTMapper, api.Scheme, fakeClient()).
SelectorParam("a=b"). SelectorParam("a=b").
SingleResourceType(). SingleResourceType().
ResourceTypeOrNameArgs("pods,services") ResourceTypeOrNameArgs(true, "pods,services")
if b.Do().Err() == nil { if b.Do().Err() == nil {
t.Errorf("unexpected non-error") t.Errorf("unexpected non-error")
@ -498,7 +498,7 @@ func TestListObject(t *testing.T) {
})). })).
SelectorParam("a=b"). SelectorParam("a=b").
NamespaceParam("test"). NamespaceParam("test").
ResourceTypeOrNameArgs("pods"). ResourceTypeOrNameArgs(true, "pods").
Flatten() Flatten()
obj, err := b.Do().Object() obj, err := b.Do().Object()
@ -531,7 +531,7 @@ func TestListObjectWithDifferentVersions(t *testing.T) {
})). })).
SelectorParam("a=b"). SelectorParam("a=b").
NamespaceParam("test"). NamespaceParam("test").
ResourceTypeOrNameArgs("pods,services"). ResourceTypeOrNameArgs(true, "pods,services").
Flatten(). Flatten().
Do().Object() Do().Object()