forked from newrelic/newrelic-php-agent
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Makefile
528 lines (438 loc) · 16.2 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
#
# Copyright 2020 New Relic Corporation. All rights reserved.
# SPDX-License-Identifier: Apache-2.0
#
#
# The top level Makefile
#
GCOV ?= gcov
GO ?= go
SHELL = /bin/bash
GCOVR ?= gcovr
GIT ?= git
include make/config.mk
include make/vendor.mk
include make/version.mk
include make/php_versions.mk
# Include the secrets file if it exists, but if it doesn't, that's OK too.
-include make/secrets.mk
# Configure an isolated workspace for the Go daemon.
export GOPATH=$(CURDIR)
export GO15VENDOREXPERIMENT=1
# Needed for Go > 1.11 to avoid building with modules by default
export GO111MODULE=auto
# GOBIN affects the behavior of go install, ensure it is unset.
unexport GOBIN
GOFLAGS := -ldflags '-X newrelic/version.Number=$(AGENT_VERSION) -X newrelic/version.Commit=$(GIT_COMMIT) -X newrelic/secrets.NewrelicCollectorHost=$(NEWRELIC_COLLECTOR_HOST) -X newrelic/secrets.NewrelicLicenseKey=$(NEWRELIC_LICENSE_KEY) -X newrelic/secrets.NewrelicCollectorKeys=$(NEWRELIC_COLLECTOR_KEYS) -X newrelic/secrets.NewrelicAccountId=$(ACCOUNT_supportability) -X newrelic/secrets.NewrelicAppId=$(APP_supportability)' $(GO_TAGS)
GCOVRFLAGS += -e "agent/tests/*" -e "axiom/tests/*" -e ".*\.h" -o
#
# Uniformly apply compiler and linker flags for the target architecture.
# This includes ensuring these flags are picked up by the agent's configure
# script. Otherwise, the agent build may fail if the agent's target
# architecture does not match axiom.
#
CFLAGS += $(MODEL_FLAGS)
LDFLAGS += $(MODEL_FLAGS)
ifeq (1,$(OPTIMIZE))
CFLAGS += -O3 -g1
LDFLAGS += -O3 -g1
else
CFLAGS += -O0 -g3 -DENABLE_TESTING_API
LDFLAGS += -O0 -g3
endif
ifeq (1,$(ENABLE_LTO))
CFLAGS += -flto
LDFLAGS += -flto
endif
ifeq (1,$(ENABLE_MULDEFS))
LDFLAGS += -Wl,--no-warn-search-mismatch -Wl,-z,muldefs
endif
#
# Sanitizers
#
# Support for sanitizers varies by compiler and platform. Generally, it's best
# to use these on Linux and with the latest version of Clang, but GCC 4.9 or
# newer works as well. Compiling with -O1 or -Og with full debug info is
# strongly recommended for usable stack traces. See the Clang user manual
# for more information and options.
#
# Common sanitizers: address, integer, memory, thread, undefined
#
ifneq (,$(SANITIZE))
CFLAGS += -fsanitize=$(SANITIZE) -fno-omit-frame-pointer
LDFLAGS += -fsanitize=$(SANITIZE)
endif
#
# When the silent flag (-s) is given, configure and libtool should
# also be quiet.
#
ifneq (,$(findstring s,$(MAKEFLAGS)))
SILENT := --silent
endif
#
# At this point the following variables should have their final values,
# and can be safely exported to recipe commands.
#
export AR CC CFLAGS CPPFLAGS LDFLAGS
.PHONY: all
all: agent daemon
#
# Print some of the build specific variables so they can be computed
# once in this file, but shared with our Jenkins build scripts. Values
# should be formatted in a shell compatible NAME=VALUE format so the
# build scripts can safely eval the output of this target. NOTE: the
# build scripts are sensitive to the variable names. If you need
# to print something that is not in this format, or should otherwise
# be ignored by the build scripts, prefix it with '#' so it will be
# treated as a comment.
#
.PHONY: echo_config
echo_config:
@echo '# Ignore me, Makefile example comment'
@echo VERSION=$(AGENT_VERSION)
@echo GIT_COMMIT=$(GIT_COMMIT)
#
# Print information where do the build dependencies come from
#
.PHONY: show-vendors
show-vendors:
@echo ""
@echo -----------------------------------------------------------------------
@echo "| Using pcre library from $(PCRE_PREFIX) (from $(origin PCRE_PREFIX))"
@echo -n "| Link to "; [ $(PCRE_STATIC) = yes ] && echo -n "static" || echo -n "shared"; echo " libpcre";
@echo "| Using protobuf-c library from $(PROTOBUF_C_PREFIX) (from $(origin PROTOBUF_C_PREFIX))"
@echo -----------------------------------------------------------------------
@echo ""
#
# Let's build an agent! Building an agent is a three step process: using phpize
# to build a configure script, using configure to build a Makefile, and then
# actually using that Makefile to build the agent extension.
#
PHPIZE := phpize
PHP_CONFIG := php-config
.PHONY: agent
agent: show-vendors agent/Makefile
$(MAKE) -C agent
agent/configure: agent/config.m4 agent/Makefile.frag
cd agent; $(PHPIZE) --clean && $(PHPIZE)
agent/Makefile: agent/configure | axiom
cd agent; ./configure $(SILENT) --enable-newrelic --with-axiom=$(realpath axiom) --with-php-config=$(PHP_CONFIG) --with-protobuf-c=$(PROTOBUF_C_PREFIX) --with-pcre=$(PCRE_PREFIX) --with-pcre-static=$(PCRE_STATIC)
#
# Installs the agent into the extension directory of the appropriate PHP
# installation. By default this is directory name formed by evaluating
# $(INSTALL_ROOT)$(EXTENSION_DIR). The former is empty by default, so to
# install the agent into the directory of your choice run the following.
#
# make agent-install EXTENSION_DIR=...
#
agent-install: agent
$(MAKE) -C agent install
.PHONY: agent-clean
agent-clean:
cd agent; [ -f Makefile ] && $(MAKE) clean; $(PHPIZE) --clean
#
# Agent tests. These just delegate to the agent Makefile: all the smarts are in
# agent/Makefile.frag.
#
.PHONY: agent-tests
agent-tests: agent/Makefile
$(MAKE) -C agent unit-tests
.PHONY: agent-check agent-run-tests
agent-check agent-run-tests: agent/Makefile
$(MAKE) -C agent run-unit-tests
.PHONY:
agent-valgrind: agent/Makefile
$(MAKE) -C agent valgrind
#
# Build the daemon and related utilities
#
#
# Minimum required version of Go is 1.5.
#
# This is defined as a rule that other rules that require Go can depend upon.
# We don't want to require Go for a general build primarily to make the PHP
# agent easier to use as a component within the C agent.
#
.PHONY: go-minimum-version
go-minimum-version:
@if $(GO) version | awk '/go1.[012345678][\. ]/ { exit 1 }'; then \
true; \
else \
echo -n 'Go 1.9 or newer required; found '; $(GO) version; false; \
fi
DAEMON_TARGETS := $(addprefix bin/,client daemon integration_runner stressor)
# Delete Go binaries before each build to force them to be re-linked. This
# ensures the version and commit variables are set correctly by the linker.
#
# The bin directory is also the target directory for the installer,
# which we need to be careful to leave in place. Therefore, the names of the
# Go binaries are made explicit. If we used a conventional `cmd` subdirectory
# for commands, we could use `go list` to determine the names.
.PHONY: daemon
daemon: go-minimum-version daemon-protobuf Makefile | bin/
@rm -rf $(DAEMON_TARGETS)
@$(GO) install $(GOFLAGS) ./...
# The -race flag enables the inegrated Go race detector. Output to stderr
.PHONY: daemon_race
daemon_race: go-minimum-version daemon-protobuf Makefile | bin/
@rm -rf $(DAEMON_TARGETS)
@$(GO) install -race $(GOFLAGS) ./...
.PHONY: daemon_test
daemon_test: go-minimum-version daemon-protobuf
@$(GO) test $(GOFLAGS) ./...
.PHONY: daemon_bench
daemon_bench: go-minimum-version daemon-protobuf
@$(GO) test $(GOFLAGS) -bench=. ./...
.PHONY: daemon_integration
daemon_integration: go-minimum-version daemon-protobuf
$(MAKE) INTEGRATION_TAGS=1 go-minimum-version
@$(GO) test $(GOFLAGS) ./...
DAEMON_COV_FILE = daemon_coverage.out
.PHONY: daemon_cover
daemon_cover: go-minimum-version daemon-protobuf
@rm -f $(DAEMON_COV_FILE)
@$(GO) test -coverprofile=$(DAEMON_COV_FILE) $(GOFLAGS) ./...
$(GO) tool cover -html=$(DAEMON_COV_FILE)
@rm -f $(DAEMON_COV_FILE)
# Note that this rule does not require the Go binary, and therefore doesn't
# depend on go-minimum-version.
.PHONY: daemon-clean
daemon-clean:
rm -f $(DAEMON_TARGETS)
rm -rf pkg/*
#
# Build the installer plus support utility. The support utility is run by
# the installer to provide some helpful operations missing from bash or
# missing from the $PATH.
#
installer: bin/newrelic-install bin/newrelic-iutil | bin/
bin/newrelic-install: agent/newrelic-install.sh Makefile VERSION | bin/
sed -e "/nrversion:=/s,UNSET,$(AGENT_VERSION)," $< > $@
chmod 755 $@
bin/newrelic-iutil: agent/install-util.c Makefile VERSION | bin/
$(CC) -DNR_VERSION="\"$(AGENT_VERSION)\"" $(CPPFLAGS) $(CFLAGS) -o $@ $< $(LDFLAGS)
#
# Directories required during builds.
#
bin/:
mkdir bin
#
# Build axiom and the axiom tests
#
.PHONY: axiom
axiom: vendor
$(MAKE) -C axiom
#
# TESTARGS is passed to every invocation of a test program.
# If you want to run each tests with 16 way thread parallelism, you would do:
# make TESTARGS=-j16 run_tests
#
# TESTARGS =
.PHONY: axiom-tests
axiom-tests: vendor
$(MAKE) -C axiom tests
.PHONY: axiom-check axiom-run-tests
axiom-check axiom-run-tests: vendor axiom/tests/cross_agent_tests
$(MAKE) -C axiom run_tests
.PHONY: axiom-valgrind
axiom-valgrind: vendor axiom/tests/cross_agent_tests
$(MAKE) -C axiom valgrind
.PHONY: tests
tests: agent-tests axiom-tests
.PHONY: check run_tests
check run_tests: agent-run-tests axiom-run-tests
.PHONY: valgrind
valgrind: agent-valgrind axiom-valgrind
axiom/tests/cross_agent_tests:
$(error Please run "git submodule update --init" to install the cross agent tests.)
.PHONY: axiom-clean
axiom-clean:
$(MAKE) -C axiom clean
#
# Go protocol buffers
#
# Run this target to compile the protocol buffers for Go. The Go protocol
# buffer plugin used for this is not vendored. It can be installed via
#
# go install google.golang.org/protobuf/cmd/protoc-gen-go
#
# Before running this script, make sure the location of protoc-gen-go is in
# your $PATH.
.PHONY: daemon-protobuf
daemon-protobuf: src/newrelic/infinite_tracing/com_newrelic_trace_v1/v1.pb.go
src/newrelic/infinite_tracing/com_newrelic_trace_v1/v1.pb.go: protocol/infinite_tracing/v1.proto
$(MAKE) vendor # Only build vendor stuff if v1.proto has changed. Otherwise
# this rule will be triggered every time the daemon is built.
$(VENDOR_PREFIX)/bin/protoc \
-I=./protocol/infinite_tracing \
--go_out="paths=source_relative,plugins=grpc:src/newrelic/infinite_tracing/com_newrelic_trace_v1" \
protocol/infinite_tracing/v1.proto
#
# Agent integration testing
#
.PHONY: integration
integration: Makefile daemon lasp-test-all integration-events-limits
for PHP in $(PHP_VERSION_LIST); do \
echo; echo "# PHP=$${PHP}"; \
env NRLAMP_PHP=$${PHP} bin/integration_runner $(INTEGRATION_ARGS) || exit 1; \
echo "# PHP=$${PHP}"; \
done
#
# Agent event limits integration testing
#
# Because of how the integration_runner connects to the collector (once before
# running any tests) we have to tell the runner the value the agent would
# have passed to it for each test. This means these tests are not testing
# the agent <-> daemon communcations of the agent's requested custom event limit
# via the daemon to the collector and daemon sending back the collectors harvest
# limit value.
#
.PHONY: test-events-limits
test-events-limits: custom-limits-tests invalid-limits-tests default-limits-tests
.PHONY: custom-limits-tests
custom-limits-tests:
@# create array with the collector response for each agent requested custom events max samples
@# currently based on fast harvest cycle being 5 seconds so ratio is 12:1
@declare -A custom_limits_tests; \
custom_limits_tests[240]=20; \
custom_limits_tests[7000]=583; \
custom_limits_tests[30000]=2500; \
custom_limits_tests[100000]=8333; \
for custom_max in "$${!custom_limits_tests[@]}"; do \
collector_limit=$${custom_limits_tests[$$custom_max]}; \
php_test_file="tests/event_limits/custom/test_custom_events_max_samples_stored_$${custom_max}_limit.php"; \
bin/integration_runner $(INTEGRATION_ARGS) \
-max_custom_events $${custom_max} \
$${php_test_file} || exit 1; \
done;
.PHONY: invalid-limits-tests
invalid-limits-tests:
@# test for invalid value (-1) and (1000000)
@# Should use default (30000) for -1 and max (100000) for 1000000
@bin/integration_runner $(INTEGRATION_ARGS) \
-max_custom_events 100000 \
tests/event_limits/custom/test_custom_events_max_samples_stored_invalid_toolarge_limit.php || exit 1;
@bin/integration_runner $(INTEGRATION_ARGS) \
-max_custom_events 30000 \
tests/event_limits/custom/test_custom_events_max_samples_stored_invalid_toosmall_limit.php || exit 1;
.PHONY: default-limits-tests
default-limits-tests:
@# also run a test where limit is set to 0
@# default value is used
@bin/integration_runner $(INTEGRATION_ARGS) \
-max_custom_events 0 \
tests/event_limits/custom/test_custom_events_max_samples_stored_0_limit.php || exit 1;
@# also run a test where no agent custom event limit is specified and verify
@# default value is used
@bin/integration_runner $(INTEGRATION_ARGS) \
-max_custom_events 30000 \
tests/event_limits/custom/test_custom_events_max_samples_stored_not_specified.php || exit 1;
#
# Code profiling
#
.PHONY: gcov
gcov: Makefile
cd agent; $(GCOV) $(GCOV_FLAGS) *.gcno
cd agent; $(GCOV) -o .libs $(GCOV_FLAGS) .libs/*.gcno
cd axiom; $(GCOV) $(GCOV_FLAGS) *.gcno
cd axiom/tests; $(GCOV) $(GCOV_FLAGS) *.gcno
# Reset code coverage line counts.
.PHONY: coverage-clean
coverage-clean:
find . -name \*.gcda | xargs rm -f
find . -name \*coverage-report*.html | xargs rm -f
find . -name \*coverage-report*.xml | xargs rm -f
# Creates a human readable coverage report, the name of the file will be coverage-report.html.
.PHONY: coverage-report-html
coverage-report-html:
$(GCOVR) -r . --html-details $(GCOVRFLAGS) coverage-report.html;
# Create and xml code coverage report. The xml report is intended for jenkins.
.PHONY: coverage-report-xml
coverage-report-xml:
$(GCOVR) -r . --xml $(GCOVRFLAGS) coverage-report.xml || true
# Compile all tests with the coverage flag and run them. Then create the html coverage report.
.PHONY: coverage
coverage:
$(MAKE) ENABLE_COVERAGE=1
.\/bin\/integration_runner
$(MAKE) ENABLE_COVERAGE=1 run_tests
$(MAKE) coverage-report-html
#
# Clean up
#
.PHONY: clean
clean: agent-clean axiom-clean daemon-clean package-clean coverage-clean vendor-clean
rm -rf releases
rm -f agent/newrelic.map agent/LicenseData/license_errors.txt
.PHONY: package-clean
package-clean:
rm -rf releases/debian releases/*.deb
rm -rf releases/redhat releases/*.rpm
rm -rf releases/newrelic-php5-*.tar.gz
#
# Testing the Language Agent Security Policy (LASP) feature.
#
.PHONY: lasp-test
lasp-test:
@if [ ! $(SUITE_LASP) ]; then echo "USAGE: make lasp-test SUITE_LASP=suite-most-secure"; exit 1; fi
@if [ "$(LICENSE_lasp_$(subst -,_,$(SUITE_LASP)))" = "" ] ; then echo "Missing license for $(SUITE_LASP)"; exit 1; fi
@if [ ! -d "tests/lasp/$(SUITE_LASP)" ]; then echo "No such suite in tests/lasp folder"; exit 1; fi
@for PHP in $(PHP_VERSION_LIST); do \
echo; echo "# PHP=$${PHP}"; \
NRLAMP_PHP=$${PHP} bin/integration_runner $(INTEGRATION_ARGS) -loglevel debug \
-license $(LICENSE_lasp_$(subst -,_,$(SUITE_LASP))) \
-security_token @tests/lasp/$(SUITE_LASP)/security-token.txt \
-supported_policies @tests/lasp/$(SUITE_LASP)/securityPolicyAgent.json tests/lasp/$(SUITE_LASP) || exit 1; \
echo "# PHP=$${PHP}"; \
done
.PHONY: lasp-test-all
lasp-test-all:
$(MAKE) lasp-test SUITE_LASP=suite-most-secure
$(MAKE) lasp-test SUITE_LASP=suite-least-secure
$(MAKE) lasp-test SUITE_LASP=suite-random-1
$(MAKE) lasp-test SUITE_LASP=suite-random-2
$(MAKE) lasp-test SUITE_LASP=suite-random-3
#
# Vendored libraries
#
export GIT
.PHONY: vendor vendor-clean
ifeq (0,$(HAVE_PROTOBUF_C))
vendor:
$(MAKE) -C vendor all
vendor-clean:
$(MAKE) -C vendor clean
else
vendor: ;
vendor-clean: ;
endif
#
# Extras
#
include make/release.mk
test-services-start:
docker compose pull $(SERVICES)
docker compose up --wait --build --remove-orphans -d $(SERVICES)
test-services-stop:
docker compose stop
#
# Docker Development Environment
#
dev-shell:
docker-compose up --build --remove-orphans -d
docker exec -it agent bash -c "sh files/set_path.sh ; bash"
dev-build:
docker-compose up --build --remove-orphans -d
docker exec -it agent bash -c "sh files/set_path.sh ; make -j4 all"
dev-unit-tests:
docker-compose up --build --remove-orphans -d
docker exec -it agent bash -c "sh files/set_path.sh ; make -j4 valgrind"
dev-integration-tests:
docker-compose up --build --remove-orphans -d
docker exec -it agent bash -c "sh files/set_path.sh ; ./bin/integration_runner -agent ./agent/.libs/newrelic.so"
dev-all:
docker-compose up --build --remove-orphans -d
docker exec -it agent bash -c "sh files/set_path.sh ; make -j4 all valgrind; ./bin/integration_runner -agent ./agent/.libs/newrelic.so"
dev-stop:
docker-compose stop
# vim: set noet ts=2 sw=2: