Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Web] Feature Request: workaround for bundlers when using onnxruntime-web #22615

Closed
fs-eire opened this issue Oct 26, 2024 · 3 comments · Fixed by #23257
Closed

[Web] Feature Request: workaround for bundlers when using onnxruntime-web #22615

fs-eire opened this issue Oct 26, 2024 · 3 comments · Fixed by #23257
Labels
api:Javascript issues related to the Javascript API feature request request for unsupported feature or enhancement platform:web issues related to ONNX Runtime web; typically submitted using template

Comments

@fs-eire
Copy link
Contributor

fs-eire commented Oct 26, 2024

Summary

This issue is created to track the status of onnxruntime-web's workaround for bundlers.

What is workaround for bundlers and why?

workaround for bundlers means extra codes that required to make onnxruntime-web as a library consumable by a web app or other libraries, while those extra codes are supposed to be unnecessary but exists because of various reasons.

What are the known issues:

  1. import.meta.url as a runtime JavaScript expression, is usually rewritten into a static string starts with "file://" by WebPack as a default behavior. A few other bundlers followed this behvior. This means it is very difficult to use ESM dynamic import in the implementation.

  2. Vite development mode does not support "real" ESM. It uses some complicated steps to transcript the ESM sources into IIFE/cjs. [js/web] make ort-web export compatible with webpack #22196 tries to fix this but the fix is incompatible with Vite release mode.

  3. Some framework (eg. SvelteKit) always load onnxruntime-web in a Node.js context even if SSR is disabled . [Web] When the "node" condition path in exports is null vite (+svelte) build will wrongly throw a "No known conditions for" error  #22361 (comment)

  4. Emscripten is not able to generate both ESM/commonJS glue file in a build. ([Feature Request] Support building for both UMD/ESM emscripten-core/emscripten#21899)

Why is it difficult to fix:

It's all about dynamic loading with bundler involved.

onnxruntime-web has 4 dynamic loading target:

  • onnxruntime-web itself as an entry of WebWorker, used in proxy worker feature
  • The WebAssembly file (.wasm suffix, generated by Emscripten)
  • The MJS file (.mjs suffix, generated by Emscripten as JavaScript glue)
  • The same MJS file above as an entry of WebWorker, used in WebAssembly Multi-threading

Considering the following requirements:

  • support ESM
  • support CSP
  • support non-bundler usage. (eg. import ort.min.js from CDN in HTML script tag)
  • support major bundlers/frameworks. ("support" means user should be able to use onnxruntime-web as a dependency of their web app when using the bundlers, with no or little extra bundler config)

It's not hard to conclude that:

  • the number of combinations of usage variants is large.
  • in some usage, certain requirements may be conflicting with others.

For example:

  • UMD format cannot be supported in CSP. This is because:

    • CSP does not allow any dynamic loading. So in order to support CSP, onnxruntime-web need to offer a way to export an "all-in-one" bundle JS that does not dynamic import anything.
    • A UMD export of onnxruntime-web need to import an ESM JavaScript (because of known issue No.4), so dynamic import is required for UMD.
  • proxy worker is not well supported when using with multi-threading. This is because:

    • proxy worker is loaded using new Worker(new URL("ort.min.mjs", import.meta.url), {type: "module"}), and webpack will use its builtin worker loader to load the worker as a static loaded module.
    • when the glue JS called from generated proxy worker module tries to create another worker, it failed to load. (detailed reason is still investigating. It's very likely because of WebPack still does not fully support ESM)

There are more case that makes onnxruntime-web not working out-of-box in different combinations of dev frameworks, bundlers, dev/prod modes and so on, and one fix for a specific usage may even break a specific existing working usage.

It seems that there is no way to make every use scenario work. Need to track issues and find workaround for each.

@fs-eire fs-eire added the platform:web issues related to ONNX Runtime web; typically submitted using template label Oct 26, 2024
@github-actions github-actions bot added the api:Javascript issues related to the Javascript API label Oct 26, 2024
@ollema
Copy link

ollema commented Oct 27, 2024

there are workarounds for issue 3 - but they involve patching this package to remove the "node": null entries so I am not sure if that can be called a workaround:
ollema/g-split@15f616c

fs-eire added a commit that referenced this issue Oct 29, 2024
### Description

This change resolves issue No.3 described in #22615
@sophies927 sophies927 added the feature request request for unsupported feature or enhancement label Oct 31, 2024
ishwar-raut1 pushed a commit to ishwar-raut1/onnxruntime that referenced this issue Nov 19, 2024
### Description

This change resolves issue No.3 described in microsoft#22615
@fs-eire
Copy link
Contributor Author

fs-eire commented Dec 5, 2024

Webpack ESM support tracking: webpack/webpack#17121

ankitm3k pushed a commit to intel/onnxruntime that referenced this issue Dec 11, 2024
### Description

This change resolves issue No.3 described in microsoft#22615
ankitm3k pushed a commit to intel/onnxruntime that referenced this issue Dec 11, 2024
### Description

This change resolves issue No.3 described in microsoft#22615
ankitm3k pushed a commit to intel/onnxruntime that referenced this issue Dec 11, 2024
### Description

This change resolves issue No.3 described in microsoft#22615
@fs-eire
Copy link
Contributor Author

fs-eire commented Dec 20, 2024

Tracking: Add #23175 for it easier to verify changes of package export.

@fs-eire fs-eire closed this as completed in 0627a6c Jan 9, 2025
guschmue pushed a commit that referenced this issue Jan 12, 2025
### Description
<!-- Describe your changes. -->

This PR tries to fix #22615. (see detailed description in the issue)

A perfect solution would be too difficult to make, because there are a
huge number of combinations of usage scenarios, including combinations
of development framework, bundler, dev/prod mode, and so on.

This PR is using the following approach:
- Introduce a new type of end to end test: export test. This type of
tests are complete web apps that use popular web development frameworks,
and the tests are using puppeteer to run the apps and check if the apps
can run without error.
  - added one nextjs based web app and one vite based web app.
- In the test, perform the following test steps:
  - `npm install` for packages built locally
- `npm run dev` to start dev server and use puppeteer to launch the
browser to test
- `npm run build && npm run start` to test prod build and use puppeteer
to launch the browser to test
- Make changes to ort-web, including:
- special handling on Webpack's behavior of rewriting `import.meta.url`
to a `file://` string
  - revise build definitions
  - fix wasm URL for proxy, if used in a bundled build
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
api:Javascript issues related to the Javascript API feature request request for unsupported feature or enhancement platform:web issues related to ONNX Runtime web; typically submitted using template
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants