In 2013, software development was completely changed by a program called Docker. It used containerization to let developers pack their apps with all the needed dependencies into one easy-to-move unit that can work in any environment.
When it comes to deployment there is a lot of things you need to consider.
- Price of the server usage
- Deployment strategy
- Rollback plan
- Continuous deployment
And it can be done in different ways.
Platforms such as Heroku come with benefits including effortless scaling and deployment; however, there are disadvantages like vendor lock-in and cost issues as utilisation increases.
Using Capistrano with AWS provides flexibility and control over deployment processes, leveraging Capistrano’s automation capabilities with AWS’s scalable infrastructure. However, setup complexity and maintenance overhead can be challenges compared to platform-as-a-service solutions like Heroku.
The simplest and most cost-effective way that I have tried is using Docker + Linode
Combining Docker with Linode offers a versatile and customisable solution for deploying applications. Docker enables containerisation for consistent deployment across environments, while Linode provides robust virtual servers with flexibility in configuration and scaling options. However, managing infrastructure and Docker containers on Linode requires more technical expertise compared to managed platforms like Heroku.
Linode is a cloud-computing platform just like AWS but cheaper.
Why Docker?
Using docker on production deployments is very effective in small to medium-scale applications.
Assuming we are running a Rails application and we are using Postgresql as its database and Sidekiq for its background Jobs.
You can spin all of this in a single docker-compose.yml and a Dockerfile.
Dockerfile
# Use the official Ruby image as the base image
FROM ruby:latest
# Set environment variables for Rails
ENV RAILS_ENV=production \
RAILS_LOG_TO_STDOUT=true \
RAILS_SERVE_STATIC_FILES=true
# Set up the working directory in the container
WORKDIR /app
# Install dependencies
RUN apt-get update -qq && \
apt-get install -y nodejs npm && \
npm install -g yarn && \
apt-get clean && \
rm -rf /var/lib/apt/lists/* /tmp/* /var/tmp/*
# Copy Gemfile and Gemfile.lock into the container
COPY Gemfile Gemfile.lock ./
# Install gems
RUN gem install bundler && \
bundle install --jobs 20 --retry 5
# Copy the rest of the application code into the container
COPY . .
# Precompile assets
RUN bundle exec rails assets:precompile
# Expose port 3000 to the Docker host
EXPOSE 3000
# Start the Rails server with Puma
CMD ["bundle", "exec", "puma", "-C", "config/puma.rb"]
Docker-compose.yml
version: '3'
services:
# PostgreSQL service
db:
image: postgres:latest
volumes:
- db-data:/var/lib/postgresql/data
environment:
POSTGRES_USER: myuser
POSTGRES_PASSWORD: mypassword
POSTGRES_DB: myapp_production
ports:
- "5432:5432"
# Redis service
redis:
image: redis:latest
ports:
- "6379:6379"
# Rails app service
web:
build:
context: .
dockerfile: Dockerfile
volumes:
- .:/app
ports:
- "80:3000"
depends_on:
- db
- redis
environment:
DATABASE_URL: "postgresql://myuser:mypassword@db/myapp_production"
REDIS_URL: "redis://redis:6379/0"
volumes:
db-data:
By running
docker compose --env-file /path/to/your/.env --build -d
Your application will be live on world-wide-web in no time.
In case you would like it done automatically, a GitHub action workflow can be triggered any time you create and publish a release tag. This workflow SSHes into your server, pulls the latest release tag, and then restarts your Docker services accordingly.
name: Deploy Latest Release
on:
release:
types: [published]
jobs:
deploy:
name: Deploy to Server
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v2
- name: Setup SSH
uses: webfactory/ssh-agent@v0.5.3
with:
ssh-private-key: $
- name: SSH into server and deploy
run: |
ssh -o StrictHostKeyChecking=no -p $ $@$ \
'cd /path/to/your/repository && \
git fetch --all && \
latest_tag=$(git describe --tags `git rev-list --tags --max-count=1`) && \
git checkout $latest_tag && \
git pull origin $latest_tag && \
docker-compose down && \
docker-compose --env-file /path/to/your/.env --build -d'
Rolling is easy as well, you just revert the code you previously deployed and create another release tag for it.
Conclusion
However, deploying with docker seems scary and requires more technical skills regarding docker and GitHub actions, it is still a possible cost-effective and efficient way of handling your continuous deployment.
Once you have it all set up, all you need is to create a release tag and that’s it.
It’s easy to monitor your container logs as well using docker container logs
What about running my application in SSL?
Stay tuned to my next blog about Dokerizing Apps with SSL.