2015-09-13 00:09:03 +00:00
|
|
|
|
package command
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"strings"
|
2018-02-23 23:56:36 +00:00
|
|
|
|
"time"
|
2017-08-22 20:13:44 +00:00
|
|
|
|
|
2018-02-23 23:56:36 +00:00
|
|
|
|
"github.com/hashicorp/nomad/api"
|
2017-08-22 20:13:44 +00:00
|
|
|
|
"github.com/hashicorp/nomad/api/contexts"
|
2018-03-16 17:43:28 +00:00
|
|
|
|
"github.com/hashicorp/nomad/nomad/structs"
|
2017-08-22 20:13:44 +00:00
|
|
|
|
"github.com/posener/complete"
|
2015-09-13 00:09:03 +00:00
|
|
|
|
)
|
|
|
|
|
|
2018-02-23 23:56:36 +00:00
|
|
|
|
var (
|
|
|
|
|
// defaultDrainDuration is the default drain duration if it is not specified
|
|
|
|
|
// explicitly
|
|
|
|
|
defaultDrainDuration = 1 * time.Hour
|
|
|
|
|
)
|
|
|
|
|
|
2015-09-13 00:09:03 +00:00
|
|
|
|
type NodeDrainCommand struct {
|
2015-09-14 20:13:52 +00:00
|
|
|
|
Meta
|
2015-09-13 00:09:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *NodeDrainCommand) Help() string {
|
|
|
|
|
helpText := `
|
2018-02-23 23:56:36 +00:00
|
|
|
|
Usage: nomad node drain [options] <node>
|
2015-09-13 00:09:03 +00:00
|
|
|
|
|
|
|
|
|
Toggles node draining on a specified node. It is required
|
|
|
|
|
that either -enable or -disable is specified, but not both.
|
2016-04-11 22:20:49 +00:00
|
|
|
|
The -self flag is useful to drain the local node.
|
2015-09-13 00:09:03 +00:00
|
|
|
|
|
2015-09-14 20:13:52 +00:00
|
|
|
|
General Options:
|
|
|
|
|
|
|
|
|
|
` + generalOptionsUsage() + `
|
|
|
|
|
|
|
|
|
|
Node Drain Options:
|
2015-09-13 00:09:03 +00:00
|
|
|
|
|
|
|
|
|
-disable
|
|
|
|
|
Disable draining for the specified node.
|
|
|
|
|
|
|
|
|
|
-enable
|
|
|
|
|
Enable draining for the specified node.
|
2016-03-24 21:43:20 +00:00
|
|
|
|
|
2018-02-23 23:56:36 +00:00
|
|
|
|
-deadline <duration>
|
|
|
|
|
Set the deadline by which all allocations must be moved off the node.
|
|
|
|
|
Remaining allocations after the deadline are forced removed from the node.
|
|
|
|
|
If unspecified, a default deadline of one hour is applied.
|
|
|
|
|
|
2018-03-06 22:16:20 +00:00
|
|
|
|
-detach
|
|
|
|
|
Return immediately instead of entering monitor mode.
|
|
|
|
|
|
2018-02-23 23:56:36 +00:00
|
|
|
|
-force
|
|
|
|
|
Force remove allocations off the node immediately.
|
|
|
|
|
|
|
|
|
|
-no-deadline
|
|
|
|
|
No deadline allows the allocations to drain off the node without being force
|
|
|
|
|
stopped after a certain deadline.
|
|
|
|
|
|
|
|
|
|
-ignore-system
|
|
|
|
|
Ignore system allows the drain to complete without stopping system job
|
2018-02-27 17:41:55 +00:00
|
|
|
|
allocations. By default system jobs are stopped last.
|
2018-02-23 23:56:36 +00:00
|
|
|
|
|
2018-03-08 19:06:30 +00:00
|
|
|
|
-keep-ineligible
|
|
|
|
|
Keep ineligible will maintain the node's scheduling ineligibility even if
|
|
|
|
|
the drain is being disabled. This is useful when an existing drain is being
|
|
|
|
|
cancelled but additional scheduling on the node is not desired.
|
|
|
|
|
|
2016-04-11 22:20:49 +00:00
|
|
|
|
-self
|
2018-02-23 23:56:36 +00:00
|
|
|
|
Set the drain status of the local node.
|
2016-04-11 22:20:49 +00:00
|
|
|
|
|
2016-03-24 21:43:20 +00:00
|
|
|
|
-yes
|
|
|
|
|
Automatic yes to prompts.
|
2015-09-13 00:09:03 +00:00
|
|
|
|
`
|
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *NodeDrainCommand) Synopsis() string {
|
2015-09-13 18:39:49 +00:00
|
|
|
|
return "Toggle drain mode on a given node"
|
2015-09-13 00:09:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-22 20:13:44 +00:00
|
|
|
|
func (c *NodeDrainCommand) AutocompleteFlags() complete.Flags {
|
2017-08-23 21:56:21 +00:00
|
|
|
|
return mergeAutocompleteFlags(c.Meta.AutocompleteFlags(FlagSetClient),
|
|
|
|
|
complete.Flags{
|
2018-03-08 19:06:30 +00:00
|
|
|
|
"-disable": complete.PredictNothing,
|
|
|
|
|
"-enable": complete.PredictNothing,
|
|
|
|
|
"-deadline": complete.PredictAnything,
|
2018-03-06 22:16:20 +00:00
|
|
|
|
"-detach": complete.PredictNothing,
|
2018-03-08 19:06:30 +00:00
|
|
|
|
"-force": complete.PredictNothing,
|
|
|
|
|
"-no-deadline": complete.PredictNothing,
|
|
|
|
|
"-ignore-system": complete.PredictNothing,
|
|
|
|
|
"-keep-ineligible": complete.PredictNothing,
|
|
|
|
|
"-self": complete.PredictNothing,
|
|
|
|
|
"-yes": complete.PredictNothing,
|
2017-08-23 21:56:21 +00:00
|
|
|
|
})
|
2017-08-22 20:13:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *NodeDrainCommand) AutocompleteArgs() complete.Predictor {
|
|
|
|
|
return complete.PredictFunc(func(a complete.Args) []string {
|
2017-08-29 21:29:32 +00:00
|
|
|
|
client, err := c.Meta.Client()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-28 05:17:51 +00:00
|
|
|
|
resp, _, err := client.Search().PrefixSearch(a.Last, contexts.Nodes, nil)
|
2017-08-22 20:13:44 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
return []string{}
|
|
|
|
|
}
|
|
|
|
|
return resp.Matches[contexts.Nodes]
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 00:09:03 +00:00
|
|
|
|
func (c *NodeDrainCommand) Run(args []string) int {
|
2018-03-06 22:16:20 +00:00
|
|
|
|
var enable, disable, detach, force,
|
2018-03-08 19:06:30 +00:00
|
|
|
|
noDeadline, ignoreSystem, keepIneligible, self, autoYes bool
|
2018-02-23 23:56:36 +00:00
|
|
|
|
var deadline string
|
2015-09-13 00:09:03 +00:00
|
|
|
|
|
2015-09-14 20:13:52 +00:00
|
|
|
|
flags := c.Meta.FlagSet("node-drain", FlagSetClient)
|
2015-09-13 00:09:03 +00:00
|
|
|
|
flags.Usage = func() { c.Ui.Output(c.Help()) }
|
|
|
|
|
flags.BoolVar(&enable, "enable", false, "Enable drain mode")
|
|
|
|
|
flags.BoolVar(&disable, "disable", false, "Disable drain mode")
|
2018-02-23 23:56:36 +00:00
|
|
|
|
flags.StringVar(&deadline, "deadline", "", "Deadline after which allocations are force stopped")
|
2018-03-06 22:16:20 +00:00
|
|
|
|
flags.BoolVar(&detach, "detach", false, "")
|
2018-02-23 23:56:36 +00:00
|
|
|
|
flags.BoolVar(&force, "force", false, "Force immediate drain")
|
|
|
|
|
flags.BoolVar(&noDeadline, "no-deadline", false, "Drain node with no deadline")
|
|
|
|
|
flags.BoolVar(&ignoreSystem, "ignore-system", false, "Do not drain system job allocations from the node")
|
2018-03-08 19:06:30 +00:00
|
|
|
|
flags.BoolVar(&keepIneligible, "keep-ineligible", false, "Do not update the nodes scheduling eligibility")
|
2016-04-11 22:20:49 +00:00
|
|
|
|
flags.BoolVar(&self, "self", false, "")
|
2016-03-24 21:43:20 +00:00
|
|
|
|
flags.BoolVar(&autoYes, "yes", false, "Automatic yes to prompts.")
|
2015-09-13 00:09:03 +00:00
|
|
|
|
|
|
|
|
|
if err := flags.Parse(args); err != nil {
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check that we got either enable or disable, but not both.
|
|
|
|
|
if (enable && disable) || (!enable && !disable) {
|
|
|
|
|
c.Ui.Error(c.Help())
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check that we got a node ID
|
2015-09-14 20:13:52 +00:00
|
|
|
|
args = flags.Args()
|
2016-04-11 22:20:49 +00:00
|
|
|
|
if l := len(args); self && l != 0 || !self && l != 1 {
|
2018-02-23 23:56:36 +00:00
|
|
|
|
c.Ui.Error("Node ID must be specified if -self isn't being used")
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate a compatible set of flags were set
|
|
|
|
|
if disable && (deadline != "" || force || noDeadline || ignoreSystem) {
|
|
|
|
|
c.Ui.Error("-disable can't be combined with flags configuring drain strategy")
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
if deadline != "" && (force || noDeadline) {
|
|
|
|
|
c.Ui.Error("-deadline can't be combined with -force or -no-deadline")
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
if force && noDeadline {
|
|
|
|
|
c.Ui.Error("-force and -no-deadline are mutually exclusive")
|
2015-09-13 00:09:03 +00:00
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-23 23:56:36 +00:00
|
|
|
|
// Parse the duration
|
|
|
|
|
var d time.Duration
|
|
|
|
|
if force {
|
|
|
|
|
d = -1 * time.Second
|
|
|
|
|
} else if noDeadline {
|
|
|
|
|
d = 0
|
|
|
|
|
} else if deadline != "" {
|
|
|
|
|
dur, err := time.ParseDuration(deadline)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to parse deadline %q: %v", deadline, err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
if dur <= 0 {
|
|
|
|
|
c.Ui.Error("A positive drain duration must be given")
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
d = dur
|
|
|
|
|
} else {
|
|
|
|
|
d = defaultDrainDuration
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 00:09:03 +00:00
|
|
|
|
// Get the HTTP client
|
2015-09-14 20:13:52 +00:00
|
|
|
|
client, err := c.Meta.Client()
|
2015-09-13 00:09:03 +00:00
|
|
|
|
if err != nil {
|
2015-09-14 20:13:52 +00:00
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error initializing client: %s", err))
|
2015-09-13 00:09:03 +00:00
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-11 22:20:49 +00:00
|
|
|
|
// If -self flag is set then determine the current node.
|
2017-09-26 22:26:33 +00:00
|
|
|
|
var nodeID string
|
2016-04-11 22:20:49 +00:00
|
|
|
|
if !self {
|
|
|
|
|
nodeID = args[0]
|
|
|
|
|
} else {
|
|
|
|
|
var err error
|
|
|
|
|
if nodeID, err = getLocalNodeID(client); err != nil {
|
|
|
|
|
c.Ui.Error(err.Error())
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-12-20 17:02:10 +00:00
|
|
|
|
// Check if node exists
|
2016-03-17 23:48:45 +00:00
|
|
|
|
if len(nodeID) == 1 {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Identifier must contain at least two characters."))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
2016-01-21 19:53:05 +00:00
|
|
|
|
|
2018-03-11 18:52:59 +00:00
|
|
|
|
nodeID = sanitizeUUIDPrefix(nodeID)
|
2016-03-17 23:48:45 +00:00
|
|
|
|
nodes, _, err := client.Nodes().PrefixList(nodeID)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error toggling drain mode: %s", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
// Return error if no nodes are found
|
|
|
|
|
if len(nodes) == 0 {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("No node(s) with prefix or id %q found", nodeID))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
if len(nodes) > 1 {
|
2018-02-27 22:43:35 +00:00
|
|
|
|
c.Ui.Error(fmt.Sprintf("Prefix matched multiple nodes\n\n%s",
|
|
|
|
|
formatNodeStubList(nodes, true)))
|
2017-07-21 00:31:07 +00:00
|
|
|
|
return 1
|
2016-03-17 23:48:45 +00:00
|
|
|
|
}
|
2016-03-24 21:43:20 +00:00
|
|
|
|
|
2016-03-17 23:48:45 +00:00
|
|
|
|
// Prefix lookup matched a single node
|
|
|
|
|
node, _, err := client.Nodes().Info(nodes[0].ID, nil)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error toggling drain mode: %s", err))
|
|
|
|
|
return 1
|
2015-12-20 17:02:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-24 21:43:20 +00:00
|
|
|
|
// Confirm drain if the node was a prefix match.
|
|
|
|
|
if nodeID != node.ID && !autoYes {
|
|
|
|
|
verb := "enable"
|
|
|
|
|
if disable {
|
|
|
|
|
verb = "disable"
|
|
|
|
|
}
|
|
|
|
|
question := fmt.Sprintf("Are you sure you want to %s drain mode for node %q? [y/N]", verb, node.ID)
|
|
|
|
|
answer, err := c.Ui.Ask(question)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to parse answer: %v", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if answer == "" || strings.ToLower(answer)[0] == 'n' {
|
|
|
|
|
// No case
|
|
|
|
|
c.Ui.Output("Canceling drain toggle")
|
|
|
|
|
return 0
|
2016-03-28 02:17:29 +00:00
|
|
|
|
} else if strings.ToLower(answer)[0] == 'y' && len(answer) > 1 {
|
2016-03-24 21:43:20 +00:00
|
|
|
|
// Non exact match yes
|
|
|
|
|
c.Ui.Output("For confirmation, an exact ‘y’ is required.")
|
|
|
|
|
return 0
|
|
|
|
|
} else if answer != "y" {
|
|
|
|
|
c.Ui.Output("No confirmation detected. For confirmation, an exact 'y' is required.")
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-23 23:56:36 +00:00
|
|
|
|
var spec *api.DrainSpec
|
|
|
|
|
if enable {
|
|
|
|
|
spec = &api.DrainSpec{
|
|
|
|
|
Deadline: d,
|
|
|
|
|
IgnoreSystemJobs: ignoreSystem,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 00:09:03 +00:00
|
|
|
|
// Toggle node draining
|
2018-03-06 22:16:20 +00:00
|
|
|
|
meta, err := client.Nodes().UpdateDrain(node.ID, spec, !keepIneligible, nil)
|
|
|
|
|
if err != nil {
|
2018-02-23 23:56:36 +00:00
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error updating drain specification: %s", err))
|
2015-09-13 00:09:03 +00:00
|
|
|
|
return 1
|
|
|
|
|
}
|
2018-02-27 21:54:27 +00:00
|
|
|
|
|
|
|
|
|
c.Ui.Output(fmt.Sprintf("Node %q drain strategy set", node.ID))
|
2018-03-06 22:16:20 +00:00
|
|
|
|
|
|
|
|
|
if enable && !detach {
|
|
|
|
|
if err := monitorDrain(c.Ui.Output, client.Nodes(), node.ID, meta.LastIndex); err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error monitoring drain: %v", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
c.Ui.Output(fmt.Sprintf("Node %q drain complete", nodeID))
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-13 00:09:03 +00:00
|
|
|
|
return 0
|
|
|
|
|
}
|
2018-03-06 22:16:20 +00:00
|
|
|
|
|
|
|
|
|
// monitorDrain monitors the node being drained and exits when the node has
|
|
|
|
|
// finished draining.
|
|
|
|
|
func monitorDrain(output func(string), nodeClient *api.Nodes, nodeID string, index uint64) error {
|
|
|
|
|
doneCh := make(chan struct{})
|
|
|
|
|
defer close(doneCh)
|
|
|
|
|
|
|
|
|
|
// Errors from either goroutine are sent here
|
|
|
|
|
errCh := make(chan error, 1)
|
|
|
|
|
|
|
|
|
|
// Monitor node changes and close chan when drain is complete
|
|
|
|
|
nodeCh := make(chan struct{})
|
|
|
|
|
go func() {
|
|
|
|
|
for {
|
|
|
|
|
q := api.QueryOptions{
|
|
|
|
|
AllowStale: true,
|
|
|
|
|
WaitIndex: index,
|
|
|
|
|
}
|
|
|
|
|
node, meta, err := nodeClient.Info(nodeID, &q)
|
|
|
|
|
if err != nil {
|
|
|
|
|
select {
|
|
|
|
|
case errCh <- err:
|
|
|
|
|
case <-doneCh:
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if node.DrainStrategy == nil {
|
|
|
|
|
close(nodeCh)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Drain still ongoing
|
|
|
|
|
index = meta.LastIndex
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
// Monitor alloc changes
|
|
|
|
|
allocCh := make(chan string, 1)
|
|
|
|
|
go func() {
|
|
|
|
|
allocs, meta, err := nodeClient.Allocations(nodeID, nil)
|
|
|
|
|
if err != nil {
|
|
|
|
|
select {
|
|
|
|
|
case errCh <- err:
|
|
|
|
|
case <-doneCh:
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
initial := make(map[string]*api.Allocation, len(allocs))
|
|
|
|
|
for _, a := range allocs {
|
|
|
|
|
initial[a.ID] = a
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for {
|
|
|
|
|
q := api.QueryOptions{
|
|
|
|
|
AllowStale: true,
|
|
|
|
|
WaitIndex: meta.LastIndex,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
allocs, meta, err = nodeClient.Allocations(nodeID, &q)
|
|
|
|
|
if err != nil {
|
|
|
|
|
select {
|
|
|
|
|
case errCh <- err:
|
|
|
|
|
case <-doneCh:
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, a := range allocs {
|
|
|
|
|
// Get previous version of alloc
|
|
|
|
|
orig, ok := initial[a.ID]
|
|
|
|
|
|
|
|
|
|
// Update local alloc state
|
|
|
|
|
initial[a.ID] = a
|
|
|
|
|
|
2018-03-16 17:43:28 +00:00
|
|
|
|
migrating := a.DesiredTransition.ShouldMigrate()
|
|
|
|
|
|
2018-03-06 22:16:20 +00:00
|
|
|
|
msg := ""
|
|
|
|
|
switch {
|
|
|
|
|
case !ok:
|
|
|
|
|
// Should only be possible if response
|
|
|
|
|
// from initial Allocations call was
|
|
|
|
|
// stale. No need to output
|
|
|
|
|
|
|
|
|
|
case orig.ClientStatus != a.ClientStatus:
|
|
|
|
|
// Alloc status has changed; output
|
|
|
|
|
msg = fmt.Sprintf("status %s -> %s", orig.ClientStatus, a.ClientStatus)
|
|
|
|
|
|
2018-03-16 17:43:28 +00:00
|
|
|
|
case migrating && !orig.DesiredTransition.ShouldMigrate():
|
|
|
|
|
// Alloc was marked for migration
|
|
|
|
|
msg = "marked for migration"
|
|
|
|
|
case migrating && (orig.DesiredStatus != a.DesiredStatus) && a.DesiredStatus == structs.AllocDesiredStatusStop:
|
|
|
|
|
// Alloc has already been marked for migration and is now being stopped
|
2018-03-06 22:16:20 +00:00
|
|
|
|
msg = "draining"
|
2018-03-16 17:43:28 +00:00
|
|
|
|
case a.NextAllocation != "" && orig.NextAllocation == "":
|
|
|
|
|
// Alloc has been replaced by another allocation
|
|
|
|
|
msg = fmt.Sprintf("replaced by allocation %q", a.NextAllocation)
|
2018-03-06 22:16:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if msg != "" {
|
|
|
|
|
select {
|
|
|
|
|
case allocCh <- fmt.Sprintf("Alloc %q %s", a.ID, msg):
|
|
|
|
|
case <-doneCh:
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
|
2018-03-16 17:43:28 +00:00
|
|
|
|
done := false
|
|
|
|
|
for !done {
|
2018-03-06 22:16:20 +00:00
|
|
|
|
select {
|
|
|
|
|
case err := <-errCh:
|
|
|
|
|
return err
|
|
|
|
|
case <-nodeCh:
|
2018-03-16 17:43:28 +00:00
|
|
|
|
done = true
|
|
|
|
|
case msg := <-allocCh:
|
|
|
|
|
output(msg)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Loop on alloc messages for a bit longer as we may have gotten the
|
|
|
|
|
// "node done" first (since the watchers run concurrently the events
|
|
|
|
|
// may be received out of order)
|
2018-03-27 21:11:33 +00:00
|
|
|
|
deadline := 500 * time.Millisecond
|
2018-03-16 17:43:28 +00:00
|
|
|
|
timer := time.NewTimer(deadline)
|
|
|
|
|
for {
|
|
|
|
|
select {
|
|
|
|
|
case err := <-errCh:
|
|
|
|
|
return err
|
2018-03-06 22:16:20 +00:00
|
|
|
|
case msg := <-allocCh:
|
|
|
|
|
output(msg)
|
2018-03-16 17:43:28 +00:00
|
|
|
|
if !timer.Stop() {
|
|
|
|
|
<-timer.C
|
|
|
|
|
}
|
|
|
|
|
timer.Reset(deadline)
|
|
|
|
|
case <-timer.C:
|
|
|
|
|
// No events within deadline, exit
|
|
|
|
|
return nil
|
2018-03-06 22:16:20 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|