forked from posit-dev/shinylive
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Makefile
377 lines (302 loc) · 13.9 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
.PHONY: all dist \
packages \
update_packages_lock retrieve_packages update_pyodide_lock_json \
pyodide_js \
pyodide_packages_local \
create_typeshed_json \
copy_pyright \
submodules submodules-pull submodules-pull-shiny submodules-pull-htmltools \
buildjs watch serve \
packages \
quarto quartoserve \
clean-packages clean distclean \
test test-watch webr \
_shinylive
.DEFAULT_GOAL := help
SHINYLIVE_VERSION = $(shell node -p "require('./package.json').version")
PYODIDE_VERSION = 0.26.3
PYODIDE_DIST_FILENAME = pyodide-$(PYODIDE_VERSION).tar.bz2
DOWNLOAD_DIR = ./downloads
R_SHINY_VERSION = 1.9.1.8002
BUILD_DIR = ./build
PACKAGE_DIR = ./packages
DIST_DIR = ./dist
SITE_DIR = ./site
SHINYLIVE_DIR = ./_shinylive
# Read htmltools and shiny versions from the package code. It's done with grep
# because if we try to load the package and read shiny.__version__, it requires
# the packages to be loadable first, which isn't possible without their
# dependencies being installed first.
HTMLTOOLS_VERSION = $(shell grep '^__version__ = ' $(PACKAGE_DIR)/py-htmltools/htmltools/__init__.py | sed -E -e 's/^__version__ = "(.*)"/\1/')
SHINY_VERSION = $(shell grep '^__version__ = ' $(PACKAGE_DIR)/py-shiny/shiny/_version.py | sed -E "s/.*['\"]([^'\"]+)['\"]/\1/")
SHINYWIDGETS_VERSION = $(shell grep '^__version__ = ' $(PACKAGE_DIR)/py-shinywidgets/shinywidgets/__init__.py | sed -E -e 's/^__version__ = "(.*)"/\1/')
FAICONS_VERSION = $(shell grep '^__version__ = ' $(PACKAGE_DIR)/py-faicons/faicons/__init__.py | sed -E -e 's/^__version__ = "(.*)"/\1/')
HTMLTOOLS_WHEEL = htmltools-$(HTMLTOOLS_VERSION)-py3-none-any.whl
SHINY_WHEEL = shiny-$(SHINY_VERSION)-py3-none-any.whl
SHINYWIDGETS_WHEEL = shinywidgets-$(SHINYWIDGETS_VERSION)-py3-none-any.whl
FAICONS_WHEEL = faicons-$(FAICONS_VERSION)-py3-none-any.whl
# Hard code these versions for now
PLOTNINE_VERSION=0.0.0
PLOTNINE_WHEEL=plotnine-$(PLOTNINE_VERSION)-py3-none-any.whl
# libsass is built in gadenbuie/libsass-python
# NOTE: Update https://github.com/gadenbuie/libsass-python/blob/dev/.github/workflows/pyodide.yml
# Pyodide, Emscripten, or Python versions change here.
LIBSASS_WHEEL=libsass-0.23.0-cp312-abi3-pyodide_2024_0_wasm32.whl
VENV = venv
PYBIN = $(VENV)/bin
# Any targets that depend on $(VENV) or $(PYBIN) will cause the venv to be
# created. To use the ven, python scripts should run with the prefix $(PYBIN),
# as in `$(PYBIN)/pip`.
$(VENV):
python3 -m venv $(VENV)
$(PYBIN): $(VENV)
define PRINT_HELP_PYSCRIPT
import re, sys
prev_line_help = None
for line in sys.stdin:
if prev_line_help is None:
match = re.match(r"^## (.*)", line)
if match:
prev_line_help = match.groups()[0]
else:
prev_line_help = None
else:
match = re.match(r'^([a-zA-Z_-]+)', line)
if match:
target = match.groups()[0]
print("%-22s %s" % (target, prev_line_help))
target = None
prev_line_help = None
endef
export PRINT_HELP_PYSCRIPT
help:
@python3 -c "$$PRINT_HELP_PYSCRIPT" < $(MAKEFILE_LIST)
## Update git submodules to commits referenced in this repository
submodules:
git submodule init
git submodule update --depth=20
cd packages/py-shiny && git fetch --tags --unshallow
cd packages/plotnine && git fetch --tags --unshallow
## Pull latest changes in git submodules
submodules-pull:
git submodule update --recursive --remote
cd packages/py-shiny && git fetch --tags
cd packages/plotnine && git fetch --tags
submodules-pull-shiny:
git submodule update --remote packages/py-shiny
cd packages/py-shiny && git fetch --tags
submodules-pull-htmltools:
git submodule update --remote packages/py-htmltools
## Build everything _except_ the shinylive.tar.gz distribution file
all: node_modules \
$(BUILD_DIR)/shinylive/style-resets.css \
$(BUILD_DIR)/shinylive/pyodide \
$(BUILD_DIR)/shinylive/webr \
pyodide_js \
pyodide_packages_local \
update_packages_lock_local \
retrieve_packages \
update_pyodide_lock_json \
create_typeshed_json \
copy_pyright \
$(BUILD_DIR)/export_template/index.html \
$(BUILD_DIR)/export_template/edit/index.html \
_shinylive
## Build shinylive distribution .tar.gz file
dist: buildjs
mkdir -p $(DIST_DIR)
ln -s $(BUILD_DIR) shinylive-$(SHINYLIVE_VERSION)
tar -chzvf $(DIST_DIR)/shinylive-$(SHINYLIVE_VERSION).tar.gz shinylive-$(SHINYLIVE_VERSION)
rm shinylive-$(SHINYLIVE_VERSION)
## Install node modules
node_modules: package.json
npm ci
$(BUILD_DIR)/shinylive/style-resets.css: src/style-resets.css
mkdir -p $(BUILD_DIR)/shinylive
cp src/style-resets.css $(BUILD_DIR)/shinylive
$(DOWNLOAD_DIR)/$(PYODIDE_DIST_FILENAME):
mkdir -p $(DOWNLOAD_DIR)
cd $(DOWNLOAD_DIR) && \
curl --fail -L -O https://github.com/pyodide/pyodide/releases/download/$(PYODIDE_VERSION)/$(PYODIDE_DIST_FILENAME)
$(BUILD_DIR)/shinylive/pyodide: $(DOWNLOAD_DIR)/$(PYODIDE_DIST_FILENAME)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
tar --exclude "*test*.tar" --exclude "node_modules" \
-xvjf $(DOWNLOAD_DIR)/$(PYODIDE_DIST_FILENAME) \
-C $(BUILD_DIR)/shinylive
$(BUILD_DIR)/shinylive/webr: webr
webr:
mkdir -p $(BUILD_DIR)/shinylive/webr
cp -r node_modules/webr/dist/. $(BUILD_DIR)/shinylive/webr
curl --fail -L https://github.com/r-wasm/shiny/releases/download/v$(R_SHINY_VERSION)/library.data.gz -o $(BUILD_DIR)/shinylive/webr/library.data.gz
curl --fail -L https://github.com/r-wasm/shiny/releases/download/v$(R_SHINY_VERSION)/library.js.metadata -o $(BUILD_DIR)/shinylive/webr/library.js.metadata
# FIXME: GitHub Pages does not cache Partial Content downloads. Here, we reduce
# the damage by forcing entire file downloads with Emscripten's lazy filesystem.
# Potentially, we can add a switch to Emscripten to disable the mechanism.
sed -i.bak 's/if(!hasByteServing)//' $(BUILD_DIR)/shinylive/webr/R.bin.js
# Copy pyodide.js and .d.ts to src/pyodide/. This is a little weird in that in
# `make all`, it comes after downloading pyodide. In the future we may be able
# to use a pyodide node module, but the one currently on npm is a bit out of
# date.
pyodide_js:
cp $(BUILD_DIR)/shinylive/pyodide/pyodide.mjs src/pyodide/pyodide.js
cp $(BUILD_DIR)/shinylive/pyodide/pyodide.d.ts src/pyodide/
cp $(BUILD_DIR)/shinylive/pyodide/ffi.d.ts src/pyodide/
## Copy local package wheels to the pyodide directory
pyodide_packages_local: $(BUILD_DIR)/shinylive/pyodide/$(HTMLTOOLS_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(SHINY_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(SHINYWIDGETS_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(FAICONS_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(PLOTNINE_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(LIBSASS_WHEEL)
$(BUILD_DIR)/shinylive/pyodide/$(HTMLTOOLS_WHEEL): $(PACKAGE_DIR)/$(HTMLTOOLS_WHEEL)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
# Remove any old copies of htmltools
rm -f $(BUILD_DIR)/shinylive/pyodide/htmltools*.whl
cp $(PACKAGE_DIR)/$(HTMLTOOLS_WHEEL) $(BUILD_DIR)/shinylive/pyodide/$(HTMLTOOLS_WHEEL)
$(BUILD_DIR)/shinylive/pyodide/$(SHINY_WHEEL): $(PACKAGE_DIR)/$(SHINY_WHEEL)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
# Remove any old copies of shiny
rm -f $(BUILD_DIR)/shinylive/pyodide/shiny*.whl
cp $(PACKAGE_DIR)/$(SHINY_WHEEL) $(BUILD_DIR)/shinylive/pyodide/$(SHINY_WHEEL)
$(BUILD_DIR)/shinylive/pyodide/$(SHINYWIDGETS_WHEEL): $(PACKAGE_DIR)/$(SHINYWIDGETS_WHEEL)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
# Remove any old copies of shinywidgets
rm -f $(BUILD_DIR)/shinylive/pyodide/shinywidgets*.whl
cp $(PACKAGE_DIR)/$(SHINYWIDGETS_WHEEL) $(BUILD_DIR)/shinylive/pyodide/$(SHINYWIDGETS_WHEEL)
$(BUILD_DIR)/shinylive/pyodide/$(FAICONS_WHEEL): $(PACKAGE_DIR)/$(FAICONS_WHEEL)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
# Remove any old copies of faicons
rm -f $(BUILD_DIR)/shinylive/pyodide/faicons*.whl
cp $(PACKAGE_DIR)/$(FAICONS_WHEEL) $(BUILD_DIR)/shinylive/pyodide/$(FAICONS_WHEEL)
$(BUILD_DIR)/shinylive/pyodide/$(PLOTNINE_WHEEL): $(PACKAGE_DIR)/$(PLOTNINE_WHEEL)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
rm -f $(BUILD_DIR)/shinylive/pyodide/plotnine*.whl
cp $(PACKAGE_DIR)/$(PLOTNINE_WHEEL) $(BUILD_DIR)/shinylive/pyodide/$(PLOTNINE_WHEEL)
$(BUILD_DIR)/shinylive/pyodide/$(LIBSASS_WHEEL): $(PACKAGE_DIR)/$(LIBSASS_WHEEL)
mkdir -p $(BUILD_DIR)/shinylive/pyodide
rm -f $(BUILD_DIR)/shinylive/pyodide/libsass*.whl
cp $(PACKAGE_DIR)/$(LIBSASS_WHEEL) $(BUILD_DIR)/shinylive/pyodide/$(LIBSASS_WHEEL)
$(BUILD_DIR)/export_template/index.html: export_template/index.html
mkdir -p $(BUILD_DIR)/export_template
cp export_template/index.html $(BUILD_DIR)/export_template/index.html
$(BUILD_DIR)/export_template/edit/index.html: export_template/edit/index.html
mkdir -p $(BUILD_DIR)/export_template/edit
cp export_template/edit/index.html $(BUILD_DIR)/export_template/edit/index.html
## Build JS resources from src/ dir
buildjs:
node_modules/.bin/tsx scripts/build.ts
## Build JS resources for production (with minification)
buildjs-prod:
node_modules/.bin/tsx scripts/build.ts --prod
## Build JS resources and watch for changes
watch:
node_modules/.bin/tsx scripts/build.ts --watch
## Build JS resources, watch for changes, and serve site
serve:
node_modules/.bin/tsx scripts/build.ts --serve
## Build JS resources for production, watch for changes, and serve site
serve-prod:
node_modules/.bin/tsx scripts/build.ts --serve --prod
## Build JS resources with webR as the default engine
buildjs-r:
node_modules/.bin/tsx scripts/build.ts --r
## Build JS resources for production with webR as the default engine
buildjs-prod-r:
node_modules/.bin/tsx scripts/build.ts --prod --r
## Build JS resources for production and serve site with webR as the default engine
serve-prod-r:
node_modules/.bin/tsx scripts/build.ts --serve --prod --r
## Build JS resources and serve site with webR as the default engine
serve-r:
node_modules/.bin/tsx scripts/build.ts --serve --r
# Build the _shinylive directory for deployment of both R and Python sites
_shinylive:
$(MAKE) buildjs-prod
cp -Lr $(SITE_DIR)/. $(SHINYLIVE_DIR)/py
$(MAKE) buildjs-prod-r
cp -Lr $(SITE_DIR)/. $(SHINYLIVE_DIR)/r
# Build htmltools, shiny, and shinywidgets. This target must be run manually after
# updating the package submodules; it will not run automatically with `make all`
# because I'm not sure how to set up the dependencies reliably.
## Build htmltools, shiny, and shinywidgets wheels
packages: clean-packages \
package-htmltools \
package-shiny \
package-shinywidgets \
package-faicons \
package-plotnine
package-htmltools: $(PACKAGE_DIR)/$(HTMLTOOLS_WHEEL)
package-shiny: $(PACKAGE_DIR)/$(SHINY_WHEEL)
package-shinywidgets: $(PACKAGE_DIR)/$(SHINYWIDGETS_WHEEL)
package-faicons: $(PACKAGE_DIR)/$(FAICONS_WHEEL)
package-plotnine: $(PACKAGE_DIR)/$(PLOTNINE_WHEEL)
$(PACKAGE_DIR)/$(HTMLTOOLS_WHEEL): $(PYBIN) $(PACKAGE_DIR)/py-htmltools
# Remove any old copies of the package
rm -f $(PACKAGE_DIR)/htmltools*.whl
$(PYBIN)/pip install -e $(PACKAGE_DIR)/py-htmltools[dev]
. $(PYBIN)/activate && cd $(PACKAGE_DIR)/py-htmltools && make install && mv dist/*.whl ../
$(PACKAGE_DIR)/$(SHINY_WHEEL): $(PYBIN) $(PACKAGE_DIR)/py-shiny
# Remove any old copies of the package
rm -f $(PACKAGE_DIR)/shiny*.whl
. $(PYBIN)/activate && cd $(PACKAGE_DIR)/py-shiny && make install && mv dist/*.whl ../
$(PACKAGE_DIR)/$(SHINYWIDGETS_WHEEL): $(PYBIN) $(PACKAGE_DIR)/py-shinywidgets
# Remove any old copies of the package
rm -f $(PACKAGE_DIR)/shinywidgets*.whl
. $(PYBIN)/activate && cd $(PACKAGE_DIR)/py-shinywidgets && make install && mv dist/*.whl ../
$(PACKAGE_DIR)/$(FAICONS_WHEEL): $(PYBIN) $(PACKAGE_DIR)/py-faicons
# Remove any old copies of the package
rm -f $(PACKAGE_DIR)/faicons*.whl
. $(PYBIN)/activate && cd $(PACKAGE_DIR)/py-faicons && make install && mv dist/*.whl ../
$(PACKAGE_DIR)/$(PLOTNINE_WHEEL): $(PYBIN) $(PACKAGE_DIR)/plotnine
rm -f $(PACKAGE_DIR)/plotnine*.whl
$(PYBIN)/pip install -e $(PACKAGE_DIR)/plotnine[build]
. $(PYBIN)/activate && cd $(PACKAGE_DIR)/plotnine && make dist && mv dist/*.whl ../$(PLOTNINE_WHEEL)
$(PACKAGE_DIR)/$(LIBSASS_WHEEL): $(PYBIN) $(PACKAGE_DIR)/$(LIBSASS_WHEEL)
rm -f $(PACKAGE_DIR)/libsass*.whl
curl --fail -L https://pkg.garrickadenbuie.com/libsass-python/$(LIBSASS_WHEEL) -o $(PACKAGE_DIR)/$(LIBSASS_WHEEL)
## Update the shinylive_lock.json file, based on shinylive_requirements.json
update_packages_lock: $(PYBIN) $(BUILD_DIR)/shinylive/pyodide
$(PYBIN)/pip install -r requirements-dev.txt
. $(PYBIN)/activate && scripts/pyodide_packages.py generate_lockfile
## Update the shinylive_lock.json file, but with local packages only
update_packages_lock_local: $(PYBIN) $(BUILD_DIR)/shinylive/pyodide
$(PYBIN)/pip install -r requirements-dev.txt
. $(PYBIN)/activate && scripts/pyodide_packages.py update_lockfile_local
## Download packages in shinylive_lock.json from PyPI
retrieve_packages: $(PYBIN) $(BUILD_DIR)/shinylive/pyodide \
$(BUILD_DIR)/shinylive/pyodide/$(HTMLTOOLS_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(SHINY_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(SHINYWIDGETS_WHEEL) \
$(BUILD_DIR)/shinylive/pyodide/$(FAICONS_WHEEL)
$(PYBIN)/pip install -r requirements-dev.txt
mkdir -p $(BUILD_DIR)/shinylive/pyodide
. $(PYBIN)/activate && scripts/pyodide_packages.py retrieve_packages
## Update pyodide/pyodide-lock.json to include packages in shinylive_lock.json
update_pyodide_lock_json: $(PYBIN)
. $(PYBIN)/activate && scripts/pyodide_packages.py update_pyodide_lock_json
## Create the typeshed.json file which will be used by the shinylive type checker
create_typeshed_json: $(PYBIN)
. $(PYBIN)/activate && scripts/create_typeshed.py
## Copy src/pyright files to build directory
copy_pyright:
mkdir -p $(BUILD_DIR)/shinylive/pyright
cp -r src/pyright/* $(BUILD_DIR)/shinylive/pyright
## Build Quarto example site in quarto/
quarto:
cd quarto && quarto render
## Build Quarto example site and serve
quartoserve:
cd quarto && quarto preview --port 8080
## Remove built wheels from the packages/ directory
clean-packages:
rm -f $(PACKAGE_DIR)/*.whl
## Remove all build files
clean:
rm -rf $(PACKAGE_DIR)/*.whl $(BUILD_DIR) $(DIST_DIR) \
$(SHINYLIVE_DIR)/py $(SHINYLIVE_DIR)/r quarto/docs/ typings/
## Remove all build files, venv/, and downloads/
distclean: clean
rm -rf $(VENV) $(DOWNLOAD_DIR)
## Run tests
test:
npm run playwright test