@gorules/quickjs 0.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/LICENSE ADDED
@@ -0,0 +1,13 @@
1
+ Copyright 2025 GoRules.io
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation
4
+ files (the “Software”), to deal in the Software without restriction, including without limitation the rights to use, copy,
5
+ modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software
6
+ is furnished to do so, subject to the following conditions:
7
+
8
+ The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
9
+
10
+ THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
11
+ WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
12
+ COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
13
+ ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
package/README.md ADDED
@@ -0,0 +1,285 @@
1
+ # @gorules/quickjs
2
+
3
+ A high-performance, sandboxed JavaScript runtime for Node.js and browsers, powered by [QuickJS](https://bellard.org/quickjs/) and built with Rust via [napi-rs](https://napi.rs/).
4
+
5
+ ## Features
6
+
7
+ - **Sandboxed Execution** - Run untrusted JavaScript code safely with isolated contexts
8
+ - **Resource Constraints** - Limit execution time, memory usage, and stack size
9
+ - **HTTP Client** - Built-in `http` library for making HTTP requests
10
+ - **Custom Modules** - Define and import ES6 modules dynamically
11
+ - **Async/Await** - Full support for asynchronous JavaScript
12
+ - **Cross-Platform** - Works on macOS, Linux, Windows, and browsers (WASM)
13
+
14
+ ## Installation
15
+
16
+ ```bash
17
+ npm install @gorules/quickjs
18
+ # or
19
+ yarn add @gorules/quickjs
20
+ ```
21
+
22
+ ## Quick Start
23
+
24
+ ```javascript
25
+ import { JavascriptRunner } from '@gorules/quickjs';
26
+ import { createFetchAdapter } from '@gorules/quickjs/fetch-adapter';
27
+
28
+ const runner = new JavascriptRunner({
29
+ fetch: createFetchAdapter(),
30
+ });
31
+
32
+ // Evaluate simple expressions
33
+ const result = await runner.evaluate('1 + 1');
34
+ console.log(result); // 2
35
+
36
+ // Evaluate complex scripts
37
+ const data = await runner.evaluate(`
38
+ const users = [
39
+ { name: 'Alice', age: 30 },
40
+ { name: 'Bob', age: 25 }
41
+ ];
42
+ users.filter(u => u.age > 26).map(u => u.name)
43
+ `);
44
+ console.log(data); // ['Alice']
45
+
46
+ // Clean up when done
47
+ runner.dispose();
48
+ ```
49
+
50
+ ## API Reference
51
+
52
+ ### JavascriptRunner
53
+
54
+ The main class for evaluating JavaScript code.
55
+
56
+ ```typescript
57
+ interface JavascriptRunnerOptions {
58
+ fetch: (url: string, options?: FetchOptions) => Promise<FetchResponse>;
59
+ modules?: Array<JavascriptModule>;
60
+ maxPoolSize?: number;
61
+ memoryLimit?: number; // Memory limit in bytes
62
+ maxStackSize?: number; // Stack size limit in bytes
63
+ maxDuration?: number; // Execution timeout in milliseconds
64
+ }
65
+
66
+ const runner = new JavascriptRunner(options);
67
+ ```
68
+
69
+ #### Methods
70
+
71
+ - `evaluate(script: string): Promise<any>` - Execute JavaScript code and return the result
72
+ - `dispose(): void` - Clean up resources
73
+
74
+ ### JavascriptModule
75
+
76
+ Define custom modules that can be imported in evaluated scripts.
77
+
78
+ ```typescript
79
+ interface NewJavascriptModuleOptions {
80
+ name: string;
81
+ source: string;
82
+ }
83
+
84
+ const module = new JavascriptModule({ name: 'myModule', source: '...' });
85
+ // or create multiple at once
86
+ const modules = JavascriptModule.fromList([...]);
87
+ ```
88
+
89
+ ## Usage Examples
90
+
91
+ ### HTTP Requests
92
+
93
+ The runtime provides a built-in `http` object for making HTTP requests:
94
+
95
+ ```javascript
96
+ const runner = new JavascriptRunner({
97
+ fetch: createFetchAdapter(),
98
+ });
99
+
100
+ const result = await runner.evaluate(`
101
+ const response = await http.get("https://api.example.com/users");
102
+ response.data
103
+ `);
104
+ ```
105
+
106
+ Supported methods:
107
+ - `http.get(url, options?)` - GET request
108
+ - `http.post(url, body?, options?)` - POST request
109
+ - `http.put(url, body?, options?)` - PUT request
110
+ - `http.delete(url, options?)` - DELETE request
111
+
112
+ Request options:
113
+ ```javascript
114
+ {
115
+ headers: { "Authorization": "Bearer token" },
116
+ params: { "q": "search", "limit": "10" },
117
+ baseURL: "https://api.example.com"
118
+ }
119
+ ```
120
+
121
+ Response format:
122
+ ```javascript
123
+ {
124
+ status: 200,
125
+ statusText: "OK",
126
+ headers: { "content-type": "application/json" },
127
+ data: { ... } // Parsed JSON or string
128
+ }
129
+ ```
130
+
131
+ ### Custom Modules
132
+
133
+ Define reusable modules that scripts can import:
134
+
135
+ ```javascript
136
+ import { JavascriptRunner, JavascriptModule } from '@gorules/quickjs';
137
+
138
+ const modules = JavascriptModule.fromList([
139
+ {
140
+ name: 'utils',
141
+ source: `
142
+ export const double = x => x * 2;
143
+ export const capitalize = s => s.charAt(0).toUpperCase() + s.slice(1);
144
+ `
145
+ },
146
+ {
147
+ name: 'api',
148
+ source: `
149
+ export async function fetchUsers() {
150
+ const response = await http.get("https://api.example.com/users");
151
+ return response.data;
152
+ }
153
+ `
154
+ }
155
+ ]);
156
+
157
+ const runner = new JavascriptRunner({
158
+ fetch: createFetchAdapter(),
159
+ modules,
160
+ });
161
+
162
+ const result = await runner.evaluate(`
163
+ const { double } = await import('utils');
164
+ const { fetchUsers } = await import('api');
165
+
166
+ const users = await fetchUsers();
167
+ users.map(u => ({ ...u, score: double(u.score) }))
168
+ `);
169
+ ```
170
+
171
+ ### Resource Constraints
172
+
173
+ Protect against runaway scripts with execution limits:
174
+
175
+ ```javascript
176
+ const runner = new JavascriptRunner({
177
+ fetch: createFetchAdapter(),
178
+ maxDuration: 1000, // 1 second timeout
179
+ memoryLimit: 10 * 1024 * 1024, // 10 MB memory limit
180
+ maxStackSize: 512 * 1024, // 512 KB stack size
181
+ });
182
+
183
+ // This will throw an error after 1 second
184
+ try {
185
+ await runner.evaluate('while(true) {}');
186
+ } catch (error) {
187
+ console.log('Script interrupted:', error.message);
188
+ }
189
+
190
+ // This will throw a memory error
191
+ try {
192
+ await runner.evaluate(`
193
+ const arr = [];
194
+ while(true) { arr.push("x".repeat(10000)); }
195
+ `);
196
+ } catch (error) {
197
+ console.log('Memory limit exceeded:', error.message);
198
+ }
199
+ ```
200
+
201
+ ### Async Operations
202
+
203
+ Full support for async/await and Promises:
204
+
205
+ ```javascript
206
+ const result = await runner.evaluate(`
207
+ async function processData() {
208
+ const [users, posts] = await Promise.all([
209
+ http.get("https://api.example.com/users"),
210
+ http.get("https://api.example.com/posts")
211
+ ]);
212
+
213
+ return {
214
+ userCount: users.data.length,
215
+ postCount: posts.data.length
216
+ };
217
+ }
218
+
219
+ await processData()
220
+ `);
221
+ ```
222
+
223
+ A `sleep` function is also available:
224
+
225
+ ```javascript
226
+ await runner.evaluate(`
227
+ await sleep(100); // Sleep for 100ms
228
+ "done"
229
+ `);
230
+ ```
231
+
232
+ ## Browser Usage (WASM)
233
+
234
+ The package includes WASM support for browser environments:
235
+
236
+ ```javascript
237
+ import { JavascriptRunner } from '@gorules/quickjs/browser';
238
+
239
+ // The WASM module loads asynchronously
240
+ const runner = new JavascriptRunner({
241
+ fetch: async (url, options) => {
242
+ const res = await fetch(url, options);
243
+ return {
244
+ status: res.status,
245
+ statusText: res.statusText,
246
+ headers: Object.fromEntries(res.headers),
247
+ data: await res.text(),
248
+ };
249
+ },
250
+ });
251
+
252
+ const result = await runner.evaluate('2 + 2');
253
+ ```
254
+
255
+ ## Execution Isolation
256
+
257
+ Each call to `evaluate()` runs in an isolated context. Variables defined in one evaluation are not accessible in subsequent evaluations:
258
+
259
+ ```javascript
260
+ await runner.evaluate('var x = 100');
261
+ const result = await runner.evaluate('typeof x');
262
+ console.log(result); // 'undefined'
263
+ ```
264
+
265
+ ## Platform Support
266
+
267
+ | Platform | Architecture | Support |
268
+ |----------|--------------|---------|
269
+ | macOS | x64, ARM64 | Native |
270
+ | Linux | x64, ARM64 (glibc) | Native |
271
+ | Linux | x64, ARM64 (musl) | Native |
272
+ | Windows | x64 | Native |
273
+ | Browser | WASM | WASM |
274
+
275
+ ## Built with Rust
276
+
277
+ The core runtime is implemented in Rust using:
278
+ - [rquickjs](https://github.com/deltalabs-ai/rquickjs) - Rust bindings for QuickJS
279
+ - [napi-rs](https://napi.rs/) - Node.js native addon framework
280
+
281
+ This provides excellent performance while maintaining memory safety.
282
+
283
+ ## License
284
+
285
+ MIT
package/browser.js ADDED
@@ -0,0 +1 @@
1
+ export * from '@gorules/quickjs-wasm32-wasi'
@@ -0,0 +1,10 @@
1
+ import type { FetchOptions, FetchResponse } from './index';
2
+
3
+ /**
4
+ * Creates a fetch adapter that wraps native fetch for use with Runner.
5
+ * Transforms the native Response object into the FetchResponse format.
6
+ */
7
+ export function createFetchAdapter(): (
8
+ url: string,
9
+ options?: FetchOptions
10
+ ) => Promise<FetchResponse>;
@@ -0,0 +1,29 @@
1
+ /**
2
+ * Creates a fetch adapter that wraps native fetch for use with Runner.
3
+ * Transforms the native Response object into the FetchResponse format.
4
+ *
5
+ * @returns {(url: string, options?: import('./index').FetchOptions) => Promise<import('./index').FetchResponse>}
6
+ */
7
+ function createFetchAdapter() {
8
+ return async (...args) => {
9
+ const [url, options] = Array.isArray(args[0]) ? args[0] : args;
10
+ const res = await fetch(url, {
11
+ method: options?.method ?? 'GET',
12
+ headers: options?.headers,
13
+ body: options?.body,
14
+ });
15
+
16
+ const data = await res.text();
17
+ const headers = {};
18
+ res.headers.forEach((v, k) => { headers[k] = v; });
19
+
20
+ return {
21
+ status: res.status,
22
+ statusText: res.statusText,
23
+ headers,
24
+ data,
25
+ };
26
+ };
27
+ }
28
+
29
+ module.exports.createFetchAdapter = createFetchAdapter;
package/index.d.ts ADDED
@@ -0,0 +1,43 @@
1
+ /* auto-generated by NAPI-RS */
2
+ /* eslint-disable */
3
+ export declare class JavascriptModule {
4
+ constructor(options: NewJavascriptModuleOptions)
5
+ static fromList(options: Array<NewJavascriptModuleOptions>): Array<JavascriptModule>
6
+ get name(): string
7
+ compiled(): Promise<boolean>
8
+ }
9
+
10
+ export declare class JavascriptRunner {
11
+ constructor(options: JavascriptRunnerOptions)
12
+ evaluate(script: string): Promise<any>
13
+ dispose(): void
14
+ }
15
+
16
+ export interface FetchOptions {
17
+ method?: string
18
+ headers?: Record<string, string>
19
+ body?: string
20
+ timeout?: number
21
+ }
22
+
23
+ export interface FetchResponse {
24
+ status: number
25
+ statusText: string
26
+ headers: Record<string, string>
27
+ data: string
28
+ }
29
+
30
+ export interface JavascriptRunnerOptions {
31
+ fetch: (url: string, options?: FetchOptions) => Promise<FetchResponse>
32
+ modules?: Array<JavascriptModule>
33
+ maxPoolSize?: number
34
+ memoryLimit?: number
35
+ maxStackSize?: number
36
+ /** Max duration in milliseconds */
37
+ maxDuration?: number
38
+ }
39
+
40
+ export interface NewJavascriptModuleOptions {
41
+ name: string
42
+ source: string
43
+ }
package/index.js ADDED
@@ -0,0 +1,576 @@
1
+ // prettier-ignore
2
+ /* eslint-disable */
3
+ // @ts-nocheck
4
+ /* auto-generated by NAPI-RS */
5
+
6
+ const { readFileSync } = require('node:fs')
7
+ let nativeBinding = null
8
+ const loadErrors = []
9
+
10
+ const isMusl = () => {
11
+ let musl = false
12
+ if (process.platform === 'linux') {
13
+ musl = isMuslFromFilesystem()
14
+ if (musl === null) {
15
+ musl = isMuslFromReport()
16
+ }
17
+ if (musl === null) {
18
+ musl = isMuslFromChildProcess()
19
+ }
20
+ }
21
+ return musl
22
+ }
23
+
24
+ const isFileMusl = (f) => f.includes('libc.musl-') || f.includes('ld-musl-')
25
+
26
+ const isMuslFromFilesystem = () => {
27
+ try {
28
+ return readFileSync('/usr/bin/ldd', 'utf-8').includes('musl')
29
+ } catch {
30
+ return null
31
+ }
32
+ }
33
+
34
+ const isMuslFromReport = () => {
35
+ let report = null
36
+ if (typeof process.report?.getReport === 'function') {
37
+ process.report.excludeNetwork = true
38
+ report = process.report.getReport()
39
+ }
40
+ if (!report) {
41
+ return null
42
+ }
43
+ if (report.header && report.header.glibcVersionRuntime) {
44
+ return false
45
+ }
46
+ if (Array.isArray(report.sharedObjects)) {
47
+ if (report.sharedObjects.some(isFileMusl)) {
48
+ return true
49
+ }
50
+ }
51
+ return false
52
+ }
53
+
54
+ const isMuslFromChildProcess = () => {
55
+ try {
56
+ return require('child_process').execSync('ldd --version', { encoding: 'utf8' }).includes('musl')
57
+ } catch (e) {
58
+ // If we reach this case, we don't know if the system is musl or not, so is better to just fallback to false
59
+ return false
60
+ }
61
+ }
62
+
63
+ function requireNative() {
64
+ if (process.env.NAPI_RS_NATIVE_LIBRARY_PATH) {
65
+ try {
66
+ return require(process.env.NAPI_RS_NATIVE_LIBRARY_PATH);
67
+ } catch (err) {
68
+ loadErrors.push(err)
69
+ }
70
+ } else if (process.platform === 'android') {
71
+ if (process.arch === 'arm64') {
72
+ try {
73
+ return require('./quickjs.android-arm64.node')
74
+ } catch (e) {
75
+ loadErrors.push(e)
76
+ }
77
+ try {
78
+ const binding = require('@gorules/quickjs-android-arm64')
79
+ const bindingPackageVersion = require('@gorules/quickjs-android-arm64/package.json').version
80
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
81
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
82
+ }
83
+ return binding
84
+ } catch (e) {
85
+ loadErrors.push(e)
86
+ }
87
+ } else if (process.arch === 'arm') {
88
+ try {
89
+ return require('./quickjs.android-arm-eabi.node')
90
+ } catch (e) {
91
+ loadErrors.push(e)
92
+ }
93
+ try {
94
+ const binding = require('@gorules/quickjs-android-arm-eabi')
95
+ const bindingPackageVersion = require('@gorules/quickjs-android-arm-eabi/package.json').version
96
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
97
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
98
+ }
99
+ return binding
100
+ } catch (e) {
101
+ loadErrors.push(e)
102
+ }
103
+ } else {
104
+ loadErrors.push(new Error(`Unsupported architecture on Android ${process.arch}`))
105
+ }
106
+ } else if (process.platform === 'win32') {
107
+ if (process.arch === 'x64') {
108
+ if (process.config?.variables?.shlib_suffix === 'dll.a' || process.config?.variables?.node_target_type === 'shared_library') {
109
+ try {
110
+ return require('./quickjs.win32-x64-gnu.node')
111
+ } catch (e) {
112
+ loadErrors.push(e)
113
+ }
114
+ try {
115
+ const binding = require('@gorules/quickjs-win32-x64-gnu')
116
+ const bindingPackageVersion = require('@gorules/quickjs-win32-x64-gnu/package.json').version
117
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
118
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
119
+ }
120
+ return binding
121
+ } catch (e) {
122
+ loadErrors.push(e)
123
+ }
124
+ } else {
125
+ try {
126
+ return require('./quickjs.win32-x64-msvc.node')
127
+ } catch (e) {
128
+ loadErrors.push(e)
129
+ }
130
+ try {
131
+ const binding = require('@gorules/quickjs-win32-x64-msvc')
132
+ const bindingPackageVersion = require('@gorules/quickjs-win32-x64-msvc/package.json').version
133
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
134
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
135
+ }
136
+ return binding
137
+ } catch (e) {
138
+ loadErrors.push(e)
139
+ }
140
+ }
141
+ } else if (process.arch === 'ia32') {
142
+ try {
143
+ return require('./quickjs.win32-ia32-msvc.node')
144
+ } catch (e) {
145
+ loadErrors.push(e)
146
+ }
147
+ try {
148
+ const binding = require('@gorules/quickjs-win32-ia32-msvc')
149
+ const bindingPackageVersion = require('@gorules/quickjs-win32-ia32-msvc/package.json').version
150
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
151
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
152
+ }
153
+ return binding
154
+ } catch (e) {
155
+ loadErrors.push(e)
156
+ }
157
+ } else if (process.arch === 'arm64') {
158
+ try {
159
+ return require('./quickjs.win32-arm64-msvc.node')
160
+ } catch (e) {
161
+ loadErrors.push(e)
162
+ }
163
+ try {
164
+ const binding = require('@gorules/quickjs-win32-arm64-msvc')
165
+ const bindingPackageVersion = require('@gorules/quickjs-win32-arm64-msvc/package.json').version
166
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
167
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
168
+ }
169
+ return binding
170
+ } catch (e) {
171
+ loadErrors.push(e)
172
+ }
173
+ } else {
174
+ loadErrors.push(new Error(`Unsupported architecture on Windows: ${process.arch}`))
175
+ }
176
+ } else if (process.platform === 'darwin') {
177
+ try {
178
+ return require('./quickjs.darwin-universal.node')
179
+ } catch (e) {
180
+ loadErrors.push(e)
181
+ }
182
+ try {
183
+ const binding = require('@gorules/quickjs-darwin-universal')
184
+ const bindingPackageVersion = require('@gorules/quickjs-darwin-universal/package.json').version
185
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
186
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
187
+ }
188
+ return binding
189
+ } catch (e) {
190
+ loadErrors.push(e)
191
+ }
192
+ if (process.arch === 'x64') {
193
+ try {
194
+ return require('./quickjs.darwin-x64.node')
195
+ } catch (e) {
196
+ loadErrors.push(e)
197
+ }
198
+ try {
199
+ const binding = require('@gorules/quickjs-darwin-x64')
200
+ const bindingPackageVersion = require('@gorules/quickjs-darwin-x64/package.json').version
201
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
202
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
203
+ }
204
+ return binding
205
+ } catch (e) {
206
+ loadErrors.push(e)
207
+ }
208
+ } else if (process.arch === 'arm64') {
209
+ try {
210
+ return require('./quickjs.darwin-arm64.node')
211
+ } catch (e) {
212
+ loadErrors.push(e)
213
+ }
214
+ try {
215
+ const binding = require('@gorules/quickjs-darwin-arm64')
216
+ const bindingPackageVersion = require('@gorules/quickjs-darwin-arm64/package.json').version
217
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
218
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
219
+ }
220
+ return binding
221
+ } catch (e) {
222
+ loadErrors.push(e)
223
+ }
224
+ } else {
225
+ loadErrors.push(new Error(`Unsupported architecture on macOS: ${process.arch}`))
226
+ }
227
+ } else if (process.platform === 'freebsd') {
228
+ if (process.arch === 'x64') {
229
+ try {
230
+ return require('./quickjs.freebsd-x64.node')
231
+ } catch (e) {
232
+ loadErrors.push(e)
233
+ }
234
+ try {
235
+ const binding = require('@gorules/quickjs-freebsd-x64')
236
+ const bindingPackageVersion = require('@gorules/quickjs-freebsd-x64/package.json').version
237
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
238
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
239
+ }
240
+ return binding
241
+ } catch (e) {
242
+ loadErrors.push(e)
243
+ }
244
+ } else if (process.arch === 'arm64') {
245
+ try {
246
+ return require('./quickjs.freebsd-arm64.node')
247
+ } catch (e) {
248
+ loadErrors.push(e)
249
+ }
250
+ try {
251
+ const binding = require('@gorules/quickjs-freebsd-arm64')
252
+ const bindingPackageVersion = require('@gorules/quickjs-freebsd-arm64/package.json').version
253
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
254
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
255
+ }
256
+ return binding
257
+ } catch (e) {
258
+ loadErrors.push(e)
259
+ }
260
+ } else {
261
+ loadErrors.push(new Error(`Unsupported architecture on FreeBSD: ${process.arch}`))
262
+ }
263
+ } else if (process.platform === 'linux') {
264
+ if (process.arch === 'x64') {
265
+ if (isMusl()) {
266
+ try {
267
+ return require('./quickjs.linux-x64-musl.node')
268
+ } catch (e) {
269
+ loadErrors.push(e)
270
+ }
271
+ try {
272
+ const binding = require('@gorules/quickjs-linux-x64-musl')
273
+ const bindingPackageVersion = require('@gorules/quickjs-linux-x64-musl/package.json').version
274
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
275
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
276
+ }
277
+ return binding
278
+ } catch (e) {
279
+ loadErrors.push(e)
280
+ }
281
+ } else {
282
+ try {
283
+ return require('./quickjs.linux-x64-gnu.node')
284
+ } catch (e) {
285
+ loadErrors.push(e)
286
+ }
287
+ try {
288
+ const binding = require('@gorules/quickjs-linux-x64-gnu')
289
+ const bindingPackageVersion = require('@gorules/quickjs-linux-x64-gnu/package.json').version
290
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
291
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
292
+ }
293
+ return binding
294
+ } catch (e) {
295
+ loadErrors.push(e)
296
+ }
297
+ }
298
+ } else if (process.arch === 'arm64') {
299
+ if (isMusl()) {
300
+ try {
301
+ return require('./quickjs.linux-arm64-musl.node')
302
+ } catch (e) {
303
+ loadErrors.push(e)
304
+ }
305
+ try {
306
+ const binding = require('@gorules/quickjs-linux-arm64-musl')
307
+ const bindingPackageVersion = require('@gorules/quickjs-linux-arm64-musl/package.json').version
308
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
309
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
310
+ }
311
+ return binding
312
+ } catch (e) {
313
+ loadErrors.push(e)
314
+ }
315
+ } else {
316
+ try {
317
+ return require('./quickjs.linux-arm64-gnu.node')
318
+ } catch (e) {
319
+ loadErrors.push(e)
320
+ }
321
+ try {
322
+ const binding = require('@gorules/quickjs-linux-arm64-gnu')
323
+ const bindingPackageVersion = require('@gorules/quickjs-linux-arm64-gnu/package.json').version
324
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
325
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
326
+ }
327
+ return binding
328
+ } catch (e) {
329
+ loadErrors.push(e)
330
+ }
331
+ }
332
+ } else if (process.arch === 'arm') {
333
+ if (isMusl()) {
334
+ try {
335
+ return require('./quickjs.linux-arm-musleabihf.node')
336
+ } catch (e) {
337
+ loadErrors.push(e)
338
+ }
339
+ try {
340
+ const binding = require('@gorules/quickjs-linux-arm-musleabihf')
341
+ const bindingPackageVersion = require('@gorules/quickjs-linux-arm-musleabihf/package.json').version
342
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
343
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
344
+ }
345
+ return binding
346
+ } catch (e) {
347
+ loadErrors.push(e)
348
+ }
349
+ } else {
350
+ try {
351
+ return require('./quickjs.linux-arm-gnueabihf.node')
352
+ } catch (e) {
353
+ loadErrors.push(e)
354
+ }
355
+ try {
356
+ const binding = require('@gorules/quickjs-linux-arm-gnueabihf')
357
+ const bindingPackageVersion = require('@gorules/quickjs-linux-arm-gnueabihf/package.json').version
358
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
359
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
360
+ }
361
+ return binding
362
+ } catch (e) {
363
+ loadErrors.push(e)
364
+ }
365
+ }
366
+ } else if (process.arch === 'loong64') {
367
+ if (isMusl()) {
368
+ try {
369
+ return require('./quickjs.linux-loong64-musl.node')
370
+ } catch (e) {
371
+ loadErrors.push(e)
372
+ }
373
+ try {
374
+ const binding = require('@gorules/quickjs-linux-loong64-musl')
375
+ const bindingPackageVersion = require('@gorules/quickjs-linux-loong64-musl/package.json').version
376
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
377
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
378
+ }
379
+ return binding
380
+ } catch (e) {
381
+ loadErrors.push(e)
382
+ }
383
+ } else {
384
+ try {
385
+ return require('./quickjs.linux-loong64-gnu.node')
386
+ } catch (e) {
387
+ loadErrors.push(e)
388
+ }
389
+ try {
390
+ const binding = require('@gorules/quickjs-linux-loong64-gnu')
391
+ const bindingPackageVersion = require('@gorules/quickjs-linux-loong64-gnu/package.json').version
392
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
393
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
394
+ }
395
+ return binding
396
+ } catch (e) {
397
+ loadErrors.push(e)
398
+ }
399
+ }
400
+ } else if (process.arch === 'riscv64') {
401
+ if (isMusl()) {
402
+ try {
403
+ return require('./quickjs.linux-riscv64-musl.node')
404
+ } catch (e) {
405
+ loadErrors.push(e)
406
+ }
407
+ try {
408
+ const binding = require('@gorules/quickjs-linux-riscv64-musl')
409
+ const bindingPackageVersion = require('@gorules/quickjs-linux-riscv64-musl/package.json').version
410
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
411
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
412
+ }
413
+ return binding
414
+ } catch (e) {
415
+ loadErrors.push(e)
416
+ }
417
+ } else {
418
+ try {
419
+ return require('./quickjs.linux-riscv64-gnu.node')
420
+ } catch (e) {
421
+ loadErrors.push(e)
422
+ }
423
+ try {
424
+ const binding = require('@gorules/quickjs-linux-riscv64-gnu')
425
+ const bindingPackageVersion = require('@gorules/quickjs-linux-riscv64-gnu/package.json').version
426
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
427
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
428
+ }
429
+ return binding
430
+ } catch (e) {
431
+ loadErrors.push(e)
432
+ }
433
+ }
434
+ } else if (process.arch === 'ppc64') {
435
+ try {
436
+ return require('./quickjs.linux-ppc64-gnu.node')
437
+ } catch (e) {
438
+ loadErrors.push(e)
439
+ }
440
+ try {
441
+ const binding = require('@gorules/quickjs-linux-ppc64-gnu')
442
+ const bindingPackageVersion = require('@gorules/quickjs-linux-ppc64-gnu/package.json').version
443
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
444
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
445
+ }
446
+ return binding
447
+ } catch (e) {
448
+ loadErrors.push(e)
449
+ }
450
+ } else if (process.arch === 's390x') {
451
+ try {
452
+ return require('./quickjs.linux-s390x-gnu.node')
453
+ } catch (e) {
454
+ loadErrors.push(e)
455
+ }
456
+ try {
457
+ const binding = require('@gorules/quickjs-linux-s390x-gnu')
458
+ const bindingPackageVersion = require('@gorules/quickjs-linux-s390x-gnu/package.json').version
459
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
460
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
461
+ }
462
+ return binding
463
+ } catch (e) {
464
+ loadErrors.push(e)
465
+ }
466
+ } else {
467
+ loadErrors.push(new Error(`Unsupported architecture on Linux: ${process.arch}`))
468
+ }
469
+ } else if (process.platform === 'openharmony') {
470
+ if (process.arch === 'arm64') {
471
+ try {
472
+ return require('./quickjs.openharmony-arm64.node')
473
+ } catch (e) {
474
+ loadErrors.push(e)
475
+ }
476
+ try {
477
+ const binding = require('@gorules/quickjs-openharmony-arm64')
478
+ const bindingPackageVersion = require('@gorules/quickjs-openharmony-arm64/package.json').version
479
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
480
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
481
+ }
482
+ return binding
483
+ } catch (e) {
484
+ loadErrors.push(e)
485
+ }
486
+ } else if (process.arch === 'x64') {
487
+ try {
488
+ return require('./quickjs.openharmony-x64.node')
489
+ } catch (e) {
490
+ loadErrors.push(e)
491
+ }
492
+ try {
493
+ const binding = require('@gorules/quickjs-openharmony-x64')
494
+ const bindingPackageVersion = require('@gorules/quickjs-openharmony-x64/package.json').version
495
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
496
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
497
+ }
498
+ return binding
499
+ } catch (e) {
500
+ loadErrors.push(e)
501
+ }
502
+ } else if (process.arch === 'arm') {
503
+ try {
504
+ return require('./quickjs.openharmony-arm.node')
505
+ } catch (e) {
506
+ loadErrors.push(e)
507
+ }
508
+ try {
509
+ const binding = require('@gorules/quickjs-openharmony-arm')
510
+ const bindingPackageVersion = require('@gorules/quickjs-openharmony-arm/package.json').version
511
+ if (bindingPackageVersion !== '0.1.3' && process.env.NAPI_RS_ENFORCE_VERSION_CHECK && process.env.NAPI_RS_ENFORCE_VERSION_CHECK !== '0') {
512
+ throw new Error(`Native binding package version mismatch, expected 0.1.3 but got ${bindingPackageVersion}. You can reinstall dependencies to fix this issue.`)
513
+ }
514
+ return binding
515
+ } catch (e) {
516
+ loadErrors.push(e)
517
+ }
518
+ } else {
519
+ loadErrors.push(new Error(`Unsupported architecture on OpenHarmony: ${process.arch}`))
520
+ }
521
+ } else {
522
+ loadErrors.push(new Error(`Unsupported OS: ${process.platform}, architecture: ${process.arch}`))
523
+ }
524
+ }
525
+
526
+ nativeBinding = requireNative()
527
+
528
+ if (!nativeBinding || process.env.NAPI_RS_FORCE_WASI) {
529
+ let wasiBinding = null
530
+ let wasiBindingError = null
531
+ try {
532
+ wasiBinding = require('./quickjs.wasi.cjs')
533
+ nativeBinding = wasiBinding
534
+ } catch (err) {
535
+ if (process.env.NAPI_RS_FORCE_WASI) {
536
+ wasiBindingError = err
537
+ }
538
+ }
539
+ if (!nativeBinding) {
540
+ try {
541
+ wasiBinding = require('@gorules/quickjs-wasm32-wasi')
542
+ nativeBinding = wasiBinding
543
+ } catch (err) {
544
+ if (process.env.NAPI_RS_FORCE_WASI) {
545
+ wasiBindingError.cause = err
546
+ loadErrors.push(err)
547
+ }
548
+ }
549
+ }
550
+ if (process.env.NAPI_RS_FORCE_WASI === 'error' && !wasiBinding) {
551
+ const error = new Error('WASI binding not found and NAPI_RS_FORCE_WASI is set to error')
552
+ error.cause = wasiBindingError
553
+ throw error
554
+ }
555
+ }
556
+
557
+ if (!nativeBinding) {
558
+ if (loadErrors.length > 0) {
559
+ throw new Error(
560
+ `Cannot find native binding. ` +
561
+ `npm has a bug related to optional dependencies (https://github.com/npm/cli/issues/4828). ` +
562
+ 'Please try `npm i` again after removing both package-lock.json and node_modules directory.',
563
+ {
564
+ cause: loadErrors.reduce((err, cur) => {
565
+ cur.cause = err
566
+ return cur
567
+ }),
568
+ },
569
+ )
570
+ }
571
+ throw new Error(`Failed to load native binding`)
572
+ }
573
+
574
+ module.exports = nativeBinding
575
+ module.exports.JavascriptModule = nativeBinding.JavascriptModule
576
+ module.exports.JavascriptRunner = nativeBinding.JavascriptRunner
package/package.json ADDED
@@ -0,0 +1,103 @@
1
+ {
2
+ "name": "@gorules/quickjs",
3
+ "version": "0.1.3",
4
+ "description": "High-performance, sandboxed JavaScript runtime for Node.js and browsers, powered by QuickJS and Rust",
5
+ "author": "GoRules <hi@gorules.io> (https://gorules.io)",
6
+ "license": "MIT",
7
+ "homepage": "https://gorules.io",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "git+https://github.com/gorules/quickjs.git"
11
+ },
12
+ "bugs": {
13
+ "url": "https://github.com/gorules/quickjs/issues"
14
+ },
15
+ "keywords": [
16
+ "quickjs",
17
+ "javascript",
18
+ "sandbox",
19
+ "runtime",
20
+ "eval",
21
+ "isolated",
22
+ "secure",
23
+ "wasm",
24
+ "webassembly",
25
+ "napi",
26
+ "rust",
27
+ "embedded",
28
+ "scripting",
29
+ "interpreter"
30
+ ],
31
+ "engines": {
32
+ "node": ">=20"
33
+ },
34
+ "main": "index.js",
35
+ "browser": "browser.js",
36
+ "types": "./index.d.ts",
37
+ "typings": "index.d.ts",
38
+ "packageManager": "yarn@4.12.0",
39
+ "files": [
40
+ "fetch-adapter.js",
41
+ "fetch-adapter.d.ts",
42
+ "index.js",
43
+ "index.d.ts",
44
+ "browser.js",
45
+ "README.md"
46
+ ],
47
+ "napi": {
48
+ "binaryName": "quickjs",
49
+ "targets": [
50
+ "x86_64-apple-darwin",
51
+ "x86_64-unknown-linux-gnu",
52
+ "x86_64-unknown-linux-musl",
53
+ "x86_64-pc-windows-msvc",
54
+ "aarch64-unknown-linux-gnu",
55
+ "aarch64-unknown-linux-musl",
56
+ "aarch64-apple-darwin",
57
+ "wasm32-wasip1-threads"
58
+ ],
59
+ "npmClient": "yarn",
60
+ "wasm": {
61
+ "browser": {
62
+ "fs": false,
63
+ "asyncInit": true
64
+ }
65
+ }
66
+ },
67
+ "scripts": {
68
+ "build": "napi build --platform --release",
69
+ "build:debug": "napi build --platform",
70
+ "test": "node --experimental-transform-types --test tests/**/*.test.mts",
71
+ "format": "prettier tests/*.mts --write",
72
+ "repl": "node --experimental-transform-types repl.mts",
73
+ "artifacts": "napi artifacts -d artifacts",
74
+ "createNpmDirs": "napi create-npm-dirs",
75
+ "release": "release-it"
76
+ },
77
+ "devDependencies": {
78
+ "@napi-rs/cli": "^3.5.0",
79
+ "@release-it/conventional-changelog": "^10.0.0",
80
+ "@types/node": "^25.0.3",
81
+ "prettier": "^3.7.4",
82
+ "release-it": "^18.1.2",
83
+ "typescript": "^5.9.3"
84
+ },
85
+ "release-it": {
86
+ "git": {
87
+ "commitMessage": "chore: release v${version}",
88
+ "tagName": "v${version}"
89
+ },
90
+ "github": {
91
+ "release": true
92
+ },
93
+ "npm": {
94
+ "publish": false
95
+ },
96
+ "plugins": {
97
+ "@release-it/conventional-changelog": {
98
+ "preset": "angular",
99
+ "infile": "CHANGELOG.md"
100
+ }
101
+ }
102
+ }
103
+ }