forked from internetstandards/Internet.nl-dashboard
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Makefile
286 lines (243 loc) · 13.7 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
SHELL=/bin/bash
# settings
app_name = dashboard
docker_image_name = internetstandards/dashboard
# configure virtualenv to be created in OS specific cache directory
ifeq ($(shell uname -s),Darwin)
# macOS cache location
CACHEDIR ?= ~/Library/Caches
else
# User customized cache location or Linux default
XDG_CACHE_HOME ?= ~/.cache
CACHEDIR ?= ${XDG_CACHE_HOME}
endif
VIRTUAL_ENV ?= ${CACHEDIR}/virtualenvs/$(notdir ${PWD})
$(info Virtualenv path: ${VIRTUAL_ENV})
# variables for environment
bin = ${VIRTUAL_ENV}/bin
env = env PATH=${bin}:$$PATH
# shortcuts for common used binaries
python = ${bin}/python
pip = ${bin}/pip
pip-compile = ${bin}/pip-compile
pip-sync = ${bin}/pip-sync
# application binary
app = ${bin}/${app_name}
$(shell test -z "$$PS1" || echo -e \nRun `make help` for available commands or use tab-completion.\n)
pysrcdirs = ${app_name}/
pysrc = $(shell find ${pysrcdirs} -name *.py)
shsrc = $(shell find * ! -path vendor\* -name *.sh)
.PHONY: test check setup run fix autofix clean mrproper test_integration requirements requirements-dev
# default action to run
all: check test setup
## Setup
# setup entire dev environment
setup: ${app} ## setup development environment and application
@test \! -z "$$PS1" || (echo -ne "Development environment is tested and ready."; \
if command -v dashboard &>/dev/null;then \
echo -e " Development shell is activated."; \
else \
echo -e "\n\nTo activate development shell run:"; \
echo -e "\n\t. ${VIRTUAL_ENV}/bin/activate$$([[ "$$SHELL" =~ "fish" ]] && echo .fish)\n"; \
echo -e "Or refer to Direnv in README.md for automatic activation."; \
fi)
# install application and all its (python) dependencies
${app}: ${VIRTUAL_ENV}/.requirements.installed | ${python}
${python} setup.py develop --no-deps
@touch $@
${VIRTUAL_ENV}/.requirements.installed: requirements.txt requirements-dev.txt | ${pip-sync}
${pip-sync} $^
@touch $@
# perform 'pip freeze' on first class requirements in .in files.
requirements.txt requirements-dev.txt: %.txt: %.in | ${pip-compile}
${pip-compile} ${pip_compile_args} --verbose --output-file $@ $<
update_requirements: pip_compile_args=--upgrade
update_requirements: _mark_outdated requirements.txt requirements-dev.txt _commit_update
_mark_outdated:
touch requirements*.in
# get latest sha for gitlab.com:icf/websecmap@master and update requirements
update_requirement_websecmap: _update_websecmap_sha requirements.txt _commit_update
_update_websecmap_sha:
sha=$(shell git ls-remote -q [email protected]:internet-cleanup-foundation/web-security-map.git master|cut -f1); \
if grep $$sha requirements.in >/dev/null; then echo -e "\nNo update for you, current sha for websecmap in requirements.in is the same as master on Gitlab.\n"; exit 1;fi; \
sed -E -i '' "s/websecmap@[a-zA-Z0-9]{40}/$$sha/" requirements.in
_commit_update: requirements.txt
git add requirements*.txt requirements*.in
git commit -m "Updated requirements."
## QA
test: .make.test ## run test suite
.make.test: ${pysrc} ${app}
# run testsuite
DJANGO_SETTINGS_MODULE=${app_name}.settings ${env} coverage run --include '${app_name}/*' \
-m pytest -k 'not integration and not system' ${testargs}
# generate coverage
${env} coverage report
# and pretty html
${env} coverage html
# ensure no model updates are commited without migrations
${env} ${app} makemigrations --check
@touch $@
check: .make.check.py .make.check.sh ## code quality checks
.make.check.py: ${pysrc} ${app}
# check code quality
${env} pylama ${pysrcdirs} --skip "**/migrations/*"
@touch $@
.make.check.sh: ${shsrc}
# shell script checks (if installed)
if command -v shellcheck &>/dev/null && ! test -z "${shsrc}";then ${env} shellcheck ${shsrc}; fi
@touch $@
autofix fix: .make.fix ## automatic fix of trivial code quality issues
.make.fix: ${pysrc} ${app}
# fix trivial pep8 style issues
${env} autopep8 -ri ${pysrcdirs}
# remove unused imports
${env} autoflake -ri --remove-all-unused-imports ${pysrcdirs}
# sort imports
${env} isort -rc ${pysrcdirs}
# do a check after autofixing to show remaining problems
${MAKE} check
@touch $@
## Running
run: ${app} ## run complete application stack (frontend, worker, broker)
# start server (this can take a while)
DEBUG=1 NETWORK_SUPPORTS_IPV6=1 ${env} ${app} devserver
run-frontend: ${app} ## only run frontend component
DEBUG=1 NETWORK_SUPPORTS_IPV6=1 ${env} ${app} runserver
app: ${app} ## perform arbitrary app commands
## For example: make app cmd=migrate
# make app cmd="loaddata development"
# make app cmd="help"
# make app cmd="report -y municipality"
# make app cmd="makemigrations"
# make app cmd="migrate"
DEBUG=1 NETWORK_SUPPORTS_IPV6=1 ${env} ${app} ${cmd}
run-worker: ${app} ## only run worker component
DEBUG=1 NETWORK_SUPPORTS_IPV6=1 ${env} ${app} celery worker -ldebug -Q storage,celery
run-broker: ## only run broker
docker run --rm --name=redis -p 6379:6379 redis
## Testing
test_integration: ${app} ## perform integration test suite
DB_NAME=test.sqlite3 ${env} pytest -v -k 'integration' ${testargs}
testcase: ${app}
# run specific testcase
# example: make test_testcase testargs=test_openstreetmaps
${env} DJANGO_SETTINGS_MODULE=${app_name}.settings DB_NAME=test.sqlite3 \
${env} pytest -k ${case}
test_datasets: ${app}
${env} /bin/sh -ec "find ${app_name}/ -path '*/fixtures/*.yaml' -print0 | \
xargs -0n1 basename -s .yaml | uniq | \
xargs -n1 ${app} test_dataset"
test_deterministic: | ${VIRTUAL_ENV}
${env} /bin/bash tools/compare_differences.sh HEAD HEAD tools/show_ratings.sh testdata
pull_image:
docker pull ${docker_image_name}
push_image:
docker push ${docker_image_name}
image: ## Create Docker image
docker build -t ${docker_image_name} .
## Housekeeping
clean: ## cleanup build artifacts, caches, databases, etc.
# remove python cache files
-find * -name __pycache__ -print0 | xargs -0 rm -rf
# remove state files
-rm -f .make.*
# remove test artifacts
-rm -rf .pytest_cache htmlcov/
# remove build artifacts
-rm -rf *.egg-info dist/ pip-wheel-metadata/
# remove runtime state files
-rm -rf *.sqlite3
clean_virtualenv: ## cleanup virtualenv and installed app/dependencies
# remove virtualenv
-rm -fr ${VIRTUAL_ENV}/
# ....
# ,/%%%(//**/(%%%%(,.
# .,*//(%%/. .,(%%(///*,.
# .*/*,...,%(. ..,,,,%%*.,*//(/*.
# .**,.......*%. .........,,******%( ,*/(/*,
# .**,...... *%. ..,,,,,,,,****/**///**%/ .*/(/*.
# .**.... %* .,,,,,,,,,,,,,**//(((((*(%. */(//.
# .**... .%. . .,,*,*******,. ,(//%* */(/*.
# ,*,.. ,% .,,. ,*,,*, .... *(/%/ ./((/,
# .*.. *( /%(%%%%(,******,/%%%(//,.//%/ *((/.
# ,*. ,(%% *%%%/%%((%%%/*(%%/%%(/(/////%%(* ./(/,
# ,*. .,%/** ((*///((***/,*,,**((/**,,****,/%% . /((/,
# *, ...,,,/(/*( ,(*,...,,*/,,*,,,,,,,,,,..,///(%%...,...... ./((/.
# ,*. ...,,**////%*%/ .((*,,*//,*..,,,,*/*,.....*//%(%/,*,,,,,,,,,,...... *((/,
# *, ..,,,,***/////%%.. (%(%(/*%%%%%%%%/,,*(%/**//*/*%(****/*****,***,,,,,... ,/((*
# ./. ..,,**,****///(((%%. (%/(%%(////**,*,,,,/((%//***%(//////////**/*******,,.... ./(//.
# .*. ..,,******,,***//((((%* /(//(%(((%(((((//**,**//**/%*(((//((/((///////////**,,,,.../((/.
# ,/... ...***////**,..**//((((((%* ,(//%((%%/*,**/((*,,*,*///*/*%((((((((((((((((((((//*/***,,,/(((,
# ,/,.........,,///((((////*////(((((((%( ,%/(%%/*****,,,,,**,/((/(%%(((((((((((((((((((((((((/////***((((,
# .(,,,......,,*/(/((((((((((((((((((((((%/ /%/(%/,,...,,,**/%%%((%%(((((%%%%%%%%%%%(((((((((((((((////((((.
# .(*,,,,,,,,**//((((((%%%%%(((%%(%%%((((%%%,.*. ./(,%%(////(((%%%%%//%(%%%%%%%%%%%%%%%%%%%%%%%%(/((((((((((((((.
# (/**,,,*****/((((%%%%%%%%(/,/(%%%%%(**((%%, *%%%%(%%(((%%%%%%%%%*,,/%(%%%%%%%%%%%%%%%%%%%%%%%%(//(%%%%%%%((%%%(
# *(/*******//(((%%%%%%%%%%%%(((%%%%%%%((%%%%. .%%%%%%%%%%%%%%**//*,,,/%(%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*
# .((///**///(((%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% .%%%%%%%%%%%///((/,.,,,*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%(.
# *%////////((%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%, (%%%%%%%%%/////*,..,,,,/%/%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*
# (%(((((((((%%%%%%%%%%%%%%%%%%%%%%%%%%%%/,.,*. /%%%%%%%(******,,.,,,,/%%* */%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%(
# .%%(((((((%%%%%%%%%%%%%%%%%%%%%%%(/,. ,. .,////%%(/*,,,,,,,*/%%(, ,* .,/%%%%%%%%%%%%%%%%%%%%(((%%%%%%%%.
# *%%%%((%%%%%%%%%%%%%%%%%%(/,. .*/(%%%%%%%%(*. .. ....,,*/(%%%%%%%%((//(((%%%%%%*
# /%%%%%%%%%%%%%%%%%%*. .... ,%%%%%(/,.*((%%%%%%(
# (%%%%%%%%%%%%%%%%.. ..... (%%%%(((((%%%%%%%(
# (%%%%%%%%%%%%%%..,. ..,. .%%%%%%%%%%%%%%%(
# (%%%%%%%%%%%%% .,, .,**//(/,.. ..,...%%%%%%%%%%%%%%(
# (%%%%%%%%%%%% .,,./%*..,,,,,,,,*/%%. .. .,,,..%%%&&&&&%%%%%(
# *%%%%%%%%%%% ..(%,..,...,,,,,,,,,,*%(. *,,,. .,,..%%%&&&&%%%%%*
# .%%%%%%%%%% .(%*.. ........,,*,,,,*(%/ .,*/(%%%%%(/,.. .(,**, .,*,.,%%%&&%%%%%%.
# %%%%%%%%%. .,,(%%%%%%%(%%(/*,,,*,,,,,%%. .,*/(%%%%(*,,,,***,,,,,***//(%%/. (*(%%%%%((((%%%%,.,**,.,%%%&%%%%%%
# /%%%%%%%* ./%%(, /(((/*,,,*/((%%%%%((///*,,.. .,,******,,,,,,,**,,,,,,,*/%%%%((/***,,.,,,,(%%(**,.(%%%%%%%%/
# .%%%%%%% *%%*/, ./%%%/*. ..,,*****,,***,********,,****,,,**,,,,,/(//**,...,,,,,*%%%*.,%%%%%%%%.
# /%%%%%/*%( .*. .,....,,,,***,. .,,,***,,,,,,,,,,,,*****,,,**/(%%%(//*****/((/***(/*,,...,,,,,,,*/%%,(%%%%%%/
# %%%%%%%, .,***,,,,,,**,,***,,,,,,,,,,,********,,,,,**/%%%((%%%%%%%%%%%%%%%%((///****,**,,,,,,***/%%%%%%%%
# .%%%%%* ,/(/*,,,,,,,,,,,,***,,,,,*********,,,,,,,**/%%%((%%%%%%%%%%%%%%/*,,,,.......,,,,,,,,,,,,,****%%(%%%.
# ,%%%%/ ./%%%(/************,,**********,,,,,,,,,*/(%%%%(%%/*,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,..,,,,,**/%%%%%,
# *%%%(,(%%%%(/*******************,,,,,,,,,**/(%%%%%%%%/,******,,,,,,,,,,,,,,,,,,,,,,,,,,****,,,,,,*,,/%%%%%*
# *%%((%%%%%%(//***********,,,,,,,,,,**/((%%%%%%%%%(***************************,,,,,,,,,,,**********,(%%%%*
# *%(%%%%%%%%%(//****,,,,,,,,,***/(%%%%%%%%%%%%%%(((((////******,,,,,*****************,,,,,,,,,,,,**%%%%*
# .%%%%%%%%%(/////*******////((%%%%%%%%%%%%%%%%%((%%%%%%%%%%%%%((//**,,,,,,,,,,****************,,,*/%%(,
# (%((%%%%%%%(((/////((((%%%%%%%%%(%%%(******/(%%%%%%(((%%%%%%%%%%%%%%((//***,,,,,,,,,,,,,,,,,,,,,(%(
# %%(%%%%%%%%%%%%%%%%%%%%%%%%(%%%(/****************/(%%%%%(((%%%%%%%%%%%%%%%%((((((//////**,,,,,,%%,
# .%%%%%%%%%%%%%%%%%%((((%%%%(*,**************************/(%%%%((%%%%%%%%%%%%%%%%%%%%%%%%%%%%%((%%.
# *%%%%((((((((%%%%%%/,,,,**,,********************************(%%%%(((%%%%%%%%%%%%%%%%%%%(((%%%%,
# ,%%%%%%%%%%%%%. .,*******************************************/%%%%%%%%%%%%%%%%((%%%%%/.
# ,%%%%%%%%%%%/ .,************************************************,, %%%%%%%%%%%%,
# *%%%%%%%%%%. .************************************************,, *%%%%%%%%%%*
# .(%%%%%%%( .,************************************************/%%%%%%%%%(.
# ,%%%%%%%/*********************************************(%%%%%%%%%%%,
# ,(%%%%%%%%(***********************************/(%%%%%%%%%%%(,
# *%%%%%%%%%%%((//***********,,,,**//(%%%%%%%%%%%%%%%%*
# ,(%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%(,
# ,/(%%%%%%%%%%%%%%%%%%%%%%%%%%%%%(/,
# ..,*//(((((((//*,..
mrproper: clean clean_virtualenv ## thorough cleanup, also removes virtualenv
## Base requirements
${pip-compile} ${pip-sync}: | ${pip}
${pip} install pip-tools
python: ${python}
${python} ${pip}:
@if ! command -v python3 &>/dev/null;then \
echo "Python 3 is not available. Please refer to installation instructions in README.md"; \
fi
# create virtualenv
python3 -mvenv ${VIRTUAL_ENV}
# ensure a recent version of pip is used to avoid errors with intalling
${VIRTUAL_ENV}/bin/pip install --upgrade pip==19.1.1
## Utility
help: ## Show this help.
@IFS=$$'\n' ; \
help_lines=(`fgrep -h "##" $(MAKEFILE_LIST) | fgrep -v fgrep | sed -e 's/\\$$//' | sed -e 's/##/:/'`); \
printf "\nRun \`make\` with any of the targets below to reach the desired target state.\n" ; \
printf "\nTargets are complementary. Eg: the \`run\` target requires \`setup\` which is automatically executed.\n\n" ; \
printf "%-30s %s\n" "target" "help" ; \
printf "%-30s %s\n" "------" "----" ; \
for help_line in $${help_lines[@]}; do \
IFS=$$':' ; \
help_split=($$help_line) ; \
help_command=`echo $${help_split[0]} | sed -e 's/^ *//' -e 's/ *$$//'` ; \
help_info=`echo $${help_split[2]} | sed -e 's/^ *//' -e 's/ *$$//'` ; \
printf '\033[36m'; \
printf "%-30s %s" $$help_command ; \
printf '\033[0m'; \
printf "%s\n" $$help_info; \
done