Build Stages
What Are Build Stages? #
Build stages is a way to group jobs, and run jobs in each stage in parallel, but run one stage after another sequentially.
In the simplest and most common use case, you can now make one job run only if several other, parallel jobs have completed successfully.
Let’s say you want to test a library like a Ruby gem or an npm package against various runtime (Ruby or Node.js) versions in parallel. And you want to release your gem or package only if all tests have passed and completed successfully. Build stages make this possible.
Of course, there are a lot more and a lot more elaborated use cases than this one. You can, for example, also use build stages to warm up dependency caches in a single job on a first stage, then use the cache on several jobs on a second stage. Or, you could generate a Docker image and push it first, then test it on several jobs in parallel. Or, you could run unit tests, deploy to staging, run smoke tests and only then deploy to production.
How Do Build Stages Work? #
The concept of build stages is powerful and flexible, yet simple and approachable:
Stages group jobs that run in parallel and different stages run sequentially.
A stage is a group of jobs that are allowed to run in parallel. However, each one of the stages runs one after another and will only proceed, if all jobs in the previous stage have passed successfully. If one job fails in one stage, all other jobs on the same stage will still complete, but all jobs in subsequent stages will be canceled and the build fails.
You can configure as many jobs per stage as you need and you can have as many stages as your delivery process requires.
In the following example, we are running two jobs on the first stage called test, and then run a single third job on the second stage called deploy:
How to Define Build Stages? #
Here’s how you’d set up the build configuration for this in your .travis.yml
file:
jobs:
include:
- stage: test
script: ./test 1
- # stage name not required, will continue to use `test`
script: ./test 2
- stage: deploy
script: ./deploy
This configuration creates the build from the screencast above. I.e., it creates
a build with three jobs, two of which start in parallel in the first stage
(named test
), while the third job on the second stage (named deploy
) starts
only after the test stage completes successfully.
Build Config Reference #
You can find more information on the build config format for build stages in our Travis CI Build Config Reference.
Naming Your Build Stages #
Stages are identified by their names, which are composed of names and emojis.
The first letter of a stage name is automatically capitalized for
aesthetic reasons, so you don’t have to deal with uppercase strings in your
.travis.yml
file.
Also, you do not have to specify the name on every single job (as shown in the
example above). The default stage is test
. Jobs that do not have a stage
name are assigned to the previous stage name, if one exists or the default
stage name, if there is no previous stage name. This means that if you set the
stage name on the first job of each stage, the build will work as expected.
For example, the following config is equivalent to the one above, but also adds a
second deploy job to the deploy
stage that deploys to a different target. As
you can see, you only need to specify the stage name once:
jobs:
include:
- script: ./test 1 # uses the default stage name "test"
- script: ./test 2
- stage: deploy
script: ./deploy target-1
- script: ./deploy target-2
Naming Your Jobs within Build Stages #
You can also name specific jobs within build stages. We recommend unique job names, but
do not enforce it (though this may change in the future). Jobs defined in the jobs.include
section can be given a name attribute as follows:
jobs:
include:
- stage: "Tests" # naming the Tests stage
name: "Unit Tests" # names the first Tests stage job
script: ./unit-tests
- script: ./integration-tests
name: "Integration Tests" # names the second Tests stage job
- stage: deploy
name: "Deploy to GCP"
script: ./deploy
Build Stages and Build Matrix Expansion #
Matrix expansion means that certain top level configuration keys expand into a matrix of jobs.
For example:
rvm:
- 2.3
- 2.4
jobs:
include:
- stage: deploy
rvm: 2.4
env:
- FOO=foo
script: ./deploy
This will run two jobs on Ruby 2.3 and 2.4 respectively first and assign these to the default stage test. The third job on the deploy stage starts only after the test stage has completed successfully.
Each job included in
jobs.include
inherits the first value of the array that defines a matrix dimension. In the example above, without explicitly settingrvm: 2.4
, theinclude
d job inheritsrvm: 2.3
.
Specifying Stage Order and Conditions #
You can specify the order for stages in the section stages
:
stages:
- compile
- test
- deploy
This is mostly useful in order to “prepend” a stage to the test
stage that
jobs resulting from the matrix expansion will be assigned to.
On the same section you can also specify conditions for stages, like so:
stages:
- compile
- test
- name: deploy
if: branch = master
See Conditional Builds, Stages, and Jobs for more details on specifying conditions.
Build Stages and Deployments #
You can combine build stages with deployments:
jobs:
include:
- script: ./test 1 # uses the default stage name "test"
- script: ./test 2
- stage: deploy
script: skip # usually you do not want to rerun any tests
deploy: &heroku
provider: heroku
# ⋮
Travis CI does not set or overwrite any of your scripts and most languages
have a default test script
defined. So in many use cases you might want to overwrite the script
step by
specifying the keyword skip
or ignore
, in other cases you might want to
overwrite other steps, such as the install
step that runs by default on
several languages.
Data Persistence between Stages and Jobs #
It is important to note that jobs do not share storage, as each job runs in a fresh VM or container. If your jobs need to share files (e.g., using build artifacts from the “Test” stage for deployment in the subsequent “Deploy” stage), you need to use an external storage mechanism such as S3 and a remote
scp
server.
See the S3 example below.
Examples #
Deploying to Heroku #
An example with 5 stages:
- Two jobs running unit tests in parallel on stage 1.
- One job deploying the application to Heroku staging.
- One job testing the staging deployment on Heroku.
- One job deploying the application to Heroku production.
- One job testing the production deployment on Heroku.
You can find more details here.
Deploying to Rubygems #
This example has two build stages:
- Two jobs that run tests against Ruby 2.2 and 2.3 respectively
- One job that publishes the gem to rubygems.org
You can find more details here.
Deploying to NPM #
This example has two build stages:
- Four jobs that run tests against Node versions 4 to 7
- One job that deploys (releases) the package to NPM
You can find more details here.
Deploying to GitHub Releases #
This example has two build stages:
- Four jobs that run tests
- One job that deploys to GitHub Releases
You can find more details here.
Combining build stages with matrix expansion #
This example has two build stages:
- Four test jobs that have been expanded from
rvm
andenv
matrix keys. - One deploy job.
You can find more details here.
Warming up a cache with expensive dependencies #
This uses two build stages in order to warm up a cache with expensive dependencies, and optimize test run times:
- One job that installs dependencies and warms up the cache for the given branch.
- Three jobs that run tests, using the cache.
You can find more details here.
Sharing a Docker image #
This example has 2 build stages:
- One job builds and pushes a Docker image
- Two jobs that pull and test the image
You can find more details here.
Sharing files between jobs via S3 #
This uses two build stages, sharing files from build stage 1 in stage 2:
- Two jobs that set up files on S3.
- One job that uses both files from stage 1.
You can find more details here.
Defining different steps on different stages #
This example has 2 build stages:
- Two jobs that run different suites of tests against Ruby 2.3.1
- One job that runs a custom deploy script that doesn’t require running the default
install
orscript
steps
You can find more details here.
Defining steps using YAML aliases #
This example uses YAML aliases to define steps. It has 3 build stages:
- Two jobs that run tests against Ruby 2.2 and 2.3
- One job that deploys to staging
- Three jobs that run test against staging
You can find more details here.