Job
A job creates one or more pods and ensures that a specified number of them successfully terminate. As pods successfully complete, the job tracks the successful completions. When a specified number of successful completions is reached, the job itself is complete. Deleting a Job will cleanup the pods it created.
A simple case is to create one Job object in order to reliably run one Pod to completion. The Job object will start a new Pod if the first pod fails or is deleted (for example due to a node hardware failure or a node reboot).
A Job can also be used to run multiple pods in parallel.
Running an example Job
Here is an example Job config. It computes π to 2000 places and prints it out. It takes around 10s to complete.
Check on the status of the job using this command:
To view completed pods of a job, use pi get pods
.
To list all the pods that belong to a job in a machine readable form, you can use a command like this:
Here, the selector is the same as the selector for the job. The --output=jsonpath
option specifies an expression that just gets the name from each pod in the returned list.
View the standard output of one of the pods:
Parallel Jobs
There are three main types of jobs:
Non-parallel Jobs
normally only one pod is started, unless the pod fails.
job is complete as soon as Pod terminates successfully.
Parallel Jobs with a fixed completion count:
specify a non-zero positive value for
completions
.the job is complete when there is one successful pod for each value in the range 1 to
completions
.not implemented yet: Each pod passed a different index in the range 1 to
completions
.
Parallel Jobs with a work queue:
do not specify
completions
, default toparallelism
.the pods must coordinate with themselves or an external service to determine what each should work on.
each pod is independently capable of determining whether or not all its peers are done, thus the entire Job is done.
when any pod terminates with success, no new pods are created.
once at least one pod has terminated with success and all pods are terminated, then the job is completed with success.
once any pod has exited with success, no other pod should still be doing any work or writing any output. They should all be
in the process of exiting.
For a Non-parallel job, you can leave both completions
and parallelism
unset. When both are unset, both are defaulted to 1.
For a Fixed Completion Count job, you should set completions
to the number of completions needed. You can set parallelism
, or leave it unset and it will default to 1.
For a Work Queue Job, you must leave completions
unset, and set parallelism
to a non-negative integer.
For more information about how to make use of the different types of job, see the job patterns section.
Controlling Parallelism
The requested parallelism (parallelism
) can be set to any non-negative value. If it is unspecified, it defaults to 1. If it is specified as 0, then the Job is effectively paused until it is increased.
Actual parallelism (number of pods running at any instant) may be more or less than requested parallelism, for a variety of reasons:
For Fixed Completion Count jobs, the actual number of pods running in parallel will not exceed the number of remaining completions. Higher values of
parallelism
are effectively ignored.For work queue jobs, no new pods are started after any pod has succeeded -- remaining pods are allowed to complete, however.
If the controller has not had time to react.
If the controller failed to create pods for any reason (lack of ResourceQuota, lack of permission, etc.), then there may be fewer pods than requested.
The controller may throttle new pod creation due to excessive previous pod failures in the same Job.
When a pod is gracefully shutdown, it takes time to stop.
Handling Pod and Container Failures
A Container in a Pod may fail for a number of reasons, such as because the process in it exited with a non-zero exit code, or the Container was killed for exceeding a memory limit, etc. If this happens, and the restart = "OnFailure"
, then the Container is re-run. Therefore, your program needs to handle the case when it is restarted locally, or else specify restart = "Never"
.
An entire Pod can also fail, for a number of reasons, such as when the pod is kicked off, or if a container of the Pod fails and the restart = "Never"
. When a Pod fails, then the Job controller starts a new Pod. Therefore, your program needs to handle the case when it is restarted in a new pod. In particular, it needs to handle temporary files, locks, incomplete output and the like caused by previous runs.
Note that even if you specify parallelism = 1
and completions = 1
and restart = "Never"
, the same program may sometimes be started twice.
If you do specify parallelism
and completions
both greater than 1, then there may be multiple pods running at once. Therefore, your pods must also be tolerant of concurrency.
Pod Backoff failure policy
There are situations where you want to fail a Job after some amount of retries due to a logical error in configuration etc. To do so, set backoffLimit
to specify the number of retries before considering a Job as failed. The back-off limit is set by default to 6. Failed Pods associated with the Job are recreated by the Job controller with an exponential back-off delay (10s, 20s, 40s ...) capped at six minutes. The back-off count is reset if no new failed Pods appear before the Job's next status check.
Note: Due to a known issue #54870, when the restart
field is set to "OnFailure
", the back-off limit may be ineffective. As a short-term workaround, set the restart policy for the embedded template to "Never
".
Job Termination and Cleanup
When a Job completes, no more Pods are created, but the Pods are not deleted either. Keeping them around allows you to still view the logs of completed pods to check for errors, warnings, or other diagnostic output. The job object also remains after it is completed so that you can view its status. It is up to the user to delete old jobs after noting their status. Delete the job with pi
(e.g. pi delete jobs/pi
). When you delete the job using pi
, all the pods it created are deleted too.
By default, a Job will run uninterrupted unless a Pod fails, at which point the Job defers to the backoffLimit
described above. Another way to terminate a Job is by setting an active deadline. Do this by setting the activeDeadlineSeconds
field of the Job to a number of seconds.
The activeDeadlineSeconds
applies to the duration of the job, no matter how many Pods are created. Once a Job reaches activeDeadlineSeconds
, the Job and all of its Pods are terminated. The result is that the job has a status with reason: DeadlineExceeded
.
Note that a Job's activeDeadlineSeconds
takes precedence over its backoffLimit
. Therefore, a Job that is retrying one or more failed Pods will not deploy additional Pods once it reaches the time limit specified by activeDeadlineSeconds
, even if the backoffLimit
is not yet reached.
Job Patterns
The Job object can be used to support reliable parallel execution of Pods. The Job object is not designed to support closely-communicating parallel processes, as commonly found in scientific computing. It does support parallel processing of a set of independent but related work items. These might be emails to be sent, frames to be rendered, files to be transcoded, ranges of keys in a NoSQL database to scan, and so on.
In a complex system, there may be multiple different sets of work items. Here we are just considering one set of work items that the user wants to manage together — a batch job.
There are several different patterns for parallel computation, each with strengths and weaknesses. The tradeoffs are:
One Job object for each work item, vs. a single Job object for all work items. The latter is better for large numbers of work items. The former creates some overhead for the user and for the system to manage large numbers of Job objects.
Number of pods created equals number of work items, vs. each pod can process multiple work items. The former typically requires less modification to existing code and containers. The latter is better for large numbers of work items, for similar reasons to the previous bullet.
Several approaches use a work queue. This requires running a queue service, and modifications to the existing program or container to make it use the work queue. Other approaches are easier to adapt to an existing containerised application.
The tradeoffs are summarized here, with columns 2 to 4 corresponding to the above tradeoffs. The pattern names are also links to examples and more detailed description.
Pattern
Single Job object
Fewer pods than work items?
Use app unmodified?
Works in Kube 1.1?
Job Template Expansion
✓
✓
Queue with Pod Per Work Item
✓
sometimes
✓
Queue with Variable Pod Count
✓
✓
✓
Single Job with Static Work Assignment
✓
✓
When you specify completions with completions
, each Pod created by the Job controller has an identical spec
. This means that all pods will have the same command line and the same image, the same volumes, and (almost) the same environment variables. These patterns are different ways to arrange for pods to work on different things.
This table shows the required settings for parallelism
and completions
for each of the patterns.Here, W
is the number of work items.
Pattern
completions
parallelism
1
should be 1
W
any
1
any
Single Job with Static Work Assignment
W
any
Last updated