2024-01-01 14:58:21 -05:00
|
|
|
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-12-07 08:21:01 -05:00
|
|
|
import { core } from "ext:core/mod.js";
|
2024-01-10 17:37:25 -05:00
|
|
|
const {
|
|
|
|
op_net_listen_udp,
|
|
|
|
op_net_listen_unixpacket,
|
|
|
|
op_runtime_memory_usage,
|
|
|
|
} = core.ensureFastOps();
|
2023-09-26 20:21:06 -04:00
|
|
|
|
2023-03-08 06:44:54 -05:00
|
|
|
import * as timers from "ext:deno_web/02_timers.js";
|
|
|
|
import * as httpClient from "ext:deno_fetch/22_http_client.js";
|
2023-04-30 05:11:37 -04:00
|
|
|
import * as console from "ext:deno_console/01_console.js";
|
2023-03-08 06:44:54 -05:00
|
|
|
import * as ffi from "ext:deno_ffi/00_ffi.js";
|
|
|
|
import * as net from "ext:deno_net/01_net.js";
|
|
|
|
import * as tls from "ext:deno_net/02_tls.js";
|
|
|
|
import * as http from "ext:deno_http/01_http.js";
|
|
|
|
import * as errors from "ext:runtime/01_errors.js";
|
|
|
|
import * as version from "ext:runtime/01_version.ts";
|
|
|
|
import * as permissions from "ext:runtime/10_permissions.js";
|
|
|
|
import * as io from "ext:deno_io/12_io.js";
|
|
|
|
import * as buffer from "ext:runtime/13_buffer.js";
|
|
|
|
import * as fs from "ext:deno_fs/30_fs.js";
|
|
|
|
import * as os from "ext:runtime/30_os.js";
|
|
|
|
import * as fsEvents from "ext:runtime/40_fs_events.js";
|
|
|
|
import * as process from "ext:runtime/40_process.js";
|
|
|
|
import * as signals from "ext:runtime/40_signals.js";
|
|
|
|
import * as tty from "ext:runtime/40_tty.js";
|
2023-02-07 14:22:46 -05:00
|
|
|
// TODO(bartlomieju): this is funky we have two `http` imports
|
2023-03-08 06:44:54 -05:00
|
|
|
import * as httpRuntime from "ext:runtime/40_http.js";
|
2023-03-22 00:13:24 -04:00
|
|
|
import * as kv from "ext:deno_kv/01_db.ts";
|
2023-11-01 14:57:55 -04:00
|
|
|
import * as cron from "ext:deno_cron/01_cron.ts";
|
feat:: External webgpu surfaces / BYOW (#21835)
This PR contains the implementation of the External webgpu surfaces /
BYOW proposal. BYOW stands for "Bring your own window".
Closes #21713
Adds `Deno.UnsafeWindowSurface` ( `--unstable-webgpu` API) to the `Deno`
namespace:
```typescript
class UnsafeWindowSurface {
constructor(
system: "cocoa" | "x11" | "win32",
winHandle: Deno.PointerValue,
displayHandle: Deno.PointerValue | null
);
getContext(type: "webgpu"): GPUCanvasContext;
present(): void;
}
```
For the initial pass, I've opted to support the three major windowing
systems. The parameters correspond to the table below:
| system | winHandle | displayHandle |
| ----------------- | ---------- | ------- |
| "cocoa" (macOS) | `NSView*` | - |
| "win32" (Windows) | `HWND` | `HINSTANCE` |
| "x11" (Linux) | Xlib `Window` | Xlib `Display*` |
Ecosystem support:
- [x] deno_sdl2 (sdl2) -
[mod.ts#L1209](https://github.com/littledivy/deno_sdl2/blob/7e177bc6524750a8849c25ce421798b2e71ec943/mod.ts#L1209)
- [x] dwm (glfw) - https://github.com/deno-windowing/dwm/issues/29
- [ ] pane (winit)
<details>
<summary>Example</summary>
```typescript
// A simple clear screen pass, colors based on mouse position.
import { EventType, WindowBuilder } from "https://deno.land/x/sdl2@0.7.0/mod.ts";
const window = new WindowBuilder("sdl2 + deno + webgpu", 640, 480).build();
const [system, windowHandle, displayHandle] = window.rawHandle();
const adapter = await navigator.gpu.requestAdapter();
const device = await adapter.requestDevice();
const context = Deno.createWindowSurface(system, windowHandle, displayHandle);
context.configure({
device: device,
format: "bgra8unorm",
height: 480,
width: 640,
});
let r = 0.0;
let g = 0.0;
let b = 0.0;
for (const event of window.events()) {
if (event.type === EventType.Quit) {
break;
} else if (event.type === EventType.Draw) {
const textureView = context.getCurrentTexture().createView();
const renderPassDescriptor: GPURenderPassDescriptor = {
colorAttachments: [
{
view: textureView,
clearValue: { r, g, b, a: 1.0 },
loadOp: "clear",
storeOp: "store",
},
],
};
const commandEncoder = device.createCommandEncoder();
const passEncoder = commandEncoder.beginRenderPass(renderPassDescriptor);
passEncoder.end();
device.queue.submit([commandEncoder.finish()]);
Deno.presentGPUCanvasContext(context);
}
if (event.type === EventType.MouseMotion) {
r = event.x / 640;
g = event.y / 480;
b = 1.0 - r - g;
}
}
```
You can find more examples in the linked tracking issue.
</details>
---------
Signed-off-by: Divy Srivastava <dj.srivastava23@gmail.com>
2024-01-19 12:19:14 -05:00
|
|
|
import * as webgpuSurface from "ext:deno_webgpu/02_surface.js";
|
2022-11-10 16:03:28 -05:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
const denoNs = {
|
|
|
|
metrics: core.metrics,
|
|
|
|
Process: process.Process,
|
|
|
|
run: process.run,
|
|
|
|
isatty: tty.isatty,
|
2023-03-04 22:37:37 -05:00
|
|
|
writeFileSync: fs.writeFileSync,
|
|
|
|
writeFile: fs.writeFile,
|
|
|
|
writeTextFileSync: fs.writeTextFileSync,
|
|
|
|
writeTextFile: fs.writeTextFile,
|
|
|
|
readTextFile: fs.readTextFile,
|
|
|
|
readTextFileSync: fs.readTextFileSync,
|
|
|
|
readFile: fs.readFile,
|
|
|
|
readFileSync: fs.readFileSync,
|
2023-02-07 14:22:46 -05:00
|
|
|
watchFs: fsEvents.watchFs,
|
|
|
|
chmodSync: fs.chmodSync,
|
|
|
|
chmod: fs.chmod,
|
|
|
|
chown: fs.chown,
|
|
|
|
chownSync: fs.chownSync,
|
|
|
|
copyFileSync: fs.copyFileSync,
|
|
|
|
cwd: fs.cwd,
|
|
|
|
makeTempDirSync: fs.makeTempDirSync,
|
|
|
|
makeTempDir: fs.makeTempDir,
|
|
|
|
makeTempFileSync: fs.makeTempFileSync,
|
|
|
|
makeTempFile: fs.makeTempFile,
|
2024-01-10 17:37:25 -05:00
|
|
|
memoryUsage: () => op_runtime_memory_usage(),
|
2023-02-07 14:22:46 -05:00
|
|
|
mkdirSync: fs.mkdirSync,
|
|
|
|
mkdir: fs.mkdir,
|
|
|
|
chdir: fs.chdir,
|
|
|
|
copyFile: fs.copyFile,
|
|
|
|
readDirSync: fs.readDirSync,
|
|
|
|
readDir: fs.readDir,
|
|
|
|
readLinkSync: fs.readLinkSync,
|
|
|
|
readLink: fs.readLink,
|
|
|
|
realPathSync: fs.realPathSync,
|
|
|
|
realPath: fs.realPath,
|
|
|
|
removeSync: fs.removeSync,
|
|
|
|
remove: fs.remove,
|
|
|
|
renameSync: fs.renameSync,
|
|
|
|
rename: fs.rename,
|
|
|
|
version: version.version,
|
2023-03-05 17:18:13 -05:00
|
|
|
build: core.build,
|
2023-02-07 14:22:46 -05:00
|
|
|
statSync: fs.statSync,
|
|
|
|
lstatSync: fs.lstatSync,
|
|
|
|
stat: fs.stat,
|
|
|
|
lstat: fs.lstat,
|
|
|
|
truncateSync: fs.truncateSync,
|
|
|
|
truncate: fs.truncate,
|
|
|
|
ftruncateSync: fs.ftruncateSync,
|
|
|
|
ftruncate: fs.ftruncate,
|
|
|
|
futime: fs.futime,
|
|
|
|
futimeSync: fs.futimeSync,
|
|
|
|
errors: errors.errors,
|
|
|
|
// TODO(kt3k): Remove this export at v2
|
|
|
|
// See https://github.com/denoland/deno/issues/9294
|
|
|
|
customInspect: console.customInspect,
|
|
|
|
inspect: console.inspect,
|
|
|
|
env: os.env,
|
|
|
|
exit: os.exit,
|
|
|
|
execPath: os.execPath,
|
|
|
|
Buffer: buffer.Buffer,
|
|
|
|
readAll: buffer.readAll,
|
|
|
|
readAllSync: buffer.readAllSync,
|
|
|
|
writeAll: buffer.writeAll,
|
|
|
|
writeAllSync: buffer.writeAllSync,
|
|
|
|
copy: io.copy,
|
|
|
|
iter: io.iter,
|
|
|
|
iterSync: io.iterSync,
|
|
|
|
SeekMode: io.SeekMode,
|
|
|
|
read: io.read,
|
|
|
|
readSync: io.readSync,
|
|
|
|
write: io.write,
|
|
|
|
writeSync: io.writeSync,
|
2023-03-04 22:37:37 -05:00
|
|
|
File: fs.File,
|
|
|
|
FsFile: fs.FsFile,
|
|
|
|
open: fs.open,
|
|
|
|
openSync: fs.openSync,
|
|
|
|
create: fs.create,
|
|
|
|
createSync: fs.createSync,
|
|
|
|
stdin: io.stdin,
|
|
|
|
stdout: io.stdout,
|
|
|
|
stderr: io.stderr,
|
|
|
|
seek: fs.seek,
|
|
|
|
seekSync: fs.seekSync,
|
2023-02-07 14:22:46 -05:00
|
|
|
connect: net.connect,
|
|
|
|
listen: net.listen,
|
|
|
|
loadavg: os.loadavg,
|
|
|
|
connectTls: tls.connectTls,
|
|
|
|
listenTls: tls.listenTls,
|
|
|
|
startTls: tls.startTls,
|
|
|
|
shutdown: net.shutdown,
|
|
|
|
fstatSync: fs.fstatSync,
|
|
|
|
fstat: fs.fstat,
|
|
|
|
fsyncSync: fs.fsyncSync,
|
|
|
|
fsync: fs.fsync,
|
|
|
|
fdatasyncSync: fs.fdatasyncSync,
|
|
|
|
fdatasync: fs.fdatasync,
|
|
|
|
symlink: fs.symlink,
|
|
|
|
symlinkSync: fs.symlinkSync,
|
|
|
|
link: fs.link,
|
|
|
|
linkSync: fs.linkSync,
|
|
|
|
permissions: permissions.permissions,
|
|
|
|
Permissions: permissions.Permissions,
|
|
|
|
PermissionStatus: permissions.PermissionStatus,
|
|
|
|
// TODO(bartlomieju): why is this not in one of extensions?
|
|
|
|
serveHttp: httpRuntime.serveHttp,
|
2023-07-04 19:04:26 -04:00
|
|
|
serve: http.serve,
|
2023-02-07 14:22:46 -05:00
|
|
|
resolveDns: net.resolveDns,
|
|
|
|
upgradeWebSocket: http.upgradeWebSocket,
|
|
|
|
utime: fs.utime,
|
|
|
|
utimeSync: fs.utimeSync,
|
|
|
|
kill: process.kill,
|
|
|
|
addSignalListener: signals.addSignalListener,
|
|
|
|
removeSignalListener: signals.removeSignalListener,
|
|
|
|
refTimer: timers.refTimer,
|
|
|
|
unrefTimer: timers.unrefTimer,
|
|
|
|
osRelease: os.osRelease,
|
|
|
|
osUptime: os.osUptime,
|
|
|
|
hostname: os.hostname,
|
|
|
|
systemMemoryInfo: os.systemMemoryInfo,
|
|
|
|
networkInterfaces: os.networkInterfaces,
|
|
|
|
consoleSize: tty.consoleSize,
|
|
|
|
gid: os.gid,
|
|
|
|
uid: os.uid,
|
2023-03-05 07:19:34 -05:00
|
|
|
Command: process.Command,
|
2023-02-13 13:25:00 -05:00
|
|
|
// TODO(bartlomieju): why is this exported?
|
2023-03-05 07:19:34 -05:00
|
|
|
ChildProcess: process.ChildProcess,
|
2023-02-07 14:22:46 -05:00
|
|
|
};
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-11-01 18:15:08 -04:00
|
|
|
// NOTE(bartlomieju): keep IDs in sync with `cli/main.rs`
|
2023-12-08 08:24:49 -05:00
|
|
|
const unstableIds = {
|
|
|
|
broadcastChannel: 1,
|
|
|
|
cron: 2,
|
|
|
|
ffi: 3,
|
|
|
|
fs: 4,
|
|
|
|
http: 5,
|
|
|
|
kv: 6,
|
|
|
|
net: 7,
|
2024-01-14 19:26:57 -05:00
|
|
|
temporal: 8,
|
|
|
|
unsafeProto: 9,
|
|
|
|
webgpu: 10,
|
|
|
|
workerOptions: 11,
|
2023-11-01 18:15:08 -04:00
|
|
|
};
|
|
|
|
|
2023-12-08 08:24:49 -05:00
|
|
|
const denoNsUnstableById = {};
|
|
|
|
|
|
|
|
// denoNsUnstableById[unstableIds.broadcastChannel] = {}
|
|
|
|
|
|
|
|
denoNsUnstableById[unstableIds.cron] = {
|
|
|
|
cron: cron.cron,
|
|
|
|
};
|
|
|
|
|
|
|
|
denoNsUnstableById[unstableIds.ffi] = {
|
|
|
|
dlopen: ffi.dlopen,
|
|
|
|
UnsafeCallback: ffi.UnsafeCallback,
|
|
|
|
UnsafePointer: ffi.UnsafePointer,
|
|
|
|
UnsafePointerView: ffi.UnsafePointerView,
|
|
|
|
UnsafeFnPointer: ffi.UnsafeFnPointer,
|
|
|
|
};
|
|
|
|
|
|
|
|
denoNsUnstableById[unstableIds.fs] = {
|
|
|
|
flock: fs.flock,
|
|
|
|
flockSync: fs.flockSync,
|
|
|
|
funlock: fs.funlock,
|
|
|
|
funlockSync: fs.funlockSync,
|
|
|
|
umask: fs.umask,
|
|
|
|
};
|
|
|
|
|
|
|
|
denoNsUnstableById[unstableIds.http] = {
|
|
|
|
HttpClient: httpClient.HttpClient,
|
|
|
|
createHttpClient: httpClient.createHttpClient,
|
|
|
|
// TODO(bartlomieju): why is it needed?
|
|
|
|
http,
|
|
|
|
};
|
|
|
|
|
|
|
|
denoNsUnstableById[unstableIds.kv] = {
|
|
|
|
openKv: kv.openKv,
|
|
|
|
AtomicOperation: kv.AtomicOperation,
|
|
|
|
Kv: kv.Kv,
|
|
|
|
KvU64: kv.KvU64,
|
|
|
|
KvListIterator: kv.KvListIterator,
|
|
|
|
};
|
|
|
|
|
|
|
|
denoNsUnstableById[unstableIds.net] = {
|
|
|
|
listenDatagram: net.createListenDatagram(
|
2024-01-10 17:37:25 -05:00
|
|
|
op_net_listen_udp,
|
|
|
|
op_net_listen_unixpacket,
|
2023-12-08 08:24:49 -05:00
|
|
|
),
|
|
|
|
};
|
|
|
|
|
|
|
|
// denoNsUnstableById[unstableIds.unsafeProto] = {}
|
|
|
|
|
feat:: External webgpu surfaces / BYOW (#21835)
This PR contains the implementation of the External webgpu surfaces /
BYOW proposal. BYOW stands for "Bring your own window".
Closes #21713
Adds `Deno.UnsafeWindowSurface` ( `--unstable-webgpu` API) to the `Deno`
namespace:
```typescript
class UnsafeWindowSurface {
constructor(
system: "cocoa" | "x11" | "win32",
winHandle: Deno.PointerValue,
displayHandle: Deno.PointerValue | null
);
getContext(type: "webgpu"): GPUCanvasContext;
present(): void;
}
```
For the initial pass, I've opted to support the three major windowing
systems. The parameters correspond to the table below:
| system | winHandle | displayHandle |
| ----------------- | ---------- | ------- |
| "cocoa" (macOS) | `NSView*` | - |
| "win32" (Windows) | `HWND` | `HINSTANCE` |
| "x11" (Linux) | Xlib `Window` | Xlib `Display*` |
Ecosystem support:
- [x] deno_sdl2 (sdl2) -
[mod.ts#L1209](https://github.com/littledivy/deno_sdl2/blob/7e177bc6524750a8849c25ce421798b2e71ec943/mod.ts#L1209)
- [x] dwm (glfw) - https://github.com/deno-windowing/dwm/issues/29
- [ ] pane (winit)
<details>
<summary>Example</summary>
```typescript
// A simple clear screen pass, colors based on mouse position.
import { EventType, WindowBuilder } from "https://deno.land/x/sdl2@0.7.0/mod.ts";
const window = new WindowBuilder("sdl2 + deno + webgpu", 640, 480).build();
const [system, windowHandle, displayHandle] = window.rawHandle();
const adapter = await navigator.gpu.requestAdapter();
const device = await adapter.requestDevice();
const context = Deno.createWindowSurface(system, windowHandle, displayHandle);
context.configure({
device: device,
format: "bgra8unorm",
height: 480,
width: 640,
});
let r = 0.0;
let g = 0.0;
let b = 0.0;
for (const event of window.events()) {
if (event.type === EventType.Quit) {
break;
} else if (event.type === EventType.Draw) {
const textureView = context.getCurrentTexture().createView();
const renderPassDescriptor: GPURenderPassDescriptor = {
colorAttachments: [
{
view: textureView,
clearValue: { r, g, b, a: 1.0 },
loadOp: "clear",
storeOp: "store",
},
],
};
const commandEncoder = device.createCommandEncoder();
const passEncoder = commandEncoder.beginRenderPass(renderPassDescriptor);
passEncoder.end();
device.queue.submit([commandEncoder.finish()]);
Deno.presentGPUCanvasContext(context);
}
if (event.type === EventType.MouseMotion) {
r = event.x / 640;
g = event.y / 480;
b = 1.0 - r - g;
}
}
```
You can find more examples in the linked tracking issue.
</details>
---------
Signed-off-by: Divy Srivastava <dj.srivastava23@gmail.com>
2024-01-19 12:19:14 -05:00
|
|
|
denoNsUnstableById[unstableIds.webgpu] = {
|
|
|
|
UnsafeWindowSurface: webgpuSurface.UnsafeWindowSurface,
|
|
|
|
};
|
2023-12-08 19:19:16 -05:00
|
|
|
|
2023-12-08 08:24:49 -05:00
|
|
|
// denoNsUnstableById[unstableIds.workerOptions] = {}
|
|
|
|
|
2023-07-13 19:29:51 -04:00
|
|
|
// when editing this list, also update unstableDenoProps in cli/tsc/99_main_compiler.js
|
2023-02-07 14:22:46 -05:00
|
|
|
const denoNsUnstable = {
|
2023-03-28 08:44:22 -04:00
|
|
|
listenDatagram: net.createListenDatagram(
|
2024-01-10 17:37:25 -05:00
|
|
|
op_net_listen_udp,
|
|
|
|
op_net_listen_unixpacket,
|
2023-03-28 08:44:22 -04:00
|
|
|
),
|
2023-02-07 14:22:46 -05:00
|
|
|
umask: fs.umask,
|
|
|
|
HttpClient: httpClient.HttpClient,
|
|
|
|
createHttpClient: httpClient.createHttpClient,
|
|
|
|
// TODO(bartlomieju): why is it needed?
|
|
|
|
http,
|
|
|
|
dlopen: ffi.dlopen,
|
|
|
|
UnsafeCallback: ffi.UnsafeCallback,
|
|
|
|
UnsafePointer: ffi.UnsafePointer,
|
|
|
|
UnsafePointerView: ffi.UnsafePointerView,
|
|
|
|
UnsafeFnPointer: ffi.UnsafeFnPointer,
|
feat:: External webgpu surfaces / BYOW (#21835)
This PR contains the implementation of the External webgpu surfaces /
BYOW proposal. BYOW stands for "Bring your own window".
Closes #21713
Adds `Deno.UnsafeWindowSurface` ( `--unstable-webgpu` API) to the `Deno`
namespace:
```typescript
class UnsafeWindowSurface {
constructor(
system: "cocoa" | "x11" | "win32",
winHandle: Deno.PointerValue,
displayHandle: Deno.PointerValue | null
);
getContext(type: "webgpu"): GPUCanvasContext;
present(): void;
}
```
For the initial pass, I've opted to support the three major windowing
systems. The parameters correspond to the table below:
| system | winHandle | displayHandle |
| ----------------- | ---------- | ------- |
| "cocoa" (macOS) | `NSView*` | - |
| "win32" (Windows) | `HWND` | `HINSTANCE` |
| "x11" (Linux) | Xlib `Window` | Xlib `Display*` |
Ecosystem support:
- [x] deno_sdl2 (sdl2) -
[mod.ts#L1209](https://github.com/littledivy/deno_sdl2/blob/7e177bc6524750a8849c25ce421798b2e71ec943/mod.ts#L1209)
- [x] dwm (glfw) - https://github.com/deno-windowing/dwm/issues/29
- [ ] pane (winit)
<details>
<summary>Example</summary>
```typescript
// A simple clear screen pass, colors based on mouse position.
import { EventType, WindowBuilder } from "https://deno.land/x/sdl2@0.7.0/mod.ts";
const window = new WindowBuilder("sdl2 + deno + webgpu", 640, 480).build();
const [system, windowHandle, displayHandle] = window.rawHandle();
const adapter = await navigator.gpu.requestAdapter();
const device = await adapter.requestDevice();
const context = Deno.createWindowSurface(system, windowHandle, displayHandle);
context.configure({
device: device,
format: "bgra8unorm",
height: 480,
width: 640,
});
let r = 0.0;
let g = 0.0;
let b = 0.0;
for (const event of window.events()) {
if (event.type === EventType.Quit) {
break;
} else if (event.type === EventType.Draw) {
const textureView = context.getCurrentTexture().createView();
const renderPassDescriptor: GPURenderPassDescriptor = {
colorAttachments: [
{
view: textureView,
clearValue: { r, g, b, a: 1.0 },
loadOp: "clear",
storeOp: "store",
},
],
};
const commandEncoder = device.createCommandEncoder();
const passEncoder = commandEncoder.beginRenderPass(renderPassDescriptor);
passEncoder.end();
device.queue.submit([commandEncoder.finish()]);
Deno.presentGPUCanvasContext(context);
}
if (event.type === EventType.MouseMotion) {
r = event.x / 640;
g = event.y / 480;
b = 1.0 - r - g;
}
}
```
You can find more examples in the linked tracking issue.
</details>
---------
Signed-off-by: Divy Srivastava <dj.srivastava23@gmail.com>
2024-01-19 12:19:14 -05:00
|
|
|
UnsafeWindowSurface: webgpuSurface.UnsafeWindowSurface,
|
2023-02-07 14:22:46 -05:00
|
|
|
flock: fs.flock,
|
|
|
|
flockSync: fs.flockSync,
|
|
|
|
funlock: fs.funlock,
|
|
|
|
funlockSync: fs.funlockSync,
|
2023-03-22 15:23:36 -04:00
|
|
|
openKv: kv.openKv,
|
2023-07-01 03:24:15 -04:00
|
|
|
AtomicOperation: kv.AtomicOperation,
|
2023-03-22 00:13:24 -04:00
|
|
|
Kv: kv.Kv,
|
|
|
|
KvU64: kv.KvU64,
|
|
|
|
KvListIterator: kv.KvListIterator,
|
2023-11-01 14:57:55 -04:00
|
|
|
cron: cron.cron,
|
2023-02-07 14:22:46 -05:00
|
|
|
};
|
|
|
|
|
2023-12-08 08:24:49 -05:00
|
|
|
export { denoNs, denoNsUnstable, denoNsUnstableById, unstableIds };
|