2023-04-10 15:36:59 +00:00
|
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
|
2015-09-17 00:35:58 +00:00
|
|
|
|
package command
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"strings"
|
2022-12-16 23:46:58 +00:00
|
|
|
|
"sync"
|
2017-08-22 20:22:29 +00:00
|
|
|
|
|
2020-06-17 19:39:50 +00:00
|
|
|
|
"github.com/hashicorp/nomad/api"
|
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
|
|
|
|
|
2020-11-19 21:38:08 +00:00
|
|
|
|
Stop an existing job. This command is used to signal allocations to shut
|
|
|
|
|
down for the given job ID. Upon successful deregistration, 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.
|
|
|
|
|
|
2023-03-09 20:00:04 +00:00
|
|
|
|
When ACLs are enabled, this command requires a token with the 'submit-job'
|
|
|
|
|
and 'read-job' capabilities for the job's namespace. The 'list-jobs'
|
|
|
|
|
capability is required to run the command with job prefixes instead of exact
|
|
|
|
|
job IDs.
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
General Options:
|
|
|
|
|
|
2020-11-19 16:15:23 +00:00
|
|
|
|
` + generalOptionsUsage(usageOptsDefault) + `
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
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.
|
2016-01-14 20:57:43 +00:00
|
|
|
|
|
2021-11-23 08:23:31 +00:00
|
|
|
|
-eval-priority
|
|
|
|
|
Override the priority of the evaluations produced as a result of this job
|
|
|
|
|
deregistration. By default, this is set to the priority of the job.
|
|
|
|
|
|
2020-06-23 19:56:04 +00:00
|
|
|
|
-global
|
|
|
|
|
Stop a multi-region job in all its regions. By default job stop will stop
|
|
|
|
|
only a single region at a time. Ignored for single-region jobs.
|
|
|
|
|
|
2021-12-13 19:54:53 +00:00
|
|
|
|
-no-shutdown-delay
|
|
|
|
|
Ignore the the group and task shutdown_delay configuration so that there is no
|
|
|
|
|
delay between service deregistration and task shutdown. Note that using
|
|
|
|
|
this flag will result in failed network connections to the allocations
|
|
|
|
|
being stopped.
|
|
|
|
|
|
|
|
|
|
-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.
|
|
|
|
|
|
2016-01-15 22:32:38 +00:00
|
|
|
|
-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{
|
2021-12-13 19:54:53 +00:00
|
|
|
|
"-detach": complete.PredictNothing,
|
|
|
|
|
"-eval-priority": complete.PredictNothing,
|
|
|
|
|
"-purge": complete.PredictNothing,
|
|
|
|
|
"-global": complete.PredictNothing,
|
|
|
|
|
"-no-shutdown-delay": complete.PredictNothing,
|
|
|
|
|
"-yes": complete.PredictNothing,
|
|
|
|
|
"-verbose": complete.PredictNothing,
|
2017-08-23 21:56:21 +00:00
|
|
|
|
})
|
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 {
|
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.Jobs, nil)
|
2017-08-22 20:22:29 +00:00
|
|
|
|
if err != nil {
|
|
|
|
|
return []string{}
|
|
|
|
|
}
|
|
|
|
|
return resp.Matches[contexts.Jobs]
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-18 16:02:11 +00:00
|
|
|
|
func (c *JobStopCommand) Name() string { return "job stop" }
|
|
|
|
|
|
2018-03-21 00:37:28 +00:00
|
|
|
|
func (c *JobStopCommand) Run(args []string) int {
|
2021-12-13 19:54:53 +00:00
|
|
|
|
var detach, purge, verbose, global, autoYes, noShutdownDelay bool
|
2021-11-23 08:23:31 +00:00
|
|
|
|
var evalPriority int
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
2018-04-18 16:02:11 +00:00
|
|
|
|
flags := c.Meta.FlagSet(c.Name(), 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, "")
|
2016-01-15 22:32:38 +00:00
|
|
|
|
flags.BoolVar(&verbose, "verbose", false, "")
|
2020-06-23 19:56:04 +00:00
|
|
|
|
flags.BoolVar(&global, "global", false, "")
|
2021-12-13 19:54:53 +00:00
|
|
|
|
flags.BoolVar(&noShutdownDelay, "no-shutdown-delay", 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, "")
|
2021-11-23 08:23:31 +00:00
|
|
|
|
flags.IntVar(&evalPriority, "eval-priority", 0, "")
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
if err := flags.Parse(args); err != nil {
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check that we got exactly one job
|
|
|
|
|
args = flags.Args()
|
2022-12-16 23:46:58 +00:00
|
|
|
|
if len(args) < 1 {
|
|
|
|
|
c.Ui.Error("This command takes at least one argument: <job>")
|
2018-04-18 16:02:11 +00:00
|
|
|
|
c.Ui.Error(commandErrorText(c))
|
2015-09-17 00:35:58 +00:00
|
|
|
|
return 1
|
|
|
|
|
}
|
2022-12-16 23:46:58 +00:00
|
|
|
|
|
|
|
|
|
var jobIDs []string
|
|
|
|
|
for _, jobID := range flags.Args() {
|
|
|
|
|
jobIDs = append(jobIDs, strings.TrimSpace(jobID))
|
|
|
|
|
}
|
2015-09-17 00:35:58 +00:00
|
|
|
|
|
|
|
|
|
// Get the HTTP client
|
|
|
|
|
client, err := c.Meta.Client()
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error initializing client: %s", err))
|
|
|
|
|
return 1
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-16 23:46:58 +00:00
|
|
|
|
statusCh := make(chan int, len(jobIDs))
|
|
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
for _, jobID := range jobIDs {
|
|
|
|
|
jobID := jobID
|
|
|
|
|
|
|
|
|
|
wg.Add(1)
|
|
|
|
|
go func() {
|
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
|
|
// Truncate the id unless full length is requested
|
|
|
|
|
length := shortId
|
|
|
|
|
if verbose {
|
|
|
|
|
length = fullId
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check if the job exists
|
2023-03-03 19:43:20 +00:00
|
|
|
|
job, err := c.JobByPrefix(client, jobID, nil)
|
2022-12-16 23:46:58 +00:00
|
|
|
|
if err != nil {
|
2023-03-03 19:43:20 +00:00
|
|
|
|
c.Ui.Error(err.Error())
|
2022-12-16 23:46:58 +00:00
|
|
|
|
statusCh <- 1
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
getConfirmation := func(question string) (int, bool) {
|
|
|
|
|
answer, err := c.Ui.Ask(question)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Failed to parse answer: %v", err))
|
|
|
|
|
return 1, false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if answer == "" || strings.ToLower(answer)[0] == 'n' {
|
|
|
|
|
// No case
|
|
|
|
|
c.Ui.Output("Cancelling job stop")
|
|
|
|
|
return 0, false
|
|
|
|
|
} else if strings.ToLower(answer)[0] == 'y' && len(answer) > 1 {
|
|
|
|
|
// Non exact match yes
|
|
|
|
|
c.Ui.Output("For confirmation, an exact ‘y’ is required.")
|
|
|
|
|
return 0, false
|
|
|
|
|
} else if answer != "y" {
|
|
|
|
|
c.Ui.Output("No confirmation detected. For confirmation, an exact 'y' is required.")
|
|
|
|
|
return 1, false
|
|
|
|
|
}
|
|
|
|
|
return 0, true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Confirm the stop if the job was a prefix match
|
|
|
|
|
// Ask for confirmation only when there's just one
|
|
|
|
|
// job that needs to be stopped. Since we're stopping
|
|
|
|
|
// jobs concurrently, we're going to skip confirmation
|
|
|
|
|
// for when multiple jobs need to be stopped.
|
|
|
|
|
if len(jobIDs) == 1 && jobID != *job.ID && !autoYes {
|
|
|
|
|
question := fmt.Sprintf("Are you sure you want to stop job %q? [y/N]", *job.ID)
|
|
|
|
|
code, confirmed := getConfirmation(question)
|
|
|
|
|
if !confirmed {
|
|
|
|
|
statusCh <- code
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Confirm we want to stop only a single region of a multiregion job
|
|
|
|
|
if len(jobIDs) == 1 && job.IsMultiregion() && !global && !autoYes {
|
|
|
|
|
question := fmt.Sprintf(
|
|
|
|
|
"Are you sure you want to stop multi-region job %q in a single region? [y/N]", *job.ID)
|
|
|
|
|
code, confirmed := getConfirmation(question)
|
|
|
|
|
if !confirmed {
|
|
|
|
|
statusCh <- code
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Invoke the stop
|
|
|
|
|
opts := &api.DeregisterOptions{Purge: purge, Global: global, EvalPriority: evalPriority, NoShutdownDelay: noShutdownDelay}
|
2023-03-03 19:43:20 +00:00
|
|
|
|
wq := &api.WriteOptions{Namespace: *job.Namespace}
|
2022-12-16 23:46:58 +00:00
|
|
|
|
evalID, _, err := client.Jobs().DeregisterOpts(*job.ID, opts, wq)
|
|
|
|
|
if err != nil {
|
|
|
|
|
c.Ui.Error(fmt.Sprintf("Error deregistering job with id %s err: %s", jobID, err))
|
|
|
|
|
statusCh <- 1
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If we are stopping a periodic job there won't be an evalID.
|
|
|
|
|
if evalID == "" {
|
|
|
|
|
statusCh <- 0
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Goroutine won't wait on monitor
|
|
|
|
|
if detach {
|
|
|
|
|
c.Ui.Output(evalID)
|
|
|
|
|
statusCh <- 0
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Start monitoring the stop eval
|
|
|
|
|
// and return result on status channel
|
|
|
|
|
mon := newMonitor(c.Ui, client, length)
|
|
|
|
|
statusCh <- mon.monitor(evalID)
|
|
|
|
|
}()
|
2020-06-23 19:56:04 +00:00
|
|
|
|
}
|
2022-12-16 23:46:58 +00:00
|
|
|
|
// users will still see
|
|
|
|
|
// errors if any while we
|
|
|
|
|
// wait for the goroutines
|
|
|
|
|
// to finish processing
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
|
|
// close the channel to ensure
|
|
|
|
|
// the range statement below
|
|
|
|
|
// doesn't go on indefinitely
|
|
|
|
|
close(statusCh)
|
|
|
|
|
|
|
|
|
|
// return a non-zero exit code
|
|
|
|
|
// if even a single job stop fails
|
|
|
|
|
for status := range statusCh {
|
|
|
|
|
if status != 0 {
|
|
|
|
|
return status
|
2020-06-23 19:56:04 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-16 23:46:58 +00:00
|
|
|
|
return 0
|
2015-09-17 00:35:58 +00:00
|
|
|
|
}
|