title | description |
---|---|
Deployments |
How systems are deployed at Artsy |
- Safe: Only successfully-tested code should be deployed. Deploying should be graceful. I.e., shouldn't impact availability of the service.
- Repeatable: The result of a deploy should be consistent no matter the state of the developer's local environment, the particulars of a code change, the state of data, volume of traffic, etc.
- Visible: It should be transparent what deploy is happening when (and by whom...). An audit trail and instrumentation can be helpful in diagnosing production issues and interpreting monitoring results.
- Simple: Other things being equal, we prefer simple deployment operations whose impact can be easily understood by developers who may have to consider how to undo a change or its impact on other systems.
- Independent: Releases should be chunked and sequenced such that any one system can be updated independently of others.
- Reversible*: We should be able to roll back a deploy while keeping the state of the system stable and consistent. (This may also include associated changes such as to dependencies or configuration.)
- Consistent*: Ideally the deployment operation is the same whether a change includes just code, updated configuration, data migrations, or dependencies.
- Fast*: Other things being equal, we prefer deploys to be fast so we can deploy regularly and focus on other things.
*Not all of these goals are routinely achieved.
- A few different approaches, largely historical or based on local teams' preferences.
- In all cases, CI deploys to staging
- In almost all cases, CircleCI is responsible for CI
- Want to deploy? Check the README of any system to be sure.
- Heroku (Radiation, Impulse, Positron, ...)
- AWS OpsWorks (Gemini, Causality)
- Kubernetes (Gravity, Force, Metaphysics, Volt, Diffusion, Doppler, ...)
E.g.:
hokusai pipeline promote
(Metaphysics, Diffusion, Vibrations...)git push heroku master
(...?)
This approach is simple and a straightforward usage of our existing tools, BUT:
- Requires a working environment (at least with
hokusai
/git
and credentials) - May have a different effect depending on the local state (e.g., of git remotes or
hokusai/*.yml
configurations) - Isn't adaptable to multi-step deploys (e.g., compiling assets)
- Doesn't enforce the exact deploy commands (though they may be documented)
- May not match the code and configuration used (by CI) to deploy to staging
E.g.:
Conveniently, jenkins jobs are visible to others. They leverage a precisely configured (and thus repeatable) environment, and even prevent tasks from overlapping. However:
- Spinning up Jenkins slaves simply to execute a deploy command feels like overkill
- Once again, the set-up for such a job may not match that used by CI to deploy to staging
E.g.:
- via
hokusai pipeline promote
(Volt, ...) - via
hokusai production deploy ...
(Kaws, Force) - via
git push heroku master
(Impulse)
This leverages the exact set-up already in use to deploy staging, and is independent of any developer's environment. The pull request also acts as a convenient record of the deploy, including commits and any discussion. It's possible, however, for deploys to end up blocked or delayed by other CI tasks. This approach also requires a little extra CircleCI configuration.
Some projects trigger a deploy by PR-ing master
to a release
branch. This risks including commits that haven't
completed a full CI run over the master
branch. To avoid this, others update a staging
or master-stable
branch after the staging deploy is complete and PR from that branch.
- It's acceptable for projects to start by recommending a basic
hokusai pipeline promote
- When setting up projects, use
hokusai setup
in combination with the templates in artsy/artsy-hokusai-templates. They provide reasonable starting points for.circleci/config.yml
and other configuration files. - Full-fledged projects should drive releases (via CircleCI) with pull requests from a
staging
branch to arelease
branch as described above.- The staging deploy must update a
staging
branch (instead of publishing astaging
tag) as a basis for release PRs. - Release PRs can usually be created from a URL like
<github project URL>/compare/release...staging?expand=1
. - See Reflection's .circleci/config.yml as an example of a full set-up for staging and production deploys.
- Note that this requires configuring a read+write key for CircleCI rather than the default read-only key.
- Since a
hokusai pipeline promote
promotes the image currently in use by staging at the time of the command, it's possible that doesn't match what was merged into therelease
branch by the PR. In the future, we'd like to avoid this race condition with an argument tohokusai pipeline promote
, possibly like--only $CIRCLE_SHA1
.
- The staging deploy must update a
In all cases:
- Announce deploys for visibility in #dev
- Document accurate deploy procedure in README
hokusai
can automatically push tags to the artsy remote via the --git-remote
argument.
E.g.:
hokusai staging deploy $SHA --git-remote upstream
pushesstaging
and, e.g.,staging--2018-10-16--16-59-42
tagshokusai pipeline promote --git-remote upstream
pushesproduction
and, e.g.,production--2018-10-16--16-59-42
tags
This can be helpful for recording each release. However, Github doesn't support creating PRs from tags like
staging
so avoid that when employing the PR-driven release approach.
Sometimes there are urgent fixes to release. We always prefer using the typical pipeline (from pull requests to
master
to staging and then to production), even in cases of reverted PRs or timely fixes. When it's absolutely
critical to avoid the delay of a full CI run or staging deploy, it's possible to simply
hokusai production deploy <tag>
where tag is an image tag (including git SHAs) and can refer to a previous
release such as production--2018-09-25--10-19-2
. Projects that depend on other release operations (e.g., to
compile assets or update configuration) may require additional steps.
You may want to migrate an application from Heroku to Kubernetes.
This involves getting Hokusai and being able to hokusai test
locally, getting Circle-CI to run tests and deploy
to a new staging K8 environment, then having Circle-CI promote from staging to a new production environment, then
switching DNS.
- See doppler#154 for an example of dockerizing a Rails app.
- See doppler#157 for setting up promotion from staging to production.