open-nomad/command/job_stop.go

180 lines
4.5 KiB
Go
Raw Normal View History

2015-09-17 00:35:58 +00:00
package command
import (
"fmt"
"strings"
2017-08-22 20:22:29 +00:00
"github.com/hashicorp/nomad/api/contexts"
"github.com/posener/complete"
2015-09-17 00:35:58 +00:00
)
2018-03-21 00:37:28 +00:00
type JobStopCommand struct {
2015-09-17 00:35:58 +00:00
Meta
}
2018-03-21 00:37:28 +00:00
func (c *JobStopCommand) Help() string {
2015-09-17 00:35:58 +00:00
helpText := `
2018-03-21 00:37:28 +00:00
Usage: nomad job stop [options] <job>
2018-03-21 00:46:24 +00:00
Alias: nomad stop
2015-09-17 00:35:58 +00:00
Stop an existing job. This command is used to signal allocations
to shut down for the given job ID. Upon successful deregistration,
2015-09-18 04:09:34 +00:00
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
2016-07-20 15:53:59 +00:00
deregister command is submitted, a new evaluation ID is printed to the
screen, which can be used to examine the evaluation using the eval-status
command.
2017-04-15 03:54:30 +00:00
-purge
Purge is used to stop the job and purge it from the system. If not set, the
job will still be queryable and will be purged by the garbage collector.
2016-03-24 21:43:20 +00:00
-yes
Automatic yes to prompts.
-verbose
Display full information.
2015-09-17 00:35:58 +00:00
`
return strings.TrimSpace(helpText)
}
2018-03-21 00:37:28 +00:00
func (c *JobStopCommand) Synopsis() string {
2015-09-17 00:35:58 +00:00
return "Stop a running job"
}
2018-03-21 00:37:28 +00:00
func (c *JobStopCommand) AutocompleteFlags() complete.Flags {
2017-08-23 21:56:21 +00:00
return mergeAutocompleteFlags(c.Meta.AutocompleteFlags(FlagSetClient),
complete.Flags{
"-detach": complete.PredictNothing,
"-purge": complete.PredictNothing,
"-yes": complete.PredictNothing,
"-verbose": complete.PredictNothing,
})
2017-08-22 20:22:29 +00:00
}
2018-03-21 00:37:28 +00:00
func (c *JobStopCommand) AutocompleteArgs() complete.Predictor {
2017-08-22 20:22:29 +00:00
return complete.PredictFunc(func(a complete.Args) []string {
client, err := c.Meta.Client()
if err != nil {
return nil
}
resp, _, err := client.Search().PrefixSearch(a.Last, contexts.Jobs, nil)
2017-08-22 20:22:29 +00:00
if err != nil {
return []string{}
}
return resp.Matches[contexts.Jobs]
})
}
2018-03-21 00:37:28 +00:00
func (c *JobStopCommand) Run(args []string) int {
2017-04-15 03:54:30 +00:00
var detach, purge, verbose, autoYes bool
2015-09-17 00:35:58 +00:00
2018-03-21 00:37:28 +00:00
flags := c.Meta.FlagSet("job stop", FlagSetClient)
2015-09-17 00:35:58 +00:00
flags.Usage = func() { c.Ui.Output(c.Help()) }
2015-09-18 04:09:34 +00:00
flags.BoolVar(&detach, "detach", false, "")
flags.BoolVar(&verbose, "verbose", false, "")
2016-03-24 21:43:20 +00:00
flags.BoolVar(&autoYes, "yes", false, "")
2017-04-15 03:54:30 +00:00
flags.BoolVar(&purge, "purge", false, "")
2015-09-17 00:35:58 +00:00
if err := flags.Parse(args); err != nil {
return 1
}
// Truncate the id unless full length is requested
length := shortId
if verbose {
length = fullId
}
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)
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
}
if len(jobs) > 1 && strings.TrimSpace(jobID) != jobs[0].ID {
c.Ui.Error(fmt.Sprintf("Prefix matched multiple jobs\n\n%s", createStatusListOutput(jobs)))
return 1
2016-03-21 20:00:14 +00:00
}
// 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.
2017-02-06 19:48:28 +00:00
if jobID != *job.ID && !autoYes {
question := fmt.Sprintf("Are you sure you want to stop job %q? [y/N]", *job.ID)
2016-03-24 21:43:20 +00:00
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
} 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
2017-04-15 03:54:30 +00:00
evalID, _, err := client.Jobs().Deregister(*job.ID, purge, nil)
2015-09-17 00:35:58 +00:00
if err != nil {
c.Ui.Error(fmt.Sprintf("Error deregistering job: %s", err))
return 1
}
// 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
mon := newMonitor(c.Ui, client, length)
return mon.monitor(evalID, false)
2015-09-17 00:35:58 +00:00
}