-
Notifications
You must be signed in to change notification settings - Fork 52
/
Copy pathMakefile
266 lines (226 loc) · 9.04 KB
/
Makefile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
#
# For normal use, VERSION should be a snapshot version. I.e. one ending in
# -SNAPSHOT, such as 35-SNAPSHOT
#
# When a version is final, do the following:
# 1) Change VERSION to a non-SNAPSHOT release: 35-SNAPSHOT -> 35
# 2) Commit the repo
# 3) `make release' to push the images to dockerhub
# 4) Change VERSION to tne next SNAPSHOT release: 35 -> 36-SNAPSHOT
# 5) Commit
# 6) Continue developing
# 7) `make snapshot' as needed to push snapshot images to dockerhub
#
VERSION := $(file < version)
RELEASE_TYPE := $(if $(filter %-SNAPSHOT, $(VERSION)),snapshot,release)
LABEL := io.trino.git.hash=$(shell git rev-parse HEAD)
DEPEND_SH=bin/depend.sh
FLAG_SH=bin/flag.sh
BUILD_SH=$(realpath bin/build.sh)
TAG_SH=$(realpath bin/tag.sh)
PUSH_SH=bin/push.sh
TEST_SH=bin/test.sh
BUILDDIR=build
DEPDIR=$(BUILDDIR)/depends
FLAGDIR=$(BUILDDIR)/flags
#
# In theory, we could just find all of the Dockerfiles and derive IMAGE_DIRS
# from that, but make's dir function includes the trailing slash, which we'd
# have to strip off to get a valid Docker image name.
#
# Also, find on Mac doesn't support -exec {} +
#
# Note that the generated .d files also include reverse dependencies so that
# you can e.g. `make hdp3.1-base.dependants' and hdp3.1-hive, and all of its
# dependent images will be rebuilt. This is used in .travis.yml to break the
# build up into pieces based on image that have a large number of direct and
# indirect children.
#
IMAGE_DIRS := $(shell find testing -type f -name Dockerfile -exec dirname {} \;)
UNLABELLED_TAGS := $(addsuffix @unlabelled,$(IMAGE_DIRS))
PARENT_CHECKS := $(addsuffix -parent-check,$(IMAGE_DIRS))
LATEST_TAGS := $(addsuffix @latest,$(IMAGE_DIRS))
VERSION_TAGS := $(addsuffix @$(VERSION),$(IMAGE_DIRS))
GIT_HASH := $(shell git rev-parse --short HEAD)
GIT_HASH_TAGS := $(addsuffix @$(GIT_HASH),$(IMAGE_DIRS))
DOCKERFILES := $(addsuffix /Dockerfile,$(IMAGE_DIRS))
DEPS := $(foreach dockerfile,$(DOCKERFILES),$(DEPDIR)/$(dockerfile:/Dockerfile=.d))
FLAGS := $(foreach dockerfile,$(DOCKERFILES),$(FLAGDIR)/$(dockerfile:/Dockerfile=.flags))
RELEASE_TAGS := $(VERSION_TAGS) $(LATEST_TAGS)
SNAPSHOT_TAGS := $(GIT_HASH_TAGS)
#
# Make a list of the Docker images we depend on, but aren't built from
# Dockerfiles in this repository. Order doesn't matter, but sort() has the
# side-effect of making the list unique.
#
EXTERNAL_DEPS = \
$(sort \
$(foreach dockerfile,$(DOCKERFILES),\
$(shell $(DEPEND_SH) -x $(dockerfile) $(call docker-tag,$(UNLABELLED_TAGS)))))
#
# Image tags in the Makefile use @ instead of : in full image:tag names. This
# is because there's no way to escape a colon in a target or prerequisite
# name[0]. docker-tag reverses this transformation for places where we need to
# interact with docker using its image:tag convention.
#
# [0] http://www.mail-archive.com/[email protected]/msg03318.html
#
# Must be a recursively expanded variable to use with $(call ...)
#
docker-tag = $(subst @,:,$(1))
#
# Resolves the image name by prefixing ghcr.io/trinodb/
#
resolved-image-name = $(addprefix ghcr.io/trinodb/,$(1))
#
# Various variables that define targets need to be .PHONY so that Make
# continues to build them if a file with a matching name somehow comes into
# existence
#
.PHONY: $(IMAGE_DIRS) $(LATEST_TAGS) $(UNLABELLED_TAGS) $(VERSION_TAGS) $(GIT_HASH_TAGS)
.PHONY: $(PARENT_CHECKS) $(IMAGE_TESTS) $(EXTERNAL_DEPS)
# By default, build all of the images.
all: images
images: $(LATEST_TAGS)
#
# Release images to Dockerhub
#
.PHONY: prepare-release release push-release snapshot build-snapshot push-snapshot
prepare-release: require-clean-repo require-on-master require-release-version
release: prepare-release push-release
push-release: $(RELEASE_TAGS)
$(PUSH_SH) $(call docker-tag,$(call resolved-image-name,$^))
snapshot: require-clean-repo require-snapshot-version push-snapshot
build-snapshot: $(SNAPSHOT_TAGS)
push-snapshot: $(SNAPSHOT_TAGS)
$(PUSH_SH) $(call docker-tag,$(call resolved-image-name,$^))
#
# Create tags without pushing. This is probably only useful for testing.
#
.PHONY: release-tags snapshot-tags
release-tags: $(RELEASE_TAGS)
snapshot-tags: $(SNAPSHOT_TAGS)
#
# Targets for sanity-checking the repo prior to doing a release or snapshot.
# Use $(shell git ...) so the output of the git command shows up on the command
# line.
#
.PHONY: require-clean-repo require-on-master
require-clean-repo:
test -z "$(shell git status --porcelain)" || ( echo "Repository is not clean"; exit 1 )
require-on-master:
test "$(shell git rev-parse --abbrev-ref HEAD)" = "master" || ( echo "Current branch must be master"; exit 1 )
require-%-version:
[ "$(RELEASE_TYPE)" = "$*" ] || ( echo "$(VERSION) is not a $* version"; exit 1 )
#
# For generating/cleaning the depends directory without building any images.
# Because the Makefile includes the .d files, and will create them if they
# don't exist, an empty target is sufficient to get make to rebuild the
# dependencies if needed. This is mostly useful for testing changes to the
# script that creates the .d files.
#
.PHONY: meta
meta:
#
# Include the dependencies for every image we know how to build. These don't
# exist in the repo, but the next rule specifies how to create them. Make will
# run that rule for every .d file in $(DEPS).
#
include $(DEPS)
include $(FLAGS)
include $(TEST_RDEPS)
$(DEPDIR)/%.d: %/Dockerfile $(DEPEND_SH)
-mkdir -p $(dir $@)
$(DEPEND_SH) -d $< $(call docker-tag,$(UNLABELLED_TAGS)) >$@
$(FLAGDIR)/%.flags: %/Dockerfile $(FLAG_SH)
-mkdir -p $(dir $@)
$(FLAG_SH) $< >$@
#
# Images in the repo that are built FROM other images in the repo are built
# from the special tag `unlabelled'. This is vestigial from the old approach:
# LABEL (from `docker build --label`) data creates a new
# layer in the image. Without building from the `unlabelled' tag, all direct or
# indirect child images have to be fully rebuilt when the LABEL data changes.
# Since we include the git hash in the LABEL data, this changes frequently.
#
# We take the approach of building the :latest tag first, then finding the
# parent of the layer containing the LABEL information, and tagging that as
# :unlabelled. This works because all of the LABEL information applied via a
# --label option(s) to `docker build' is put in a single layer at the top of
# the resulting stack of layers.
#
# However
# - the above no longer worked, because `docker build --label` does not currently
# result with a separate layer that could be used for tagging.
# - the history is retained above as an explanation what the ":unlabelled" was
# supposed to mean.
# - TODO replace :unlabelled with something more appropriate.
$(UNLABELLED_TAGS): %@unlabelled: %/Dockerfile %@latest
$(TAG_SH) $*:latest $(call docker-tag,$@)
#
# We don't need to specify any (real) dependencies other than the Dockerfile
# for the image because these are .PHONY targets. In particular, if the DBFLAGS
# for an image have changed without the Dockerfile changing, it's OK because
# we'll invoke docker build for the image anyway and let Docker figure out if
# anything has changed that requires a rebuild.
#
$(LATEST_TAGS): %@latest: %/Dockerfile %-parent-check
@echo
@echo "Building [$@] image using buildkit"
@echo
cd $* && time $(BUILD_SH) $(call docker-tag,$@) ${BUILD_ARGS} $(DBFLAGS_$*) --label $(LABEL)
$(VERSION_TAGS): %@$(VERSION): %@latest
$(TAG_SH) $(call docker-tag,$^) $(call docker-tag,$@)
$(TAG_SH) $(call docker-tag,$^) $(call docker-tag,$(call resolved-image-name,$^))
$(TAG_SH) $(call docker-tag,$@) $(call docker-tag,$(call resolved-image-name,$@))
$(GIT_HASH_TAGS): %@$(GIT_HASH): %@latest
$(TAG_SH) $(call docker-tag,$^) $(call docker-tag,$@)
$(TAG_SH) $(call docker-tag,$^) $(call docker-tag,$(call resolved-image-name,$^))
$(TAG_SH) $(call docker-tag,$@) $(call docker-tag,$(call resolved-image-name,$@))
#
# Verify that the parent image specified in the Dockerfile is either
# 1. External
# 2. Has the tag :unlabelled
#
$(PARENT_CHECKS): %-parent-check: %/Dockerfile $(DEPEND_SH)
$(DEPEND_SH) -p unlabelled $< $(call docker-tag,$(UNLABELLED_TAGS))
#
# This makes it possible it possible to type `make testing/image' without
# specifying @latest
#
$(IMAGE_DIRS): %: %@latest
#
# Static pattern rule to pull docker images that are external dependencies of
# this repository.
#
$(EXTERNAL_DEPS): %:
docker pull $(call docker-tag,$@)
#
# Targets and variables for creating the dependency graph of the docker images
# as an image file.
#
GVDIR=$(BUILDDIR)/graphviz
GVWHOLE=$(GVDIR)/dependency_graph.gv
DEPENDENCY_GRAPH=$(GVDIR)/dependency_graph.svg
GVFRAGS=$(addprefix $(GVDIR)/,$(addsuffix .gv.frag,$(IMAGE_DIRS)))
.PHONY: graph
graph: $(DEPENDENCY_GRAPH)
$(DEPENDENCY_GRAPH): $(GVWHOLE) Makefile
dot -T svg $(filter %.gv,$^) > $@
$(GVWHOLE): $(GVFRAGS) Makefile
echo "digraph {" >$@
echo 'size="14!" pack=true packmode="array2"' >>$@
cat $(filter %.gv.frag,$^) >>$@
echo "}" >>$@
$(GVFRAGS): $(GVDIR)/%.gv.frag: %/Dockerfile $(DEPEND_SH)
-mkdir -p $(dir $@)
$(DEPEND_SH) -g $< $(call docker-tag,$(UNLABELLED_TAGS)) >$@
.PHONY: test
test:
$(TEST_SH) $(IMAGE_TO_TEST)
.PHONY: clean
clean:
-rm -r $(BUILDDIR)
.PHONY: list
list:
@echo $(IMAGE_DIRS)