-
Notifications
You must be signed in to change notification settings - Fork 60
/
Makefile
360 lines (297 loc) · 12.8 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
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
##########################################################################
# Configuration:
# Git version & date which are injected into built binaries.
export FLOW_VERSION = $(shell git describe --dirty --tags)
DATE = $(shell date +%F-%T-%Z)
# Configured Rust installation path of built release targets.
# Caller may override with a CARGO_TARGET_DIR environment variable.
# See: https://doc.rust-lang.org/cargo/reference/environment-variables.html
CARGO_TARGET_DIR ?= target
UNAME := $(shell uname -sp)
# Unfortunately, cargo's build cache get's completely invalidated when you switch between the
# default target and an explicit --target argument. We work around this by setting an explicit
# target. Thus, when running `cargo build` (without an
# explicit --target), the artifacts will be output to target/$TARGET/. This allows
# developers to omit the --target in most cases, and still be able to run make commands that can use
# the same build cache.
# See: https://github.com/rust-lang/cargo/issues/8899
ifeq ($(UNAME),Darwin arm)
export CARGO_BUILD_TARGET=aarch64-apple-darwin
PACKAGE_ARCH=arm64-darwin
ETCD_ARCH=darwin-arm64
ETCD_SHASUM=33094133a771b2d086dc04f2ede41c249258947042de72132af127972880171f
ETCD_EXT=zip
export CARGO_TARGET_X86_64_UNKNOWN_LINUX_MUSL_RUSTFLAGS=-C linker=musl-gcc
else ifeq ($(UNAME),Darwin i386)
export CARGO_BUILD_TARGET=x86_64-apple-darwin
PACKAGE_ARCH=x86-darwin
ETCD_ARCH=darwin-amd64
ETCD_SHASUM=8bd279948877cfb730345ecff2478f69eaaa02513c2a43384ba182c9985267bd
ETCD_EXT=zip
export CARGO_TARGET_X86_64_UNKNOWN_LINUX_MUSL_RUSTFLAGS=-C linker=musl-gcc
else
export CARGO_BUILD_TARGET=x86_64-unknown-linux-gnu
PACKAGE_ARCH=x86-linux
ETCD_ARCH=linux-amd64
ETCD_SHASUM=7910a2fdb1863c80b885d06f6729043bff0540f2006bf6af34674df2636cb906
ETCD_EXT=tar.gz
endif
RUSTBIN = ${CARGO_TARGET_DIR}/${CARGO_BUILD_TARGET}/release
RUST_MUSL_BIN = ${CARGO_TARGET_DIR}/x86_64-unknown-linux-musl/release
# Location to place intermediate files and output artifacts
# during the build process. Note the go tool ignores directories
# with leading '.' or '_'.
WORKDIR = $(realpath .)/.build
# Packaged build outputs.
PKGDIR = ${WORKDIR}/package
# Deno and Etcd release we pin within Flow distributions.
DENO_VERSION = v1.32.1
ETCD_VERSION = v3.5.5
# PROTOC_INC_GO_MODULES are Go modules which must be resolved and included
# with `protoc` invocations
PROTOC_INC_GO_MODULES = \
google.golang.org/protobuf \
github.com/gogo/protobuf \
go.gazette.dev/core
# Targets of Go protobufs which must be compiled.
GO_PROTO_TARGETS = \
./go/protocols/capture/capture.pb.go \
./go/protocols/derive/derive.pb.go \
./go/protocols/flow/flow.pb.go \
./go/protocols/materialize/materialize.pb.go \
./go/protocols/ops/ops.pb.go \
./go/protocols/runtime/runtime.pb.go
# GO_MODULE_PATH expands a $(module), like "go.gazette.dev/core", to the local path
# of its respository as currently specified by go.mod. The `go list` tool
# is used to map submodules to corresponding go.mod versions and paths.
GO_MODULE_PATH = $(shell go list -f '{{ .Dir }}' -m $(module))
##########################################################################
# Configure Go build & test behaviors.
# Tell the go-sqlite3 package to link against a pre-built library
# rather than statically compiling in its own definition.
# This will use the static library definitions provided by libbindings.a.
GO_BUILD_TAGS += libsqlite3
# Targets which Go targets rely on in order to build.
GO_BUILD_DEPS = \
${RUSTBIN}/libbindings.a \
${RUSTBIN}/librocks-exp/librocksdb.a \
crates/bindings/flow_bindings.h
##########################################################################
# Build rules:
.PHONY: default
default: linux-binaries package
# Rules for protocols
.PHONY: protoc-gen-gogo
protoc-gen-gogo:
go mod download
go install github.com/gogo/protobuf/protoc-gen-gogo
# Run the protobuf compiler to generate message and gRPC service implementations.
# Invoke protoc with local and third-party include paths set.
%.pb.go: %.proto protoc-gen-gogo
PATH=$$PATH:$(shell go env GOPATH)/bin ;\
protoc -I . $(foreach module, $(PROTOC_INC_GO_MODULES), -I$(GO_MODULE_PATH)) \
--gogo_out=paths=source_relative,Mgoogle/protobuf/any.proto=github.com/gogo/protobuf/types,Mgoogle/protobuf/timestamp.proto=github.com/gogo/protobuf/types,plugins=grpc:. $*.proto
go-protobufs: $(GO_PROTO_TARGETS)
rust-protobufs:
cargo build --release -p proto-gazette -p proto-flow -p proto-grpc --all-features
${PKGDIR}/bin/deno:
curl -L -o /tmp/deno.zip \
https://github.com/denoland/deno/releases/download/${DENO_VERSION}/deno-${CARGO_BUILD_TARGET}.zip \
&& unzip /tmp/deno.zip -d /tmp \
&& rm /tmp/deno.zip \
&& mkdir -p ${PKGDIR}/bin/ \
&& mv /tmp/deno ${PKGDIR}/bin/
# `etcd` is used for testing, and packaged as a release artifact.
${PKGDIR}/bin/etcd:
curl -L -o /tmp/etcd.${ETCD_EXT} \
https://github.com/etcd-io/etcd/releases/download/${ETCD_VERSION}/etcd-${ETCD_VERSION}-${ETCD_ARCH}.${ETCD_EXT} \
&& echo "${ETCD_SHASUM} /tmp/etcd.${ETCD_EXT}" | sha256sum -c - \
&& if [ "${ETCD_EXT}" = "zip" ]; then \
unzip /tmp/etcd.${ETCD_EXT} -d /tmp; \
else \
tar --extract --file /tmp/etcd.${ETCD_EXT} --directory /tmp/; \
fi \
&& mkdir -p ${PKGDIR}/bin/ \
&& mv /tmp/etcd-${ETCD_VERSION}-${ETCD_ARCH}/etcd /tmp/etcd-${ETCD_VERSION}-${ETCD_ARCH}/etcdctl ${PKGDIR}/bin/ \
&& chown ${UID}:${UID} ${PKGDIR}/bin/etcd ${PKGDIR}/bin/etcdctl \
&& rm -r /tmp/etcd-${ETCD_VERSION}-${ETCD_ARCH}/ \
&& rm /tmp/etcd.${ETCD_EXT} \
&& $@ --version; \
# Rule for building Go targets.
# go-install rules never correspond to actual files, and are always re-run each invocation.
go-install/%: ${RUSTBIN}/libbindings.a crates/bindings/flow_bindings.h
MBP=go.gazette.dev/core/mainboilerplate ;\
./go.sh \
build -o ${PKGDIR}/bin/$(@F) \
-v --tags "${GO_BUILD_TAGS}" \
-ldflags "-X $${MBP}.Version=${FLOW_VERSION} -X $${MBP}.BuildDate=${DATE}" $*
${PKGDIR}/bin/gazette: go-install/go.gazette.dev/core/cmd/gazette
${PKGDIR}/bin/gazctl: go-install/go.gazette.dev/core/cmd/gazctl
${PKGDIR}/bin/flowctl-go: $(GO_BUILD_DEPS) $(GO_PROTO_TARGETS) go-install/github.com/estuary/flow/go/flowctl-go
${PKGDIR}/bin/fetch-open-graph:
cd fetch-open-graph && go build -o ${PKGDIR}/
# `sops` is used for encrypt/decrypt of connector configurations.
${PKGDIR}/bin/sops:
go install go.mozilla.org/sops/v3/cmd/[email protected]
cp $(shell go env GOPATH)/bin/sops $@
########################################################################
# Rust outputs:
# The & here declares that this single invocation will produce all of the files on the left hand
# side. flow_bindings.h is generated by the bindings build.rs.
.PHONY: ${RUSTBIN}/libbindings.a crates/bindings/flow_bindings.h
${RUSTBIN}/libbindings.a crates/bindings/flow_bindings.h &:
cargo build --release --locked -p bindings
.PHONY: ${RUSTBIN}/librocks-exp/librocksdb.a
${RUSTBIN}/librocks-exp/librocksdb.a:
cargo build --release --locked -p librocks-exp
.PHONY: ${RUSTBIN}/agent
${RUSTBIN}/agent:
cargo build --release --locked -p agent
.PHONY: ${RUSTBIN}/flowctl
${RUSTBIN}/flowctl:
cargo build --release --locked -p flowctl
# Statically linked binaries using MUSL:
.PHONY: ${RUST_MUSL_BIN}/flow-connector-init
${RUST_MUSL_BIN}/flow-connector-init:
cargo build --target x86_64-unknown-linux-musl --release --locked -p connector-init
.PHONY: ${RUST_MUSL_BIN}/flow-parser
${RUST_MUSL_BIN}/flow-parser:
cargo build --target x86_64-unknown-linux-musl --release --locked -p parser
.PHONY: ${RUST_MUSL_BIN}/flow-schemalate
${RUST_MUSL_BIN}/flow-schemalate:
cargo build --target x86_64-unknown-linux-musl --release --locked -p schemalate
########################################################################
# Final output packaging:
GNU_TARGETS = \
${PKGDIR}/bin/agent \
${PKGDIR}/bin/deno \
${PKGDIR}/bin/etcd \
${PKGDIR}/bin/flowctl \
${PKGDIR}/bin/flowctl-go \
${PKGDIR}/bin/gazette \
${PKGDIR}/bin/sops
MUSL_TARGETS = \
${PKGDIR}/bin/flow-connector-init \
${PKGDIR}/bin/flow-parser \
${PKGDIR}/bin/flow-schemalate
.PHONY: linux-gnu-binaries
ifeq ($(UNAME),Darwin arm)
linux-gnu-binaries: $(GNU_TARGETS)
codesign -f -s - .build/package/bin/gazette
codesign -f -s - .build/package/bin/flowctl-go
else
linux-gnu-binaries: $(GNU_TARGETS)
endif
.PHONY: linux-musl-binaries
linux-musl-binaries: | ${PKGDIR}
cargo build --target x86_64-unknown-linux-musl --release --locked -p connector-init -p parser -p schemalate
cp -f target/x86_64-unknown-linux-musl/release/flow-connector-init .build/package/bin/
cp -f target/x86_64-unknown-linux-musl/release/flow-parser .build/package/bin/
cp -f target/x86_64-unknown-linux-musl/release/flow-schemalate .build/package/bin/
.PHONY: linux-binaries
linux-binaries: linux-gnu-binaries linux-musl-binaries
${PKGDIR}/flow-$(PACKAGE_ARCH).tar.gz:
rm -f $@
cd ${PKGDIR}/bin && tar -zcf ../flow-$(PACKAGE_ARCH).tar.gz *
.PHONY: package
package: ${PKGDIR}/flow-$(PACKAGE_ARCH).tar.gz
${PKGDIR}:
mkdir -p ${PKGDIR}/bin
mkdir ${PKGDIR}/lib
# The following binaries are statically linked, so come from a different subdirectory
${PKGDIR}/bin/flow-connector-init: ${RUST_MUSL_BIN}/flow-connector-init | ${PKGDIR}
cp ${RUST_MUSL_BIN}/flow-connector-init $@
${PKGDIR}/bin/flow-parser: ${RUST_MUSL_BIN}/flow-parser | ${PKGDIR}
cp ${RUST_MUSL_BIN}/flow-parser $@
${PKGDIR}/bin/flow-schemalate: ${RUST_MUSL_BIN}/flow-schemalate | ${PKGDIR}
cp ${RUST_MUSL_BIN}/flow-schemalate $@
${PKGDIR}/bin/flowctl: ${RUSTBIN}/flowctl | ${PKGDIR}
cp ${RUSTBIN}/flowctl $@
# Control-plane binaries
${PKGDIR}/bin/agent: ${RUSTBIN}/agent | ${PKGDIR}
cp ${RUSTBIN}/agent $@
##########################################################################
# Make targets used by CI:
# We use LLVM (lld) for faster linking. See RUSTFLAGS in .github/workflows/main.yml
.PHONY: extra-ci-runner-setup
extra-ci-runner-setup:
sudo apt install -y \
build-essential \
clang \
libsqlite3-dev \
lld \
musl \
musl-dev \
musl-tools \
python3-poetry \
python3-venv
.PHONY: print-versions
print-versions:
echo "Resolved repository version: ${FLOW_VERSION}" \
&& /usr/bin/ld.lld --version \
&& cargo version --verbose \
&& docker --version \
&& go version \
&& jq --version \
&& node --version \
&& npm --version \
&& rustc --version \
.PHONY: install-tools
install-tools: ${PKGDIR}/bin/deno ${PKGDIR}/bin/etcd ${PKGDIR}/bin/sops
.PHONY: rust-gnu-test
rust-gnu-test:
PATH=${PKGDIR}/bin:$$PATH ;\
cargo test --release --locked --workspace --exclude parser --exclude schemalate --exclude connector-init
.PHONY: rust-musl-test
rust-musl-test:
cargo test --release --locked --target x86_64-unknown-linux-musl --package parser --package schemalate --package connector-init
# `go` test targets must have PATH-based access to tools (etcd & sops),
# because the `go` tool compiles tests as binaries within a temp directory,
# and these binaries cannot expect `sops` to be co-located alongside.
.PHONY: go-test-fast
go-test-fast: $(GO_BUILD_DEPS) | ${PKGDIR}/bin/deno ${PKGDIR}/bin/etcd ${PKGDIR}/bin/sops
PATH=${PKGDIR}/bin:$$PATH ;\
./go.sh test --tags "${GO_BUILD_TAGS}" --count=1 ./go/...
.PHONY: go-test-ci
go-test-ci:
PATH=${PKGDIR}/bin:$$PATH ;\
GORACE="halt_on_error=1" ;\
./go.sh test --tags "${GO_BUILD_TAGS}" --race --count=15 --failfast ./go/...
.PHONY: data-plane-test-setup
data-plane-test-setup:
ifeq ($(SKIP_BUILD),true)
data-plane-test-setup:
@echo "testing using pre-built binaries:"
@ls -al ${PKGDIR}/bin/
${PKGDIR}/bin/flowctl raw json-schema > flow.schema.json
else
data-plane-test-setup: ${PKGDIR}/bin/flowctl-go ${PKGDIR}/bin/flowctl ${PKGDIR}/bin/flow-connector-init ${PKGDIR}/bin/gazette ${PKGDIR}/bin/deno ${PKGDIR}/bin/etcd ${PKGDIR}/bin/sops flow.schema.json
endif
.PHONY: catalog-test
catalog-test: data-plane-test-setup
${PKGDIR}/bin/flowctl-go test --source examples/flow.yaml $(ARGS)
.PHONY: end-to-end-test
end-to-end-test: data-plane-test-setup
./tests/run-all.sh
flow.schema.json: ${PKGDIR}/bin/flowctl
${PKGDIR}/bin/flowctl raw json-schema > $@
# These docker targets intentionally don't depend on any upstream targets. This is because the
# upstream targes are all PHONY as well, so there would be no way to prevent them from running twice if you
# invoke e.g. `make package` followed by `make docker-image`. If the `docker-image` target depended
# on the `package` target, it would not skip the package step when you invoke `docker-image`.
# For now, the github action workflow manually invokes make to perform each of these tasks.
.PHONY: docker-image
docker-image:
docker build \
--file .devcontainer/release.Dockerfile \
--tag ghcr.io/estuary/flow:${FLOW_VERSION} \
--tag ghcr.io/estuary/flow:dev \
${PKGDIR}/
.PHONY: docker-push
docker-push:
docker push ghcr.io/estuary/flow:${FLOW_VERSION}
# This is used by the GH Action workflow to push the 'dev' tag.
# It is invoked only for builds on the master branch.
.PHONY: docker-push-dev
docker-push-dev:
docker push ghcr.io/estuary/flow:dev