MAKEFLAGS += --no-builtin-rules --no-builtin-variables --warn-undefined-variables
unexport MAKEFLAGS
.DEFAULT_GOAL := all
.DELETE_ON_ERROR:
.SUFFIXES:
SHELL := bash
.SHELLFLAGS := -eu -o pipefail -c
escape = $(subst ','\'',$(1))
define noexpand
ifeq ($$(origin $(1)),environment)
$(1) := $$(value $(1))
endif
ifeq ($$(origin $(1)),environment override)
$(1) := $$(value $(1))
endif
ifeq ($$(origin $(1)),command line)
override $(1) := $$(value $(1))
endif
endef
.PHONY: DO
DO:
PROJECT := math-server
TOOLSET ?= auto
CONFIGURATION ?= Debug
BOOST_VERSION ?= 1.72.0
BOOST_LIBRARIES := --with-filesystem --with-program_options --with-regex --with-test
CMAKE_FLAGS ?= -D MATH_SERVER_TESTS=ON
this_dir := $(dir $(realpath $(firstword $(MAKEFILE_LIST))))
src_dir := $(this_dir)
ifdef CI
build_dir := $(this_dir)../build
else
build_dir := $(this_dir).build
endif
boost_dir := $(build_dir)/boost
cmake_dir := $(build_dir)/cmake
DESTDIR ?= $(build_dir)/install
# Enable buildx support:
export DOCKER_CLI_EXPERIMENTAL := enabled
# Target platforms (used by buildx):
DOCKER_PLATFORMS := linux/amd64,linux/armhf
# In case buildx isn't installed (e.g. on Ubuntu):
BUILDX_VERSION := v0.4.2
# Docker Hub credentials:
DOCKER_USERNAME := egortensin
$(eval $(call noexpand,TOOLSET))
$(eval $(call noexpand,CONFIGURATION))
$(eval $(call noexpand,BOOST_VERSION))
$(eval $(call noexpand,CMAKE_FLAGS))
ifdef DOCKER_PASSWORD
$(eval $(call noexpand,DOCKER_PASSWORD))
endif
$(eval $(call noexpand,DESTDIR))
.PHONY: all
all: build
.PHONY: clean
clean:
rm -rf -- '$(call escape,$(build_dir))'
$(boost_dir)/:
cd cmake && python3 -m project.boost.download \
--cache '$(call escape,$(build_dir))' \
-- \
'$(call escape,$(BOOST_VERSION))' \
'$(call escape,$(boost_dir))'
.PHONY: deps
ifdef CI
deps:
cd cmake && python3 -m project.ci.boost -- $(BOOST_LIBRARIES)
else
deps: $(boost_dir)/
cd cmake && python3 -m project.boost.build \
--toolset '$(call escape,$(TOOLSET))' \
--configuration '$(call escape,$(CONFIGURATION))' \
-- \
'$(call escape,$(boost_dir))' \
$(BOOST_LIBRARIES)
endif
.PHONY: build
build:
ifdef CI
cd cmake && python3 -m project.ci.cmake --install -- $(CMAKE_FLAGS)
else
cd cmake && python3 -m project.cmake.build \
--toolset '$(call escape,$(TOOLSET))' \
--configuration '$(call escape,$(CONFIGURATION))' \
--build '$(call escape,$(cmake_dir))' \
--install '$(call escape,$(DESTDIR))' \
--boost '$(call escape,$(boost_dir))' \
-- \
'$(call escape,$(src_dir))' \
$(CMAKE_FLAGS)
endif
.PHONY: install
install: build
.PHONY: test
test:
cd -- '$(call escape,$(cmake_dir))' && ctest -C '$(call escape,$(CONFIGURATION))' --verbose
.PHONY: docker/login
docker/login:
ifndef DOCKER_PASSWORD
$(error Please define DOCKER_PASSWORD)
endif
@echo '$(call escape,$(DOCKER_PASSWORD))' \
| docker login --username '$(call escape,$(DOCKER_USERNAME))' --password-stdin
.PHONY: docker/build
# Build using Compose by default.
docker/build: compose/build
.PHONY: docker/clean
docker/clean:
docker system prune --all --force --volumes
.PHONY: docker/push
# Push multi-arch images by default.
docker/push: buildx/push
.PHONY: pull
pull:
docker-compose pull
.PHONY: up
up:
docker-compose up -d server
.PHONY: client
client:
docker-compose run --rm client
.PHONY: down
down:
docker-compose down --volumes
# `docker build` has weak support for multiarch repos (you need to use multiple
# Dockerfile's, create a manifest manually, etc.).
.PHONY: docker/check-build
docker/check-build:
ifndef FORCE
$(warning Going to build natively; consider `docker buildx build` instead)
endif
# `docker push` would replace the multiarch repo with a single image by default
# (you'd have to create a manifest and push it instead).
.PHONY: docker/check-push
docker/check-push:
ifndef FORCE
$(error Please use `docker buildx build --push` instead)
endif
.PHONY: compose/build
# `docker-compose build` has the same problems as `docker build`.
compose/build: docker/check-build
docker-compose build
.PHONY: compose/push
# `docker-compose push` has the same problems as `docker push`.
compose/push: docker/check-push compose/build
docker-compose push
# The simple way to build multiarch repos is `docker buildx`.
binfmt_image := docker/binfmt:66f9012c56a8316f9244ffd7622d7c21c1f6f28d
.PHONY: fix-binfmt
fix-binfmt:
docker run --rm --privileged '$(call escape,$(binfmt_image))'
curl := curl --silent --show-error --location --dump-header - --connect-timeout 20
buildx_url := https://github.com/docker/buildx/releases/download/$(BUILDX_VERSION)/buildx-$(BUILDX_VERSION).linux-amd64
.PHONY: buildx/install
buildx/install:
mkdir -p -- ~/.docker/cli-plugins/
$(curl) --output ~/.docker/cli-plugins/docker-buildx -- '$(call escape,$(buildx_url))'
chmod +x -- ~/.docker/cli-plugins/docker-buildx
.PHONY: buildx/create
buildx/create: fix-binfmt
docker buildx create --use --name '$(call escape,$(PROJECT))_builder'
.PHONY: buildx/rm
buildx/rm:
docker buildx rm '$(call escape,$(PROJECT))_builder'
buildx/build/%: DO
docker buildx build \
-f '$*/Dockerfile' \
-t '$(call escape,$(DOCKER_USERNAME))/math-$*' \
--platform '$(call escape,$(DOCKER_PLATFORMS))' \
--progress plain .
.PHONY: buildx/build
buildx/build: buildx/build/client buildx/build/server
buildx/push/%: DO
docker buildx build \
-f '$*/Dockerfile' \
-t '$(call escape,$(DOCKER_USERNAME))/math-$*' \
--platform '$(call escape,$(DOCKER_PLATFORMS))' \
--progress plain \
--push .
.PHONY: buildx/push
buildx/push: buildx/push/client buildx/push/server