The project publishes several static subparts to a single Cloudflare Pages project
named imgui-bundle, reachable at https://
| Path | Contents |
|---|---|
doc/ | Jupyter-book documentation (this book); |
/playground/ | Pyodide playground (Python examples in the browser) |
/min_pyodide_app/ | Minimal Pyodide template (single-file demo) |
/explorer/ | Dear ImGui Bundle Explorer (Emscripten build with OpenCV) |
/local_wheels/ | Shared imgui_bundle-*.whl, loaded by the pyodide subparts |
doc/assets/imgui_bundle_book.pdf | PDF export of the documentation |
A _headers file (pyodide_projects/cf_headers) scopes
Cross-Origin-Opener-Policy / Cross-Origin-Embedder-Policy to /explorer/*
only; site-wide COI would break the playground (which pulls Pyodide and
CodeMirror from CDNs that don’t set Cross-Origin-Resource-Policy). The same
file also sets Content-Encoding: gzip on /explorer/*.data because those
emscripten asset bundles (application/octet-stream) aren’t auto-compressed
by the CF edge; they are pre-gzipped by cf_stage.
How the deploy works¶
Cloudflare Pages replaces the whole site on every upload, so we maintain a
local composed staging directory (pyodide_projects/_cf_staging/, gitignored)
that holds the union of all subparts. just cf_stage refreshes the staging
tree; just cf_deploy uploads it via wrangler.
The trade-off: on every deploy the other subparts must already be present
(built) on disk. cf_stage does not build them — it only copies:
Pyodide wheel:
just pyodide_buildmust have run (output inpyodide_projects/projects/local_wheels/).Ibex binaries:
just ibex_buildmust have run (output inbuild_ibex_ems/bin/).Jupyter-book docs:
just doc_build_cfmust have run (output indocs/book/_build/html/).
One-time setup¶
Install wrangler (needs Node.js 18+):
npm install -g wranglerCreate an API token at https://
dash .cloudflare .com /profile /api -tokens using the Edit Cloudflare Workers template (it covers Pages too). Export credentials in your shell (or your shell rc):
export CLOUDFLARE_API_TOKEN=... export CLOUDFLARE_ACCOUNT_ID=...Verify:
wrangler whoamishould print the account.
Local workflow¶
# Build every subpart (slow: ibex with OpenCV takes 30-60 min).
# `cf_stage_prepare` runs: pyodide_setup_local_build + pyodide_build
# + ibex_build + doc_build_cf
just cf_stage_prepare
# Stage everything into _cf_staging, then deploy
just cf_stage
just cf_deploy
# Test the composed site locally before deploying
just cf_serve_local # http://localhost:8765/
# → COOP/COEP headers are applied only to /explorer/* (mirrors prod)
# → Content-Encoding: gzip is applied to /explorer/*.data
# Inspect the staging tree (gitignored)
ls pyodide_projects/_cf_staging/
# Nuke staging (next deploy re-uses on-disk build outputs)
just cf_cleanCI workflow¶
.github/workflows/cf_pages_deploy.yml mirrors the local flow and is
triggered manually from the Actions tab (workflow_dispatch). It runs:
just pyodide_setup_local_build+just pyodide_build(builds the wheel)just ibex_build(builds Emscripten binaries; slowest step, typically 30–60 min becauseIMMVISION_FETCH_OPENCV=ONrebuilds OpenCV for wasm)just doc_build_cf(jupyter-book HTML + PDF, needs Typst)just cf_stage(composes the staging tree)npm install -g wrangler+just cf_deploy(uploads)
Required GitHub Actions secrets: CLOUDFLARE_API_TOKEN,
CLOUDFLARE_ACCOUNT_ID (same values used locally).
Pyodide wheel: deploy URL and filename references¶
The Pyodide wheel built by just pyodide_build is rsync’d into
_cf_staging/local_wheels/ by cf_stage and ends up at:
https://imgui-bundle.pages.dev/local_wheels/imgui_bundle-<VERSION>-cp313-cp313-pyemscripten_2025_0_wasm32.whlThis URL points at the latest Pages deploy, not a versioned archive.
Cloudflare Pages replaces the entire site on every upload, so previous
wheels disappear when a new version is deployed. We do not promise
stability for this URL: it is convenient for quick experiments, but
docs / demos that ship to end users tell readers to download the wheel
and self-host it (GitHub release assets work as a download source even
though they cannot be passed to micropip.install directly — no CORS).
When to update wheel filename references¶
Every time pyproject.toml / CMakeLists.txt version changes (or the wheel
platform tag changes per the runbook in
ci_scripts/pyodide_local_build/config_versions_pyodide.sh), the hardcoded
wheel filenames in HTML/JS/doc pages must be updated. Find them with:
rg "imgui_bundle.*\.whl" --glob '!external' --glob '!builds' --glob '!dist' --glob '!local_wheels' --glob '!.pyodide_build'Glob-only references (*pyemscripten*.whl in justfile and the GitHub
workflows) do not need updating on a version bump — only on a tag rename.
Related¶
Local traineq deploy (kept as fallback):
just pyodide_deploy_imgui_bundle_online,just ibex_deploy