Made by:
Joshua Holmes
jpholmes.com
github.com/joshua-holmes
[email protected]
This project serves as a template that will allow you to toss your Ruby on Rails (can optionally include React frontend) application into a folder, configure a few settings, then quickly have a Dockerized app with it's own contained environment! From there, the app can be deployed on a VPS (Virtual Private Server) with almost no environment configuration. Using Docker also makes it easier to manage multiple apps running on the same VPS, which makes it very cost-effective when you have more than a few apps deployed. This template aims to make Docker as easy to setup as it is to use!
Note: If you are using React Router Dom with your app, please visit that section below, as there are a few extra steps for deployment. The steps are not Docker specific. It would be the same steps, regardless of the deployment method.
- Copy and paste your React on Rails application (or make a new one) inside
ror/
so thatror/
is the root directory of your RoR app. - Run
bin/dockerize -p <PORT_NUMBER>
in your project's local root directory where<PORT_NUMBER>
is a port number you want your Docker Container to run on. Typey
or press enter when prompted to overwrite. This step will preserve database names and will not overwrite them, but will overwrite other settings. If you have custom settings in ror/config/database.yml, other than database names, pressn
instead and follow the instructions in yellow. Here is an example:
# (example) in your local repo, run:
bin/dockerize -p 3001
- Upload your project to GitHub and then pull it down onto your server. Now in your project's server repo, run
bin/server-setup -p '<DB_PASSWORD>' -m <MASTERKEY>
. The masterkey is found in your local project repo:ror/config/master.key
. The password is your database password and you are creating a new one here. Notice the password is wrapped in single quotes. That is to prevent bash from trying to interpret special characters, such as '$
'. Optionally, you can also add a username for your database with the -u flag. Here is an example of the entire command:
# (example) in your server repo, run:
bin/server-setup -u username -p 'YourNewSecureP@$$w0rD' -m fakeMasterKey1234abcd5678efgh
- Copy and paste your React on Rails application (or make a new one) inside
ror/
so thatror/
is the root directory of your RoR app. - In
.docker/
copy theexample.env
file into the same directory and rename it.env
. You do not need to change any variables here in your local repository, but you will need to configure this file on your server repo. - Copy and paste the contents of
.docker/example.database.yml
intoror/config/database.yml
and replace the contents inside, being careful of any database configurations you have already set for your project (if any). Preserve the database names from your original file if one of the databases has data that needs to be preserved on your local machine for development. - Open
docker-compose.yml
file in this project's root directory and change the first port number only, to specify which port you want your app to run on. For example change it to"4000:3000"
if you want to run the app on port 4000. Do not change the second number. - Skip this step if you do not have a React frontend! If you want to test your dockerization locally, open your terminal in the root directory of your project and enter
bin/build-frontend
. Then runsudo docker compose up --build
. Keep in mind that, when it is done loading, the port displayed in your terminal is the internal port number, not the external one. That means that if you set your port to 4000 in the previous step, the terminal will still display port 3000. In this case, you would ignore the terminal and visit port 4000 by going to http://0.0.0.0:4000 and your app should be up. - Your app is now Dockerized!!
- You can now upload your app to GitHub and then pull it down onto your server. Once on your server, go to
.docker/
directory in your server project repo and repeat step 2 above. This time, choose a secure password and set the correct variable to your chosen password. Be sure the password is wrapped in single quotes, as shown in the example file. - Now copy the master key from your local repo in
ror/config/master.key
and paste that key either intoror/config/master.key
on your server repo, or uncomment theRAILS_MASTER_KEY
variable in the.docker/.env
and paste the key in that line in on your server repo.
You can run your app now!! Make sure Docker and Docker Compose are installed on your server, then run sudo docker compose up --build -d
from your project's root directory to build the image on your server and run your app. To see your app, visit http://<YOUR_SERVERS_PUBLIC_IP>:<PORT>
. To take down your app, run sudo docker compose down
in the root of your server' project repo. If your app is running and you've made changes to it and want to update the app on the server, I've included a useful script that will automatically run git pull
, bring your app offline, rebuild the Docker image, and bring it back online, all in one command! Simply run bin/update-app
from the root directory of your project.
When you update your app, make sure you run bin/build-frontend
on your local repo before pushing it up to GitHub and updating your server. RoR's static frontend files go with the rest of your project via git, and the reason for that is because it is not likely that your server has the npm
package manager, which is required to build the frontend.
Warning: When you run your app, the command line will tell you it is running on port 3000, even if this is not true. Just don't fall for it's tricks and visit the port you specified earlier. It is referring to the internal Docker port, not the external one which is the one you use when you visit your app.
Another warning: You may need to enable the port on your server firewall before being able to view the app by directly typing in the IP address and port in your browser. If you use Nginx or Apache, changing firewall settings will not be necessary because all traffic typically comes in through port 80 or 443, then is redirected to your app's port internally.
If you are new to VPS server deployment and are wondering how you get your domain name connected, and more, here are some next steps to look into:
- Buy a domain from Name Cheap or Google Domains. Usually this is around $10-$12 per year.
- Go into the DNS settings and create an A record that points to your server's public IP address (the IP should be provided by your server platform provider. Usually it's in your web portal after you login.).
- Install a routing software like Nginx to forward requests to the appropriate port that your app is running on.
- Install Let's Encrypt on your server to install an SSL Certificate so your server can accept HTTPS and so your browser sees your site as safe.
This is an explanation of the steps taken above and is more for your education than it is necessary to read.
- The
ror/
directory stands for Ruby on Rails or React on Rails. Your React app should then be inside theclient/
folder inside of your ruby app directory. - The .docker/.env file contains environment variables that get set in your Docker Container. These variables are loaded into both the database's container and the app's container. By sharing the same environment variables, they can share useful information, like the database's password, active user and the Rails environment. The master key is also stored here. This master key is used for encrypting data and should not be public, which is why git ignores it and also why you must manually load the master key onto the server's repo, even though your local repo already has it. Rails is able to view the master key when it is stored in either the
.docker/.env
file as an environment variable, or in it's ownmaster.key
file in theror/config/
directory. Either option works for Rails, so how you store the key is entirely up to you. If you need to generate a new one, deleteror/config/master.key
andror/config/credentials.yml.enc
, then runrails credentials:edit
command in your Rails directory. That will generate a newmaster.key
file, but will also recreate the credentials file. Since the credentials file is uploaded via git (which is a good thing), be sure that you update all other repos after issuing that command. Themaster.key
file is ignored by git (which is also a good thing), so don't expose that file publicly. - The
ror/config/database.yml
file is a configuration file for you database. In order for it to work with Docker, there are a few modifications that need to be made: A host needs to be specified and a username and password needs to be required and passed in, for security. Anything enclosed in<%= %>
is actually ruby code. The username and password useENV["VAR"]
instead ofENV.fetch("VAR")
to fetch environment variables because the first option returnsnil
if the variable is not found, while the second one throws an error. By using the first option for username and password, it allows the username and password to be optional, since those environment variables will be established in the Docker Container (via the.docker/.env
file), and are not likely established on your computer. Essentially, a password is not required when you are running it locally, but is required when it is running on your server through Docker. Similar story with the host key. If thePOSTGRES_PASSWORD
environment variable is not found, the program assumes that you are on your local machine and does not need to specify a host. But when the database and app are running through separate containers, a host name is needed. - The
docker-compose.yml
file is wheredocker compose
gets it's instructions to build and run your app. Probably the most important part of this file is the port number. To adjust the port number that the app is running on, you must change the first of the two numbers. So entering"4000:3000"
would run the app on port 4000. The reason the second number can't be changed is because that number is specifying the internal port number. This is the port number that Rails is technically running on inside the Docker Container, but then Docker forwards this internal port to an external port, which can be accessed outside the container. This external port number is the first number (port 4000 in my example case above). Inside this file, you can also see things like theenv_file
key, which is where the.env
file is referenced, which contains the environment variables for the container. Additionally, thedockerfile
key-value pair points to the Dockerfile, which contains instructions on how to build the app. Feel free to open that up and take a look if you are curious. - There are more custom scripts located in the
bin/
folder. Feel free to open them up with a text editor to see what's inside! Keep in mind, these are bash scripts, not ruby scripts. - Once the app is dockerized, it can be taken to the server to be deployed.
This forwards all unknown GET
requests to fallback#index
, which returns your React app. Without these steps, refreshing the page while on any page except the home page will result in a 404 error. This error is not Docker-specific, but instead is React on Rails-specific when using React Router Dom.
- Create a controller called
fallback_controller.rb
inror/app/controllers/
and paste this code in:
class FallbackController < ActionController::Base
def index
render file: 'public/index.html'
end
end
- Open the routes file, found in
ror/config/routes.rb
, and add this line -get "*path", to: "fallback#index", constraints: ->(req) { !req.xhr? && req.format.html? }
- to the bottom of the main code block so it looks something like this:
Rails.application.routes.draw do
...
get "*path", to: "fallback#index", constraints: ->(req) { !req.xhr? && req.format.html? }
end
- Docker commands should always be run with
sudo
because the Docker daemon is being run with admin priviledges. There is a way to get around it, but to me it just seems easier to usesudo
. Feel free to Google around if that's the direction you want to go. - Some installations of Docker Compose come with an alias, allowing you to use the
docker-compose
command instead ofdocker compose
. Others do not, so I usedocker compose
in this instructional set, as well as in my custom bash scripts. - A Docker Image is like a screenshot of your app with it's environment included. The image even includes it's own Linux distro and file system. However, since it is a screenshot, the original cannot be modified. If you make a change, you must take a new screenshot.
- A Docker Container runs a Docker Image. It allows code to be stored in memory so your app can be dynamic. It is similar to a virtual machine, but instead of the environment being stored in the VM, it is stored in the image. Containers are much more performant than VMs.
- A Docker Volume is a way of syncing data between a specified folder on your container with a folder on your local machine. The reason these are necessary is because, remember, data cannot be persisted to Docker Images, since images cannot be modified. Since this is the case, the Docker Container stores the data in memory and syncs it with a folder on the local machine, so when the docker is shut off, the data can be accessed next time a connected container is run. The best use for these volumes is to persist data in a database. In this project, the database in the Docker Container is syncing with our project folder
.docker/volumes/database/
. If this data is ever needed to be accessed in the future, that is where you can find it. - A Docker Network ties two or more Docker Containers together and allows them to interact with each other. In the example app included with this project, the PostgreSQL database is running in a container, and the RoR app is running in a seperate container. They are able to communicate with each other through the Docker Network.
- Docker Compose is a solution for allowing you to configure Docker settings in a YAML file, instead of typing those settings in the command every time you run a container. For example, after specifying the port, environment variables, network, and more, the command to run my container might look like this:
sudo docker -p 3000:3000 -e POSTGRES_USER=rails -e POSTGRES_PASSWORD=1234 -e RAILS_ENVIRONMENT=production --network host ...
etc, etc. By placing my settings in a YAML file, Docker Compose can read the file and apply those settings automatically.
Note: When using docker compose
commands, be sure you are in the root directory of the project you want to manipulate. Docker Compose uses the docker-compose.yml
file in the root directory to know which containers to apply your commands to.
sudo docker compose build
will build a docker image with your app in it and save it to a mysterious directory on your computer. Images do not ever change, so anytime you update your app, you must build a new image to replace the old. This command can be used for that. Everytime you build an image, it replaces the old automatically.sudo docker compose up
will take the image built from the directory you are in and run it on the port specified in thedocker-compose.yml
file! If you add the--build
flag, it will build a new image first, then run that image. It essentailly functions likesudo docker compose build
immediately followed bysudo docker compose up
. If you add the-d
flag, it runs your app in the background, so you don't need to stare at a terminal the whole time your app is up. My favorite command to use when updating my app on my server issudo docker compose up --build -d
because it builds my app, then runs it in the background.sudo docker compose down
will take your app offline.sudo docker ps
will show you all containers that are online and listening on one of your ports.sudo docker ps -a
will show you all containers that are online or offline.sudo docker images
will show all images saved on your system.sudo docker stop <CONTAINER_ID>
takes a container offline, using the container's ID.sudo docker rm <CONTAINER_ID>
deletes a container from your local storage. If you login to root usingsu
and then typing your password, you can usedocker rm $(docker ps -aq)
to delete all containers on your system. Noticesudo
is not used. That is because as root, you don't needsudo
to elevate your privileges, since root already has highest privileges.sudo docker rmi <IMAGE_ID>
deletes an image from your local storage. Before deleting images, make sure a container is not reliant on that image. If you login to root usingsu
and then typing your password, you can usedocker rmi $(docker images -aq)
to delete all images on your system. Noticesudo
is not used here either.
- If you change the database username or password in the
.docker/.env
, the containerized app will fail to connect to the database in.docker/volumes/database
. If this occurs during development and the database contents are disposable, simply delete the database folder usingsudo rm -fr .docker/volumes/database
from the project's root directory. This will delete the databases and the data stored in them, allowing Docker to rebuild the databases with your new username and password. - If
ror/config/master.key
cannot be found, runrails credentials:edit
insideror/
to generate a new one. - If you need a new master key, delete
ror/config/master.key
andror/config/credentials.yml.enc
, then runrails credentials:edit
insideror/
to generate a new ones.