Browse Source
Merge
Merge 78f3a6f25d into 5c365e7090
committed by
GitHub
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
1 changed files with 174 additions and 0 deletions
@ -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 |
||||
|
} |
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue