Automate deployments with Capistrano
Why automate?
Let’s suppose you have a static HTML website built with Jekyll and you deploy this on an Ubuntu 20.04 server. If you manually deploy the website, the flow for every release will probably look something like this:
- Push changes to remote Git repo (e.g.,
git push origin master
) - SSH into server (e.g.,
ssh sajad@123.44.55.66
) - cd into project folder and run
git pull
to pull latest updates - Install dependencies (e.g.,
bundle install
) - Run a build task to build static HTML pages, CSS and JS files ( e.g.,
bundle exec jekyll build
).
That is tedious. Let’s see how you can automate those steps so that the flow becomes:
- Push changes to remote Git repo (e.g.,
git push origin master
) - Run deploy command (e.g.,
bundle exec cap deploy production
)
That’s what Capistrano tries to do. It lets you encapsulate a bunch of steps (SSH into server, git pull, install dependencies, etc.) into a single command. Let’s take a look.
Install capistrano gem
Add to Gemfile.
Execute:
Capify project
This should create the following files:
Let’s assume you only have a production server so go ahead and delete the config/deploy/staging.rb
file.
Configure config/deploy.rb
The config/deploy.rb
file is used to set global configuration that is shared between multiple environments (e.g., staging, ci, production, etc). Edit config/deploy.rb
so it looks like this:
We’ll use the custom app:ls
task to test that Capistrano can successfully SSH into our servers.
Configure config/deploy/production.rb
In Capistrano, a server represents a single server (e.g., EC2 instance or Digital Ocean droplet). Each server can have one or more roles such as web
, app
, or db
. For the example Jekyll website, you can use a single app
role.
Edit config/deploy/production.rb
:
Before you proceed further, you’ll want to ensure you’ve configured SSH access to your server so that running ssh <ssh-user>@<ipv4-address>
logs you into the server.
Test Capistrano configuration
Run the following command:
You should see the list of files and directories in the home directory of the default deploy user in the production
server. If you get any errors, it’s probably because your SSH access isn’t configured correctly..
Test deploy
Run the following command to run a dry run (simulation):
If this is successful, go ahead and deploy for real:
(Optional) Setup rbenv
If you’re not using rbenv, skip to the next section.
Add the following to your Gemfile
:
Execute:
Add the following to your Capfile
:
Edit config/deploy.rb
:
Setup bundler
Add the following to your Gemfile
:
Execute:
Add the following to your Capfile
:
Edit config/deploy.rb
so that the .bundle
directory is configured as a persistent directory:
Check that capistrano/bundler
has been configured properly:
You should see a bunch of tasks listed:
Execute a dry run to ensure your deployment is still working:
You should see bundler tasks (e.g., bundler:install
) in the deploy log.
Add custom task
Let’s add a custom app:build
task that runs bundle exec jekyll build
. Edit config/deploy.rb
:
Notice the new app:build
task that will run bundle exec jekyll build
. We’ve also configured this task to run after bundler:install
.
Configure Jekyll
Next, edit config/deploy/production.rb
to set the JEKYLL_ENV
environment variable:
You’ll also want to setup the .jekyll-cache
directory as a shared directory. A shared directory will be reused between different releases so that each release symlinks to the same shared directory.
Edit config/deploy.rb
and update the append :linked_dirs
line:
Run a dry run to make sure the app:build
task is configured to execute after bundle:install
:
Deploy
Finally, you’re ready to deploy:
(Optional) Quick deployment
Create a bin/deploy
bash script:
Make it executable:
Add the following:
Now, you can deploy and get notified of deployment all with a single command:
Sources
Thanks for your comment . Once it's approved, it will appear here.
Leave a comment