cmd/client/command/delete.go (150 lines of code) (raw):
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 command
import (
"fmt"
"strconv"
"strings"
"github.com/spf13/cobra"
)
type DeleteOptions struct {
namespace string
cluster string
shard int
}
var deleteOptions DeleteOptions
var DeleteCommand = &cobra.Command{
Use: "delete",
Short: "Delete a resource",
Example: `
# Delete a namespace
kvctl delete namespace <namespace>
# Delete a cluster in the namespace
kvctl delete cluster <cluster> -n <namespace>
# Delete a shard in the cluster
kvctl delete shard <shard> -n <namespace> -c <cluster>
# Delete a node in the cluster
kvctl delete node <node_id> -n <namespace> -c <cluster> --shard <shard>
`,
PreRunE: deletePreRun,
RunE: func(cmd *cobra.Command, args []string) error {
resource := strings.ToLower(args[0])
if len(args) < 2 {
return fmt.Errorf("missing resource name")
}
host, _ := cmd.Flags().GetString("host")
client := newClient(host)
switch resource {
case ResourceNamespace:
namespace := args[1]
return deleteNamespace(client, namespace)
case ResourceCluster:
deleteOptions.cluster = args[1]
return deleteCluster(client, &deleteOptions)
case ResourceShard:
shard, err := strconv.Atoi(args[1])
if err != nil {
return err
}
deleteOptions.shard = shard
return deleteShard(client, &deleteOptions)
case ResourceNode:
nodeID := args[1]
return deleteNode(client, &deleteOptions, nodeID)
default:
return fmt.Errorf("unsupported resource type %s", resource)
}
},
SilenceUsage: true,
SilenceErrors: true,
}
func deletePreRun(_ *cobra.Command, args []string) error {
if len(args) == 0 {
return fmt.Errorf("missing resource type")
}
resource := strings.ToLower(args[0])
if resource == ResourceNamespace {
return nil
}
if deleteOptions.namespace == "" {
return fmt.Errorf("missing namespace, please specify the namespace via -n or --namespace option")
}
if resource == ResourceCluster {
return nil
}
if deleteOptions.cluster == "" {
return fmt.Errorf("missing cluster, please specify the cluster via -c or --cluster option")
}
if resource == ResourceShard {
return nil
}
if deleteOptions.shard == -1 {
return fmt.Errorf("missing shard, please specify the shard via -s or --shard option")
}
if deleteOptions.shard < 0 {
return fmt.Errorf("invalid shard %d", deleteOptions.shard)
}
return nil
}
func deleteNamespace(client *client, namespace string) error {
rsp, err := client.restyCli.R().
SetPathParam("namespace", namespace).
Delete("/namespaces/{namespace}")
if err != nil {
return err
}
if rsp.IsError() {
return unmarshalError(rsp.Body())
}
printLine("delete namespace: %s successfully.", namespace)
return nil
}
func deleteCluster(client *client, options *DeleteOptions) error {
rsp, err := client.restyCli.R().
SetPathParams(map[string]string{
"namespace": options.namespace,
"cluster": options.cluster,
}).Delete("/namespaces/{namespace}/clusters/{cluster}")
if err != nil {
return err
}
if rsp.IsError() {
return unmarshalError(rsp.Body())
}
printLine("delete cluster: %s successfully.", options.cluster)
return nil
}
func deleteShard(client *client, options *DeleteOptions) error {
rsp, err := client.restyCli.R().
SetPathParam("namespace", options.namespace).
SetPathParam("cluster", options.cluster).
SetPathParam("shard", strconv.Itoa(options.shard)).
Delete("/namespaces/{namespace}/clusters/{cluster}/shards/{shard}")
if err != nil {
return err
}
if rsp.IsError() {
return unmarshalError(rsp.Body())
}
printLine("delete shard %d successfully.", options.shard)
return nil
}
func deleteNode(client *client, options *DeleteOptions, nodeID string) error {
rsp, err := client.restyCli.R().
SetPathParam("namespace", options.namespace).
SetPathParam("cluster", options.cluster).
SetPathParam("shard", strconv.Itoa(options.shard)).
SetPathParam("node", nodeID).
Delete("/namespaces/{namespace}/clusters/{cluster}/shards/{shard}/nodes/{node}")
if err != nil {
return err
}
if rsp.IsError() {
return unmarshalError(rsp.Body())
}
printLine("delete node: %s successfully.", nodeID)
return nil
}
func init() {
DeleteCommand.Flags().StringVarP(&deleteOptions.namespace, "namespace", "n", "", "The namespace")
DeleteCommand.Flags().StringVarP(&deleteOptions.cluster, "cluster", "c", "", "The cluster")
DeleteCommand.Flags().IntVarP(&deleteOptions.shard, "shard", "s", -1, "The shard")
}