mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
Support node label update.
Signed-off-by: Dong Chen <dongluo.chen@docker.com>
This commit is contained in:
parent
e6a97db2c9
commit
e1165cdfd1
6 changed files with 77 additions and 14 deletions
|
@ -21,8 +21,9 @@ func newAcceptCommand(dockerCli *client.DockerCli) *cobra.Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
func runAccept(dockerCli *client.DockerCli, nodes []string) error {
|
func runAccept(dockerCli *client.DockerCli, nodes []string) error {
|
||||||
accept := func(node *swarm.Node) {
|
accept := func(node *swarm.Node) error {
|
||||||
node.Spec.Membership = swarm.NodeMembershipAccepted
|
node.Spec.Membership = swarm.NodeMembershipAccepted
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
success := func(nodeID string) {
|
success := func(nodeID string) {
|
||||||
fmt.Fprintf(dockerCli.Out(), "Node %s accepted in the swarm.\n", nodeID)
|
fmt.Fprintf(dockerCli.Out(), "Node %s accepted in the swarm.\n", nodeID)
|
||||||
|
|
|
@ -21,8 +21,9 @@ func newDemoteCommand(dockerCli *client.DockerCli) *cobra.Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
func runDemote(dockerCli *client.DockerCli, nodes []string) error {
|
func runDemote(dockerCli *client.DockerCli, nodes []string) error {
|
||||||
demote := func(node *swarm.Node) {
|
demote := func(node *swarm.Node) error {
|
||||||
node.Spec.Role = swarm.NodeRoleWorker
|
node.Spec.Role = swarm.NodeRoleWorker
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
success := func(nodeID string) {
|
success := func(nodeID string) {
|
||||||
fmt.Fprintf(dockerCli.Out(), "Manager %s demoted in the swarm.\n", nodeID)
|
fmt.Fprintf(dockerCli.Out(), "Manager %s demoted in the swarm.\n", nodeID)
|
||||||
|
|
|
@ -4,18 +4,37 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
"github.com/docker/docker/opts"
|
||||||
|
runconfigopts "github.com/docker/docker/runconfig/opts"
|
||||||
"github.com/docker/engine-api/types/swarm"
|
"github.com/docker/engine-api/types/swarm"
|
||||||
)
|
)
|
||||||
|
|
||||||
type nodeOptions struct {
|
type nodeOptions struct {
|
||||||
|
annotations
|
||||||
role string
|
role string
|
||||||
membership string
|
membership string
|
||||||
availability string
|
availability string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type annotations struct {
|
||||||
|
name string
|
||||||
|
labels opts.ListOpts
|
||||||
|
}
|
||||||
|
|
||||||
|
func newNodeOptions() *nodeOptions {
|
||||||
|
return &nodeOptions{
|
||||||
|
annotations: annotations{
|
||||||
|
labels: opts.NewListOpts(nil),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (opts *nodeOptions) ToNodeSpec() (swarm.NodeSpec, error) {
|
func (opts *nodeOptions) ToNodeSpec() (swarm.NodeSpec, error) {
|
||||||
var spec swarm.NodeSpec
|
var spec swarm.NodeSpec
|
||||||
|
|
||||||
|
spec.Annotations.Name = opts.annotations.name
|
||||||
|
spec.Annotations.Labels = runconfigopts.ConvertKVStringsToMap(opts.annotations.labels.GetAll())
|
||||||
|
|
||||||
switch swarm.NodeRole(strings.ToLower(opts.role)) {
|
switch swarm.NodeRole(strings.ToLower(opts.role)) {
|
||||||
case swarm.NodeRoleWorker:
|
case swarm.NodeRoleWorker:
|
||||||
spec.Role = swarm.NodeRoleWorker
|
spec.Role = swarm.NodeRoleWorker
|
||||||
|
|
|
@ -21,8 +21,9 @@ func newPromoteCommand(dockerCli *client.DockerCli) *cobra.Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
func runPromote(dockerCli *client.DockerCli, nodes []string) error {
|
func runPromote(dockerCli *client.DockerCli, nodes []string) error {
|
||||||
promote := func(node *swarm.Node) {
|
promote := func(node *swarm.Node) error {
|
||||||
node.Spec.Role = swarm.NodeRoleManager
|
node.Spec.Role = swarm.NodeRoleManager
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
success := func(nodeID string) {
|
success := func(nodeID string) {
|
||||||
fmt.Fprintf(dockerCli.Out(), "Node %s promoted to a manager in the swarm.\n", nodeID)
|
fmt.Fprintf(dockerCli.Out(), "Node %s promoted to a manager in the swarm.\n", nodeID)
|
||||||
|
|
|
@ -5,6 +5,8 @@ import (
|
||||||
|
|
||||||
"github.com/docker/docker/api/client"
|
"github.com/docker/docker/api/client"
|
||||||
"github.com/docker/docker/cli"
|
"github.com/docker/docker/cli"
|
||||||
|
"github.com/docker/docker/opts"
|
||||||
|
runconfigopts "github.com/docker/docker/runconfig/opts"
|
||||||
"github.com/docker/engine-api/types/swarm"
|
"github.com/docker/engine-api/types/swarm"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"github.com/spf13/pflag"
|
"github.com/spf13/pflag"
|
||||||
|
@ -12,7 +14,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func newUpdateCommand(dockerCli *client.DockerCli) *cobra.Command {
|
func newUpdateCommand(dockerCli *client.DockerCli) *cobra.Command {
|
||||||
var opts nodeOptions
|
nodeOpts := newNodeOptions()
|
||||||
|
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "update [OPTIONS] NODE",
|
Use: "update [OPTIONS] NODE",
|
||||||
|
@ -24,9 +26,12 @@ func newUpdateCommand(dockerCli *client.DockerCli) *cobra.Command {
|
||||||
}
|
}
|
||||||
|
|
||||||
flags := cmd.Flags()
|
flags := cmd.Flags()
|
||||||
flags.StringVar(&opts.role, flagRole, "", "Role of the node (worker/manager)")
|
flags.StringVar(&nodeOpts.role, flagRole, "", "Role of the node (worker/manager)")
|
||||||
flags.StringVar(&opts.membership, flagMembership, "", "Membership of the node (accepted/rejected)")
|
flags.StringVar(&nodeOpts.membership, flagMembership, "", "Membership of the node (accepted/rejected)")
|
||||||
flags.StringVar(&opts.availability, flagAvailability, "", "Availability of the node (active/pause/drain)")
|
flags.StringVar(&nodeOpts.availability, flagAvailability, "", "Availability of the node (active/pause/drain)")
|
||||||
|
flags.Var(&nodeOpts.annotations.labels, flagLabelAdd, "Add or update a node label (key=value)")
|
||||||
|
labelKeys := opts.NewListOpts(nil)
|
||||||
|
flags.Var(&labelKeys, flagLabelRemove, "Remove a node label if exists")
|
||||||
return cmd
|
return cmd
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,7 +42,7 @@ func runUpdate(dockerCli *client.DockerCli, flags *pflag.FlagSet, nodeID string)
|
||||||
return updateNodes(dockerCli, []string{nodeID}, mergeNodeUpdate(flags), success)
|
return updateNodes(dockerCli, []string{nodeID}, mergeNodeUpdate(flags), success)
|
||||||
}
|
}
|
||||||
|
|
||||||
func updateNodes(dockerCli *client.DockerCli, nodes []string, mergeNode func(node *swarm.Node), success func(nodeID string)) error {
|
func updateNodes(dockerCli *client.DockerCli, nodes []string, mergeNode func(node *swarm.Node) error, success func(nodeID string)) error {
|
||||||
client := dockerCli.Client()
|
client := dockerCli.Client()
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
|
@ -47,7 +52,10 @@ func updateNodes(dockerCli *client.DockerCli, nodes []string, mergeNode func(nod
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
mergeNode(&node)
|
err = mergeNode(&node)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
err = client.NodeUpdate(ctx, node.ID, node.Version, node.Spec)
|
err = client.NodeUpdate(ctx, node.ID, node.Version, node.Spec)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -57,22 +65,51 @@ func updateNodes(dockerCli *client.DockerCli, nodes []string, mergeNode func(nod
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func mergeNodeUpdate(flags *pflag.FlagSet) func(*swarm.Node) {
|
func mergeNodeUpdate(flags *pflag.FlagSet) func(*swarm.Node) error {
|
||||||
return func(node *swarm.Node) {
|
return func(node *swarm.Node) error {
|
||||||
spec := &node.Spec
|
spec := &node.Spec
|
||||||
|
|
||||||
if flags.Changed(flagRole) {
|
if flags.Changed(flagRole) {
|
||||||
str, _ := flags.GetString(flagRole)
|
str, err := flags.GetString(flagRole)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
spec.Role = swarm.NodeRole(str)
|
spec.Role = swarm.NodeRole(str)
|
||||||
}
|
}
|
||||||
if flags.Changed(flagMembership) {
|
if flags.Changed(flagMembership) {
|
||||||
str, _ := flags.GetString(flagMembership)
|
str, err := flags.GetString(flagMembership)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
spec.Membership = swarm.NodeMembership(str)
|
spec.Membership = swarm.NodeMembership(str)
|
||||||
}
|
}
|
||||||
if flags.Changed(flagAvailability) {
|
if flags.Changed(flagAvailability) {
|
||||||
str, _ := flags.GetString(flagAvailability)
|
str, err := flags.GetString(flagAvailability)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
spec.Availability = swarm.NodeAvailability(str)
|
spec.Availability = swarm.NodeAvailability(str)
|
||||||
}
|
}
|
||||||
|
if spec.Annotations.Labels == nil {
|
||||||
|
spec.Annotations.Labels = make(map[string]string)
|
||||||
|
}
|
||||||
|
if flags.Changed(flagLabelAdd) {
|
||||||
|
labels := flags.Lookup(flagLabelAdd).Value.(*opts.ListOpts).GetAll()
|
||||||
|
for k, v := range runconfigopts.ConvertKVStringsToMap(labels) {
|
||||||
|
spec.Annotations.Labels[k] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if flags.Changed(flagLabelRemove) {
|
||||||
|
keys := flags.Lookup(flagLabelRemove).Value.(*opts.ListOpts).GetAll()
|
||||||
|
for _, k := range keys {
|
||||||
|
// if a key doesn't exist, fail the command explicitly
|
||||||
|
if _, exists := spec.Annotations.Labels[k]; !exists {
|
||||||
|
return fmt.Errorf("key %s doesn't exist in node's labels", k)
|
||||||
|
}
|
||||||
|
delete(spec.Annotations.Labels, k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -80,4 +117,6 @@ const (
|
||||||
flagRole = "role"
|
flagRole = "role"
|
||||||
flagMembership = "membership"
|
flagMembership = "membership"
|
||||||
flagAvailability = "availability"
|
flagAvailability = "availability"
|
||||||
|
flagLabelAdd = "label-add"
|
||||||
|
flagLabelRemove = "label-rm"
|
||||||
)
|
)
|
||||||
|
|
|
@ -19,6 +19,8 @@ Update a node
|
||||||
Options:
|
Options:
|
||||||
--availability string Availability of the node (active/pause/drain)
|
--availability string Availability of the node (active/pause/drain)
|
||||||
--help Print usage
|
--help Print usage
|
||||||
|
--label-add value Add or update a node label (key=value) (default [])
|
||||||
|
--label-rm value Remove a node label if exists (default [])
|
||||||
--membership string Membership of the node (accepted/rejected)
|
--membership string Membership of the node (accepted/rejected)
|
||||||
--role string Role of the node (worker/manager)
|
--role string Role of the node (worker/manager)
|
||||||
```
|
```
|
||||||
|
|
Loading…
Reference in a new issue