Skip to content

Containerization practical example. A 3 tier web application with a React front end client, node.js api and a database. Providing a realistic microservice system to containerize.

Notifications You must be signed in to change notification settings

msrxse/docker-showcase

Repository files navigation

Docker showcase web application

Minimal 3 tier web application

  • React frontend: Uses react query to load data from the two apis and display the result
  • Node JS API: Do have / and /ping endpoints. / queries the Database for the current time, and /ping returns pong
  • Postgres Database: An empty PostgreSQL database with no tables or data. Used to show how to set up connectivity. The API applications execute SELECT NOW() as now; to determine the current time to return.

Running the Application

While the whole point of this app is that you probably won't want/need to run the application locally, we can do so as a starting point.

The Makefile contains the commands to start each application.

Postgres

It's way more convenient to run postgres in a container, so we will do that.

make run-postgres will start postgres in a container and publish port 5432 from the container to your localhost.

api-node

To run the node api you will need to run npm install to install the dependencies.

After installing the dependencies, make run-api-node will run the api in development mode with nodemon for restarting the app when you make source code changes.

client-react

Like api-node, you will first need to install the dependencies with npm install

After installing the dependencies, make run-client-react will use vite to run the react app in development mode.

Running Containers (with Docker)

There are two primary ways to run docker containers, with docker run and docker compose up.

Docker run takes a single container image and runs a container based on it, while docker compose takes a specification of 1 or more services and can build container images for them and/or run containers from those images.

Generally docker run is preferable for one off quick use cases while docker compose is preferable if you are developing a containerized application with more than one service.

individual docker run commands

The portion of the Makefile labeled ### DOCKER CLI COMMANDS shows the commands can would use to build and run all of these services. To build the images and then run them you can execute:

make docker-build-all
make docker-run-all

You will notice that each of the run commands has a bunch of options used to ensure the configuration works properly.

  • Uses the default docker bridge network
  • -d = detach mode
  • Uses --link to enable easy host name for network connections
  • Publishing ports (-p option) useful to connect to each service individually from host, but only necessary to connect to the frontend
  • Named containers make it easier to reference (e.g. with link), but does require removing them to avoid naming conflict
  • Restart policy allows docker to restart the container (for example if database weren't up yet causing one of the api servers to crash)
  • -v option on client-react-vite because vite.config needs to be different on vite than on nginx

At this point the add would be running at port 5173 (http://localhost:5173/). And the backend at http://localhost:5173/api/node or just http://127.0.0.1:3000/

docker compose

Using docker compose allows encoding all of the logic from the docker build and docker run commands into a single file. Docker compose also manages naming of the container images and containers, attaching to logs from all the containers at runtime, etc...

The docker-compose.yml file and the portion of the Makefile labeled ### DOCKER COMPOSE COMMANDS shows how you can use docker compose to build and run the services. To build and run them you can execute

make compose-up-build

As you can see, this is much simpler than needing to execute all of the individual build/run commands and provides a clear way to specify the entire application stack in a single file!

Development Workflow

Development Environment

Because we are running our application within containers, we need a way to quickly iterate and make changes to them.

We want our development environment to have the following attributes:

  1. Easy/simple to set up: Using docker compose, we can define the entire environment with a single yaml file. To get started, team members can issue a single command make compose-up-build or make compose-up-build-debug depending if they want to run the debugger or not.

  2. Ability to iterate without rebuilding the container image: In order to avoid having to rebuild the container image with every single change, we can use a bind mount to mount the code from our host into the container filesystem. For example:

      - type: bind
        source: api-node/
        target: /usr/src/app/
  1. Automatic reloading of the application:

    • React Client: We are using Vite for the react client which handles this automatically
    • Node API: We added nodemon as a development dependency and specify the Docker CMD to use it
  2. Use a debugger:

    • React Client: For a react app, you can use the browser developer tools + extensions to debug. I did include react-query-devtools to help debug react query specific things. It is also viewed from within the browser.

    • Node API: To enable debugging for a NodeJS application we can run the app with the --inspect flag. The debug session can then be accessed via a websocket on port 9229. The additional considerations in this case are to specify that the debugger listen for requests from 0.0.0.0 (any) and to publish port 9229 from the container to localhost.


      These modifications to the configuration (overridden commands + port publishing) are specified in docker-compose-debug.yml. By passing both docker-compose-dev.yml AND docker-compose-debug.yml to the docker compose up command (See: make compose-up-debug-build) Docker combines the two files, taking the config from the latter and overlaying it onto the former.

      The ./api-node/README.md show a launch.json configuration you can use to connect to these remote debuggers using VSCode. The key setting is substitutePath such that you can set breakpoints on your local system that get recognized within the container.

  3. Executing tests: We also need the ability to execute our test suites within containers. Again, we can create a custom docker-compose-test.yml overlay which modifies the container commands to execute our tests. To build the api images and execute their tests, you can execute make run-tests which will use the test compose file along with the dev compose file to do so.

Links

About

Containerization practical example. A 3 tier web application with a React front end client, node.js api and a database. Providing a realistic microservice system to containerize.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published