Here's a run down on what's different. You can also use this as a guide to Dockerize an existing Rails app.
- Core:
- Use PostgreSQL (
-d postgresql)
as the primary SQL database - Use Redis as the cache back-end
- Use Sidekiq as a background worker through Active Job
- Use a standalone Action Cable process
- Use PostgreSQL (
- App Features:
- Add
pages
controller with a home page - Add
up
controller with 2 health check related actions
- Add
- Config:
- Log to STDOUT so that Docker can consume and deal with log output
- Credentials are removed (secrets are loaded in with an
.env
file) - Extract a bunch of configuration settings into environment variables
- Rewrite
config/database.yml
to use environment variables .yarnc
sets a customnode_modules/
directoryconfig/initializers/rack_mini_profiler.rb
to enable profiling Hotwire Turbo Driveconfig/initializers/assets.rb
references a customnode_modules/
directoryconfig/routes.rb
has Sidekiq's dashboard ready to be used but commented out for safetyProcifile.dev
has been removed since Docker Compose handles this for us
- Assets:
- Use esbuild (
-j esbuild
) and TailwindCSS (-c tailwind
) - Add
postcss-import
support fortailwindcss
by using the--postcss
flag - Add ActiveStorage JavaScript package
- Use esbuild (
- Public:
- Custom
502.html
andmaintenance.html
pages - Generate favicons using modern best practices
- Custom
Besides the Rails app itself, a number of new Docker related files were added
to the project which would be any file having *docker*
in its name. Also
GitHub Actions have been set up.
You'll need to have Docker installed. It's available on Windows, macOS and most distros of Linux. If you're new to Docker and want to learn it in detail check out the additional resources links near the bottom of this README.
You'll also need to enable Docker Compose v2 support if you're using Docker Desktop. On native Linux without Docker Desktop you can install it as a plugin to Docker. It's been generally available for a while now and very stable. This project uses a specific Docker Compose profiles feature that only works with Docker Compose v2.
git clone https://github.com/skipmaple/Paul.git
cd Paul
# Optionally checkout a specific tag, such as: git checkout 0.4.0
cp .env.example .env
The first time you run this it's going to take 5-10 minutes depending on your internet connection speed and computer's hardware specs. That's because it's going to download a few Docker images and build the Ruby + Yarn dependencies.
docker compose up --build
Now that everything is built and running we can treat it like any other Rails app.
Did you receive an error about a port being in use? Chances are it's because
something on your machine is already running on port 8000. Check out the docs
in the .env
file for the DOCKER_WEB_PORT
variable to fix this.
Did you receive a permission denied error? Chances are you're running native
Linux and your uid:gid
aren't 1000:1000
(you can verify this by running
id
). Check out the docs in the .env
file to customize the UID
and GID
variables to fix this.
# You can run this from a 2nd terminal.
./run rails db:setup
We'll go over that ./run
script in a bit!
Visit http://localhost:8000 in your favorite browser.
# You can run this from the same terminal as before.
./run test
You can also run ./run test -b
with does the same thing but builds your JS
and CSS bundles. This could come in handy in fresh environments such as CI
where your assets haven't changed and you haven't visited the page in a
browser.
# Stop the containers and remove a few Docker related resources associated to this project.
docker compose down
You can start things up again with docker compose up
and unlike the first
time it should only take seconds.
I recommend checking out most files and searching the code base for TODO:
,
but please review the .env
and run
files before diving into the rest of the
code and customizing it. Also, you should hold off on changing anything until
we cover how to customize this example app's name with an automated script
(coming up next in the docs).
This file is ignored from version control so it will never be commit. There's a number of environment variables defined here that control certain options and behavior of the application. Everything is documented there.
Feel free to add new variables as needed. This is where you should put all of your secrets as well as configuration that might change depending on your environment (specific dev boxes, CI, production, etc.).
You can run ./run
to get a list of commands and each command has
documentation in the run
file itself.
It's a shell script that has a number of functions defined to help you interact
with this project. It's basically a Makefile
except with less
limitations.
For example as a shell script it allows us to pass any arguments to another
program.
This comes in handy to run various Docker commands because sometimes these commands can be a bit long to type. Feel free to add as many convenience functions as you want. This file's purpose is to make your experience better!
If you get tired of typing ./run
you can always create a shell alias with
alias run=./run
in your ~/.bash_aliases
or equivalent file. Then you'll be
able to run run
instead of ./run
.
This won't take as long as before because Docker can re-use most things. We'll also need to setup our database since a new one will be created for us by Docker.
docker compose up --build
# Then in a 2nd terminal once it's up and ready.
./run rails db:setup
If you get an error upping the project related to RuntimeError: invalid bytecode
then you have old tmp/
files sitting around related to the old
project name, you can run ./run clean
to clear all temporary files and fix
the error.
It's always a good idea to make sure things are in a working state before adding custom changes.
# You can run this from the same terminal as before.
./run test
If everything passes now you can optionally git add -A && git commit -m "Initial commit"
and start customizing your app. Alternatively you can wait
until you develop more of your app before committing anything. It's up to you!
Let's say you've customized your app and it's time to make a change to your
Gemfile
or package.json
file.
Without Docker you'd normally run bundle install
or yarn install
. With
Docker it's basically the same thing and since these commands are in our
Dockerfile
we can get away with doing a docker compose build
but don't run
that just yet.
You can run ./run bundle:outdated
or ./run yarn:outdated
to get a list of
outdated dependencies based on what you currently have installed. Once you've
figured out what you want to update, go make those updates in your Gemfile
and / or package.json
file.
Then to update your dependencies you can run ./run bundle:install
or ./run yarn:install
. That'll make sure any lock files get copied from Docker's image
(thanks to volumes) into your code repo and now you can commit those files to
version control like usual.
Alternatively for updating your gems based on specific version ranges defined
in your Gemfile
you can run ./run bundle:update
which will install the
latest versions of your gems and then write out a new lock file.
You can check out the run
file to see what these commands do in more detail.
You'll want to run docker compose build
since it will use any existing lock
files if they exist. You can also check out the complete CI test pipeline in
the run
file under the ci:test
function.
This is usually a non-issue since you'll be pulling down pre-built images from
a Docker registry but if you decide to build your Docker images directly on
your server you could run docker compose build
as part of your deploy
pipeline.
..