2015-09-17 00:35:58 +00:00
|
|
|
|
package command
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"strings"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type StopCommand struct {
|
|
|
|
|
Meta
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *StopCommand) Help() string {
|
|
|
|
|
helpText := `
|
|
|
|
|
Usage: nomad stop [options] <job>
|
|
|
|
|
|
|
|
|
|
Stop an existing job. This command is used to signal allocations
|
2015-09-18 04:09:34 +00:00
|
|
|
|
to shut down for the given job ID. Upon successful deregistraion,
|
|
|
|
|
an interactive monitor session will start to display log lines as
|
|
|
|
|
the job unwinds its allocations and completes shutting down. It
|
|
|
|
|
is safe to exit the monitor early using ctrl+c.
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
General Options:
|
|
|
|
|
|
|
|
|
|
` + generalOptionsUsage() + `
|
|
|
|
|
|
|
|
|
|
Stop Options:
|
|
|
|
|
|
2015-09-18 04:09:34 +00:00
|
|
|
|
-detach
|
|
|
|
|
Return immediately instead of entering monitor mode. After the
|
|
|
|
|
deregister command is submitted, a new evaluation ID is printed
|
|
|
|
|
to the screen, which can be used to call up a monitor later if
|
|
|
|
|
needed using the eval-monitor command.
|
2016-01-14 20:57:43 +00:00
|
|
|
|
|
2016-03-24 21:43:20 +00:00
|
|
|
|
-yes
|
|
|
|
|
Automatic yes to prompts.
|
|
|
|
|
|
2016-01-15 22:32:38 +00:00
|
|
|
|
-verbose
|
|
|
|
|
Display full information.
|
2015-09-17 00:35:58 +00:00
|
|
|
|
`
|
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *StopCommand) Synopsis() string {
|
|
|
|
|
return "Stop a running job"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *StopCommand) Run(args []string) int {
|
2016-03-24 21:43:20 +00:00
|
|
|
|
var detach, verbose, autoYes bool
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
flags := c.Meta.FlagSet("stop", FlagSetClient)
|
|
|
|
|
flags.Usage = func() { c.Ui.Output(c.Help()) }
|
2015-09-18 04:09:34 +00:00
|
|
|
|
flags.BoolVar(&detach, "detach", false, "")
|
2016-01-15 22:32:38 +00:00
|
|
|
|
flags.BoolVar(&verbose, "verbose", false, "")
|
2016-03-24 21:43:20 +00:00
|
|
|
|
flags.BoolVar(&autoYes, "yes", false, "")
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
if err := flags.Parse(args); err != nil {
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-14 20:57:43 +00:00
|
|
|
|
// Truncate the id unless full length is requested
|
2016-01-15 22:32:38 +00:00
|
|
|
|
length := shortId
|
|
|
|
|
if verbose {
|
|
|
|
|
length = fullId
|
2016-01-14 20:57:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-09-17 00:35:58 +00:00
|
|
|
|
// Check that we got exactly one job
|
|
|
|
|
args = flags.Args()
|
|
|
|
|
if len(args) != 1 {
|
|
|
|
|
c.Ui.Error(c.Help())
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
jobID := args[0]
|
|
|
|
|
|
|
|
|
|
// Get the HTTP client
|
|
|
|
|
client, err := c.Meta.Client()
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error initializing client: %s", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check if the job exists
|
2016-03-21 20:00:14 +00:00
|
|
|
|
jobs, _, err := client.Jobs().PrefixList(jobID)
|
2015-12-20 11:14:59 +00:00
|
|
|
|
if err != nil {
|
2016-03-21 20:00:14 +00:00
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error deregistering job: %s", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
if len(jobs) == 0 {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("No job(s) with prefix or id %q found", jobID))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
2016-04-13 23:05:58 +00:00
|
|
|
|
if len(jobs) > 1 && strings.TrimSpace(jobID) != jobs[0].ID {
|
2016-03-21 20:00:14 +00:00
|
|
|
|
out := make([]string, len(jobs)+1)
|
|
|
|
|
out[0] = "ID|Type|Priority|Status"
|
|
|
|
|
for i, job := range jobs {
|
|
|
|
|
out[i+1] = fmt.Sprintf("%s|%s|%d|%s",
|
|
|
|
|
job.ID,
|
|
|
|
|
job.Type,
|
|
|
|
|
job.Priority,
|
|
|
|
|
job.Status)
|
2015-12-24 10:46:59 +00:00
|
|
|
|
}
|
2016-03-21 20:00:14 +00:00
|
|
|
|
c.Ui.Output(fmt.Sprintf("Prefix matched multiple jobs\n\n%s", formatList(out)))
|
|
|
|
|
return 0
|
|
|
|
|
}
|
|
|
|
|
// Prefix lookup matched a single job
|
|
|
|
|
job, _, err := client.Jobs().Info(jobs[0].ID, nil)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error deregistering job: %s", err))
|
|
|
|
|
return 1
|
2015-09-17 00:35:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-24 21:43:20 +00:00
|
|
|
|
// Confirm the stop if the job was a prefix match.
|
|
|
|
|
if jobID != job.ID && !autoYes {
|
|
|
|
|
question := fmt.Sprintf("Are you sure you want to stop job %q? [y/N]", job.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("Cancelling job stop")
|
|
|
|
|
return 0
|
2016-03-28 02:21:23 +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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-17 00:35:58 +00:00
|
|
|
|
// Invoke the stop
|
2015-12-20 11:14:59 +00:00
|
|
|
|
evalID, _, err := client.Jobs().Deregister(job.ID, nil)
|
2015-09-17 00:35:58 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error deregistering job: %s", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-07 23:25:08 +00:00
|
|
|
|
// If we are stopping a periodic job there won't be an evalID.
|
|
|
|
|
if evalID == "" {
|
|
|
|
|
return 0
|
|
|
|
|
}
|
|
|
|
|
|
2015-09-18 04:09:34 +00:00
|
|
|
|
if detach {
|
|
|
|
|
c.Ui.Output(evalID)
|
|
|
|
|
return 0
|
2015-09-17 00:35:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2015-09-18 04:09:34 +00:00
|
|
|
|
// Start monitoring the stop eval
|
2016-01-14 20:57:43 +00:00
|
|
|
|
mon := newMonitor(c.Ui, client, length)
|
2016-01-07 23:25:08 +00:00
|
|
|
|
return mon.monitor(evalID, false)
|
2015-09-17 00:35:58 +00:00
|
|
|
|
}
|