Docker Stack with Ghost and MySQL
A docker compose stack with the blogging platform Ghost CMS and a MySQL Database. The docker stack will be deployed in production mode.
A docker compose stack with the blogging platform Ghost CMS and a MySQL Database. The docker stack will be deployed in production
mode.
What is Ghost CMS?
Ghost CMS is a blogging platform where most you as a writer can focus more on writing than coding or configuring the settings. I personally use Ghost for my own website/blogging needs. To learn more visit the official Ghost Page.
Requirements
Using this repo will require a few things:
- Docker Desktop installed, how to here.
- Nginx Proxy Manager Installed, how to here.
- A domain or subdomain.
- STMP settings for your transactional emails: signups, password resets and admin invites.
- Optional - Mailgun settings for your bulk emails, Newsletters.
Install
1. Get the Docker Compose
Clone the repo, download the docker-compose.yml or copy the following into a YAML file.
version: "3.3"
networks:
frontend:
external: true
backend:
services:
ghost:
image: ghost:latest
restart: always
container_name: ghost-app
depends_on:
- db
environment:
url: https://domain.com
database__client: mysql
database__connection__host: ghost-db
database__connection__port: 3306
database__connection__user: ghostuser
database__connection__password: DBPASSWORD
database__connection__database: ghost
mail__transport: SMTP
mail__options__host: host.domain.com
mail__options__port: 587
mail__options__auth__user: smtp.username
mail__options__auth__pass: smtp.password
mail__from: Example Corpo <mail@domain.com>
volumes:
- ./content:/var/lib/ghost/content
networks:
- frontend
- backend
db:
image: mysql:latest
restart: always
container_name: ghost-db
environment:
MYSQL_ROOT_PASSWORD: DBPASSWORDROOT
MYSQL_DATABASE: ghost
MYSQL_USER: ghostuser
MYSQL_PASSWORD: DBPASSWORD
volumes:
- ./db:/var/lib/mysql
networks:
- backend
Note: if you are downloading the docker-compose.yml or making your own, its in your own best interest to nest that YAML file in its own folder for simpler docker compose up -d
.
Below we are going further into variables and general configurations.
2. Networks
networks:
frontend:
external: true
backend:
We have two networks defined here: frontend
and backend
. You'll notice that frontend
is external, that's so that it can talk to the NGINX Proxy Manager reverse proxy that was set up earlier.
The network backend
is strictly for the Ghost application to talk to its MySQL database. The external: true
hash states the network already exist and for the docker service to assign the container to it that network.
3. Ghost CMS Application
services:
ghost:
image: ghost:latest
restart: always
container_name: ghost-app
depends_on:
- db
environment:
url: https://domain.com
database__client: mysql
database__connection__host: ghost-db
database__connection__port: 3306
database__connection__user: ghostuser
database__connection__password: DBPASSWORD
database__connection__database: ghost
mail__transport: SMTP
mail__options__host: host.domain.com
mail__options__port: 587
mail__options__auth__user: smtp.username
mail__options__auth__pass: smtp.password
mail__from: Example Corpo <mail@domain.com>
volumes:
- ./content:/var/lib/ghost/content
networks:
- frontend
- backend
Lets break it down even more:
depends_on:
- db
The docker service will start the database container before the main app. This is to ensure that the dependencies needed for the app are avail when the app is spun up.
environment:
url: https://domain.com
You'll want to put your sub domain or domain here, whatever the exact address is of your blog. Make sure you state your connection type: HTTP or HTTPS, just for the record you should always go for HTTPS, as it's more secure.
database__client: mysql
database__connection__host: ghost-db
database__connection__port: 3306
database__connection__user: ghostuser
database__connection__password: DBPASSWORD
database__connection__database: ghost
This section is the client configuration so that ghost-app
can access your database container, ghost-db
securely.database_connection_host: ghost-db
Where our ghost database is hosted, in this case the MySQL container ghost-db
.database__connection__port: 3306
Is the default MySQL connection port.database__connection__user: ghostuser
Defines the user of the databasedatabase__connection__password: DBPASSWORD
The password associated with the user
Make sure you change the database_connection_password
environment variable to something more secure.database__connection__database: ghost
The name of the database inside MySQL.
mail__transport: SMTP
mail__options__host: host.domain.com
mail__options__port: 587
mail__options__auth__user: smtp.username
mail__options__auth__pass: smtp.password
mail__from: Example Corpo <mail@domain.com>
This section is going to be dealing with the transactional emails. Password resets, newsletter signups and collaboration invites are all examples of transactional emails. Ghost does bulk email services through the Mailgun API, which can also act as a transactional service. However this configuration based on my own hosting does not use the Mailgun API. You can add the Mailgun feature later on through the Ghost admin panel.
mail__transport: SMTP
Transactional emails are always SMTP.mail__options__host: host.domain.com
Your email service STMP host domain.mail__options__port: 587
SMTP emails are usually sent on the 587 port.mail__options__auth__user: smtp.username
The username for your SMTP service.mail__options__auth__pass: smtp.password
Password for your SMTP usernamemail__from: Example Corpo <mail@domain.com>
What the person receiving would see it's from.
An example configuration would look like this if you were using iCloud
mail__transport: SMTP
mail__options__host: smtp.mail.me.com
mail__options__port: 587
mail__options__auth__user: noreply.you@icloud.com
mail__options__auth__pass: cwla-ktrm-pzou-dznn #App specific password
mail__from: your.domain.com <noreply.you@icloud.com>
Deployment
This section deals with setting up ghost behind the Nginx Proxy Manager you should've set up as a requirement. Findout how
Nginix Proxy Manager
Once you've logged into your instance of Nginx Proxy Manager, select the Proxy Host
button. Then select the Add Proxy Host
button. The following window will appear.
Domain Names
- The domain or the subdomain you want your Ghost Blog to be directed to.scheme
- Leave it at HTTP, we'll upgrade it to HTTPS in a bit.Forward Hostname/IP
- If you are using the docker compose above then it would be ghost-app
. The Nginx will direct all traffic directed at your Domain Name
to the Ghost docker container named ghost-app
.Forward Port
- 2368 is the default port that ghost uses to talk to the internet. But through the power of reverse proxies we will not need to expose the port but rather nginx will take care of connecting either 80 or 443 to port 2368.
Enable the toggles: Cache Assets
, Block Common Exploits
, and Web-sockets Support
.
Next we are going to click on the SSL
tab.
In this section we are going to upgrade to HTTPS and get ourselves a SSL Certificate.SSL Certificate
- We are going to select "request a new SSL Certificate", this will get a new SSL Certificate through Let's Encrypt.Force SSL
- Turn that onHTTP/2 Support
- Turn that onHSTS Enabled
- Turn that onHSTS SubDomains
- Turn that on
Press the save button and your Ghost Blog site should now be reachable at the domain you used.
Getting to Ghost Admin Panel
To get access to your ghost blog go to your site: https://your.domain.com/ghost.
Follow the prompts and remember to use a secure password.
For further documentation please refer to the official ghost documents here