cmd/client/command/failover.go (79 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" "github.com/spf13/cobra" ) type FailoverOptions struct { namespace string cluster string preferred string } var failoverOptions FailoverOptions var FailoverCommand = &cobra.Command{ Use: "failover", Short: "Failover the master of a shard", Example: ` # Failover the master of a shard kvctl failover shard <shard_index> -n <namespace> -c <cluster> # Failover the master of a shard with preferred slave kvctl failover shard <shard_index> --preferred <node_id> -n <namespace> -c <cluster> `, PreRunE: failoverPreRun, RunE: func(cmd *cobra.Command, args []string) error { if len(args) < 2 { return fmt.Errorf("missing shard index, plese specify the shard index") } host, _ := cmd.Flags().GetString("host") client := newClient(host) resource := args[0] switch resource { case "shard": shardIndex, err := strconv.Atoi(args[1]) if err != nil { return fmt.Errorf("invalid shard index: %s", args[1]) } return failoverShard(client, &failoverOptions, shardIndex) default: return fmt.Errorf("unsupported resource type: %s", resource) } }, } func failoverPreRun(cmd *cobra.Command, args []string) error { if len(args) < 1 { return fmt.Errorf("missing resource name, must be [shard]") } if failoverOptions.namespace == "" { return fmt.Errorf("namespace is required") } if failoverOptions.cluster == "" { return fmt.Errorf("cluster is required") } return nil } func failoverShard(client *client, options *FailoverOptions, shardIndex int) error { rsp, err := client.restyCli.R(). SetPathParam("namespace", options.namespace). SetPathParam("cluster", options.cluster). SetPathParam("shard", strconv.Itoa(shardIndex)). Post("/namespaces/{namespace}/clusters/{cluster}/shards/{shard}/failover") if err != nil { return err } if rsp.IsError() { return unmarshalError(rsp.Body()) } var result struct { NewMasterID string `json:"new_master_id"` } if err := unmarshalData(rsp.Body(), &result); err != nil { return err } printLine("failover shard %d successfully, new master id: %s.", shardIndex, result.NewMasterID) return nil } func init() { FailoverCommand.Flags().StringVarP(&failoverOptions.namespace, "namespace", "n", "", "The namespace of the cluster") FailoverCommand.Flags().StringVarP(&failoverOptions.cluster, "cluster", "c", "", "The name of the cluster") FailoverCommand.Flags().StringVarP(&failoverOptions.preferred, "preferred", "", "", "The preferred slave node id") }