-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathMakefile
273 lines (244 loc) · 8.47 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
# prepare env file and tmp dir
$(shell mkdir -p .tmp/out && touch .env && git init -q)
# import custom makefiles
-include Makefile.project
# import and use env vars if exist
-include .env.project
include .env
export
# check required project vars
ifndef PROJECT_DISPLAY_NAME
PROJECT_DISPLAY_NAME := Example App
$(shell echo "\nPROJECT_DISPLAY_NAME=$(PROJECT_DISPLAY_NAME)" >> .env.project)
$(info Created `PROJECT_DISPLAY_NAME` variable in `.env.project` file!)
endif
ifndef PROJECT_VERSION
PROJECT_VERSION := 0.0.1
$(shell echo "\nPROJECT_VERSION=$(PROJECT_VERSION)" >> .env.project)
$(info Created `PROJECT_VERSION` variable in `.env.project` file!)
endif
-include .env.project
# project default vars
ifdef PROJECT_EXTRA_BUILD_ARGS
PROJECT_EXTRA_BUILD_ARGS +=
endif
.DEFAULT_GOAL = info
PROJECT_SHORT_NAME ?= $(PROJECT_DISPLAY_NAME)
PROJECT_SHORT_NAME := $(shell echo $(PROJECT_SHORT_NAME) | sed 's/ //g' | sed 's/-/ /g' | tr '[:upper:]' '[:lower:]')
PROJECT_COMMIT_SHORT := $(shell git rev-parse --is-inside-work-tree > /dev/null 2>&1 && git rev-parse --verify HEAD > /dev/null 2>&1 && (commit=$$(git rev-parse --short HEAD); status=$$(git status -s); if [ -n "$$status" ]; then echo $$commit-modified; else echo $$commit; fi) || echo "no-commit")
PROJECT_BUILD_ARGS ?= "$(PROJECT_EXTRA_BUILD_ARGS)-X main.Version=$(PROJECT_VERSION) -X main.Commit=$(PROJECT_COMMIT_SHORT) -X \"main.DisplayName=$(PROJECT_DISPLAY_NAME)\" -X main.ShortName=$(PROJECT_SHORT_NAME)"
PROJECT_BUILDALL_OS ?= linux darwin windows
PROJECT_BUILDALL_ARCH ?= arm amd64 386 arm64
# general default vars
PORT ?= 8080
HOST ?= 0.0.0.0
GOOS ?= $(shell go env GOOS)
GOARCH ?= $(shell go env GOARCH)
GOCACHE ?= $(shell if [ -d "$$(go env GOCACHE)" ]; then realpath "$$(go env GOCACHE)"; else mkdir -p .tmp/.cache/go-build && realpath ".tmp/.cache/go-build"; fi)
##@ These environment variables control various project configurations, including build, run, and deployment settings.
##@ They are loaded from the `.env.projects` file, which is overwritten the `.env` file variables.
##@
##@ Makefile vars
##@
##@ PROJECT_DISPLAY_NAME: projects full name,
##@: default adds 'Example App' to '.env.project'
##@ PROJECT_VERSION: semver like project version,
##@: default adds '0.0.1' to '.env.project'
##@ PROJECT_SHORT_NAME: short spaceless lowercase name,
##@: default is $PROJECT_DISPLAY_NAME
##@ PROJECT_COMMIT_SHORT: commit short hash,
##@: default is loaded via git
##@ PROJECT_BUILD_ARGS: build args,
##@: default is project base vars + $PROJECT_EXTRA_BUILD_ARGS
##@ PROJECT_EXTRA_BUILD_ARGS: additional build args,
##@: default is empty
##@ PROJECT_BUILDALL_OS: defines 'buildall' os targets
##@: default is 'linux darwin windows'
##@ PROJECT_BUILDALL_ARCH: defines 'buildall' arch targets
##@: default is 'arm amd64 386 arm64'
##@
##@ Docker vars
##@
##@ DOCKER_PORT: container bind port or range (80-84),
##@: default is $PORT else 8080
##@ DOCKER_HOST: container bind host,
##@: default is $HOST else 0.0.0.0
##@ DOCKER_ARGS: container arguments for app executions,
##@: default is $ARGS
##@
##@ General vars
##@
##@ PORT: main app bind port if needed,
##@: default is empty
##@ HOST: main app bind host if needed,
##@: default is empty
##@ ARGS: arguments for app executions,
##@: default is $ARGS
##@ GOOS: target build operating system,
##@: default build system os
##@ GOARCH: target build architecture,
##@: default build system arch
##@ GOCACHE: container go cache dir,
##@: default global go cache if is dir
##@: else creates .tmp/.cache/go-build
##@
##@ Misc commands
##@
.PHONY: help
help: ##@ prints a command help message
@make -s info
@grep -F -h "##@" $(MAKEFILE_LIST) | grep -F -v grep -F | sed -e 's/\\$$//' | awk 'BEGIN {FS = ":*[[:space:]]*##@[[:space:]]*"}; \
{ \
if($$2 == "") \
printf "\n"; \
else if($$0 ~ /^#/) { \
split($$2, arr, ": "); \
if (length(arr) > 1) \
printf " \033[34m%s\033[0m %s\n", arr[1], arr[2]; \
else \
printf "%s\n", $$2; \
} \
else if($$1 == "") \
printf "%-15s^- %s\n", "", $$2; \
else \
printf " \033[34m%-16s\033[0m %s\n", $$1, $$2; \
}'
@printf "\nUsage: make <command>\n"
.PHONY: info
info: ##@ prints a project info message
@echo "$(PROJECT_DISPLAY_NAME) version $(PROJECT_VERSION), build $(PROJECT_COMMIT_SHORT)"
.PHONY: vars
vars: ##@ prints some vars for debugging
@echo "\nProject\n"
@env |grep "^PROJECT_" || true
@echo "\nGo\n"
@env |grep "^GO" || true
@echo "\nDocker\n"
@env |grep "^DOCKER_" || true
@echo "\nGeneral\n"
@echo "ARGS: '$(ARGS)'"
@echo "PORT: '$(PORT)'"
@echo "HOST: '$(HOST)'"
.PHONY: env
env: ##@ prints env vars for debugging
@env
.PHONY: clean
clean: ##@ cleans up generated files and docker cache
@rm -fr .tmp bin
@if command -v go 2>&1 >/dev/null; then \
echo "cleanup go..."; \
go clean; \
go clean -cache -fuzzcache; \
fi
@if command -v docker 2>&1 >/dev/null; then \
echo "cleanup docker containers and images..."; \
docker rm -f dev-local-$(PROJECT_SHORT_NAME)-bash > /dev/null 2>&1 \
docker rm -f dev-local-$(PROJECT_SHORT_NAME) > /dev/null 2>&1 \
docker image prune -f; \
fi
@echo "cleanup done!"
##@
##@ Go commands
##@
.PHONY: run
run: ##@ runs the main.go file using go run
@go run main.go $(ARGS)
.PHONY: build
build: ##@ uses go to build the app with build args
@touch .env
go build \
-ldflags=$(PROJECT_BUILD_ARGS) \
-o bin
@chmod +x bin
.PHONY: buildall
buildall: ##@ cross-compilation for all GOOS/GOARCH combinations
@echo "Prepare..."
@echo "Selected operating systems: $(PROJECT_BUILDALL_OS)"
@echo "Selected architectures: $(PROJECT_BUILDALL_ARCH)"
@set -- $(PROJECT_BUILDALL_OS) && \
OS_ARRAY=$$@ && \
set -- $(PROJECT_BUILDALL_ARCH) && \
ARCH_ARRAY=$$@ && \
set -- $$(go tool dist list | tr '\n' ' ') && \
TARGET_ARRAY=$$@ && \
FILTERED_TARGETS="" && \
for target in $$TARGET_ARRAY; do \
target_os=$$(echo $$target | cut -d '/' -f1); \
target_arch=$$(echo $$target | cut -d '/' -f2); \
if [ -z "$$PROJECT_BUILDALL_OS" ] || echo "$$OS_ARRAY" | grep -qw "$$target_os"; then \
if [ -z "$$PROJECT_BUILDALL_ARCH" ] || echo "$$ARCH_ARRAY" | grep -qw "$$target_arch"; then \
FILTERED_TARGETS="$$FILTERED_TARGETS $$target"; \
fi; \
fi; \
done && \
FILTERED_TARGETS=$${FILTERED_TARGETS#?} && \
if [ -z "$$FILTERED_TARGETS" ]; then \
echo "Error: No matching targets found for the selected OS and architectures"; \
echo "- os: $$PROJECT_BUILDALL_OS"; \
echo "- arch: $$PROJECT_BUILDALL_ARCH"; \
echo "- targets: $$TARGET_ARRAY"; \
exit 1; \
fi && \
echo "\nBuild for targets:\n$$FILTERED_TARGETS" && \
rm -rf .tmp/out-bak && \
mv .tmp/out .tmp/out-bak || true && \
echo "\nRun test build-system build..." && \
make -s build || { echo "Test system-build build failed!"; exit 1; } && \
echo "Start build processes..." && \
for target in $$FILTERED_TARGETS; do \
GOOS=$$(echo $$target | cut -d'/' -f1) && \
GOARCH=$$(echo $$target | cut -d'/' -f2) && \
( \
go build \
-ldflags=$(PROJECT_BUILD_ARGS) \
-o .tmp/out/$(PROJECT_SHORT_NAME)-$$GOOS-$$GOARCH && \
chmod +x .tmp/out/$(PROJECT_SHORT_NAME)-$$GOOS-$$GOARCH \
) && echo "- $$GOOS/$$GOARCH build!" || { echo "Build failed for $$GOOS/$$GOARCH!"; exit 1; } & \
done && \
wait
@echo "All build processes finished."
.PHONY: gi
gi: ##@ installs the build binary globally
@sudo cp bin /usr/local/bin/$(PROJECT_SHORT_NAME)
.PHONY: gu
gu: ##@ uninstalls the build binary globally
@sudo rm -f /usr/local/bin/$(PROJECT_SHORT_NAME)
.PHONY: up
up: ##@ updates dependencies recursively using go get
@echo "Update go deps recursively..."
go get -t -u ./...
.PHONY: i
i: ##@ makes the go.mod matches the source code in the module
@echo "Install go deps recursively..."
go mod tidy
.PHONY: test
test: ##@ runs all GO tests recursively without coverage
@echo "Run go tests recursively..."
go test ./...
.PHONY: cover
cover: ##@ generates a raw and html test coverage report
@echo "Run go tests recursively..."
go test -coverprofile .tmp/cover.out ./...
go tool cover -html=.tmp/cover.out -o .tmp/cover.html
@echo "cover.out and cover.html generated in .tmp!"
.PHONY: init
init: ##@ infos, deps install, test and build
@make -s info
@make -s i
@make -s test
@make -s build
.PHONY: dev
dev: ##@ runs the app in watch mode
@echo "Install air if needed and run it..."
@go install github.com/air-verse/air@v1
@air
##@
##@ Docker targets
##@
.PHONY: docker
docker: ##@ runs a shell in the container
@docker rm -f dev-$(PROJECT_SHORT_NAME) || > /dev/null 2>&1
docker compose run -P --rm -it --build \
--name dev-$(PROJECT_SHORT_NAME) \
--entrypoint bash \
local