-
Notifications
You must be signed in to change notification settings - Fork 9
/
Makefile
324 lines (267 loc) · 11.4 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
# Makefile to configure and run Plone instance
##############################################################################
# SETUP MAKE
## Defensive settings for make: https://tech.davis-hansson.com/p/make/
SHELL:=bash
.ONESHELL:
# for Makefile debugging purposes add -x to the .SHELLFLAGS
.SHELLFLAGS:=-eu -o pipefail -O inherit_errexit -c
.SILENT:
.DELETE_ON_ERROR:
MAKEFLAGS+=--warn-undefined-variables
MAKEFLAGS+=--no-builtin-rules
# Colors
# OK=Green, warn=yellow, error=red
ifeq ($(TERM),)
# no colors if not in terminal
MARK_COLOR=
OK_COLOR=
WARN_COLOR=
ERROR_COLOR=
NO_COLOR=
else
MARK_COLOR=`tput setaf 6`
OK_COLOR=`tput setaf 2`
WARN_COLOR=`tput setaf 3`
ERROR_COLOR=`tput setaf 1`
NO_COLOR=`tput sgr0`
endif
##############################################################################
# SETTINGS AND VARIABLE
# adjust to your project needs
PROJECT_NAME=plone.formwidget.geolocation
IMAGE_NAME=${PROJECT_NAME}
CONSTRAINTS_IN=constraints.txt
CONSTRAINTS_MXDEV=constraints-mxdev.txt
PIP_REQUIREMENTS_IN_FILE=requirements.txt
ADDONBASE=./
ADDONFOLDER=${ADDONBASE}
INSTANCE_YAML=instance.yaml
INSTANCE_FOLDER=instance
PIP_PARAMS= --pre
##############################################################################
# targets and prerequisites
# target has to be one file, otherwise step gets executes for each file separate
PREPARE_PREREQUISITES=${PIP_REQUIREMENTS_IN_FILE} ${CONSTRAINTS_IN} mx.ini ${ADDONBASE}setup.cfg
PREPARE_TARGET=requirements-mxdev.txt
INSTALL_PREREQUSISTES=${PREPARE_TARGET}
INSTALL_TARGET=.installed.txt
INSTANCE_PREREQUISITES=${INSTALL_TARGET} ${INSTANCE_YAML}
INSTANCE_TARGET=${INSTANCE_FOLDER}/etc/zope.ini ${INSTANCE_FOLDER}/etc/zope.conf ${INSTANCE_FOLDER}/etc/site.zcml
TEST_PREREQUISITES=${INSTALL_TARGET}
RUN_PREREQUISITES=${INSTANCE_TARGET}
##############################################################################
# CONVENIENCE
# install and run
.PHONY: all # full install, test and run
all:style test run
# Add the following 'help' target to your Makefile
# And add help text after each target name starting with '\#\#'
.PHONY: help
help: ## This help message
@echo "${OK_COLOR}This is the Makefile for ${WARN_COLOR}${PROJECT_NAME}${NO_COLOR}"
@echo
@echo "${WARN_COLOR}Additional parameters:${NO_COLOR}"
@echo "${MARK_COLOR}PYTHON${NO_COLOR}: python interpreter to be used (default: python3)"
@echo "${MARK_COLOR}VENV${NO_COLOR}: [on|off] whether to create a Python virtual environment or not (default: on)"
@echo "${MARK_COLOR}VENV_FOLDER${NO_COLOR}: location of the virtual environment (default: ./venv)"
@echo
@echo "${WARN_COLOR}Targets:${NO_COLOR}"
@grep -E '^[a-zA-Z_-]+:.*?## .*$$' $(MAKEFILE_LIST) | sort | awk 'BEGIN {FS = ":.*?## "}; {printf "\033[36m%-30s\033[0m %s\n", $$1, $$2}'
##############################################################################
# BASE
SENTINELFOLDER=.make-sentinels/
SENTINEL=${SENTINELFOLDER}ABOUT.txt
${SENTINEL}:
@mkdir -p ${SENTINELFOLDER}
@echo "Sentinels for the Makefile process." > ${SENTINEL}
# PYTHON, VENV, PIP
# venv and pybin
PYTHON?=python3
VENV?=on
ifeq ("${VENV}", "on")
VENV_FOLDER?=./venv
PYBIN=${VENV_FOLDER}/bin/
else
VENV_FOLDER?=
ifneq ("${VENV_FOLDER}", "")
PYBIN=${VENV_FOLDER}/bin/
PYTHON=${PYBIN}python
else
PYBIN=
endif
endif
# installed?
ifeq (, $(shell which $(PYTHON) ))
$(error "PYTHON=$(PYTHON) not found in $(PATH)")
endif
# version ok?
PYTHON_VERSION_MIN=3.7
PYTHON_VERSION_OK=$(shell $(PYTHON) -c 'import sys; print(int(sys.version_info[0:2] >= tuple(map(int, "$(PYTHON_VERSION_MIN)".split(".")))))' )
ifeq ($(PYTHON_VERSION_OK),0)
$(error "Need python $(PYTHON_VERSION) >= $(PYTHON_VERSION_MIN)")
endif
VENV_SENTINEL=${SENTINELFOLDER}venv.sentinel
${VENV_SENTINEL}: ${SENTINEL}
ifeq ("${VENV}", "on")
@echo "$(OK_COLOR)Setup Python Virtual Environment under '${VENV_FOLDER}' $(NO_COLOR)"
@${PYTHON} -m venv ${VENV_FOLDER}
else
@echo "$(OK_COLOR)Use current local or global Python: `which ${PYTHON}` $(NO_COLOR)"
endif
@touch ${VENV_SENTINEL}
PIP_SENTINEL=${SENTINELFOLDER}pip.sentinel
${PIP_SENTINEL}: ${VENV_SENTINEL} ${CONSTRAINTS_IN} ${SENTINEL}
@echo "$(OK_COLOR)Install pip $(NO_COLOR)"
@${PYBIN}pip install "pip>=22" wheel
@touch ${PIP_SENTINEL}
##############################################################################
# MXDEV
MXDEV_SENTINEL=${SENTINELFOLDER}pip-mxdev.sentinel
${MXDEV_SENTINEL}: ${PIP_SENTINEL}
@echo "$(OK_COLOR)Install mxdev$(NO_COLOR)"
@${PYBIN}pip install mxdev libvcs==0.11.1
@touch ${MXDEV_SENTINEL}
.PHONY: prepare
prepare: ${PREPARE_TARGET} ## prepare sources and dependencies
${PREPARE_PREREQUISITES}:
@touch $@
${PREPARE_TARGET}: ${MXDEV_SENTINEL} ${PREPARE_PREREQUISITES}
@echo "$(OK_COLOR)Prepare sources and dependencies$(NO_COLOR)"
@${PYBIN}mxdev -c mx.ini
.PHONY: install
install: ${INSTALL_TARGET} ## pip install all dependencies and scripts
${INSTALL_TARGET}: ${PREPARE_TARGET}
@echo "$(OK_COLOR)Install dependencies and scripts$(NO_COLOR)"
@${PYBIN}pip install -r ${PREPARE_TARGET} ${PIP_PARAMS}
@${PYBIN}pip freeze >${INSTALL_TARGET}
##############################################################################
# INSTANCE
COOKIECUTTER_SENTINEL=${SENTINELFOLDER}pip-cookiecutter.sentinel
${COOKIECUTTER_SENTINEL}:
@echo "$(OK_COLOR)Install cookiecutter$(NO_COLOR)"
@${PYBIN}pip install git+https://github.com/cookiecutter/cookiecutter.git#egg=cookiecutter
@touch ${COOKIECUTTER_SENTINEL}
${INSTANCE_YAML}:
@touch ${INSTANCE_YAML}
.PHONY: instance
instance: ${INSTANCE_TARGET} ## create configuration for an zope (plone) instance
${INSTANCE_TARGET}: ${INSTANCE_PREREQUISITES} ${COOKIECUTTER_SENTINEL}
@echo "$(OK_COLOR)Create Plone/Zope configuration$(NO_COLOR)"
@${PYBIN}cookiecutter -f --no-input --config-file ${INSTANCE_YAML} https://github.com/bluedynamics/cookiecutter-zope-instance
##############################################################################
# TESTING
TESTRUNNER_SENTINEL=${SENTINELFOLDER}pip-testrunner.sentinel
${TESTRUNNER_SENTINEL}: ${PIP_SENTINEL}
@echo "$(OK_COLOR)Install zope.testrunner$(NO_COLOR)"
@${PYBIN}pip install -c ${CONSTRAINTS_MXDEV} zope.testrunner
@touch ${TESTRUNNER_SENTINEL}
.PHONY: test
test: ${TEST_PREREQUISITES} ${TESTRUNNER_SENTINEL} ## run tests
@echo "$(OK_COLOR)Run addon tests$(NO_COLOR)"
@${PYBIN}zope-testrunner --auto-color --auto-progress --test-path=${ADDONFOLDER}
.PHONY: test-ignore-warnings
test-ignore-warnings: ${TEST_PREREQUISITES} ${TESTRUNNER_SENTINEL} ## run tests (hide warnings)
@echo "$(OK_COLOR)Run addon tests$(NO_COLOR)"
@PYTHONWARNINGS=ignore ${PYBIN}zope-testrunner --auto-color --auto-progress --test-path=${ADDONFOLDER}
##############################################################################
# CODE FORMATTING
BLACK_SENTINEL=${SENTINELFOLDER}pip-black.sentinel
${BLACK_SENTINEL}: ${PREPARE_TARGET}
@echo "$(OK_COLOR)Install black$(NO_COLOR)"
@${PYBIN}pip install -c ${CONSTRAINTS_MXDEV} black
@touch ${BLACK_SENTINEL}
ISORT_SENTINEL=${SENTINELFOLDER}pip-isort.sentinel
${ISORT_SENTINEL}: ${PREPARE_TARGET}
@echo "$(OK_COLOR)Install isort$(NO_COLOR)"
@${PYBIN}pip install -c ${CONSTRAINTS_MXDEV} isort
@touch ${ISORT_SENTINEL}
ZPRETTY_SENTINEL=${SENTINELFOLDER}pip-zpretty.sentinel
${ZPRETTY_SENTINEL}: ${PREPARE_TARGET}
@echo "$(OK_COLOR)Install zpretty$(NO_COLOR)"
@${PYBIN}pip install -c ${CONSTRAINTS_MXDEV} "zpretty>=2.2.0"
@touch ${ZPRETTY_SENTINEL}
.PHONY: apply-style-black
apply-style-black: ${BLACK_SENTINEL} ## apply/format code style black (to Python files)
@echo "$(OK_COLOR)Apply style black rules to code in ${ADDONFOLDER}/*$(NO_COLOR)"
@${PYBIN}black ${ADDONFOLDER}
.PHONY: apply-style-isort
apply-style-isort: ${ISORT_SENTINEL} ## apply/format code style isort (sorted imports in Python files)
@echo "$(OK_COLOR)Apply style isort rules to code in ${ADDONFOLDER}/*$(NO_COLOR)"
@${PYBIN}isort ${ADDONFOLDER}
.PHONY: apply-style-zpretty
apply-style-zpretty: ${ZPRETTY_SENTINEL} ## apply/format code style zpretty (to XML/ZCML files)
@echo "$(OK_COLOR)Apply style zpretty rules to code in ${ADDONFOLDER}/*$(NO_COLOR)"
@find ${ADDONFOLDER} -name '*.zcml' -exec ${PYBIN}zpretty -iz {} +
@find ${ADDONFOLDER} -name "*.xml"|grep -v locales|xargs ${PYBIN}zpretty -ix
.PHONY: style ## apply code styles black, isort and zpretty
style: apply-style-black apply-style-isort ## apply-style-zpretty
.PHONY: format ## alias for "style"
FORMATTING: style
.PHONY: lint-black
lint-black: ${BLACK_SENTINEL} ## lint code-style black (to Python files)
@echo "$(OK_COLOR)Lint black rules to code in ${ADDONFOLDER}/*$(NO_COLOR)"
@${PYBIN}black --check ${ADDONFOLDER}
.PHONY: lint-isort
lint-isort: ${ISORT_SENTINEL} ## lint code-style isort (sorted imports in Python files)
@echo "$(OK_COLOR)Apply style isort rules to code in ${ADDONFOLDER}/*$(NO_COLOR)"
@${PYBIN}isort --check-only ${ADDONFOLDER}
.PHONY: lint-zpretty
lint-zpretty: ${ZPRETTY_SENTINEL} ## lint code-style zpretty (to XML/ZCML files)
@echo "$(OK_COLOR)Apply style zpretty rules to code in ${ADDONFOLDER}/*$(NO_COLOR)"
@find ${ADDONFOLDER} -name '*.zcml' -exec ${PYBIN}zpretty --check -z {} +
@find ${ADDONFOLDER} -name '*.xml'|grep -v locales|xargs ${PYBIN}zpretty --check -x
.PHONY: lint ## lint all: check if complies with code-styles black, isort and zpretty
lint: lint-black lint-isort ## lint-zpretty
##############################################################################
# RUN
.PHONY: run
run: ${RUN_PREREQUISITES} ## run/start Plone
@echo "$(OK_COLOR)Run Plone$(NO_COLOR)"
@${PYBIN}runwsgi -v instance/etc/zope.ini
##############################################################################
# NODE
.PHONY: node
node: # yarn install
@echo "$(OK_COLOR)Yarn install$(NO_COLOR)"
yarn install
##############################################################################
# CLEAN
.PHONY: clean-venv
clean-venv: ## remove Python virtual environment
ifeq ("${VENV}", "on")
@echo "$(OK_COLOR)Remove Virtualenv.$(NO_COLOR)"
rm -rf ${VENV_FOLDER} ${SENTINELFOLDER}/pip*.sentinel ${VENV_SENTINEL}
else:
@echo "$(OK_WARN)No self-created Python virtualenv at '${VENV_FOLDER}'! Nothing to do.$(NO_COLOR)"
endif
.PHONY: clean-pyc
clean-pyc: ## remove Python file artifacts
@echo "$(OK_COLOR)Remove Python file artifacts (like byte-code) of code in current directory.$(NO_COLOR)"
find . -name '*.py[c|o]' -delete
find . -name '*.mo' -delete
find . -name '*~' -exec rm -f {} +
find . -name '__pycache__' -exec rm -fr {} +
.PHONY: clean-make
clean-make: ## remove make artifact @echo "$(OK_COLOR)Remove Plone/Zope configuration (keeps data) and sentinel files.$(NO_COLOR)"
rm -rf ${INSTALL_PREREQUSISTES} ${INSTANCE_TARGET} ${SENTINELFOLDER}
.PHONY: clean-instance
clean-instance: ## remove instance configuration (keeps data)
@echo "$(OK_COLOR)Remove Plone/Zope configuration (keeps data) and sentinel files.$(NO_COLOR)"
rm -f ${INSTANCE_TARGET}
.PHONY: clean-node
clean-node: ## remove instance configuration (keeps data)
@echo "$(OK_COLOR)Remove npm/yarn installation.$(NO_COLOR)"
rm -rf ${ADDONBASE}/node_modules
.PHONY: clean
clean: clean-venv clean-pyc clean-make clean-instance clean-node ## clean all (except local database and pip installed packages)
##############################################################################
# DOCKER/CONTAINER
# this needs a Dockerfile, which is not provided by plone-kickstarter
.PHONY: build-image
build-image: ## Build Docker Image
ifneq ("$(wildcard Dockerfile)", "")
@docker build . -t $(IMAGE_NAME) -f Dockerfile
else
@echo "$(ERROR_COLOR)A 'Dockerfile' is required to build an image.$(NO_COLOR)"
endif