From c40048681292abad7d79c66bb1dc3559c2f905b5 Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Mon, 9 Oct 2023 18:25:58 +0900 Subject: [PATCH 1/9] feat: disposable Deno resources This commit implements Symbol.dispose and Symbol.asyncDispose for the relevant resources. --- ext/fs/30_fs.js | 6 +++++- ext/http/00_serve.js | 22 +++++++++++++++------- ext/http/01_http.js | 27 ++++++++++++++++++++++----- ext/io/12_io.js | 5 +++++ ext/kv/01_db.ts | 5 +++++ ext/kv/lib.rs | 2 +- ext/net/01_net.js | 10 ++++++++++ ext/web/00_infra.js | 5 +++++ runtime/js/40_fs_events.js | 6 ++++++ runtime/js/40_process.js | 18 ++++++++++-------- runtime/js/99_main.js | 22 +++++++++++++++++----- 11 files changed, 101 insertions(+), 27 deletions(-) diff --git a/ext/fs/30_fs.js b/ext/fs/30_fs.js index d2a656a23e1f8c..830703230930c5 100644 --- a/ext/fs/30_fs.js +++ b/ext/fs/30_fs.js @@ -34,7 +34,7 @@ import { ReadableStreamPrototype, writableStreamForRid, } from "ext:deno_web/06_streams.js"; -import { pathFromURL } from "ext:deno_web/00_infra.js"; +import { pathFromURL, SymbolDispose } from "ext:deno_web/00_infra.js"; function chmodSync(path, mode) { ops.op_fs_chmod_sync(pathFromURL(path), mode); @@ -669,6 +669,10 @@ class FsFile { } return this.#writable; } + + [SymbolDispose]() { + core.tryClose(this.rid); + } } function checkOpenOptions(options) { diff --git a/ext/http/00_serve.js b/ext/http/00_serve.js index 2e0c62fefa940a..c2b3e86d63f6dd 100644 --- a/ext/http/00_serve.js +++ b/ext/http/00_serve.js @@ -36,6 +36,7 @@ import { } from "ext:deno_web/06_streams.js"; import { listen, listenOptionApiName, TcpConn } from "ext:deno_net/01_net.js"; import { listenTls } from "ext:deno_net/02_tls.js"; +import { SymbolAsyncDispose } from "ext:deno_web/00_infra.js"; const { ArrayPrototypePush, ObjectHasOwn, @@ -343,6 +344,7 @@ class CallbackContext { fallbackHost; serverRid; closed; + /** @type {Promise | undefined} */ closing; listener; @@ -682,22 +684,25 @@ function serveHttpOn(context, callback) { PromisePrototypeCatch(callback(req), promiseErrorHandler); } - if (!context.closed && !context.closing) { - context.closed = true; - await op_http_close(rid, false); + if (!context.closing && !context.closed) { + context.closing = op_http_close(rid, false); context.close(); } + + await context.closing; + context.close(); + context.closed = true; })(); return { finished, async shutdown() { - if (!context.closed && !context.closing) { + if (!context.closing && !context.closed) { // Shut this HTTP server down gracefully - context.closing = true; - await op_http_close(context.serverRid, true); - context.closed = true; + context.closing = op_http_close(rid, true); } + await context.closing; + context.closed = true; }, ref() { ref = true; @@ -711,6 +716,9 @@ function serveHttpOn(context, callback) { core.unrefOp(currentPromise[promiseIdSymbol]); } }, + [SymbolAsyncDispose]() { + return this.shutdown(); + }, }; } diff --git a/ext/http/01_http.js b/ext/http/01_http.js index 705e616e4ad443..cfe0f710e8ad56 100644 --- a/ext/http/01_http.js +++ b/ext/http/01_http.js @@ -44,6 +44,7 @@ import { ReadableStreamPrototype, } from "ext:deno_web/06_streams.js"; import { serve } from "ext:deno_http/00_serve.js"; +import { SymbolDispose } from "ext:deno_web/00_infra.js"; const { ArrayPrototypeIncludes, ArrayPrototypeMap, @@ -69,6 +70,9 @@ const { const connErrorSymbol = Symbol("connError"); const _deferred = Symbol("upgradeHttpDeferred"); +/** @type {(self: HttpConn, rid: number) => boolean} */ +let deleteManagedResource; + class HttpConn { #rid = 0; #closed = false; @@ -79,7 +83,12 @@ class HttpConn { // that were created during lifecycle of this request. // When the connection is closed these resources should be closed // as well. - managedResources = new SafeSet(); + #managedResources = new SafeSet(); + + static { + deleteManagedResource = (self, rid) => + SetPrototypeDelete(self.#managedResources, rid); + } constructor(rid, remoteAddr, localAddr) { this.#rid = rid; @@ -123,7 +132,7 @@ class HttpConn { } const { 0: streamRid, 1: method, 2: url } = nextRequest; - SetPrototypeAdd(this.managedResources, streamRid); + SetPrototypeAdd(this.#managedResources, streamRid); /** @type {ReadableStream | undefined} */ let body = null; @@ -167,13 +176,21 @@ class HttpConn { if (!this.#closed) { this.#closed = true; core.close(this.#rid); - for (const rid of new SafeSetIterator(this.managedResources)) { - SetPrototypeDelete(this.managedResources, rid); + for (const rid of new SafeSetIterator(this.#managedResources)) { + SetPrototypeDelete(this.#managedResources, rid); core.close(rid); } } } + [SymbolDispose]() { + core.tryClose(this.#rid); + for (const rid of new SafeSetIterator(this.#managedResources)) { + SetPrototypeDelete(this.#managedResources, rid); + core.tryClose(rid); + } + } + [SymbolAsyncIterator]() { // deno-lint-ignore no-this-alias const httpConn = this; @@ -395,7 +412,7 @@ function createRespondWith( abortController.abort(error); throw error; } finally { - if (SetPrototypeDelete(httpConn.managedResources, streamRid)) { + if (deleteManagedResource(httpConn, streamRid)) { core.close(streamRid); } } diff --git a/ext/io/12_io.js b/ext/io/12_io.js index 1bb8f9fba90083..65e11087827a7a 100644 --- a/ext/io/12_io.js +++ b/ext/io/12_io.js @@ -11,6 +11,7 @@ import { readableStreamForRid, writableStreamForRid, } from "ext:deno_web/06_streams.js"; +import { SymbolDispose } from "ext:deno_web/00_infra.js"; const { Uint8Array, ArrayPrototypePush, @@ -255,6 +256,10 @@ class Stdin { const cbreak = !!(options.cbreak ?? false); ops.op_stdin_set_raw(mode, cbreak); } + + [SymbolDispose]() { + core.tryClose(this.rid); + } } class Stdout { diff --git a/ext/kv/01_db.ts b/ext/kv/01_db.ts index 6e8a571f0c4186..34678261a7ba88 100644 --- a/ext/kv/01_db.ts +++ b/ext/kv/01_db.ts @@ -12,6 +12,7 @@ const { SymbolToStringTag, Uint8ArrayPrototype, } = globalThis.__bootstrap.primordials; +import { SymbolDispose } from "ext:deno_web/00_infra.js"; const core = Deno.core; const ops = core.ops; @@ -299,6 +300,10 @@ class Kv { close() { core.close(this.#rid); } + + [SymbolDispose]() { + core.tryClose(this.#rid); + } } class AtomicOperation { diff --git a/ext/kv/lib.rs b/ext/kv/lib.rs index 3056e4660f2814..35e8409b717643 100644 --- a/ext/kv/lib.rs +++ b/ext/kv/lib.rs @@ -45,7 +45,7 @@ const MAX_TOTAL_MUTATION_SIZE_BYTES: usize = 800 * 1024; const MAX_TOTAL_KEY_SIZE_BYTES: usize = 80 * 1024; deno_core::extension!(deno_kv, - deps = [ deno_console ], + deps = [ deno_console, deno_web ], parameters = [ DBH: DatabaseHandler ], ops = [ op_kv_database_open, diff --git a/ext/net/01_net.js b/ext/net/01_net.js index f2bf5e7dfa2c64..05aca07e54450f 100644 --- a/ext/net/01_net.js +++ b/ext/net/01_net.js @@ -9,6 +9,8 @@ import { writableStreamForRid, } from "ext:deno_web/06_streams.js"; import * as abortSignal from "ext:deno_web/03_abort_signal.js"; +import { SymbolDispose } from "ext:deno_web/00_infra.js"; + const primordials = globalThis.__bootstrap.primordials; const { ArrayPrototypeFilter, @@ -160,6 +162,10 @@ class Conn { (id) => core.unrefOp(id), ); } + + [SymbolDispose]() { + core.tryClose(this.#rid); + } } class TcpConn extends Conn { @@ -249,6 +255,10 @@ class Listener { core.close(this.rid); } + [SymbolDispose]() { + core.tryClose(this.#rid); + } + [SymbolAsyncIterator]() { return this; } diff --git a/ext/web/00_infra.js b/ext/web/00_infra.js index e9d5dd48b593e3..d553f1dc62e758 100644 --- a/ext/web/00_infra.js +++ b/ext/web/00_infra.js @@ -458,6 +458,11 @@ function pathFromURL(pathOrUrl) { // it in unit tests internals.pathFromURL = pathFromURL; +// deno-lint-ignore prefer-primordials +export const SymbolDispose = Symbol.dispose ?? Symbol("Symbol.dispose"); +// deno-lint-ignore prefer-primordials +export const SymbolAsyncDispose = Symbol.asyncDispose ?? Symbol("Symbol.asyncDispose"); + export { ASCII_ALPHA, ASCII_ALPHANUMERIC, diff --git a/runtime/js/40_fs_events.js b/runtime/js/40_fs_events.js index 4c2f5fc9a5920d..b10f6fd10e93cf 100644 --- a/runtime/js/40_fs_events.js +++ b/runtime/js/40_fs_events.js @@ -9,6 +9,8 @@ const { PromiseResolve, SymbolAsyncIterator, } = primordials; +import { SymbolDispose } from "ext:deno_web/00_infra.js"; + class FsWatcher { #rid = 0; @@ -51,6 +53,10 @@ class FsWatcher { [SymbolAsyncIterator]() { return this; } + + [SymbolDispose]() { + core.tryClose(this.#rid); + } } function watchFs( diff --git a/runtime/js/40_process.js b/runtime/js/40_process.js index 664a4b303deeb7..f5a40b9ca9c4e2 100644 --- a/runtime/js/40_process.js +++ b/runtime/js/40_process.js @@ -18,7 +18,7 @@ const { } = primordials; import { FsFile } from "ext:deno_fs/30_fs.js"; import { readAll } from "ext:deno_io/12_io.js"; -import { assert, pathFromURL } from "ext:deno_web/00_infra.js"; +import { assert, pathFromURL, SymbolAsyncDispose } from "ext:deno_web/00_infra.js"; import * as abortSignal from "ext:deno_web/03_abort_signal.js"; import { readableStreamCollectIntoUint8Array, @@ -201,7 +201,6 @@ class ChildProcess { #rid; #waitPromiseId; #waitComplete = false; - #unrefed = false; #pid; get pid() { @@ -216,7 +215,6 @@ class ChildProcess { return this.#stdin; } - #stdoutRid; #stdout = null; get stdout() { if (this.#stdout == null) { @@ -225,7 +223,6 @@ class ChildProcess { return this.#stdout; } - #stderrRid; #stderr = null; get stderr() { if (this.#stderr == null) { @@ -254,12 +251,10 @@ class ChildProcess { } if (stdoutRid !== null) { - this.#stdoutRid = stdoutRid; this.#stdout = readableStreamForRidUnrefable(stdoutRid); } if (stderrRid !== null) { - this.#stderrRid = stderrRid; this.#stderr = readableStreamForRidUnrefable(stderrRid); } @@ -324,15 +319,22 @@ class ChildProcess { ops.op_spawn_kill(this.#rid, signo); } + async [SymbolAsyncDispose]() { + try { + ops.op_spawn_kill(this.#rid, "SIGTERM") + } catch { + // ignore errors from killing the process (such as ESRCH or BadResource) + } + await this.#status; + } + ref() { - this.#unrefed = false; core.refOp(this.#waitPromiseId); if (this.#stdout) readableStreamForRidUnrefableRef(this.#stdout); if (this.#stderr) readableStreamForRidUnrefableRef(this.#stderr); } unref() { - this.#unrefed = true; core.unrefOp(this.#waitPromiseId); if (this.#stdout) readableStreamForRidUnrefableUnref(this.#stdout); if (this.#stderr) readableStreamForRidUnrefableUnref(this.#stderr); diff --git a/runtime/js/99_main.js b/runtime/js/99_main.js index ccc61036acbf9f..17326520b1e9c1 100644 --- a/runtime/js/99_main.js +++ b/runtime/js/99_main.js @@ -70,11 +70,23 @@ import { windowOrWorkerGlobalScope, workerRuntimeGlobalProperties, } from "ext:runtime/98_global_scope.js"; - -// deno-lint-ignore prefer-primordials -Symbol.dispose ??= Symbol("Symbol.dispose"); -// deno-lint-ignore prefer-primordials -Symbol.asyncDispose ??= Symbol("Symbol.asyncDispose"); +import { SymbolAsyncDispose, SymbolDispose } from "ext:deno_web/00_infra.js"; + +if (Symbol.dispose) throw "V8 supports Symbol.dispose now, no need to shim it!"; +ObjectDefineProperties(Symbol, { + dispose: { + value: SymbolAsyncDispose, + enumerable: false, + writable: false, + configurable: false, + }, + asyncDispose: { + value: SymbolDispose, + enumerable: false, + writable: false, + configurable: false, + }, +}); let windowIsClosing = false; let globalThis_; From f65268226b58b8b12950730ae7a8345f5662799e Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Tue, 10 Oct 2023 16:33:41 +0900 Subject: [PATCH 2/9] first test --- cli/tests/unit/files_test.ts | 28 ++++++++++++++++++++++++++++ cli/tsc/99_main_compiler.js | 6 +++++- cli/tsc/dts/lib.deno.ns.d.ts | 16 +++++++++++----- cli/tsc/dts/lib.deno.unstable.d.ts | 4 +++- ext/io/12_io.js | 5 ----- ext/net/lib.deno_net.d.ts | 6 ++++-- runtime/js/99_main.js | 4 ++-- 7 files changed, 53 insertions(+), 16 deletions(-) diff --git a/cli/tests/unit/files_test.ts b/cli/tests/unit/files_test.ts index 873c70c8627eb3..213131f06d615b 100644 --- a/cli/tests/unit/files_test.ts +++ b/cli/tests/unit/files_test.ts @@ -824,3 +824,31 @@ Deno.test( assertEquals(res, "hello \uFFFD"); }, ); + +Deno.test( + { permissions: { read: true } }, + async function fsFileExplicitResourceManagement() { + let file2: Deno.FsFile; + + { + using file = await Deno.open("cli/tests/testdata/assets/hello.txt"); + file2 = file; + + const stat = file.statSync(); + assert(stat.isFile); + } + + assertThrows(() => file2.statSync(), Deno.errors.BadResource); + }, +); + +Deno.test( + { permissions: { read: true } }, + async function fsFileExplicitResourceManagementManualClose() { + using file = await Deno.open("cli/tests/testdata/assets/hello.txt"); + + file.close(); + + assertThrows(() => file.statSync(), Deno.errors.BadResource); + }, +); diff --git a/cli/tsc/99_main_compiler.js b/cli/tsc/99_main_compiler.js index e246d5d0c36dfe..a2079c01edf332 100644 --- a/cli/tsc/99_main_compiler.js +++ b/cli/tsc/99_main_compiler.js @@ -1138,6 +1138,7 @@ delete Object.prototype.__proto__; `${ASSETS_URL_PREFIX}${specifier}`, ts.ScriptTarget.ESNext, ), + `failed to load '${ASSETS_URL_PREFIX}${specifier}'`, ); } // this helps ensure as much as possible is in memory that is re-usable @@ -1148,7 +1149,10 @@ delete Object.prototype.__proto__; options: SNAPSHOT_COMPILE_OPTIONS, host, }); - assert(ts.getPreEmitDiagnostics(TS_SNAPSHOT_PROGRAM).length === 0); + assert( + ts.getPreEmitDiagnostics(TS_SNAPSHOT_PROGRAM).length === 0, + "lib.d.ts files have errors", + ); // remove this now that we don't need it anymore for warming up tsc sourceFileCache.delete(buildSpecifier); diff --git a/cli/tsc/dts/lib.deno.ns.d.ts b/cli/tsc/dts/lib.deno.ns.d.ts index 6784ee0529b884..a731081731d12c 100644 --- a/cli/tsc/dts/lib.deno.ns.d.ts +++ b/cli/tsc/dts/lib.deno.ns.d.ts @@ -2,6 +2,7 @@ /// /// +/// /// /** Deno provides extra properties on `import.meta`. These are included here @@ -2177,7 +2178,8 @@ declare namespace Deno { WriterSync, Seeker, SeekerSync, - Closer { + Closer, + Disposable { /** The resource ID associated with the file instance. The resource ID * should be considered an opaque reference to resource. */ readonly rid: number; @@ -2451,6 +2453,8 @@ declare namespace Deno { * ``` */ close(): void; + + [Symbol.dispose](): void; } /** @@ -3831,7 +3835,7 @@ declare namespace Deno { * * @category File System */ - export interface FsWatcher extends AsyncIterable { + export interface FsWatcher extends AsyncIterable, Disposable { /** The resource id. */ readonly rid: number; /** Stops watching the file system and closes the watcher resource. */ @@ -4284,7 +4288,7 @@ declare namespace Deno { * * @category Sub Process */ - export class ChildProcess { + export class ChildProcess implements Disposable { get stdin(): WritableStream; get stdout(): ReadableStream; get stderr(): ReadableStream; @@ -4307,6 +4311,8 @@ declare namespace Deno { /** Ensure that the status of the child process does not block the Deno * process from exiting. */ unref(): void; + + [Symbol.dispose](): void; } /** @@ -5258,7 +5264,7 @@ declare namespace Deno { * requests on the HTTP server connection. * * @category HTTP Server */ - export interface HttpConn extends AsyncIterable { + export interface HttpConn extends AsyncIterable, Disposable { /** The resource ID associated with this connection. Generally users do not * need to be aware of this identifier. */ readonly rid: number; @@ -5911,7 +5917,7 @@ declare namespace Deno { * * @category HTTP Server */ - export interface Server { + export interface Server extends Disposable { /** A promise that resolves once server finishes - eg. when aborted using * the signal passed to {@linkcode ServeOptions.signal}. */ diff --git a/cli/tsc/dts/lib.deno.unstable.d.ts b/cli/tsc/dts/lib.deno.unstable.d.ts index 782e8eba42271b..e984b8c0f6a089 100644 --- a/cli/tsc/dts/lib.deno.unstable.d.ts +++ b/cli/tsc/dts/lib.deno.unstable.d.ts @@ -1724,7 +1724,7 @@ declare namespace Deno { * * @category KV */ - export class Kv { + export class Kv implements Disposable { /** * Retrieve the value and versionstamp for the given key from the database * in the form of a {@linkcode Deno.KvEntryMaybe}. If no value exists for @@ -1920,6 +1920,8 @@ declare namespace Deno { * operations immediately. */ close(): void; + + [Symbol.dispose](): void; } /** **UNSTABLE**: New API, yet to be vetted. diff --git a/ext/io/12_io.js b/ext/io/12_io.js index 65e11087827a7a..1bb8f9fba90083 100644 --- a/ext/io/12_io.js +++ b/ext/io/12_io.js @@ -11,7 +11,6 @@ import { readableStreamForRid, writableStreamForRid, } from "ext:deno_web/06_streams.js"; -import { SymbolDispose } from "ext:deno_web/00_infra.js"; const { Uint8Array, ArrayPrototypePush, @@ -256,10 +255,6 @@ class Stdin { const cbreak = !!(options.cbreak ?? false); ops.op_stdin_set_raw(mode, cbreak); } - - [SymbolDispose]() { - core.tryClose(this.rid); - } } class Stdout { diff --git a/ext/net/lib.deno_net.d.ts b/ext/net/lib.deno_net.d.ts index 030157d6368501..79e15563a6f121 100644 --- a/ext/net/lib.deno_net.d.ts +++ b/ext/net/lib.deno_net.d.ts @@ -2,6 +2,7 @@ /// /// +/// declare namespace Deno { /** @category Network */ @@ -24,7 +25,8 @@ declare namespace Deno { * * @category Network */ - export interface Listener extends AsyncIterable { + export interface Listener + extends AsyncIterable, Disposable { /** Waits for and resolves to the next connection to the `Listener`. */ accept(): Promise; /** Close closes the listener. Any pending accept promises will be rejected @@ -57,7 +59,7 @@ declare namespace Deno { export type TlsListener = Listener; /** @category Network */ - export interface Conn extends Reader, Writer, Closer { + export interface Conn extends Reader, Writer, Closer, Disposable { /** The local address of the connection. */ readonly localAddr: Addr; /** The remote address of the connection. */ diff --git a/runtime/js/99_main.js b/runtime/js/99_main.js index 17326520b1e9c1..5e50329e6259e9 100644 --- a/runtime/js/99_main.js +++ b/runtime/js/99_main.js @@ -75,13 +75,13 @@ import { SymbolAsyncDispose, SymbolDispose } from "ext:deno_web/00_infra.js"; if (Symbol.dispose) throw "V8 supports Symbol.dispose now, no need to shim it!"; ObjectDefineProperties(Symbol, { dispose: { - value: SymbolAsyncDispose, + value: SymbolDispose, enumerable: false, writable: false, configurable: false, }, asyncDispose: { - value: SymbolDispose, + value: SymbolAsyncDispose, enumerable: false, writable: false, configurable: false, From 7a5b7f2d059c3f933148ca68a26ab4357f62ba7f Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Tue, 10 Oct 2023 18:42:02 +0900 Subject: [PATCH 3/9] more tests --- cli/tests/unit/files_test.ts | 5 ++--- cli/tests/unit/net_test.ts | 31 +++++++++++++++++++++++++++++++ 2 files changed, 33 insertions(+), 3 deletions(-) diff --git a/cli/tests/unit/files_test.ts b/cli/tests/unit/files_test.ts index 213131f06d615b..3e0390ae6e583d 100644 --- a/cli/tests/unit/files_test.ts +++ b/cli/tests/unit/files_test.ts @@ -846,9 +846,8 @@ Deno.test( { permissions: { read: true } }, async function fsFileExplicitResourceManagementManualClose() { using file = await Deno.open("cli/tests/testdata/assets/hello.txt"); - file.close(); - - assertThrows(() => file.statSync(), Deno.errors.BadResource); + assertThrows(() => file.statSync(), Deno.errors.BadResource); // definitely closed + // calling [Symbol.dispose] after manual close is a no-op }, ); diff --git a/cli/tests/unit/net_test.ts b/cli/tests/unit/net_test.ts index 2a98b5e26f3e0a..db99d2480a8a21 100644 --- a/cli/tests/unit/net_test.ts +++ b/cli/tests/unit/net_test.ts @@ -1242,3 +1242,34 @@ Deno.test({ const listener = Deno.listen({ hostname: "localhost", port: "0" }); listener.close(); }); + +Deno.test( + { permissions: { net: true } }, + async function listenerExplicitResourceManagement() { + let done: Promise; + + { + using listener = Deno.listen({ port: listenPort }); + + done = assertRejects( + () => listener.accept(), + Deno.errors.BadResource, + ); + } + + await done; + }, +); + +Deno.test( + { permissions: { net: true } }, + async function listenerExplicitResourceManagementManualClose() { + using listener = Deno.listen({ port: listenPort }); + listener.close(); + await assertRejects( // definitely closed + () => listener.accept(), + Deno.errors.BadResource, + ); + // calling [Symbol.dispose] after manual close is a no-op + }, +); From a15363042992e7d848f073c715b3d03e3f6cc6d7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bartek=20Iwa=C5=84czuk?= Date: Tue, 31 Oct 2023 12:52:58 +0100 Subject: [PATCH 4/9] format --- ext/web/00_infra.js | 3 ++- runtime/js/40_process.js | 10 +++++++--- 2 files changed, 9 insertions(+), 4 deletions(-) diff --git a/ext/web/00_infra.js b/ext/web/00_infra.js index d553f1dc62e758..7ce1857b79936c 100644 --- a/ext/web/00_infra.js +++ b/ext/web/00_infra.js @@ -461,7 +461,8 @@ internals.pathFromURL = pathFromURL; // deno-lint-ignore prefer-primordials export const SymbolDispose = Symbol.dispose ?? Symbol("Symbol.dispose"); // deno-lint-ignore prefer-primordials -export const SymbolAsyncDispose = Symbol.asyncDispose ?? Symbol("Symbol.asyncDispose"); +export const SymbolAsyncDispose = Symbol.asyncDispose ?? + Symbol("Symbol.asyncDispose"); export { ASCII_ALPHA, diff --git a/runtime/js/40_process.js b/runtime/js/40_process.js index f5a40b9ca9c4e2..4ddc4a02a8cd04 100644 --- a/runtime/js/40_process.js +++ b/runtime/js/40_process.js @@ -18,7 +18,11 @@ const { } = primordials; import { FsFile } from "ext:deno_fs/30_fs.js"; import { readAll } from "ext:deno_io/12_io.js"; -import { assert, pathFromURL, SymbolAsyncDispose } from "ext:deno_web/00_infra.js"; +import { + assert, + pathFromURL, + SymbolAsyncDispose, +} from "ext:deno_web/00_infra.js"; import * as abortSignal from "ext:deno_web/03_abort_signal.js"; import { readableStreamCollectIntoUint8Array, @@ -320,8 +324,8 @@ class ChildProcess { } async [SymbolAsyncDispose]() { - try { - ops.op_spawn_kill(this.#rid, "SIGTERM") + try { + ops.op_spawn_kill(this.#rid, "SIGTERM"); } catch { // ignore errors from killing the process (such as ESRCH or BadResource) } From 862c60a3e6cb0dcd349ba095c6fd8d301cb8237a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bartek=20Iwa=C5=84czuk?= Date: Tue, 31 Oct 2023 13:00:48 +0100 Subject: [PATCH 5/9] lint --- ext/web/00_infra.js | 1 + runtime/js/99_main.js | 1 + 2 files changed, 2 insertions(+) diff --git a/ext/web/00_infra.js b/ext/web/00_infra.js index 7ce1857b79936c..44108160415dfb 100644 --- a/ext/web/00_infra.js +++ b/ext/web/00_infra.js @@ -32,6 +32,7 @@ const { StringPrototypeSubstring, StringPrototypeToLowerCase, StringPrototypeToUpperCase, + Symbol, TypeError, } = primordials; import { URLPrototype } from "ext:deno_url/00_url.js"; diff --git a/runtime/js/99_main.js b/runtime/js/99_main.js index 5e50329e6259e9..ac1f52e749ad63 100644 --- a/runtime/js/99_main.js +++ b/runtime/js/99_main.js @@ -72,6 +72,7 @@ import { } from "ext:runtime/98_global_scope.js"; import { SymbolAsyncDispose, SymbolDispose } from "ext:deno_web/00_infra.js"; +// deno-lint-ignore prefer-primordials if (Symbol.dispose) throw "V8 supports Symbol.dispose now, no need to shim it!"; ObjectDefineProperties(Symbol, { dispose: { From 4898970510ccdbb9bd11e99aabc544fe0f0a00a4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bartek=20Iwa=C5=84czuk?= Date: Tue, 31 Oct 2023 13:42:07 +0100 Subject: [PATCH 6/9] Update ext/http/00_serve.js MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Bartek IwaƄczuk --- ext/http/00_serve.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ext/http/00_serve.js b/ext/http/00_serve.js index b0d3d64ae72419..b99f28c0fb76a1 100644 --- a/ext/http/00_serve.js +++ b/ext/http/00_serve.js @@ -688,7 +688,7 @@ function serveHttpOn(context, callback) { async shutdown() { if (!context.closing && !context.closed) { // Shut this HTTP server down gracefully - context.closing = op_http_close(rid, true); + context.closing = op_http_close(context.serverRid, true); } await context.closing; context.closed = true; From dc8a1a3bfb6293b9ea3d7259e0dc499750eb8c7f Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Wed, 1 Nov 2023 16:05:31 +0100 Subject: [PATCH 7/9] add more tests --- cli/tests/unit/command_test.ts | 36 +++++++++++++++++++++++++++ cli/tests/unit/fs_events_test.ts | 30 +++++++++++++++++++++++ cli/tests/unit/http_test.ts | 18 ++++++++++++++ cli/tests/unit/kv_test.ts | 27 ++++++++++++++++++++ cli/tests/unit/serve_test.ts | 42 +++++++++++++++++++++++++++++++- cli/tsc/dts/lib.deno.ns.d.ts | 2 +- 6 files changed, 153 insertions(+), 2 deletions(-) diff --git a/cli/tests/unit/command_test.ts b/cli/tests/unit/command_test.ts index 5f56a0c229af63..ed3da29270a4d1 100644 --- a/cli/tests/unit/command_test.ts +++ b/cli/tests/unit/command_test.ts @@ -256,6 +256,42 @@ Deno.test( }, ); +Deno.test( + { permissions: { run: true, read: true } }, + // deno lint bug, see https://github.com/denoland/deno_lint/issues/1206 + // deno-lint-ignore require-await + async function childProcessExplicitResourceManagement() { + let dead = undefined; + { + const command = new Deno.Command(Deno.execPath(), { + args: ["eval", "setTimeout(() => {}, 10000)"], + stdout: "null", + stderr: "null", + }); + await using child = command.spawn(); + child.status.then(({ signal }) => { + dead = signal; + }); + } + + assertEquals(dead, "SIGTERM"); + }, +); + +Deno.test( + { permissions: { run: true, read: true } }, + async function childProcessExplicitResourceManagementManualClose() { + const command = new Deno.Command(Deno.execPath(), { + args: ["eval", "setTimeout(() => {}, 10000)"], + stdout: "null", + stderr: "null", + }); + await using child = command.spawn(); + child.kill("SIGTERM"); + await child.status; + }, +); + Deno.test( { permissions: { run: true, read: true } }, async function commandKillFailed() { diff --git a/cli/tests/unit/fs_events_test.ts b/cli/tests/unit/fs_events_test.ts index 9330f20072aaeb..86adeb4d77322f 100644 --- a/cli/tests/unit/fs_events_test.ts +++ b/cli/tests/unit/fs_events_test.ts @@ -107,3 +107,33 @@ Deno.test( assertEquals(events, []); }, ); + +Deno.test( + { permissions: { read: true, write: true } }, + async function watchFsExplicitResourceManagement() { + let res; + { + const testDir = await makeTempDir(); + using iter = Deno.watchFs(testDir); + + res = iter[Symbol.asyncIterator]().next(); + } + + const { done } = await res; + assert(done); + }, +); + +Deno.test( + { permissions: { read: true, write: true } }, + async function watchFsExplicitResourceManagementManualClose() { + const testDir = await makeTempDir(); + using iter = Deno.watchFs(testDir); + + const res = iter[Symbol.asyncIterator]().next(); + + iter.close(); + const { done } = await res; + assert(done); + }, +); diff --git a/cli/tests/unit/http_test.ts b/cli/tests/unit/http_test.ts index 10414cab376ee8..8c7bf9974178c3 100644 --- a/cli/tests/unit/http_test.ts +++ b/cli/tests/unit/http_test.ts @@ -2817,6 +2817,24 @@ Deno.test({ }, }); +Deno.test( + async function httpConnExplicitResourceManagement() { + let promise; + + { + const listen = Deno.listen({ port: listenPort }); + promise = fetch(`http://localhost:${listenPort}/`).catch(() => null); + const serverConn = await listen.accept(); + listen.close(); + + using _httpConn = Deno.serveHttp(serverConn); + } + + const response = await promise; + assertEquals(response, null); + }, +); + function chunkedBodyReader(h: Headers, r: BufReader): Deno.Reader { // Based on https://tools.ietf.org/html/rfc2616#section-19.4.6 const tp = new TextProtoReader(r); diff --git a/cli/tests/unit/kv_test.ts b/cli/tests/unit/kv_test.ts index 4e3ce538509541..0bfc754819b5d9 100644 --- a/cli/tests/unit/kv_test.ts +++ b/cli/tests/unit/kv_test.ts @@ -2100,3 +2100,30 @@ Deno.test({ db.close(); }, }); + +Deno.test( + { permissions: { read: true } }, + async function kvExplicitResourceManagement() { + let kv2: Deno.Kv; + + { + using kv = await Deno.openKv(":memory:"); + kv2 = kv; + + const res = await kv.get(["a"]); + assertEquals(res.versionstamp, null); + } + + await assertRejects(() => kv2.get(["a"]), Deno.errors.BadResource); + }, +); + +Deno.test( + { permissions: { read: true } }, + async function kvExplicitResourceManagementManualClose() { + using kv = await Deno.openKv(":memory:"); + kv.close(); + await assertRejects(() => kv.get(["a"]), Deno.errors.BadResource); + // calling [Symbol.dispose] after manual close is a no-op + }, +); diff --git a/cli/tests/unit/serve_test.ts b/cli/tests/unit/serve_test.ts index 2e560af9957f3b..c2495649c9719e 100644 --- a/cli/tests/unit/serve_test.ts +++ b/cli/tests/unit/serve_test.ts @@ -48,7 +48,12 @@ function onListen( async function makeServer( handler: (req: Request) => Response | Promise, ): Promise< - { finished: Promise; abort: () => void; shutdown: () => Promise } + { + finished: Promise; + abort: () => void; + shutdown: () => Promise; + [Symbol.asyncDispose](): PromiseLike; + } > { const ac = new AbortController(); const listeningPromise = deferred(); @@ -69,6 +74,9 @@ async function makeServer( async shutdown() { await server.shutdown(); }, + [Symbol.asyncDispose]() { + return server[Symbol.asyncDispose](); + }, }; } @@ -296,6 +304,38 @@ Deno.test( }, ); +Deno.test( + { permissions: { net: true, write: true, read: true } }, + async function httpServerExplicitResourceManagement() { + let f; + + { + await using _server = await makeServer(async (_req) => { + return new Response((await makeTempFile(1024 * 1024)).readable); + }); + + f = await fetch(`http://localhost:${servePort}`); + } + + assertEquals((await f.text()).length, 1048576); + }, +); + +Deno.test( + { permissions: { net: true, write: true, read: true } }, + async function httpServerExplicitResourceManagementManualClose() { + await using server = await makeServer(async (_req) => { + return new Response((await makeTempFile(1024 * 1024)).readable); + }); + + const f = await fetch(`http://localhost:${servePort}`); + + await server.shutdown(); + + assertEquals((await f.text()).length, 1048576); + }, +); + Deno.test( { permissions: { read: true, run: true } }, async function httpServerUnref() { diff --git a/cli/tsc/dts/lib.deno.ns.d.ts b/cli/tsc/dts/lib.deno.ns.d.ts index 36d86f469383b5..08c8e9e9d83695 100644 --- a/cli/tsc/dts/lib.deno.ns.d.ts +++ b/cli/tsc/dts/lib.deno.ns.d.ts @@ -5917,7 +5917,7 @@ declare namespace Deno { * * @category HTTP Server */ - export interface HttpServer extends Disposable { + export interface HttpServer extends AsyncDisposable { /** A promise that resolves once server finishes - eg. when aborted using * the signal passed to {@linkcode ServeOptions.signal}. */ From bd4df5e6dc00a327fbac07eca9d5ae86d2b20472 Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Wed, 1 Nov 2023 18:40:06 +0100 Subject: [PATCH 8/9] fix tests on macos --- cli/tests/unit/serve_test.ts | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/cli/tests/unit/serve_test.ts b/cli/tests/unit/serve_test.ts index c2495649c9719e..9f6bd4aa167962 100644 --- a/cli/tests/unit/serve_test.ts +++ b/cli/tests/unit/serve_test.ts @@ -307,17 +307,18 @@ Deno.test( Deno.test( { permissions: { net: true, write: true, read: true } }, async function httpServerExplicitResourceManagement() { - let f; + let dataPromise; { await using _server = await makeServer(async (_req) => { return new Response((await makeTempFile(1024 * 1024)).readable); }); - f = await fetch(`http://localhost:${servePort}`); + const resp = await fetch(`http://localhost:${servePort}`); + dataPromise = resp.arrayBuffer(); } - assertEquals((await f.text()).length, 1048576); + assertEquals((await dataPromise).byteLength, 1048576); }, ); @@ -328,11 +329,14 @@ Deno.test( return new Response((await makeTempFile(1024 * 1024)).readable); }); - const f = await fetch(`http://localhost:${servePort}`); + const resp = await fetch(`http://localhost:${servePort}`); - await server.shutdown(); + const [_, data] = await Promise.all([ + server.shutdown(), + resp.arrayBuffer(), + ]); - assertEquals((await f.text()).length, 1048576); + assertEquals(data.byteLength, 1048576); }, ); From 6cfdfbf442c1c87c895009beb4cab6e862232dfd Mon Sep 17 00:00:00 2001 From: Luca Casonato Date: Wed, 1 Nov 2023 19:38:58 +0100 Subject: [PATCH 9/9] fix on windows --- cli/tests/unit/command_test.ts | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/cli/tests/unit/command_test.ts b/cli/tests/unit/command_test.ts index ed3da29270a4d1..299c70b9bfca54 100644 --- a/cli/tests/unit/command_test.ts +++ b/cli/tests/unit/command_test.ts @@ -274,7 +274,11 @@ Deno.test( }); } - assertEquals(dead, "SIGTERM"); + if (Deno.build.os == "windows") { + assertEquals(dead, null); + } else { + assertEquals(dead, "SIGTERM"); + } }, );