mirror of https://github.com/knative/client.git
add --prune & --prune-all options to delete all the unreferenced revisions (#1217)
This commit is contained in:
parent
d68cde6ebe
commit
2ad8dfe483
|
|
@ -17,6 +17,10 @@
|
||||||
|===
|
|===
|
||||||
| | Description | PR
|
| | Description | PR
|
||||||
|
|
||||||
|
| 🎁
|
||||||
|
| Add `--prune` & `--prune-all` options to delete the unreferenced revisions
|
||||||
|
| https://github.com/knative/client/pull/1217[#1217]
|
||||||
|
|
||||||
| ✨
|
| ✨
|
||||||
| Update CRDs API version to `v1`
|
| Update CRDs API version to `v1`
|
||||||
| https://github.com/knative/client/issues/1248[#1248]
|
| https://github.com/knative/client/issues/1248[#1248]
|
||||||
|
|
|
||||||
|
|
@ -12,6 +12,12 @@ kn revision delete NAME [NAME ...]
|
||||||
|
|
||||||
# Delete a revision 'svc1-abcde' in default namespace
|
# Delete a revision 'svc1-abcde' in default namespace
|
||||||
kn revision delete svc1-abcde
|
kn revision delete svc1-abcde
|
||||||
|
|
||||||
|
# Delete all unreferenced revisions
|
||||||
|
kn revision delete --prune-all
|
||||||
|
|
||||||
|
# Delete all unreferenced revisions for a given service 'mysvc'
|
||||||
|
kn revision delete --prune mysvc
|
||||||
```
|
```
|
||||||
|
|
||||||
### Options
|
### Options
|
||||||
|
|
@ -20,6 +26,8 @@ kn revision delete NAME [NAME ...]
|
||||||
-h, --help help for delete
|
-h, --help help for delete
|
||||||
-n, --namespace string Specify the namespace to operate in.
|
-n, --namespace string Specify the namespace to operate in.
|
||||||
--no-wait Do not wait for 'revision delete' operation to be completed. (default true)
|
--no-wait Do not wait for 'revision delete' operation to be completed. (default true)
|
||||||
|
--prune string Remove unreferenced revisions for a given service in a namespace.
|
||||||
|
--prune-all Remove all unreferenced revisions in a namespace.
|
||||||
--wait Wait for 'revision delete' operation to be completed.
|
--wait Wait for 'revision delete' operation to be completed.
|
||||||
--wait-timeout int Seconds to wait before giving up on waiting for revision to be deleted. (default 600)
|
--wait-timeout int Seconds to wait before giving up on waiting for revision to be deleted. (default 600)
|
||||||
```
|
```
|
||||||
|
|
|
||||||
|
|
@ -139,3 +139,17 @@ func RevisionListWithService(r *KnRunResultCollector, serviceNames ...string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RevisionDeleteWithPruneOption verifies removeing all unreferenced revisions for a given service in sync mode
|
||||||
|
func RevisionDeleteWithPruneOption(r *KnRunResultCollector, serviceName, revName string) {
|
||||||
|
out := r.KnTest().Kn().Run("revision", "delete", "--prune", serviceName)
|
||||||
|
assert.Check(r.T(), util.ContainsAll(out.Stdout, "Revision", "deleted", revName, "namespace", r.KnTest().Kn().Namespace()))
|
||||||
|
r.AssertNoError(out)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RevisionDeleteWithPruneAllOption verifies removeing all unreferenced revision in sync mode
|
||||||
|
func RevisionDeleteWithPruneAllOption(r *KnRunResultCollector, revName1, revName2 string) {
|
||||||
|
out := r.KnTest().Kn().Run("revision", "delete", "--prune-all")
|
||||||
|
assert.Check(r.T(), util.ContainsAll(out.Stdout, "Revision", "deleted", revName1, revName1, "namespace", r.KnTest().Kn().Namespace()))
|
||||||
|
r.AssertNoError(out)
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -21,24 +21,39 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
|
|
||||||
"knative.dev/client/pkg/kn/commands"
|
"knative.dev/client/pkg/kn/commands"
|
||||||
|
v1 "knative.dev/client/pkg/serving/v1"
|
||||||
|
servingv1 "knative.dev/serving/pkg/apis/serving/v1"
|
||||||
)
|
)
|
||||||
|
|
||||||
// NewRevisionDeleteCommand represent 'revision delete' command
|
// NewRevisionDeleteCommand represent 'revision delete' command
|
||||||
func NewRevisionDeleteCommand(p *commands.KnParams) *cobra.Command {
|
func NewRevisionDeleteCommand(p *commands.KnParams) *cobra.Command {
|
||||||
var waitFlags commands.WaitFlags
|
var waitFlags commands.WaitFlags
|
||||||
|
// prune filter, used with "-p"
|
||||||
|
var pruneFilter string
|
||||||
|
var pruneAll bool
|
||||||
RevisionDeleteCommand := &cobra.Command{
|
RevisionDeleteCommand := &cobra.Command{
|
||||||
Use: "delete NAME [NAME ...]",
|
Use: "delete NAME [NAME ...]",
|
||||||
Short: "Delete revisions",
|
Short: "Delete revisions",
|
||||||
Example: `
|
Example: `
|
||||||
# Delete a revision 'svc1-abcde' in default namespace
|
# Delete a revision 'svc1-abcde' in default namespace
|
||||||
kn revision delete svc1-abcde`,
|
kn revision delete svc1-abcde
|
||||||
|
|
||||||
|
# Delete all unreferenced revisions
|
||||||
|
kn revision delete --prune-all
|
||||||
|
|
||||||
|
# Delete all unreferenced revisions for a given service 'mysvc'
|
||||||
|
kn revision delete --prune mysvc`,
|
||||||
RunE: func(cmd *cobra.Command, args []string) error {
|
RunE: func(cmd *cobra.Command, args []string) error {
|
||||||
if len(args) < 1 {
|
prune := cmd.Flags().Changed("prune")
|
||||||
|
argsLen := len(args)
|
||||||
|
if argsLen < 1 && !pruneAll && !prune {
|
||||||
return errors.New("'kn revision delete' requires one or more revision name")
|
return errors.New("'kn revision delete' requires one or more revision name")
|
||||||
}
|
}
|
||||||
|
if argsLen > 0 && pruneAll {
|
||||||
|
return errors.New("'kn revision delete' with --prune-all flag requires no arguments")
|
||||||
|
}
|
||||||
|
|
||||||
namespace, err := p.GetNamespace(cmd)
|
namespace, err := p.GetNamespace(cmd)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
|
@ -47,6 +62,21 @@ func NewRevisionDeleteCommand(p *commands.KnParams) *cobra.Command {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
// Create list filters
|
||||||
|
var params []v1.ListConfig
|
||||||
|
if prune {
|
||||||
|
params = append(params, v1.WithService(pruneFilter))
|
||||||
|
}
|
||||||
|
if prune || pruneAll {
|
||||||
|
args, err = getUnreferencedRevisionNames(params, client)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if len(args) == 0 {
|
||||||
|
fmt.Fprintf(cmd.OutOrStdout(), "No unreferenced revisions found.\n")
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
errs := []string{}
|
errs := []string{}
|
||||||
for _, name := range args {
|
for _, name := range args {
|
||||||
|
|
@ -67,7 +97,27 @@ func NewRevisionDeleteCommand(p *commands.KnParams) *cobra.Command {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
flags := RevisionDeleteCommand.Flags()
|
||||||
|
flags.StringVar(&pruneFilter, "prune", "", "Remove unreferenced revisions for a given service in a namespace.")
|
||||||
|
flags.BoolVar(&pruneAll, "prune-all", false, "Remove all unreferenced revisions in a namespace.")
|
||||||
commands.AddNamespaceFlags(RevisionDeleteCommand.Flags(), false)
|
commands.AddNamespaceFlags(RevisionDeleteCommand.Flags(), false)
|
||||||
waitFlags.AddConditionWaitFlags(RevisionDeleteCommand, commands.WaitDefaultTimeout, "delete", "revision", "deleted")
|
waitFlags.AddConditionWaitFlags(RevisionDeleteCommand, commands.WaitDefaultTimeout, "delete", "revision", "deleted")
|
||||||
return RevisionDeleteCommand
|
return RevisionDeleteCommand
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Return unreferenced revision names
|
||||||
|
func getUnreferencedRevisionNames(lConfig []v1.ListConfig, client v1.KnServingClient) ([]string, error) {
|
||||||
|
revisionList, err := client.ListRevisions(lConfig...)
|
||||||
|
if err != nil {
|
||||||
|
return []string{}, err
|
||||||
|
}
|
||||||
|
// Sort revisions by namespace, service, generation (in this order)
|
||||||
|
sortRevisions(revisionList)
|
||||||
|
revisionNames := []string{}
|
||||||
|
for _, revision := range revisionList.Items {
|
||||||
|
if revision.GetRoutingState() != servingv1.RoutingStateActive {
|
||||||
|
revisionNames = append(revisionNames, revision.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return revisionNames, nil
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,172 @@
|
||||||
|
/*
|
||||||
|
Copyright 2020 The Knative 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 revision
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"gotest.tools/v3/assert"
|
||||||
|
clientservingv1 "knative.dev/client/pkg/serving/v1"
|
||||||
|
"knative.dev/client/pkg/util"
|
||||||
|
"knative.dev/client/pkg/util/mock"
|
||||||
|
"knative.dev/serving/pkg/apis/serving"
|
||||||
|
servingv1 "knative.dev/serving/pkg/apis/serving/v1"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRevisionDeletePruneAllMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
|
||||||
|
// Wait for delete event
|
||||||
|
r.DeleteRevision("foo", mock.Any(), nil)
|
||||||
|
r.DeleteRevision("bar", mock.Any(), nil)
|
||||||
|
r.DeleteRevision("baz", mock.Any(), nil)
|
||||||
|
|
||||||
|
revision1 := createMockRevisionWithParams("foo", "svc1", "1", "50", "")
|
||||||
|
revision1.Labels[serving.RoutingStateLabelKey] = string(servingv1.RoutingStateReserve)
|
||||||
|
revision2 := createMockRevisionWithParams("bar", "svc2", "1", "50", "")
|
||||||
|
revision2.Labels[serving.RoutingStateLabelKey] = string(servingv1.RoutingStateReserve)
|
||||||
|
revision3 := createMockRevisionWithParams("baz", "svc3", "1", "50", "")
|
||||||
|
revision3.Labels[serving.RoutingStateLabelKey] = string(servingv1.RoutingStateReserve)
|
||||||
|
revisionList := &servingv1.RevisionList{Items: []servingv1.Revision{*revision1, *revision2, *revision3}}
|
||||||
|
r.ListRevisions(mock.Any(), revisionList, nil)
|
||||||
|
|
||||||
|
output, err := executeRevisionCommand(client, "delete", "--prune-all")
|
||||||
|
fmt.Println(output)
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Assert(t, util.ContainsAll(output, "Revision", "deleted", "foo", "bar", "baz", "default"))
|
||||||
|
r.Validate()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeleteCheckErrorForNotFoundRevisionsMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
|
||||||
|
r.DeleteRevision("foo", mock.Any(), nil)
|
||||||
|
r.DeleteRevision("bar", mock.Any(), errors.New("revisions.serving.knative.dev \"bar\" not found"))
|
||||||
|
r.DeleteRevision("baz", mock.Any(), errors.New("revisions.serving.knative.dev \"baz\" not found"))
|
||||||
|
|
||||||
|
output, err := executeRevisionCommand(client, "delete", "foo", "bar", "baz")
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("Expected revision not found error, returned nil")
|
||||||
|
}
|
||||||
|
assert.Assert(t, util.ContainsAll(output, "'foo' deleted", "\"bar\" not found", "\"baz\" not found"))
|
||||||
|
|
||||||
|
r.Validate()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeletePruneWithArgMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
r.DeleteRevision("foo", mock.Any(), nil)
|
||||||
|
r.DeleteRevision("baz", mock.Any(), nil)
|
||||||
|
r.DeleteRevision("bar", mock.Any(), nil)
|
||||||
|
|
||||||
|
revision1 := createMockRevisionWithParams("foo", "svc1", "1", "50", "")
|
||||||
|
revision1.Labels[serving.RoutingStateLabelKey] = string(servingv1.RoutingStateReserve)
|
||||||
|
revision2 := createMockRevisionWithParams("bar", "svc1", "1", "50", "")
|
||||||
|
revision2.Labels[serving.RoutingStateLabelKey] = string(servingv1.RoutingStateActive)
|
||||||
|
revision3 := createMockRevisionWithParams("baz", "svc1", "1", "50", "")
|
||||||
|
revision3.Labels[serving.RoutingStateLabelKey] = string(servingv1.RoutingStateActive)
|
||||||
|
revisionList := &servingv1.RevisionList{Items: []servingv1.Revision{*revision1, *revision2, *revision3}}
|
||||||
|
r.ListRevisions(mock.Any(), revisionList, nil)
|
||||||
|
|
||||||
|
output, err := executeRevisionCommand(client, "delete", "--prune", "svc1")
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Assert(t, util.ContainsAll(output, "Revision", "deleted", "foo", "default"))
|
||||||
|
assert.Assert(t, util.ContainsNone(output, "bar", "baz"))
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeletePruneErrorFromArgMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
revisionList := &servingv1.RevisionList{Items: []servingv1.Revision{}}
|
||||||
|
r.ListRevisions(mock.Any(), revisionList, nil)
|
||||||
|
|
||||||
|
_, err := executeRevisionCommand(client, "delete", "--prune")
|
||||||
|
assert.Error(t, err, "flag needs an argument: --prune")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeletePruneNoRevisionsMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
revisionList := &servingv1.RevisionList{Items: []servingv1.Revision{}}
|
||||||
|
r.ListRevisions(mock.Any(), revisionList, nil)
|
||||||
|
|
||||||
|
output, err := executeRevisionCommand(client, "delete", "--prune", "mysvc")
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Assert(t, util.ContainsAll(output, "No", "unreferenced", "revisions", "found"))
|
||||||
|
|
||||||
|
r.Validate()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeleteNoNameMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
|
||||||
|
_, err := executeRevisionCommand(client, "delete")
|
||||||
|
assert.ErrorContains(t, err, "'kn revision delete' requires one or more revision name")
|
||||||
|
|
||||||
|
r.Validate()
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeletePruneAllNoRevisionsMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
revisionList := &servingv1.RevisionList{Items: []servingv1.Revision{}}
|
||||||
|
r.ListRevisions(mock.Any(), revisionList, nil)
|
||||||
|
|
||||||
|
output, err := executeRevisionCommand(client, "delete", "--prune-all")
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Assert(t, util.ContainsAll(output, "No", "unreferenced", "revisions", "found"))
|
||||||
|
|
||||||
|
r.Validate()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRevisionDeletePruneAllErrorFromArgMock(t *testing.T) {
|
||||||
|
// New mock client
|
||||||
|
client := clientservingv1.NewMockKnServiceClient(t)
|
||||||
|
// Recording:
|
||||||
|
r := client.Recorder()
|
||||||
|
revisionList := &servingv1.RevisionList{Items: []servingv1.Revision{}}
|
||||||
|
r.ListRevisions(mock.Any(), revisionList, nil)
|
||||||
|
|
||||||
|
_, err := executeRevisionCommand(client, "delete", "--prune-all", "mysvc")
|
||||||
|
assert.Error(t, err, "'kn revision delete' with --prune-all flag requires no arguments")
|
||||||
|
}
|
||||||
|
|
@ -25,9 +25,7 @@ import (
|
||||||
clienttesting "k8s.io/client-go/testing"
|
clienttesting "k8s.io/client-go/testing"
|
||||||
|
|
||||||
"knative.dev/client/pkg/kn/commands"
|
"knative.dev/client/pkg/kn/commands"
|
||||||
clientservingv1 "knative.dev/client/pkg/serving/v1"
|
|
||||||
"knative.dev/client/pkg/util"
|
"knative.dev/client/pkg/util"
|
||||||
"knative.dev/client/pkg/util/mock"
|
|
||||||
"knative.dev/client/pkg/wait"
|
"knative.dev/client/pkg/wait"
|
||||||
servingv1 "knative.dev/serving/pkg/apis/serving/v1"
|
servingv1 "knative.dev/serving/pkg/apis/serving/v1"
|
||||||
)
|
)
|
||||||
|
|
@ -104,23 +102,3 @@ func getRevisionDeleteEvents(name string) []watch.Event {
|
||||||
{Type: watch.Deleted, Object: &servingv1.Revision{ObjectMeta: metav1.ObjectMeta{Name: name}}},
|
{Type: watch.Deleted, Object: &servingv1.Revision{ObjectMeta: metav1.ObjectMeta{Name: name}}},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRevisionDeleteCheckErrorForNotFoundRevisionsMock(t *testing.T) {
|
|
||||||
// New mock client
|
|
||||||
client := clientservingv1.NewMockKnServiceClient(t)
|
|
||||||
|
|
||||||
// Recording:
|
|
||||||
r := client.Recorder()
|
|
||||||
|
|
||||||
r.DeleteRevision("foo", mock.Any(), nil)
|
|
||||||
r.DeleteRevision("bar", mock.Any(), errors.New("revisions.serving.knative.dev \"bar\" not found."))
|
|
||||||
r.DeleteRevision("baz", mock.Any(), errors.New("revisions.serving.knative.dev \"baz\" not found."))
|
|
||||||
|
|
||||||
output, err := executeRevisionCommand(client, "delete", "foo", "bar", "baz")
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("Expected revision not found error, returned nil")
|
|
||||||
}
|
|
||||||
assert.Assert(t, util.ContainsAll(output, "'foo' deleted", "\"bar\" not found", "\"baz\" not found"))
|
|
||||||
|
|
||||||
r.Validate()
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -59,6 +59,25 @@ func TestRevision(t *testing.T) {
|
||||||
nonexistRevision := "hello-nonexist"
|
nonexistRevision := "hello-nonexist"
|
||||||
test.RevisionMultipleDelete(r, existRevision1, existRevision2, nonexistRevision)
|
test.RevisionMultipleDelete(r, existRevision1, existRevision2, nonexistRevision)
|
||||||
|
|
||||||
|
t.Log("update hello service and increase revision count to 4")
|
||||||
|
test.ServiceUpdate(r, "hello", "--env", "TARGET=kn", "--port", "8888")
|
||||||
|
t.Log("delete all unreferenced revision from hello service and return no error")
|
||||||
|
unRefRevision := test.FindRevisionByGeneration(r, "hello", 3)
|
||||||
|
test.RevisionDeleteWithPruneOption(r, "hello", unRefRevision)
|
||||||
|
|
||||||
|
t.Log("update hello service and increase revision count to 5")
|
||||||
|
test.ServiceUpdate(r, "hello", "--env", "TARGET=kn", "--port", "9000")
|
||||||
|
t.Log("create hello service and return no error")
|
||||||
|
test.ServiceCreate(r, "hello1")
|
||||||
|
t.Log("update hello1 service and increase revision count to 2")
|
||||||
|
test.ServiceUpdate(r, "hello1", "--env", "TARGET=kn", "--port", "8888")
|
||||||
|
t.Log("delete all unreferenced revisions return no error")
|
||||||
|
unRefRevision1 := test.FindRevisionByGeneration(r, "hello", 4)
|
||||||
|
unRefRevision2 := test.FindRevisionByGeneration(r, "hello1", 1)
|
||||||
|
test.RevisionDeleteWithPruneAllOption(r, unRefRevision1, unRefRevision2)
|
||||||
|
t.Log("delete hello1 service and return no error")
|
||||||
|
test.ServiceDelete(r, "hello1")
|
||||||
|
|
||||||
t.Log("delete latest revision from hello service and return no error")
|
t.Log("delete latest revision from hello service and return no error")
|
||||||
revName = test.FindRevision(r, "hello")
|
revName = test.FindRevision(r, "hello")
|
||||||
test.RevisionDelete(r, revName)
|
test.RevisionDelete(r, revName)
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue