UNPKG

@vmg-anysphere/napi-rs-cli

Version:
217 lines (194 loc) 6.09 kB
export const createWasiBrowserBinding = ( wasiFilename: string, initialMemory = 4000, maximumMemory = 65536, fs = false, asyncInit = false, buffer = false, ) => { const fsImport = fs ? buffer ? `import { memfs, Buffer } from '@napi-rs/wasm-runtime/fs'` : `import { memfs } from '@napi-rs/wasm-runtime/fs'` : '' const bufferImport = buffer && !fs ? `import { Buffer } from 'buffer'` : '' const wasiCreation = fs ? ` export const { fs: __fs, vol: __volume } = memfs() const __wasi = new __WASI({ version: 'preview1', fs: __fs, preopens: { '/': '/', }, })` : ` const __wasi = new __WASI({ version: 'preview1', })` const workerFsHandler = fs ? ` worker.addEventListener('message', __wasmCreateOnMessageForFsProxy(__fs))\n` : '' const emnapiInjectBuffer = buffer ? '__emnapiContext.feature.Buffer = Buffer' : '' const emnapiInstantiateImport = asyncInit ? `instantiateNapiModule as __emnapiInstantiateNapiModule` : `instantiateNapiModuleSync as __emnapiInstantiateNapiModuleSync` const emnapiInstantiateCall = asyncInit ? `await __emnapiInstantiateNapiModule` : `__emnapiInstantiateNapiModuleSync` return `import { createOnMessage as __wasmCreateOnMessageForFsProxy, getDefaultContext as __emnapiGetDefaultContext, ${emnapiInstantiateImport}, WASI as __WASI, } from '@napi-rs/wasm-runtime' ${fsImport} ${bufferImport} ${wasiCreation} const __wasmUrl = new URL('./${wasiFilename}.wasm', import.meta.url).href const __emnapiContext = __emnapiGetDefaultContext() ${emnapiInjectBuffer} const __sharedMemory = new WebAssembly.Memory({ initial: ${initialMemory}, maximum: ${maximumMemory}, shared: true, }) const __wasmFile = await fetch(__wasmUrl).then((res) => res.arrayBuffer()) const { instance: __napiInstance, module: __wasiModule, napiModule: __napiModule, } = ${emnapiInstantiateCall}(__wasmFile, { context: __emnapiContext, asyncWorkPoolSize: 4, wasi: __wasi, onCreateWorker() { const worker = new Worker(new URL('./wasi-worker-browser.mjs', import.meta.url), { type: 'module', }) ${workerFsHandler} return worker }, overwriteImports(importObject) { importObject.env = { ...importObject.env, ...importObject.napi, ...importObject.emnapi, memory: __sharedMemory, } return importObject }, beforeInit({ instance }) { for (const name of Object.keys(instance.exports)) { if (name.startsWith('__napi_register__')) { instance.exports[name]() } } }, }) ` } export const createWasiBinding = ( wasmFileName: string, packageName: string, initialMemory = 4000, maximumMemory = 65536, ) => `/* eslint-disable */ /* prettier-ignore */ /* auto-generated by NAPI-RS */ const __nodeFs = require('node:fs') const __nodePath = require('node:path') const { WASI: __nodeWASI } = require('node:wasi') const { Worker } = require('node:worker_threads') const { createOnMessage: __wasmCreateOnMessageForFsProxy, getDefaultContext: __emnapiGetDefaultContext, instantiateNapiModuleSync: __emnapiInstantiateNapiModuleSync, } = require('@napi-rs/wasm-runtime') const __rootDir = __nodePath.parse(process.cwd()).root const __wasi = new __nodeWASI({ version: 'preview1', env: process.env, preopens: { [__rootDir]: __rootDir, } }) const __emnapiContext = __emnapiGetDefaultContext() const __sharedMemory = new WebAssembly.Memory({ initial: ${initialMemory}, maximum: ${maximumMemory}, shared: true, }) let __wasmFilePath = __nodePath.join(__dirname, '${wasmFileName}.wasm') const __wasmDebugFilePath = __nodePath.join(__dirname, '${wasmFileName}.debug.wasm') if (__nodeFs.existsSync(__wasmDebugFilePath)) { __wasmFilePath = __wasmDebugFilePath } else if (!__nodeFs.existsSync(__wasmFilePath)) { try { __wasmFilePath = __nodePath.resolve('${packageName}-wasm32-wasi') } catch { throw new Error('Cannot find ${wasmFileName}.wasm file, and ${packageName}-wasm32-wasi package is not installed.') } } const { instance: __napiInstance, module: __wasiModule, napiModule: __napiModule } = __emnapiInstantiateNapiModuleSync(__nodeFs.readFileSync(__wasmFilePath), { context: __emnapiContext, asyncWorkPoolSize: (function() { const threadsSizeFromEnv = Number(process.env.NAPI_RS_ASYNC_WORK_POOL_SIZE ?? process.env.UV_THREADPOOL_SIZE) // NaN > 0 is false if (threadsSizeFromEnv > 0) { return threadsSizeFromEnv } else { return 4 } })(), reuseWorker: true, wasi: __wasi, onCreateWorker() { const worker = new Worker(__nodePath.join(__dirname, 'wasi-worker.mjs'), { env: process.env, }) worker.onmessage = ({ data }) => { __wasmCreateOnMessageForFsProxy(__nodeFs)(data) } // The main thread of Node.js waits for all the active handles before exiting. // But Rust threads are never waited without \`thread::join\`. // So here we hack the code of Node.js to prevent the workers from being referenced (active). // According to https://github.com/nodejs/node/blob/19e0d472728c79d418b74bddff588bea70a403d0/lib/internal/worker.js#L415, // a worker is consist of two handles: kPublicPort and kHandle. { const kPublicPort = Object.getOwnPropertySymbols(worker).find(s => s.toString().includes("kPublicPort") ); if (kPublicPort) { worker[kPublicPort].ref = () => {}; } const kHandle = Object.getOwnPropertySymbols(worker).find(s => s.toString().includes("kHandle") ); if (kHandle) { worker[kHandle].ref = () => {}; } worker.unref(); } return worker }, overwriteImports(importObject) { importObject.env = { ...importObject.env, ...importObject.napi, ...importObject.emnapi, memory: __sharedMemory, } return importObject }, beforeInit({ instance }) { for (const name of Object.keys(instance.exports)) { if (name.startsWith('__napi_register__')) { instance.exports[name]() } } }, }) `