mirror of
https://github.com/denoland/deno.git
synced 2025-01-12 00:54:02 -05:00
feat(unstable): Add Deno.Conn.ref()/unref() (#17170)
This commit adds "Deno.Conn.ref()" and "Deno.Conn.unref()" methods. These methods can be used to make connection block or not block the event loop from finishing. Refing/unrefing only influences "read" operations - ie. scheduling writes to a connection _do_ keep event loop alive. Required for https://github.com/denoland/deno/issues/16710
This commit is contained in:
parent
3baf1a4bd3
commit
2b15b07753
3 changed files with 110 additions and 16 deletions
|
@ -906,6 +906,55 @@ Deno.test({
|
||||||
listener.close();
|
listener.close();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Deno.test(
|
||||||
|
{ permissions: { net: true, read: true, run: true } },
|
||||||
|
async function netConnUnref() {
|
||||||
|
const listener = Deno.listen({ port: 3500 });
|
||||||
|
const intervalId = setInterval(() => {}); // This keeps event loop alive.
|
||||||
|
|
||||||
|
const program = execCode(`
|
||||||
|
async function main() {
|
||||||
|
const conn = await Deno.connect({ port: 3500 });
|
||||||
|
conn.unref();
|
||||||
|
await conn.read(new Uint8Array(10)); // The program exits here
|
||||||
|
throw new Error(); // The program doesn't reach here
|
||||||
|
}
|
||||||
|
main();
|
||||||
|
`);
|
||||||
|
const conn = await listener.accept();
|
||||||
|
const [statusCode, _output] = await program;
|
||||||
|
conn.close();
|
||||||
|
listener.close();
|
||||||
|
clearInterval(intervalId);
|
||||||
|
assertEquals(statusCode, 0);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
Deno.test(
|
||||||
|
{ permissions: { net: true, read: true, run: true } },
|
||||||
|
async function netConnUnrefReadable() {
|
||||||
|
const listener = Deno.listen({ port: 3500 });
|
||||||
|
const intervalId = setInterval(() => {}); // This keeps event loop alive.
|
||||||
|
|
||||||
|
const program = execCode(`
|
||||||
|
async function main() {
|
||||||
|
const conn = await Deno.connect({ port: 3500 });
|
||||||
|
conn.unref();
|
||||||
|
const reader = conn.readable.getReader();
|
||||||
|
await reader.read(); // The program exits here
|
||||||
|
throw new Error(); // The program doesn't reach here
|
||||||
|
}
|
||||||
|
main();
|
||||||
|
`);
|
||||||
|
const conn = await listener.accept();
|
||||||
|
const [statusCode, _output] = await program;
|
||||||
|
conn.close();
|
||||||
|
listener.close();
|
||||||
|
clearInterval(intervalId);
|
||||||
|
assertEquals(statusCode, 0);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
Deno.test({ permissions: { net: true } }, async function netTcpReuseAddr() {
|
Deno.test({ permissions: { net: true } }, async function netTcpReuseAddr() {
|
||||||
const listener1 = Deno.listen({
|
const listener1 = Deno.listen({
|
||||||
hostname: "127.0.0.1",
|
hostname: "127.0.0.1",
|
||||||
|
|
|
@ -4,8 +4,12 @@
|
||||||
((window) => {
|
((window) => {
|
||||||
const core = window.Deno.core;
|
const core = window.Deno.core;
|
||||||
const { BadResourcePrototype, InterruptedPrototype, ops } = core;
|
const { BadResourcePrototype, InterruptedPrototype, ops } = core;
|
||||||
const { readableStreamForRid, writableStreamForRid } =
|
const {
|
||||||
window.__bootstrap.streams;
|
readableStreamForRidUnrefable,
|
||||||
|
readableStreamForRidUnrefableRef,
|
||||||
|
readableStreamForRidUnrefableUnref,
|
||||||
|
writableStreamForRid,
|
||||||
|
} = window.__bootstrap.streams;
|
||||||
const {
|
const {
|
||||||
Error,
|
Error,
|
||||||
ObjectPrototypeIsPrototypeOf,
|
ObjectPrototypeIsPrototypeOf,
|
||||||
|
@ -19,17 +23,6 @@
|
||||||
|
|
||||||
const promiseIdSymbol = SymbolFor("Deno.core.internalPromiseId");
|
const promiseIdSymbol = SymbolFor("Deno.core.internalPromiseId");
|
||||||
|
|
||||||
async function read(
|
|
||||||
rid,
|
|
||||||
buffer,
|
|
||||||
) {
|
|
||||||
if (buffer.length === 0) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
const nread = await core.read(rid, buffer);
|
|
||||||
return nread === 0 ? null : nread;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function write(rid, data) {
|
async function write(rid, data) {
|
||||||
return await core.write(rid, data);
|
return await core.write(rid, data);
|
||||||
}
|
}
|
||||||
|
@ -46,6 +39,8 @@
|
||||||
#rid = 0;
|
#rid = 0;
|
||||||
#remoteAddr = null;
|
#remoteAddr = null;
|
||||||
#localAddr = null;
|
#localAddr = null;
|
||||||
|
#unref = false;
|
||||||
|
#pendingReadPromiseIds = [];
|
||||||
|
|
||||||
#readable;
|
#readable;
|
||||||
#writable;
|
#writable;
|
||||||
|
@ -72,8 +67,25 @@
|
||||||
return write(this.rid, p);
|
return write(this.rid, p);
|
||||||
}
|
}
|
||||||
|
|
||||||
read(p) {
|
async read(buffer) {
|
||||||
return read(this.rid, p);
|
if (buffer.length === 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
const promise = core.read(this.rid, buffer);
|
||||||
|
const promiseId = promise[promiseIdSymbol];
|
||||||
|
if (this.#unref) core.unrefOp(promiseId);
|
||||||
|
this.#pendingReadPromiseIds.push(promiseId);
|
||||||
|
let nread;
|
||||||
|
try {
|
||||||
|
nread = await promise;
|
||||||
|
} catch (e) {
|
||||||
|
throw e;
|
||||||
|
} finally {
|
||||||
|
this.#pendingReadPromiseIds = this.#pendingReadPromiseIds.filter((id) =>
|
||||||
|
id !== promiseId
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return nread === 0 ? null : nread;
|
||||||
}
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
|
@ -86,7 +98,10 @@
|
||||||
|
|
||||||
get readable() {
|
get readable() {
|
||||||
if (this.#readable === undefined) {
|
if (this.#readable === undefined) {
|
||||||
this.#readable = readableStreamForRid(this.rid);
|
this.#readable = readableStreamForRidUnrefable(this.rid);
|
||||||
|
if (this.#unref) {
|
||||||
|
readableStreamForRidUnrefableUnref(this.#readable);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return this.#readable;
|
return this.#readable;
|
||||||
}
|
}
|
||||||
|
@ -97,6 +112,22 @@
|
||||||
}
|
}
|
||||||
return this.#writable;
|
return this.#writable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ref() {
|
||||||
|
this.#unref = false;
|
||||||
|
if (this.#readable) {
|
||||||
|
readableStreamForRidUnrefableRef(this.#readable);
|
||||||
|
}
|
||||||
|
this.#pendingReadPromiseIds.forEach((id) => core.refOp(id));
|
||||||
|
}
|
||||||
|
|
||||||
|
unref() {
|
||||||
|
this.#unref = true;
|
||||||
|
if (this.#readable) {
|
||||||
|
readableStreamForRidUnrefableUnref(this.#readable);
|
||||||
|
}
|
||||||
|
this.#pendingReadPromiseIds.forEach((id) => core.unrefOp(id));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class TcpConn extends Conn {
|
class TcpConn extends Conn {
|
||||||
|
|
14
ext/net/lib.deno_net.d.ts
vendored
14
ext/net/lib.deno_net.d.ts
vendored
|
@ -61,6 +61,20 @@ declare namespace Deno {
|
||||||
* callers should just use `close()`. */
|
* callers should just use `close()`. */
|
||||||
closeWrite(): Promise<void>;
|
closeWrite(): Promise<void>;
|
||||||
|
|
||||||
|
/** **UNSTABLE**: New API, yet to be vetted.
|
||||||
|
*
|
||||||
|
* Make the connection block the event loop from finishing.
|
||||||
|
*
|
||||||
|
* Note: the connection 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 connection not block the event loop from finishing.
|
||||||
|
*/
|
||||||
|
unref(): void;
|
||||||
|
|
||||||
readonly readable: ReadableStream<Uint8Array>;
|
readonly readable: ReadableStream<Uint8Array>;
|
||||||
readonly writable: WritableStream<Uint8Array>;
|
readonly writable: WritableStream<Uint8Array>;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue