diff --git a/cli/dts/lib.deno.unstable.d.ts b/cli/dts/lib.deno.unstable.d.ts index 2e935643d4..564a7a0739 100644 --- a/cli/dts/lib.deno.unstable.d.ts +++ b/cli/dts/lib.deno.unstable.d.ts @@ -1296,6 +1296,22 @@ declare namespace Deno { alpnProtocols?: string[]; } + export interface Listener extends AsyncIterable { + /** **UNSTABLE**: new API, yet to be vetted. + * + * Make the listener block the event loop from finishing. + * + * Note: the listener blocks the event loop from finishing by default. + * This method is only meaningful after `.unref()` is called. + */ + ref(): void; + /** **UNSTABLE**: new API, yet to be vetted. + * + * Make the listener not block the event loop from finishing. + */ + unref(): void; + } + /** **UNSTABLE**: New API should be tested first. * * Acquire an advisory file-system lock for the provided file. `exclusive` diff --git a/cli/tests/unit/net_test.ts b/cli/tests/unit/net_test.ts index 6769f6301a..c7a081bf4c 100644 --- a/cli/tests/unit/net_test.ts +++ b/cli/tests/unit/net_test.ts @@ -7,6 +7,7 @@ import { assertThrows, deferred, delay, + execCode, } from "./test_util.ts"; let isCI: boolean; @@ -807,3 +808,79 @@ Deno.test( assertEquals(res, "hello world!"); }, ); + +Deno.test( + { permissions: { read: true, run: true } }, + async function netListenUnref() { + const [statusCode, _output] = await execCode(` + async function main() { + const listener = Deno.listen({ port: 3500 }); + listener.unref(); + await listener.accept(); // This doesn't block the program from exiting + } + main(); + `); + assertEquals(statusCode, 0); + }, +); + +Deno.test( + { permissions: { read: true, run: true } }, + async function netListenUnref() { + const [statusCode, _output] = await execCode(` + async function main() { + const listener = Deno.listen({ port: 3500 }); + await listener.accept(); + listener.unref(); + await listener.accept(); // The program exits here + throw new Error(); // The program doesn't reach here + } + main(); + const conn = await Deno.connect({ port: 3500 }); + conn.close(); + `); + assertEquals(statusCode, 0); + }, +); + +Deno.test( + { permissions: { read: true, run: true, net: true } }, + async function netListenUnrefAndRef() { + const p = execCode(` + async function main() { + const listener = Deno.listen({ port: 3500 }); + listener.unref(); + listener.ref(); // This restores 'ref' state of listener + await listener.accept(); + console.log("accepted") + } + main(); + `); + // TODO(kt3k): This is racy. Find a correct way to + // wait for the server to be ready + setTimeout(async () => { + const conn = await Deno.connect({ port: 3500 }); + conn.close(); + }, 200); + const [statusCode, output] = await p; + assertEquals(statusCode, 0); + assertEquals(output.trim(), "accepted"); + }, +); + +Deno.test( + { permissions: { net: true } }, + async function netListenUnrefConcurrentAccept() { + const timer = setTimeout(() => {}, 1000); + const listener = Deno.listen({ port: 3500 }); + listener.accept().catch(() => {}); + listener.unref(); + // Unref'd listener still causes Busy error + // on concurrent accept calls. + await assertRejects(async () => { + await listener.accept(); // The program exits here + }, Deno.errors.Busy); + listener.close(); + clearTimeout(timer); + }, +); diff --git a/cli/tests/unit/test_util.ts b/cli/tests/unit/test_util.ts index b18ad95500..36ef8c0e73 100644 --- a/cli/tests/unit/test_util.ts +++ b/cli/tests/unit/test_util.ts @@ -30,7 +30,7 @@ export function pathToAbsoluteFileUrl(path: string): URL { const decoder = new TextDecoder(); -export async function execCode(code: string) { +export async function execCode(code: string): Promise<[number, string]> { const p = Deno.run({ cmd: [ Deno.execPath(), diff --git a/ext/net/01_net.js b/ext/net/01_net.js index 11d6fbfacf..a17c667246 100644 --- a/ext/net/01_net.js +++ b/ext/net/01_net.js @@ -6,14 +6,18 @@ const { BadResourcePrototype, InterruptedPrototype } = core; const { ReadableStream, WritableStream } = window.__bootstrap.streams; const { + Error, ObjectPrototypeIsPrototypeOf, PromiseResolve, + Symbol, SymbolAsyncIterator, - Error, - Uint8Array, + SymbolFor, TypedArrayPrototypeSubarray, + Uint8Array, } = window.__bootstrap.primordials; + const promiseIdSymbol = SymbolFor("Deno.core.internalPromiseId"); + async function read( rid, buffer, @@ -191,9 +195,16 @@ class UnixConn extends Conn {} + // Use symbols for method names to hide these in stable API. + // TODO(kt3k): Remove these symbols when ref/unref become stable. + const listenerRef = Symbol("listenerRef"); + const listenerUnref = Symbol("listenerUnref"); + class Listener { #rid = 0; #addr = null; + #unref = false; + #promiseId = null; constructor(rid, addr) { this.#rid = rid; @@ -208,15 +219,21 @@ return this.#addr; } - async accept() { - const res = await opAccept(this.rid, this.addr.transport); - if (this.addr.transport == "tcp") { - return new TcpConn(res.rid, res.remoteAddr, res.localAddr); - } else if (this.addr.transport == "unix") { - return new UnixConn(res.rid, res.remoteAddr, res.localAddr); - } else { - throw new Error("unreachable"); + accept() { + const promise = opAccept(this.rid, this.addr.transport); + this.#promiseId = promise[promiseIdSymbol]; + if (this.#unref) { + this.#unrefOpAccept(); } + return promise.then((res) => { + if (this.addr.transport == "tcp") { + return new TcpConn(res.rid, res.remoteAddr, res.localAddr); + } else if (this.addr.transport == "unix") { + return new UnixConn(res.rid, res.remoteAddr, res.localAddr); + } else { + throw new Error("unreachable"); + } + }); } async next() { @@ -247,6 +264,27 @@ [SymbolAsyncIterator]() { return this; } + + [listenerRef]() { + this.#unref = false; + this.#refOpAccept(); + } + + [listenerUnref]() { + this.#unref = true; + this.#unrefOpAccept(); + } + + #refOpAccept() { + if (typeof this.#promiseId === "number") { + core.refOp(this.#promiseId); + } + } + #unrefOpAccept() { + if (typeof this.#promiseId === "number") { + core.unrefOp(this.#promiseId); + } + } } class Datagram { @@ -304,14 +342,14 @@ } } - function listen({ hostname, ...options }) { + function listen({ hostname, ...options }, constructor = Listener) { const res = opListen({ transport: "tcp", hostname: typeof hostname === "undefined" ? "0.0.0.0" : hostname, ...options, }); - return new Listener(res.rid, res.localAddr); + return new constructor(res.rid, res.localAddr); } async function connect(options) { @@ -335,6 +373,8 @@ UnixConn, opConnect, listen, + listenerRef, + listenerUnref, opListen, Listener, shutdown, diff --git a/ext/net/04_net_unstable.js b/ext/net/04_net_unstable.js index e22c9bf937..fcdb3c547c 100644 --- a/ext/net/04_net_unstable.js +++ b/ext/net/04_net_unstable.js @@ -7,9 +7,9 @@ function listen(options) { if (options.transport === "unix") { const res = net.opListen(options); - return new net.Listener(res.rid, res.localAddr); + return new Listener(res.rid, res.localAddr); } else { - return net.listen(options); + return net.listen(options, Listener); } } @@ -41,9 +41,20 @@ } } + class Listener extends net.Listener { + ref() { + this[net.listenerRef](); + } + + unref() { + this[net.listenerUnref](); + } + } + window.__bootstrap.netUnstable = { connect, listenDatagram, listen, + Listener, }; })(this); diff --git a/runtime/js/90_deno_ns.js b/runtime/js/90_deno_ns.js index 7ab0108a8b..ddaecd7c9f 100644 --- a/runtime/js/90_deno_ns.js +++ b/runtime/js/90_deno_ns.js @@ -132,6 +132,7 @@ listen: __bootstrap.netUnstable.listen, connect: __bootstrap.netUnstable.connect, listenDatagram: __bootstrap.netUnstable.listenDatagram, + Listener: __bootstrap.netUnstable.Listener, umask: __bootstrap.fs.umask, futime: __bootstrap.fs.futime, futimeSync: __bootstrap.fs.futimeSync,