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.

Docker Stack with Ghost and MySQL
Banff, Alberta

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:

  1. Docker Desktop installed, how to here.
  2. Nginx Proxy Manager Installed, how to here.
  3. A domain or subdomain.
  4. STMP settings for your transactional emails: signups, password resets and admin invites.
  5. 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 database
database__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 username
mail__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.

New Proxy Host - Details Window

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.

New Proxy Host - SSL Window

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 on
HTTP/2 Support - Turn that on
HSTS Enabled - Turn that on
HSTS 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


GitHub - ravencore17/Docker-Stack-Ghost-MySQL: A docker compose stack with Ghost CMS and a MySQL Database.
A docker compose stack with Ghost CMS and a MySQL Database. - GitHub - ravencore17/Docker-Stack-Ghost-MySQL: A docker compose stack with Ghost CMS and a MySQL Database.