Browse Source

Merge 78f3a6f25d into 5c365e7090

pull/8271/merge
Lisandro Pin 8 hours ago
committed by GitHub
parent
commit
58f3bf2e48
No known key found for this signature in database GPG Key ID: B5690EEEBB952194
  1. 174
      weed/shell/command_volume_server_state.go

174
weed/shell/command_volume_server_state.go

@ -0,0 +1,174 @@
package shell
import (
"context"
"flag"
"fmt"
"io"
"slices"
"strings"
"github.com/seaweedfs/seaweedfs/weed/operation"
"github.com/seaweedfs/seaweedfs/weed/pb"
"github.com/seaweedfs/seaweedfs/weed/pb/volume_server_pb"
"google.golang.org/protobuf/proto"
)
func init() {
Commands = append(Commands, &commandVolumeServerState{})
}
type commandVolumeServerState struct {
env *CommandEnv
writer io.Writer
}
func (c *commandVolumeServerState) Name() string {
return "volumeServer.state"
}
func (c *commandVolumeServerState) Help() string {
return `query/update volume server state settings
volumeServer.state [-nodes <host:port>] [..state flags...]
This command display volume server state flags for the provided
list of nodes; if empty, all nodes in the topology are queried.
For example:
volumeServer.state --nodes 192.168.10.111:9000,192.168.10.112:9000
Additionally, if any flags are provided, these are applied
to the selected node(s). The command will display the resulting
state for each node *after* the state is updated. For exmaple...
volumeServer.state --nodes 192.168.10.111:9000 --maintenanceOn
...will set the specified volume server to maintenance mode.
`
}
func (c *commandVolumeServerState) HasTag(CommandTag) bool {
return false
}
func (c *commandVolumeServerState) write(format string, a ...any) {
format = strings.TrimRight(format, " ")
if len(format) == 0 {
format = "\n"
}
fmt.Fprintf(c.writer, format, a...)
last := format[len(format)-1:]
if last != "\n" && last != "\r" {
fmt.Fprint(c.writer, "\n")
}
}
func (c *commandVolumeServerState) bool2Str(b bool) string {
if b {
return "YES"
}
return "no"
}
func (c *commandVolumeServerState) Do(args []string, commandEnv *CommandEnv, writer io.Writer) (err error) {
vsStateCommand := flag.NewFlagSet(c.Name(), flag.ContinueOnError)
nodesStr := vsStateCommand.String("nodes", "", "comma-separated list of volume servers <host>:<port> (optional)")
maintenanceOn := vsStateCommand.Bool("maintenanceOn", false, "enables maintenance mode on the server")
maintenanceOff := vsStateCommand.Bool("maintenanceOff", false, "disables maintenance mode on the server")
if err = vsStateCommand.Parse(args); err != nil {
return err
}
if *maintenanceOn && *maintenanceOff {
return fmt.Errorf("--maintenanceOn and --maintenanceOff are mutually exclusive")
}
volumeServerAddrs := []pb.ServerAddress{}
if *nodesStr != "" {
for _, addr := range strings.Split(*nodesStr, ",") {
volumeServerAddrs = append(volumeServerAddrs, pb.ServerAddress(addr))
}
} else {
dns, err := collectDataNodes(commandEnv, 0)
if err != nil {
return err
}
for _, dn := range dns {
volumeServerAddrs = append(volumeServerAddrs, pb.ServerAddress(dn.Address))
}
}
slices.Sort(volumeServerAddrs)
if len(volumeServerAddrs) == 0 {
return fmt.Errorf("no volume servers specified")
}
c.env = commandEnv
c.writer = writer
for _, addr := range volumeServerAddrs {
state, err := c.getVolumeServerState(addr)
if err != nil {
return fmt.Errorf("failed to load state from %v: %v", addr, err)
}
stateOrig := &volume_server_pb.VolumeServerState{}
proto.Merge(stateOrig, state)
// apply updates, if any
switch {
case *maintenanceOn:
state.Maintenance = true
case *maintenanceOff:
state.Maintenance = false
}
// update volume server state if settings changed
if !proto.Equal(state, stateOrig) {
state, err = c.setVolumeServerState(addr, state)
if err != nil {
return fmt.Errorf("failed to update state for %v: %v", addr, err)
}
}
c.write("%v\t -> Maintenance mode: %s\n", addr.String(), c.bool2Str(state.GetMaintenance()))
}
return
}
func (c *commandVolumeServerState) getVolumeServerState(volumeServerAddress pb.ServerAddress) (*volume_server_pb.VolumeServerState, error) {
var state *volume_server_pb.VolumeServerState
err := operation.WithVolumeServerClient(false, volumeServerAddress, c.env.option.GrpcDialOption, func(volumeServerClient volume_server_pb.VolumeServerClient) error {
res, err := volumeServerClient.GetState(context.Background(), &volume_server_pb.GetStateRequest{})
if err != nil {
return err
}
state = res.GetState()
return nil
})
return state, err
}
func (c *commandVolumeServerState) setVolumeServerState(volumeServerAddress pb.ServerAddress, state *volume_server_pb.VolumeServerState) (*volume_server_pb.VolumeServerState, error) {
var stateAfter *volume_server_pb.VolumeServerState
err := operation.WithVolumeServerClient(false, volumeServerAddress, c.env.option.GrpcDialOption, func(volumeServerClient volume_server_pb.VolumeServerClient) error {
res, err := volumeServerClient.SetState(context.Background(), &volume_server_pb.SetStateRequest{
State: state,
})
if err != nil {
return err
}
stateAfter = res.GetState()
return nil
})
return stateAfter, err
}
Loading…
Cancel
Save