This project provides a reusable Makefile for architectures with multiple software components that use docker extensively and are organized as monorepositories.
Everything else apart from the Makefile is just us showcasing its usage and capabilities.
At Enspirit we have embraced docker, docker-compose and monorepositories. Our team is usually rotating among multiple software projects.
We want to have reproducible builds and tooling allowing us to work the same way on all these different projects. We also want only one command to be needed for building then starting a software after a fresh git clone, even for newcomers.
This Makefile is an open-source consolidation of years of work to achieve that goal. Why don't you see for yourself:
git clone git@github.com:enspirit/makefile-for-monorepos.git
cd makefile-for-monorepos
make up
n.b. You need docker, docker-compose and make (~> 3.81) installed locally.
- it manages builds of all components' docker images
- it allows managing inter-component dependencies
- it only rebuilds images whose source code or dependencies have changed
- it manages lifecycles of the various components (start, stop, restart, ...)
- it pushes and pulls images from repositories
- it provides extension points for standard rules (e.g. unit testing, cleaning, ...)
- it also allows the extension of the Makefile with ad-hoc rules per component
Copy our Makefile from this repo and place it in your own monorepository project. Your source code organization needs to follow the conventions below.
We all love those one-liner installation methods, so here you go:
n.b. make sure you are inside the folder of your monorepo project.
Option 1, with wget:
wget https://raw.githubusercontent.com/enspirit/makefile-for-monorepos/1.0.3/Makefile
Option 2, with curl:
curl https://raw.githubusercontent.com/enspirit/makefile-for-monorepos/1.0.3/Makefile -o Makefile
By following some conventions our Makefile adds some magic. Let's consider the file structure of this very repository:
monorepo
├── base # A component used as a base for others (dependency)
│ └── Dockerfile # ... with its Dockerfile
├── api # An api component
│ ├── Dockerfile # ... with its Dockerfile extending the base image
│ ├── env #
│ │ └── devel.env # Default env vars for the devel environment
│ ├── makefile.mk # Extensions and ad-hoc rules for the component
│ └── ... #
├── frontend # Another component
│ └── Dockerfile #
├── .env # Main environment variables (e.g. COMPOSE_FILE)
├── docker-compose.base.yml #
├── docker-compose.devel.yml # Orchestration with docker-compose files
├── docker-compose.testing.yml #
├── Makefile # Our reusable Makefile
└── config.mk # Specific configuration and global ad-hoc rules
The Makefile provides tooling organized in three layers:
-
Builds: a folder at level one will be considered a component of the architecture as soon as it includes a Dockerfile. It is the case above for base, api and frontend. For all of them you magically get all the component image rules and component test rules.
-
Lifecycle: All services defined in the docker-compose files currently enabled by the COMPOSE_FILE variable automatically get the component lifecycle rules.
-
Extension: It provides 'magic' but it is still based on make so you can configure or override things and even extend them globally or on a component basis using
config.mk
andmakefile.mk
files. This allows you to extend rules without changing the original Makefile so that you can get bugfixes and improvements
The three layers are independent of each other. In particular the Builds layer only requires docker, so the build tooling works even if you use another orchestrator than docker-compose.
The first time you run one of the Makefile's rules, it will create a config.mk where you can configure the name of your project (it defaults to your project's folder name). That name will be used as a prefix for all the subsequent images built.
Let's take this repository as an example. We've used "monorepo" as a project name and we have 4 components: api, base, frontend and tests. The images built will be tagged monorepo/api:latest, monorepo/base:latest, etc...
You can override other things in your config.mk
if our defaults are not to your taste. It's as simple as adding a line specifying which variable you want to override and providing its new value.
For instance to use your own private docker registry:
PROJECT := monorepo
DOCKER_REGISTRY := my.private.registry
Here is the list of variables you can override:
# Specify which docker tag is built (defaults to 'latest')
DOCKER_TAG :=
# Which command is used to build docker images (defaults to 'docker build')
DOCKER_BUILD :=
# Which command is used for docker-compose (defaults to 'docker-compose')
DOCKER_COMPOSE :=
# Docker build extra options for all builds (optional)
DOCKER_BUILD_ARGS :=
# Which command is used to scan docker images (defaults to 'docker scan')
DOCKER_SCAN :=
# Docker scan extra options (optional)
DOCKER_SCAN_ARGS :=
# Should docker scan fail on errors (true/false, defaults to 'true')
# When running the `make images.scan` rule with this setting to true
# the scan will stop at the first image with vulnerabilities
DOCKER_SCAN_FAIL_ON_ERR :=
It is important to note that these variables can also be overriden by exporting environment variables, e.g. DOCKER_TAG=test make images
.
You can add global rules in your config.mk
and ad-hoc component rules in makefile.mk
files.
As soon as you create one of them it will be included automatically. You can see an example of that in tests/makefile.mk where we created a tests.run
rule that runs the unit tests for this project.
N.B. We recommend keeping all custom rules prefixed with the name of the component.
make clean
: removes the sentinel files (see Sentinel files)make images
: builds all the docker images for the repo's componentsmake images.push
: pushes all images to the docker registry (after building them if necessary)make images.pull
: pulls all images from the docker registrymake images.scan
: scan all images for vulnerabilities
make up
: build images for components that have changed then force-starts the docker-compose projectmake down
: stops the docker-compose projectmake start
: starts the docker-compose projectmake restart
: restarts the docker-compose projectmake ps
: alias for docker-compose ps
make tests
: Runs all tests (equivalent totests.unit
thentests.integration
)make tests.unit
: Runs all unit tests on all componentsmake tests.integration
: Runs all integration tests on all components
For every docker component in your repo, you can run:
make {component}.clean
: removes the component's sentinel files (see Sentinel files)make {component}.image
: builds the component's docker imagemake {component}.image.pull
: pulls the component image from the docker registrymake {component}.image.push
: pushes the image to the registry,⚠️ it also rebuilds the component if any files or dependencies have changedmake {component}.image.scan
: scans the component image for vulnerabilities
make {component}.on
: starts the componentmake {component}.off
: stops the componentmake {component}.up
: forces the recreation of the container,⚠️ it also rebuilds the component's image if any files or dependencies have changedmake {component}.down
: stops the componentmake {component}.restart
: restarts the component (equivalent ofoff
thenon
)make {component}.logs
: tails the logs of the componentmake {component}.bash
: gets a bash on the component
make {component}.up
and make {component}.on
behave slightly differently: the former will first rebuild the image when needed (if files or dependencies have changed) while the latter simply starts the component using the last known image.
make {component}.down
and make {component}.off
behave exactly the same way, they stop the component. It is just for consistency: if we can start things with both up
and on
one would expect to have both down
and off
.
make {component}.tests
: Runs all tests for the component (equivalent to{component}.tests.unit
then{component}.tests.integration
)make {component}.tests.unit
: Runs all the component's unit testsmake {component}.tests.integration
: Runs all the component's integration tests
These three rules are placeholders. Their recipes must be implemented in the components' makefile.mk
.
For example consider the api/makefile.mk file from this repository:
api.tests.unit::
@docker run monorepo/api npm run test:unit
api.tests.integration::
@docker run monorepo/api npm run test:integration
Sometimes components depend on each other. There is an example of this in this repository: the api component depends on the base component since its Dockerfile uses the base component as a base image.
This can be expressed in the component's makefile.mk
like we do in api/makefile.mk by defining a variable {component}_DEPS
that lists such dependencies.
In our example it means that make will know that the base component has to be built before api. Not only that, but also any rebuild of base should retrigger a build of api.
The make {component}.bash
rule assumes that your component's image has bash
installed. Not all of them do (e.g: alpine, so the rule may not always work out of the box.
You can override the shell that is run by creating a {component}_SHELL
override in the component's makefile.mk
. See our example.
First a little quote from make's user manual:
A rule appears in the makefile and says when and how to remake certain files, called the rule's targets (most often only one per rule). It lists the other files that are the prerequisites of the target, and the recipe to use to create or update the target.
Now let's look at a makefile such as:
component.image: component/Dockerfile
docker build -t component component/
component.image
is the rule, component/Dockerfile
is a prerequisite and the recipe is the second line with docker build ...
. But what about the target?
Well, since docker build
does not produce any file we don't really have a real target here, it is what is called a phony target.
And the 'problem' with a phony target is that its recipe will be executed every time the rule is called. That is: the example makefile would re-run docker build every single time.
What's the problem? docker build has a cache mechanism.
Yes and with buildkit it's getting even better.
But on monorepos with a lot of components we can save precious seconds by not re-running 'docker build' for components that haven't changed. That also allows us to rely on make up
to decide which components need rebuildind before starting the docker-compose project.
Ok, now what about those sentinels?
Sentinels are files that we can use to tell make when a rebuild is needed. Let's look at an example:
component.image: component/.built
component/.built: component/Dockerfile
docker build -t component component/
touch component/.built
Here as you can see we have now two rules: component.image
and component/.built
.
component.image
has one prerequisite: component/.built
.
component/.built
also has one prerequisite: component/Dockerfile
.
You can also see that the first rule does not have a recipe and that our second rule 'touches' the component/.built
after building the docker image.
Now what does this do?
Well let's look again at some of make's user manual:
A normal prerequisite makes two statements: first, it imposes an order in which recipes will be invoked: the recipes for all prerequisites of a target will be completed before the recipe for the target is run. Second, it imposes a dependency relationship: if any prerequisite is newer than the target, then the target is considered out-of-date and must be rebuilt.
In other terms this means that since the recipe that builds our docker image finishes by touching the sentinel file, make will know:
- if the file is there, it doesn't have to build the image again
- if our sentinel file is older than our Dockerfile (the prerequisite for our sentinel), then we do need a rebuild.
Our Makefile is using this in the following ways:
make {component}.image
uses a prerequisite:.build/{component}/Dockerfile.built
(a sentinel file)- For every component we generate a
.build/{component}/Dockerfile.built
rule. - That rule produces the sentinel file after building the image, like in the example above.
- When listing inter component dependencies we generate additional prerequisites that use the dependencies' sentinels as prerequisites. This way we can ensure that our dependent images rebuild when their dependencies change.
make {component}.push
is another example of such usage. It lists.build/{component}/Dockerfile.pushed
as a prerequisite. The rule for it, in turn, lists.build/{component}/Dockerfile.built
as a prerequisite. This means that if you already pushed your image and that none of its files (nor dependencies) have changed: there is no need to push it again.