-
Notifications
You must be signed in to change notification settings - Fork 146
/
Copy pathMakefile
858 lines (712 loc) · 29.1 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
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
include $(CURDIR)/make/env.mk
PLATFORM ?= linux/amd64
ROX_PROJECT=apollo
TESTFLAGS=-race -p 4
BASE_DIR=$(CURDIR)
BENCHTIME ?= 1x
BENCHTIMEOUT ?= 20m
BENCHCOUNT ?= 1
podman =
# docker --version might not contain any traces of podman in the latest
# version, search for more output
ifneq (,$(findstring podman,$(shell docker --version 2>/dev/null)))
podman = yes
endif
ifneq (,$(findstring Podman,$(shell docker version 2>/dev/null)))
podman = yes
endif
ifdef podman
# Disable selinux for local podman builds.
DOCKER_OPTS=--security-opt label=disable
else
# Podman DTRT by running processes unprivileged in containers,
# but it's UID mapping is more nuanced. Only set user for vanilla docker.
DOCKER_OPTS=--user "$(shell id -u)"
endif
# Set to empty string to echo some command lines which are hidden by default.
SILENT ?= @
# UNIT_TEST_IGNORE ignores a set of file patterns from the unit test make command.
# the pattern is passed to: grep -Ev
# usage: "path/to/ignored|another/path"
# TODO: [ROX-19070] Update postgres store test generation to work for foreign keys
UNIT_TEST_IGNORE := "stackrox/rox/sensor/tests|stackrox/rox/operator/tests|stackrox/rox/central/reports/config/store/postgres|stackrox/rox/central/complianceoperator/v2/scanconfigurations/store/postgres|stackrox/rox/central/auth/store/postgres|stackrox/rox/scanner/e2etests"
ifeq ($(TAG),)
TAG=$(shell git describe --tags --abbrev=10 --dirty --long --exclude '*-nightly-*')$(MAIN_TAG_SUFFIX)
endif
# Set expiration on Quay.io for non-release tags.
ifeq ($(findstring x,$(TAG)),x)
QUAY_TAG_EXPIRATION=13w
else
QUAY_TAG_EXPIRATION=never
endif
ROX_PRODUCT_BRANDING ?= STACKROX_BRANDING
# ROX_IMAGE_FLAVOR is an ARG used in Dockerfiles that defines the default registries for main, scanner, and collector images.
# ROX_IMAGE_FLAVOR valid values are: development_build, rhacs, opensource.
ROX_IMAGE_FLAVOR ?= $(shell \
if [[ "$(ROX_PRODUCT_BRANDING)" == "STACKROX_BRANDING" ]]; then \
echo "opensource"; \
else \
echo "development_build"; \
fi)
DEFAULT_IMAGE_REGISTRY := quay.io/stackrox-io
ifeq ($(ROX_PRODUCT_BRANDING),RHACS_BRANDING)
DEFAULT_IMAGE_REGISTRY := quay.io/rhacs-eng
endif
GOBUILD := $(CURDIR)/scripts/go-build.sh
DOCKERBUILD := $(CURDIR)/scripts/docker-build.sh
GO_TEST_OUTPUT_PATH=$(CURDIR)/test-output/test.log
GOPATH_VOLUME_NAME := stackrox-rox-gopath
GOCACHE_VOLUME_NAME := stackrox-rox-gocache
GENERATE_PATH ?= ./...
# If git branch name contains substring "-debug", a debug build will be made, unless overridden by environment variable.
ifneq (,$(findstring -debug,$(shell git rev-parse --abbrev-ref HEAD)))
DEBUG_BUILD ?= yes
endif
DEBUG_BUILD ?= no
# Figure out whether to use standalone Docker volume for GOPATH/Go build cache, or bind
# mount one from the host filesystem.
# The latter is painfully slow on Mac OS X with Docker Desktop, so we default to using a
# standalone volume in that case, and to bind mounting otherwise.
UNAME_S := $(shell uname -s)
UNAME_M := $(shell uname -m)
BUILD_IMAGE := quay.io/stackrox-io/apollo-ci:$(shell sed 's/\s*\#.*//' BUILD_IMAGE_VERSION)
ifneq ($(UNAME_M),x86_64)
BUILD_IMAGE = docker.io/library/golang:1.22
endif
CENTRAL_DB_DOCKER_ARGS :=
ifeq ($(GOARCH),s390x)
CENTRAL_DB_DOCKER_ARGS := \
--build-arg="BASE_IMAGE=ubi9-minimal" \
--build-arg="BASE_TAG=9.2" \
--build-arg="RPMS_REGISTRY=quay.io" \
--build-arg="RPMS_BASE_IMAGE=centos/centos" \
--build-arg="RPMS_BASE_TAG=stream9"
endif
# By default, assume we are going to use a bind mount volume instead of a standalone one.
BIND_GOCACHE ?= 1
BIND_GOPATH ?= 1
# Only resort to local volumes on X86_64 Darwin, since on ARM Darwin the permissions of the
# standalone volume will be mapped to root instead of the local user, making the build fail.
# An alternative is to chown the directory of the standalone volume within the container.
ifeq ($(UNAME_S),Darwin)
ifneq ($(UNAME_M),arm64)
BIND_GOCACHE ?= 0
BIND_GOPATH ?= 0
endif
endif
ifeq ($(BIND_GOCACHE),1)
GOCACHE_VOLUME_SRC := $(CURDIR)/linux-gocache
else
GOCACHE_VOLUME_SRC := $(GOCACHE_VOLUME_NAME)
endif
ifeq ($(BIND_GOPATH),1)
GOPATH_VOLUME_SRC := $(GOPATH)
else
GOPATH_VOLUME_SRC := $(GOPATH_VOLUME_NAME)
endif
LOCAL_VOLUME_ARGS := -v $(CURDIR):/src:delegated -v $(GOCACHE_VOLUME_SRC):/linux-gocache:delegated -v $(GOPATH_VOLUME_SRC):/go:delegated
GOPATH_WD_OVERRIDES := -w /src -e GOPATH=/go -e GOCACHE=/linux-gocache -e GIT_CONFIG_COUNT=1 -e GIT_CONFIG_KEY_0=safe.directory -e GIT_CONFIG_VALUE_0='/src'
null :=
space := $(null) $(null)
comma := ,
.PHONY: all
all: deps style test image
#####################################################################
###### Binaries we depend on (need to be defined on top) ############
#####################################################################
include make/gotools.mk
$(call go-tool, BUF_BIN, github.com/bufbuild/buf/cmd/buf, tools/proto)
$(call go-tool, GOLANGCILINT_BIN, github.com/golangci/golangci-lint/cmd/golangci-lint, tools/linters)
$(call go-tool, EASYJSON_BIN, github.com/mailru/easyjson/easyjson)
$(call go-tool, ROXVET_BIN, ./tools/roxvet)
$(call go-tool, STRINGER_BIN, golang.org/x/tools/cmd/stringer)
$(call go-tool, MOCKGEN_BIN, go.uber.org/mock/mockgen)
$(call go-tool, GO_JUNIT_REPORT_BIN, github.com/jstemmer/go-junit-report/v2, tools/test)
$(call go-tool, PROTOLOCK_BIN, github.com/nilslice/protolock/cmd/protolock, tools/linters)
$(call go-tool, GOVULNCHECK_BIN, golang.org/x/vuln/cmd/govulncheck, tools/linters)
$(call go-tool, IMAGE_PREFETCHER_DEPLOY_BIN, github.com/stackrox/image-prefetcher/deploy, tools/test)
$(call go-tool, PROMETHEUS_METRIC_PARSER_BIN, github.com/stackrox/prometheus-metric-parser, tools/test)
###########
## Style ##
###########
.PHONY: style
style: golangci-lint style-slim
.PHONY: style-slim
style-slim: \
blanks \
check-service-protos \
newlines \
no-large-files \
openshift-ci-style \
proto-style \
qa-tests-style \
roxvet \
shell-style \
storage-protos-compatible \
ui-lint
GOLANGCILINT_FLAGS := --verbose --print-resources-usage
.PHONY: golangci-lint-cache-status
golangci-lint-cache-status: $(GOLANGCILINT_BIN) deps
@echo '+ $@'
@echo "Checking golangci-lint cache status"
$(GOLANGCILINT_BIN) cache status
.PHONY: golangci-lint
golangci-lint: $(GOLANGCILINT_BIN) deps
ifdef CI
@echo '+ $@'
@echo 'The environment indicates we are in CI; running linters in check mode.'
@echo 'If this fails, run `make style`.'
$(GOLANGCILINT_BIN) --version
@echo "Running with no tags and no tests..."
@# The first run is meant to have limited scope to warmup the cache.
@# Adding it as first allowed to shorten the runtime of the following runs to about 5 min each
$(GOLANGCILINT_BIN) run $(GOLANGCILINT_FLAGS) --tests=false
@echo "Running with no tags..."
@# We need to enable unused linter here as it will not work without tests or in release tag.
$(GOLANGCILINT_BIN) run $(GOLANGCILINT_FLAGS) --enable=unused
@echo "Running with release tags..."
@# We use --tests=false because some unit tests don't compile with release tags,
@# since they use functions that we don't define in the release build. That's okay.
$(GOLANGCILINT_BIN) run $(GOLANGCILINT_FLAGS) --build-tags "$(subst $(comma),$(space),$(RELEASE_GOTAGS))" --tests=false
else
$(GOLANGCILINT_BIN) run $(GOLANGCILINT_FLAGS) --fix --enable=unused
$(GOLANGCILINT_BIN) run $(GOLANGCILINT_FLAGS) --fix --build-tags "$(subst $(comma),$(space),$(RELEASE_GOTAGS))" --tests=false
endif
.PHONY: proto-style
proto-style: $(BUF_BIN) deps
@echo "+ $@"
$(BUF_BIN) format --exit-code --diff -w
.PHONY: qa-tests-style
qa-tests-style:
@echo "+ $@"
make -C qa-tests-backend/ style
.PHONY: ui-lint
ui-lint:
@echo "+ $@"
make -C ui lint
.PHONY: openshift-ci-style
openshift-ci-style:
@echo "+ $@"
make -C .openshift-ci/ style
.PHONY: shell-style
shell-style:
@echo "+ $@"
$(SILENT)$(BASE_DIR)/scripts/style/shellcheck.sh
.PHONY: update-shellcheck-skip
update-shellcheck-skip:
@echo "+ $@"
$(SILENT)rm -f scripts/style/shellcheck_skip.txt
$(SILENT)$(BASE_DIR)/scripts/style/shellcheck.sh update_failing_list
.PHONY: fast-central-build
fast-central-build: central-build-nodeps
.PHONY: central-build-nodeps
central-build-nodeps:
@echo "+ $@"
$(GOBUILD) central
.PHONY: config-controller-build-nodeps
config-controller-build-nodeps:
@echo "+ $@"
$(GOBUILD) config-controller
.PHONY: fast-central
fast-central: deps
@echo "+ $@"
docker run $(DOCKER_OPTS) -e CGO_ENABLED --rm $(GOPATH_WD_OVERRIDES) $(LOCAL_VOLUME_ARGS) $(BUILD_IMAGE) make fast-central-build
$(SILENT)$(BASE_DIR)/scripts/k8s/kill-pod.sh central
# fast is a dev mode options when using local dev
# it will automatically restart Central if there are any changes
.PHONY: fast
fast: fast-central
.PHONY: fast-sensor
fast-sensor: sensor-build-dockerized
.PHONY: fast-sensor-kubernetes
fast-sensor-kubernetes: sensor-kubernetes-build-dockerized
$(SILENT)$(BASE_DIR)/scripts/k8s/kill-pod.sh sensor
.PHONY: fast-migrator
fast-migrator:
@echo "+ $@"
docker run $(DOCKER_OPTS) -e CGO_ENABLED --rm $(GOPATH_WD_OVERRIDES) $(LOCAL_VOLUME_ARGS) $(BUILD_IMAGE) make fast-migrator-build
.PHONY: fast-migrator-build
fast-migrator-build: migrator-build-nodeps
.PHONY: migrator-build-nodeps
migrator-build-nodeps:
@echo "+ $@"
$(GOBUILD) migrator
.PHONY: check-service-protos
check-service-protos:
@echo "+ $@"
$(SILENT)$(BASE_DIR)/tools/check-service-protos/run.sh
.PHONY: no-large-files
no-large-files:
$(BASE_DIR)/tools/detect-large-files.sh "$(BASE_DIR)/tools/allowed-large-files"
# adding the uptodate flag will inform us if the protos and lock are out of date if they pass the compatibility check.
.PHONY: storage-protos-compatible
storage-protos-compatible: $(PROTOLOCK_BIN)
@echo "+ $@"
$(SILENT)$(PROTOLOCK_BIN) status -lockdir=$(BASE_DIR)/proto/storage -protoroot=$(BASE_DIR)/proto/storage --uptodate true
.PHONY: update-storage-protolock
update-storage-protolock: $(PROTOLOCK_BIN)
@echo "+ $@"
$(SILENT)$(PROTOLOCK_BIN) commit -lockdir=$(BASE_DIR)/proto/storage -protoroot=$(BASE_DIR)/proto/storage
.PHONY: blanks
blanks:
@echo "+ $@"
ifdef CI
$(SILENT)git grep -L '^// Code generated by .* DO NOT EDIT\.' -- '*.go' | xargs -n 1000 $(BASE_DIR)/tools/import_validate.py
else
$(SILENT)git grep -L '^// Code generated by .* DO NOT EDIT\.' -- '*.go' | xargs -n 1000 $(BASE_DIR)/tools/fix-blanks.sh
endif
.PHONY: newlines
newlines:
@echo "+ $@"
ifdef CI
$(SILENT)git grep --cached -Il '' | xargs $(BASE_DIR)/tools/check-newlines.sh
else
$(SILENT)git grep --cached -Il '' | xargs $(BASE_DIR)/tools/check-newlines.sh --fix
endif
.PHONY: init-githooks
init-githooks:
@echo "+ $@"
./tools/githooks/install-hooks.sh tools/githooks/pre-commit
.PHONY: dev
dev: install-dev-tools
@echo "+ $@"
#####################################
## Generated Code and Dependencies ##
#####################################
PROTO_GENERATED_SRCS = $(GENERATED_PB_SRCS) $(GENERATED_VT_SRCS) $(GENERATED_COMPAT_SRCS) $(GENERATED_API_SRCS) $(GENERATED_API_GW_SRCS)
include make/protogen.mk
.PHONY: go-easyjson-srcs
go-easyjson-srcs: $(EASYJSON_BIN)
@echo "+ $@"
@# Files are ordered such that repeated runs of `make go-easyjson-srcs` don't create diffs.
$(SILENT)$(EASYJSON_BIN) pkg/compliance/compress/compress.go
.PHONY: clean-easyjson-srcs
clean-easyjson-srcs:
@echo "+ $@"
$(SILENT)find . -name '*_easyjson.go' -exec rm {} \;
.PHONY: go-generated-srcs
go-generated-srcs: deps clean-easyjson-srcs go-easyjson-srcs $(MOCKGEN_BIN) $(STRINGER_BIN)
@echo "+ $@"
PATH="$(GOTOOLS_BIN):$(PATH):$(BASE_DIR)/tools/generate-helpers" MOCKGEN_BIN="$(MOCKGEN_BIN)" go generate -v -x $(GENERATE_PATH)
proto-generated-srcs: $(PROTO_GENERATED_SRCS) $(GENERATED_API_SWAGGER_SPECS) $(GENERATED_API_SWAGGER_SPECS_V2) inject-proto-tags cleanup-swagger-json-gotags
@echo "+ $@"
$(SILENT)touch proto-generated-srcs
$(SILENT)$(MAKE) clean-obsolete-protos
clean-proto-generated-srcs:
@echo "+ $@"
git clean -xdf generated
.PHONY: config-controller-gen
config-controller-gen:
make -C config-controller/ manifests
make -C config-controller/ generate
cp config-controller/config/crd/bases/config.stackrox.io_securitypolicies.yaml image/templates/helm/stackrox-central/crds
.PHONY: generated-srcs
generated-srcs: go-generated-srcs config-controller-gen
deps: $(shell find $(BASE_DIR) -name "go.sum")
@echo "+ $@"
$(SILENT)touch deps
%/go.sum: %/go.mod
$(SILENT)cd $*
@echo "+ $@"
$(SILENT)$(eval GOMOCK_REFLECT_DIRS=`find . -type d -name 'gomock_reflect_*'`)
$(SILENT)test -z $(GOMOCK_REFLECT_DIRS) || { echo "Found leftover gomock directories. Please remove them and rerun make deps!"; echo $(GOMOCK_REFLECT_DIRS); exit 1; }
$(SILENT)go mod tidy
ifdef CI
$(SILENT)GOTOOLCHAIN=local go mod tidy || { >&2 echo "Go toolchain does not match with installed Go version. This is a compatibility check that prevents breaking downstream builds. If you really need to update the toolchain version, ask in #forum-acs-golang" ; exit 1 ; }
$(SILENT)git diff --exit-code -- go.mod go.sum || { echo "go.mod/go.sum files were updated after running 'go mod tidy', run this command on your local machine and commit the results." ; exit 1 ; }
go mod verify
endif
$(SILENT)go mod download
$(SILENT)touch $@
.PHONY: clean-deps
clean-deps:
@echo "+ $@"
$(SILENT)rm -f deps
.PHONY: clean-obsolete-protos
clean-obsolete-protos:
@echo "+ $@"
$(BASE_DIR)/tools/clean_autogen_protos.py --protos $(BASE_DIR)/proto --generated $(BASE_DIR)/generated
###########
## Build ##
###########
HOST_OS:=linux
ifeq ($(UNAME_S),Darwin)
HOST_OS:=darwin
endif
.PHONY: build-prep
build-prep: deps
mkdir -p bin/{darwin_amd64,darwin_arm64,linux_amd64,linux_arm64,linux_ppc64le,linux_s390x,windows_amd64}
.PHONY: cli-build
cli-build: cli-linux cli-darwin cli-windows
.PHONY: cli-install
cli-install:
# Workaround a bug on MacOS
rm -f $(GOPATH)/bin/roxctl
# Copy the user's specific OS into gopath
mkdir -p $(GOPATH)/bin
cp bin/$(HOST_OS)_$(GOARCH)/roxctl $(GOPATH)/bin/roxctl
chmod u+w $(GOPATH)/bin/roxctl
.PHONY: cli
cli: cli-build cli-install
cli-linux: cli_linux-amd64 cli_linux-arm64 cli_linux-ppc64le cli_linux-s390x
cli-darwin: cli_darwin-amd64 cli_darwin-arm64
cli-windows: cli_windows-amd64
cli_%: build-prep
$(eval w := $(subst -, ,$*))
$(eval os := $(firstword $(w)))
$(eval arch := $(lastword $(w)))
ifdef SKIP_CLI_BUILD
test -f bin/$(os)_$(arch)/roxctl || RACE=0 CGO_ENABLED=0 GOOS=$(os) GOARCH=$(arch) $(GOBUILD) ./roxctl
else
RACE=0 CGO_ENABLED=0 GOOS=$(os) GOARCH=$(arch) $(GOBUILD) ./roxctl
endif
.PHONY: cli_host-arch
cli_host-arch: cli_$(HOST_OS)-$(GOARCH)
upgrader: bin/$(HOST_OS)_$(GOARCH)/upgrader
bin/$(HOST_OS)_$(GOARCH)/upgrader: build-prep
GOOS=$(HOST_OS) GOARCH=$(GOARCH) $(GOBUILD) ./sensor/upgrader
bin/$(HOST_OS)_$(GOARCH)/admission-control: build-prep
GOOS=$(HOST_OS) GOARCH=$(GOARCH) $(GOBUILD) ./sensor/admission-control
.PHONY: build-volumes
build-volumes:
$(SILENT)mkdir -p $(CURDIR)/linux-gocache
$(SILENT)docker volume inspect $(GOPATH_VOLUME_NAME) >/dev/null 2>&1 || docker volume create $(GOPATH_VOLUME_NAME)
$(SILENT)docker volume inspect $(GOCACHE_VOLUME_NAME) >/dev/null 2>&1 || docker volume create $(GOCACHE_VOLUME_NAME)
.PHONY: main-build
main-build: build-prep main-build-dockerized
@echo "+ $@"
.PHONY: sensor-build-dockerized
sensor-build-dockerized: build-volumes
@echo "+ $@"
docker run $(DOCKER_OPTS) --rm -e CI -e BUILD_TAG -e GOTAGS -e DEBUG_BUILD -e CGO_ENABLED $(GOPATH_WD_OVERRIDES) $(LOCAL_VOLUME_ARGS) $(BUILD_IMAGE) make sensor-build
.PHONY: sensor-kubernetes-build-dockerized
sensor-kubernetes-build-dockerized: build-volumes
@echo "+ $@"
docker run $(DOCKER_OPTS) -e CI -e BUILD_TAG -e GOTAGS -e DEBUG_BUILD -e CGO_ENABLED $(GOPATH_WD_OVERRIDES) $(LOCAL_VOLUME_ARGS) $(BUILD_IMAGE) make sensor-kubernetes-build
.PHONY: sensor-build
sensor-build:
$(GOBUILD) sensor/kubernetes sensor/admission-control
CGO_ENABLED=0 $(GOBUILD) sensor/upgrader
.PHONY: sensor-kubernetes-build
sensor-kubernetes-build:
$(GOBUILD) sensor/kubernetes
.PHONY: main-build-dockerized
main-build-dockerized: build-volumes
@echo "+ $@"
docker run $(DOCKER_OPTS) -i -e RACE -e CI -e BUILD_TAG -e SHORTCOMMIT -e GOTAGS -e DEBUG_BUILD -e CGO_ENABLED --rm $(GOPATH_WD_OVERRIDES) $(LOCAL_VOLUME_ARGS) $(BUILD_IMAGE) make main-build-nodeps
.PHONY: main-build-nodeps
main-build-nodeps: central-build-nodeps migrator-build-nodeps config-controller-build-nodeps
$(GOBUILD) sensor/kubernetes sensor/admission-control compliance/cmd/compliance
$(GOBUILD) sensor/upgrader
$(GOBUILD) sensor/init-tls-certs
ifndef CI
CGO_ENABLED=0 $(GOBUILD) roxctl
endif
.PHONY: scale-build
scale-build: build-prep
@echo "+ $@"
CGO_ENABLED=0 $(GOBUILD) scale/profiler scale/chaos
.PHONY: webhookserver-build
webhookserver-build: build-prep
@echo "+ $@"
CGO_ENABLED=0 $(GOBUILD) webhookserver
.PHONY: syslog-build
syslog-build:build-prep
@echo "+ $@"
CGO_ENABLED=0 $(GOBUILD) qa-tests-backend/test-images/syslog
.PHONY: gendocs
gendocs: $(GENERATED_API_DOCS)
@echo "+ $@"
# We don't need to do anything here, because the $(MERGED_API_SWAGGER_SPEC) and $(MERGED_API_SWAGGER_SPEC_V2) targets
# already perform validation.
.PHONY: swagger-docs
swagger-docs: $(MERGED_API_SWAGGER_SPEC) $(MERGED_API_SWAGGER_SPEC_V2)
@echo "+ $@"
UNIT_TEST_PACKAGES ?= ./...
.PHONY: test-prep
test-prep:
@echo "+ $@"
$(SILENT)mkdir -p test-output
.PHONY: go-unit-tests
go-unit-tests: build-prep test-prep
set -o pipefail ; \
CGO_ENABLED=1 GOEXPERIMENT=cgocheck2 MUTEX_WATCHDOG_TIMEOUT_SECS=30 GOTAGS=$(GOTAGS),test scripts/go-test.sh -timeout 15m -race -cover -coverprofile test-output/coverage.out -v \
$(shell git ls-files -- '*_test.go' | sed -e 's@^@./@g' | xargs -n 1 dirname | sort | uniq | xargs go list| grep -v '^github.com/stackrox/rox/tests$$' | grep -Ev $(UNIT_TEST_IGNORE)) \
| tee $(GO_TEST_OUTPUT_PATH)
# Exercise the logging package for all supported logging levels to make sure that initialization works properly
@echo "Run log tests"
for encoding in console json; do \
for level in debug info warn error fatal panic; do \
LOGENCODING=$$encoding LOGLEVEL=$$level CGO_ENABLED=1 GOEXPERIMENT=cgocheck2 MUTEX_WATCHDOG_TIMEOUT_SECS=30 GOTAGS=$(GOTAGS),test scripts/go-test.sh -p 4 -race -v ./pkg/logging/... | grep -v "iteration"; \
done; \
done
.PHONY: sensor-integration-test
sensor-integration-test: build-prep test-prep
set -eo pipefail ; \
rm -rf $(GO_TEST_OUTPUT_PATH); \
for package in $(shell git ls-files ./sensor/tests | grep '_test.go' | xargs -n 1 dirname | uniq | sort | sed -e 's/sensor\/tests\///'); do \
CGO_ENABLED=1 GOEXPERIMENT=cgocheck2 MUTEX_WATCHDOG_TIMEOUT_SECS=30 LOGLEVEL=debug GOTAGS=$(GOTAGS),test scripts/go-test.sh -p 4 -race -cover -coverprofile test-output/coverage.out -v ./sensor/tests/$$package \
| tee -a $(GO_TEST_OUTPUT_PATH); \
done \
sensor-pipeline-benchmark: build-prep test-prep
LOGLEVEL="panic" go test -bench=. -run=^# -benchtime=30s -count=5 ./sensor/tests/pipeline | tee $(CURDIR)/test-output/pipeline.results.txt
.PHONY: go-postgres-unit-tests
go-postgres-unit-tests: build-prep test-prep
@# The -p 1 passed to go test is required to ensure that tests of different packages are not run in parallel, so as to avoid conflicts when interacting with the DB.
set -o pipefail ; \
CGO_ENABLED=1 GOEXPERIMENT=cgocheck2 MUTEX_WATCHDOG_TIMEOUT_SECS=30 GOTAGS=$(GOTAGS),test,sql_integration scripts/go-test.sh -p 1 -race -cover -coverprofile test-output/coverage.out -v \
$(shell git grep -rl "//go:build sql_integration" central pkg migrator tools | sed -e 's@^@./@g' | xargs -n 1 dirname | sort | uniq | xargs go list -tags sql_integration | grep -v '^github.com/stackrox/rox/tests$$' | grep -Ev $(UNIT_TEST_IGNORE)) \
| tee $(GO_TEST_OUTPUT_PATH)
.PHONY: go-postgres-bench-tests
go-postgres-bench-tests: build-prep test-prep
@# The -p 1 passed to go test is required to ensure that tests of different packages are not run in parallel, so as to avoid conflicts when interacting with the DB.
set -o pipefail ; \
CGO_ENABLED=1 GOEXPERIMENT=cgocheck2 MUTEX_WATCHDOG_TIMEOUT_SECS=30 GOTAGS=$(GOTAGS),test,sql_integration scripts/go-test.sh -p 1 -run=nonthing -bench=. -benchtime=$(BENCHTIME) -benchmem -timeout $(BENCHTIMEOUT) -count $(BENCHCOUNT) -v \
$(shell git grep -rl "testing.B" central pkg migrator tools | sed -e 's@^@./@g' | xargs -n 1 dirname | sort | uniq | xargs go list -tags sql_integration | grep -v '^github.com/stackrox/rox/tests$$' | grep -Ev $(UNIT_TEST_IGNORE)) \
| tee $(GO_TEST_OUTPUT_PATH)
.PHONY: shell-unit-tests
shell-unit-tests:
@echo "+ $@"
$(SILENT)mkdir -p shell-test-output
bats --print-output-on-failure --verbose-run --recursive --report-formatter junit --output shell-test-output \
scripts \
tests/e2e/bats
.PHONY: ui-build
ui-build:
ifdef SKIP_UI_BUILD
test -d ui/build || make -C ui build
else
make -C ui build
endif
.PHONY: ui-test
ui-test:
make -C ui test
.PHONY: ui-component-tests
ui-component-tests:
make -C ui test-component
.PHONY: test
test: go-unit-tests ui-test shell-unit-tests
.PHONY: integration-unit-tests
integration-unit-tests: build-prep test-prep
set -o pipefail ; \
GOTAGS=$(GOTAGS),test,integration scripts/go-test.sh -count=1 -v \
$(shell go list ./... | grep "registries\|scanners\|notifiers") \
| tee $(GO_TEST_OUTPUT_PATH)
.PHONY: generate-junit-reports
generate-junit-reports: junit-reports/report.xml
$(GO_TEST_OUTPUT_PATH):
@echo "The test output log cannot be created via a direct Makefile rule. You must make the desired test targets"
@echo "first to ensure the file's existence."
@exit 1
junit-reports/report.xml: $(GO_TEST_OUTPUT_PATH) $(GO_JUNIT_REPORT_BIN)
@mkdir -p junit-reports
$(SILENT)$(GO_JUNIT_REPORT_BIN) <"$<" >"$@"
###########
## Image ##
###########
# image is an alias for main-image
.PHONY: image
image: main-image
.PHONY: all-builds
all-builds: cli main-build clean-image $(MERGED_API_SWAGGER_SPEC) $(MERGED_API_SWAGGER_SPEC_V2) ui-build
.PHONY: main-image
main-image: all-builds
make docker-build-main-image
.PHONY: docker-build-main-image
docker-build-main-image: copy-binaries-to-image-dir central-db-image
$(DOCKERBUILD) \
-t stackrox/main:$(TAG) \
-t $(DEFAULT_IMAGE_REGISTRY)/main:$(TAG) \
--build-arg DEBUG_BUILD="$(DEBUG_BUILD)" \
--build-arg ROX_PRODUCT_BRANDING=$(ROX_PRODUCT_BRANDING) \
--build-arg TARGET_ARCH=$(GOARCH) \
--build-arg ROX_IMAGE_FLAVOR=$(ROX_IMAGE_FLAVOR) \
--build-arg LABEL_VERSION=$(TAG) \
--build-arg LABEL_RELEASE=$(TAG) \
--build-arg QUAY_TAG_EXPIRATION=$(QUAY_TAG_EXPIRATION) \
$(CENTRAL_DB_DOCKER_ARGS) \
--file image/rhel/Dockerfile \
image/rhel
@echo "Built main image for RHEL with tag: $(TAG), image flavor: $(ROX_IMAGE_FLAVOR)"
@echo "You may wish to: export MAIN_IMAGE_TAG=$(TAG)"
.PHONY: docker-build-roxctl-image
docker-build-roxctl-image:
cp -f bin/linux_$(GOARCH)/roxctl image/roxctl/roxctl-linux
$(DOCKERBUILD) \
-t stackrox/roxctl:$(TAG) \
-t $(DEFAULT_IMAGE_REGISTRY)/roxctl:$(TAG) \
-f image/roxctl/Dockerfile \
--label quay.expires-after=$(QUAY_TAG_EXPIRATION) \
image/roxctl
.PHONY: copy-go-binaries-to-image-dir
copy-go-binaries-to-image-dir:
cp bin/linux_$(GOARCH)/central image/rhel/bin/central
cp bin/linux_$(GOARCH)/config-controller image/rhel/bin/config-controller
ifdef CI
cp bin/linux_amd64/roxctl image/rhel/bin/roxctl-linux-amd64
cp bin/linux_arm64/roxctl image/rhel/bin/roxctl-linux-arm64
cp bin/linux_ppc64le/roxctl image/rhel/bin/roxctl-linux-ppc64le
cp bin/linux_s390x/roxctl image/rhel/bin/roxctl-linux-s390x
cp bin/darwin_amd64/roxctl image/rhel/bin/roxctl-darwin-amd64
cp bin/darwin_arm64/roxctl image/rhel/bin/roxctl-darwin-arm64
cp bin/windows_amd64/roxctl.exe image/rhel/bin/roxctl-windows-amd64.exe
else
ifneq ($(HOST_OS),linux)
cp bin/linux_$(GOARCH)/roxctl image/rhel/bin/roxctl-linux-$(GOARCH)
endif
cp bin/$(HOST_OS)_amd64/roxctl image/rhel/bin/roxctl-$(HOST_OS)-amd64
endif
cp bin/linux_$(GOARCH)/migrator image/rhel/bin/migrator
cp bin/linux_$(GOARCH)/kubernetes image/rhel/bin/kubernetes-sensor
cp bin/linux_$(GOARCH)/init-tls-certs image/rhel/bin/init-tls-certs
cp bin/linux_$(GOARCH)/upgrader image/rhel/bin/sensor-upgrader
cp bin/linux_$(GOARCH)/admission-control image/rhel/bin/admission-control
cp bin/linux_$(GOARCH)/compliance image/rhel/bin/compliance
# Workaround to bug in lima: https://github.com/lima-vm/lima/issues/602
find image/rhel/bin -not -path "*/.*" -type f -exec chmod +x {} \;
.PHONY: copy-binaries-to-image-dir
copy-binaries-to-image-dir: copy-go-binaries-to-image-dir
cp -r ui/build image/rhel/ui/
ifdef CI
$(SILENT)[ -d image/rhel/THIRD_PARTY_NOTICES ] || { echo "image/rhel/THIRD_PARTY_NOTICES dir not found! It is required for CI-built images."; exit 1; }
else
$(SILENT)[ -f image/rhel/THIRD_PARTY_NOTICES ] || mkdir -p image/rhel/THIRD_PARTY_NOTICES
endif
$(SILENT)[ -d image/rhel/docs ] || { echo "Generated docs not found in image/rhel/docs. They are required for build."; exit 1; }
.PHONY: scale-image
scale-image: scale-build clean-image
cp bin/linux_$(GOARCH)/profiler scale/image/rhel/bin/profiler
cp bin/linux_$(GOARCH)/chaos scale/image/rhel/bin/chaos
chmod +w scale/image/rhel/bin/*
docker build \
-t stackrox/scale:$(TAG) \
-t quay.io/rhacs-eng/scale:$(TAG) \
-f scale/image/Dockerfile scale
webhookserver-image: webhookserver-build
-mkdir webhookserver/bin
cp bin/linux_$(GOARCH)/webhookserver webhookserver/bin/webhookserver
chmod +w webhookserver/bin/webhookserver
docker build \
-t stackrox/webhookserver:1.2 \
-t quay.io/rhacs-eng/webhookserver:1.2 \
-f webhookserver/Dockerfile webhookserver
syslog-image: syslog-build
-mkdir qa-tests-backend/test-images/syslog/bin
cp bin/linux_$(GOARCH)/syslog qa-tests-backend/test-images/syslog/bin/syslog
chmod +w qa-tests-backend/test-images/syslog/bin/syslog
docker build \
-t stackrox/qa:syslog_server_1_0 \
-t quay.io/rhacs-eng/qa:syslog_server_1_0 \
-f qa-tests-backend/test-images/syslog/Dockerfile qa-tests-backend/test-images/syslog
.PHONY: central-db-image
central-db-image:
$(DOCKERBUILD) \
-t stackrox/central-db:$(TAG) \
-t $(DEFAULT_IMAGE_REGISTRY)/central-db:$(TAG) \
$(CENTRAL_DB_DOCKER_ARGS) \
--file image/postgres/Dockerfile \
image/postgres
@echo "Built central-db image with tag $(TAG)"
###########
## Clean ##
###########
.PHONY: clean
clean: clean-image
@echo "+ $@"
.PHONY: clean-image
clean-image:
@echo "+ $@"
git clean -xf image/bin image/rhel/bin
git clean -xdf image/ui image/rhel/ui image/rhel/docs
rm -f $(CURDIR)/image/rhel/bundle.tar.gz $(CURDIR)/image/postgres/bundle.tar.gz
rm -rf $(CURDIR)/image/rhel/scripts
.PHONY: tag
tag:
@echo $(TAG)
.PHONY: shortcommit
shortcommit:
ifdef SHORTCOMMIT
@echo $(SHORTCOMMIT)
else
@git rev-parse --short HEAD
endif
.PHONY: image-flavor
image-flavor:
@echo $(ROX_IMAGE_FLAVOR)
.PHONY: default-image-registry
default-image-registry:
@echo $(DEFAULT_IMAGE_REGISTRY)
.PHONY: product-branding
product-branding:
@echo $(ROX_PRODUCT_BRANDING)
.PHONY: ossls-audit
ossls-audit: deps
ossls version
ossls audit
.PHONY: ossls-notice
ossls-notice: deps
ossls version
ossls audit --export image/rhel/THIRD_PARTY_NOTICES
.PHONY: collector-tag
collector-tag:
@echo "$$(cat COLLECTOR_VERSION)$(COLLECTOR_TAG_SUFFIX)"
.PHONY: scanner-tag
scanner-tag:
@echo "$$(cat SCANNER_VERSION)$(SCANNER_TAG_SUFFIX)"
.PHONY: clean-dev-tools
clean-dev-tools: gotools-clean
@echo "+ $@"
.PHONY: reinstall-dev-tools
reinstall-dev-tools: clean-dev-tools
@echo "+ $@"
$(SILENT)$(MAKE) install-dev-tools
.PHONY: install-dev-tools
install-dev-tools: gotools-all
@echo "+ $@"
.PHONY: roxvet
roxvet: skip-dirs := operator/pkg/clientset
roxvet: $(ROXVET_BIN)
@echo "+ $@"
@# TODO(ROX-7574): Add options to ignore specific files or paths in roxvet
$(SILENT)go list -e ./... \
| $(foreach d,$(skip-dirs),grep -v '$(d)' |) \
xargs -n 1000 go vet -vettool "$(ROXVET_BIN)" -donotcompareproto -gogoprotofunctions -tags "sql_integration test_e2e test race destructive integration scanner_db_integration compliance externalbackups"
$(SILENT)go list -e ./... \
| $(foreach d,$(skip-dirs),grep -v '$(d)' |) \
xargs -n 1000 go vet -vettool "$(ROXVET_BIN)"
##########
## Misc ##
##########
.PHONY: clean-offline-bundle
clean-offline-bundle:
$(SILENT)find scripts/offline-bundle -name '*.img' -delete -o -name '*.tgz' -delete -o -name 'bin' -type d -exec rm -r "{}" \;
.PHONY: offline-bundle
offline-bundle: clean-offline-bundle
$(SILENT)./scripts/offline-bundle/create.sh
.PHONY: check-debugger
check-debugger:
/usr/bin/env DEBUG_BUILD="$(DEBUG_BUILD)" BUILD_TAG="$(BUILD_TAG)" TAG="$(TAG)" ./scripts/check-debugger.sh
ifeq ($(DEBUG_BUILD),yes)
$(warning Warning: DEBUG_BUILD is enabled. Don not use this for production builds)
endif
.PHONY: policyutil
policyutil:
@echo "+ $@"
CGO_ENABLED=0 GOOS=$(HOST_OS) $(GOBUILD) ./tools/policyutil
go install ./tools/policyutil
.PHONY: mitre
mitre:
@echo "+ $@"
CGO_ENABLED=0 GOOS=$(HOST_OS) $(GOBUILD) ./tools/mitre
go install ./tools/mitre
.PHONY: bootstrap_migration
bootstrap_migration:
$(SILENT)if [[ "x${DESCRIPTION}" == "x" ]]; then echo "Please set a description for your migration in the DESCRIPTION environment variable"; else go run tools/generate-helpers/bootstrap-migration/main.go --root . --description "${DESCRIPTION}" ;fi
.PHONY: image-prefetcher-deploy-bin
image-prefetcher-deploy-bin: $(IMAGE_PREFETCHER_DEPLOY_BIN) ## download and install
.PHONY: print-image-prefetcher-deploy-bin
print-image-prefetcher-deploy-bin:
@echo $(IMAGE_PREFETCHER_DEPLOY_BIN)
.PHONY: prometheus-metric-parser
prometheus-metric-parser: $(PROMETHEUS_METRIC_PARSER_BIN)
@echo $(PROMETHEUS_METRIC_PARSER_BIN)