Managing multiple Micro-services in Development using Docker Compose

Tushar Tuteja
5 min readJul 4, 2020

--

Part 1: The Problem

Any growing tech project, is usually accompanied by an ever growing number of micro services, and with it a growing number of problems.

They each have their own database layers that they connect to (Redis, Mongo, Mysql etc.). With different micro-services connecting to different db versions.

When a new developer joins the team, he/she wastes a lot of time installing all the dependencies, and hates it.

And lastly, we want to keep our dev stack as close to production stack as possible.

Consider the Following Project Architecture

Our Project Architecture

Service A is dependent on

  1. Redis
  2. MySql
  3. Service B
  4. Service C

And So on..(You get my point)

Now Imagine, starting all of the above, with proper language versions and database versions. Starting all the services above, is definitely not joy.

There should be an easier way to start our ecosystem of services, with one command/click, should be easy to setup, should be fast and finally should be fun.

Here comes Docker and Docker Compose to help you with it. To understand more of Docker and Docker Containers go here https://www.docker.com/.

In an Ideal Scenario if I have to test functioning of Service A, I’ll use unit and integrations tests and I’ll mock all the API calls to Service B and Service C.

Part 2 Using Docker Compose

Docker compose exactly lets you do that. It gives you a way to write the above configuration in a yaml file. And using one command everything boots up like magic.

One file to run them all, one file to bind them under one network. My Precious!!!

As soon I hit docker-compose up.

docker-compose up
GIF of Terminal

Everything starts running

  1. Mysql DB for service A
  2. Mysql for Service C
  3. Redis for Service A
  4. Mongo Db for Service B.
  5. Service B
  6. Service C
  7. Service A

High Definition Youtube Link: https://www.youtube.com/watch?v=HJ0D7Ns5tBU

TL;DR go to this repo, and experience it for yourself. (Only Pre Req — Docker should be installed)

Repo: https://github.com/tushartuteja/multiple-microservices-docker-compose

Let’s go to localhost:8080, localhost:8081, and localhost:8082 to see what happens.

Service A is running smoothly
Service B is running smoothly
And So is Service C

Part 2.2 Understanding Service A and it Dependencies.

Service A is dependent on four things.

  1. If you notice lines, 11–13 and 15–17, Service A is calling Service B and Service C.
  2. Notice lines, 19–29, Service A is calling Redis.
  3. Notice Lines 31–48, Service A is calling mysql.

The responses for the above are logged to console. (See Image Below)

We can see that first service A called Redis, it worked and printed “Reply Ok”, Then it called Mysql and printed “The solution is: 2”.

Then finally Service C and Service B were called, and their response was printed.

Part 3: Understanding docker-compose.yaml

One file to run them all, one file to bind them under one network. My Precious!!!

A. Version

This is the version of Docker-compose file format and this version also tells which docker engine is compatible for the file. More on this here.

B. Services

This tells docker compose that which all different services are to be run. We have listed out 7 such services. 3 Web Services and 4 Databases.

C. Build

This tells which dockerfile to use to form our image. We are using this in our three web services. We are using pre defined images for all DBs from Docker Hub.

D. Dockerfile

Let’s look at Dockerfile of Service A.

This tells the docker, to start with an alpine image of nodejs. After that create a directory myapp, copy package.json in that directory, run npm install, expose 3000 port and finally run our dev-start.sh file.

E. ports

Let’s notice for Service A

ports:
- "8080:3000"

Ports let you expose multiple ports from your container to a parent machine. The above line says, expose port 3000 at port 8080 of the parent machine.

Notice how we navigated to localhost:8080 in the above image.

You may expose multiple ports, it takes an array as input.

F. Volumes

volumes:
- ./service_a:/myapp

We are attaching service_a folder of our code as a volume to our container, which would be mounted at /myapp.

This way if we change any code in our service_a directory, that is instantly detected in our container and we don’t need to restart our container. They are always in sync. Neat! (I Know)

G. depends_upon

depends_on:
- service_a_db
- service_a_redis
- service_b
- service_c

This tells docker-compose that before you can start Service A, you need to start service_a_db, service_a_redis, service_b and service_c. As Service A depends on it.

It’s let you do one more thing. Suppose you have to just start Service B.

You can just run the following command.

docker-compose up service_b

This would just run Service B and its dependencies and nothing else.

H. Environment Variables

Let’s notice environment variables part of mysql

environment:
MYSQL_ROOT_PASSWORD: service_a
MYSQL_DATABASE: "db"
MYSQL_USER: "user"
MYSQL_PASSWORD: "pass"

The above would form environment variables for our mysql container. Notice how we used the above credentials to login to mysql in service A.

I. Network

When all of the containers run, they become a part of their own virtual network. This what makes it possible for Service A to connect to mysql with a host name of service_a_db.

The following are all hosts for Service A.

  1. service_a_db
  2. service_b
  3. service_c

But None can be accessed from you machine. Try visiting these hosts from your machine.

Part 4: Conclusion.

As a tech lead, this was my de-facto way of managing projects. Whenever a developer was onboarded, he ran only one command and boom! everything was set up on day one. From day one, a new developer was able to code right away.

I have found this way much more efficient, as it also removes any dependencies in “production environment”, since we used docker to run our app in production. This brought a seamless experience between us and our dev-ops team.

If you are using rails, read this https://medium.com/better-programming/setting-up-rails-with-postgres-using-docker-426c853e8590

More on Docker Here https://www.docker.com/

More on Docker Compose Here https://docs.docker.com/compose/

Note: It takes time the first time you run it, as it builds all those docker images, downloads any dependencies. But after that it runs like a breeze.(Look at the video above)

--

--

Tushar Tuteja
Tushar Tuteja

Responses (1)