2024-12-31 14:12:39 -05:00
|
|
|
// Copyright 2018-2025 the Deno authors. MIT license.
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2021-07-06 08:38:12 -04:00
|
|
|
// @ts-check
|
|
|
|
/// <reference path="../../core/internal.d.ts" />
|
|
|
|
|
2024-06-18 17:47:05 -04:00
|
|
|
import { core, primordials } from "ext:core/mod.js";
|
2023-02-07 14:22:46 -05:00
|
|
|
const {
|
2023-11-12 19:04:11 -05:00
|
|
|
ArrayPrototypeEvery,
|
|
|
|
ArrayPrototypePush,
|
perf(ext/http): recover memory for serve and optimize AbortController (#23559)
Max rps without a signal is unchanged, however we can drastically reduce
memory usage by not creating the signal until needed, and we can
optimize the rps in the case where the signal is created.
With a quick memory benchmark, it looks like this helps pretty
drastically with # of GCs when benchmarking w/wrk:
- 1.42.4: 1763
- canary: 1093
- this patch: 874
This branch:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 87.33us 439.95us 20.68ms 99.67%
Req/Sec 66.70k 6.39k 74.11k 83.66%
1340255 requests in 10.10s, 191.73MB read
Requests/sec: 132696.90
Transfer/sec: 18.98MB
cpu: Apple M2 Pro
runtime: deno 1.43.0 (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 986.5 ns/iter 1,013,682.6 (878.2 ns … 1.18 µs) 1.01 µs 1.18 µs 1.18 µs
newAbortController 18 ns/iter 55,541,104.1 (15.6 ns … 42.62 ns) 17.71 ns 25.05 ns 26.27 ns
newAbortControllerSignal 18.66 ns/iter 53,578,966.7 (16.49 ns … 32.16 ns) 18.71 ns 25.67 ns 26.39 ns
newAbortControllerSignalOnAbort 106.49 ns/iter 9,390,164.9 (97.87 ns … 120.61 ns) 108.6 ns 114.24 ns 115.89 ns
newAbortControllerSignalAddEventListener 86.92 ns/iter 11,504,880.2 (81.88 ns … 103.15 ns) 90 ns 98.28 ns 99.55 ns
newAbortControllerSignalOnAbortNoListener 3.01 µs/iter 331,964.4 (2.97 µs … 3.1 µs) 3.06 µs 3.1 µs 3.1 µs
newAbortControllerSignalOnAbortAbort 3.26 µs/iter 306,662.6 (3.22 µs … 3.36 µs) 3.27 µs 3.36 µs 3.36 µs
```
Latest canary:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 72.86us 71.23us 4.47ms 99.05%
Req/Sec 64.66k 5.54k 72.48k 82.18%
1299015 requests in 10.10s, 185.83MB read
Requests/sec: 128616.02
Transfer/sec: 18.40MB
cpu: Apple M2 Pro
runtime: deno 1.43.0+bc4aa5f (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 1.25 µs/iter 800,005.2 (1.01 µs … 4.18 µs) 1.16 µs 4.18 µs 4.18 µs
newAbortController 18.56 ns/iter 53,868,204.3 (16.04 ns … 38.73 ns) 18.38 ns 26.1 ns 26.63 ns
newAbortControllerSignal 18.72 ns/iter 53,430,746.1 (16.13 ns … 36.71 ns) 18.71 ns 26.19 ns 26.98 ns
newAbortControllerSignalOnAbort 193.91 ns/iter 5,156,992.4 (184.25 ns … 211.41 ns) 194.96 ns 207.87 ns 209.4 ns
newAbortControllerSignalAddEventListener 171.45 ns/iter 5,832,569.2 (153 ns … 182.03 ns) 176.17 ns 180.75 ns 181.05 ns
newAbortControllerSignalOnAbortNoListener 3.07 µs/iter 326,263.3 (2.98 µs … 3.17 µs) 3.08 µs 3.17 µs 3.17 µs
newAbortControllerSignalOnAbortAbort 3.32 µs/iter 301,344.6 (3.29 µs … 3.4 µs) 3.33 µs 3.4 µs 3.4 µs
```
2024-04-25 14:52:24 -04:00
|
|
|
FunctionPrototypeApply,
|
2023-11-19 03:13:38 -05:00
|
|
|
ObjectPrototypeIsPrototypeOf,
|
2023-04-14 16:23:28 -04:00
|
|
|
SafeSet,
|
2023-02-07 14:22:46 -05:00
|
|
|
SafeSetIterator,
|
2023-11-12 19:04:11 -05:00
|
|
|
SafeWeakRef,
|
|
|
|
SafeWeakSet,
|
2023-02-07 14:22:46 -05:00
|
|
|
SetPrototypeAdd,
|
|
|
|
SetPrototypeDelete,
|
|
|
|
Symbol,
|
2023-11-19 03:13:38 -05:00
|
|
|
SymbolFor,
|
2023-02-07 14:22:46 -05:00
|
|
|
TypeError,
|
2023-11-12 19:04:11 -05:00
|
|
|
WeakRefPrototypeDeref,
|
|
|
|
WeakSetPrototypeAdd,
|
|
|
|
WeakSetPrototypeHas,
|
2023-02-07 14:22:46 -05:00
|
|
|
} = primordials;
|
2024-01-10 17:37:25 -05:00
|
|
|
|
|
|
|
import * as webidl from "ext:deno_webidl/00_webidl.js";
|
2024-03-22 12:21:05 -04:00
|
|
|
import { assert } from "./00_infra.js";
|
2024-01-10 17:37:25 -05:00
|
|
|
import { createFilteredInspectProxy } from "ext:deno_console/01_console.js";
|
|
|
|
import {
|
|
|
|
defineEventHandler,
|
|
|
|
Event,
|
|
|
|
EventTarget,
|
|
|
|
listenerCount,
|
|
|
|
setIsTrusted,
|
2024-03-22 12:21:05 -04:00
|
|
|
} from "./02_event.js";
|
2024-06-18 17:47:05 -04:00
|
|
|
import { clearTimeout, refTimer, unrefTimer } from "./02_timers.js";
|
2023-02-07 14:22:46 -05:00
|
|
|
|
2023-11-12 19:04:11 -05:00
|
|
|
// Since WeakSet is not a iterable, WeakRefSet class is provided to store and
|
|
|
|
// iterate objects.
|
|
|
|
// To create an AsyncIterable using GeneratorFunction in the internal code,
|
|
|
|
// there are many primordial considerations, so we simply implement the
|
|
|
|
// toArray method.
|
|
|
|
class WeakRefSet {
|
|
|
|
#weakSet = new SafeWeakSet();
|
|
|
|
#refs = [];
|
|
|
|
|
|
|
|
add(value) {
|
|
|
|
if (WeakSetPrototypeHas(this.#weakSet, value)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
WeakSetPrototypeAdd(this.#weakSet, value);
|
|
|
|
ArrayPrototypePush(this.#refs, new SafeWeakRef(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
has(value) {
|
|
|
|
return WeakSetPrototypeHas(this.#weakSet, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
toArray() {
|
|
|
|
const ret = [];
|
|
|
|
for (let i = 0; i < this.#refs.length; ++i) {
|
|
|
|
const value = WeakRefPrototypeDeref(this.#refs[i]);
|
|
|
|
if (value !== undefined) {
|
|
|
|
ArrayPrototypePush(ret, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
const add = Symbol("[[add]]");
|
|
|
|
const signalAbort = Symbol("[[signalAbort]]");
|
|
|
|
const remove = Symbol("[[remove]]");
|
2024-09-04 02:15:34 -04:00
|
|
|
const runAbortSteps = Symbol("[[runAbortSteps]]");
|
2023-02-07 14:22:46 -05:00
|
|
|
const abortReason = Symbol("[[abortReason]]");
|
|
|
|
const abortAlgos = Symbol("[[abortAlgos]]");
|
2023-11-12 19:04:11 -05:00
|
|
|
const dependent = Symbol("[[dependent]]");
|
|
|
|
const sourceSignals = Symbol("[[sourceSignals]]");
|
|
|
|
const dependentSignals = Symbol("[[dependentSignals]]");
|
2023-02-07 14:22:46 -05:00
|
|
|
const signal = Symbol("[[signal]]");
|
|
|
|
const timerId = Symbol("[[timerId]]");
|
|
|
|
|
|
|
|
const illegalConstructorKey = Symbol("illegalConstructorKey");
|
|
|
|
|
|
|
|
class AbortSignal extends EventTarget {
|
perf(ext/http): recover memory for serve and optimize AbortController (#23559)
Max rps without a signal is unchanged, however we can drastically reduce
memory usage by not creating the signal until needed, and we can
optimize the rps in the case where the signal is created.
With a quick memory benchmark, it looks like this helps pretty
drastically with # of GCs when benchmarking w/wrk:
- 1.42.4: 1763
- canary: 1093
- this patch: 874
This branch:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 87.33us 439.95us 20.68ms 99.67%
Req/Sec 66.70k 6.39k 74.11k 83.66%
1340255 requests in 10.10s, 191.73MB read
Requests/sec: 132696.90
Transfer/sec: 18.98MB
cpu: Apple M2 Pro
runtime: deno 1.43.0 (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 986.5 ns/iter 1,013,682.6 (878.2 ns … 1.18 µs) 1.01 µs 1.18 µs 1.18 µs
newAbortController 18 ns/iter 55,541,104.1 (15.6 ns … 42.62 ns) 17.71 ns 25.05 ns 26.27 ns
newAbortControllerSignal 18.66 ns/iter 53,578,966.7 (16.49 ns … 32.16 ns) 18.71 ns 25.67 ns 26.39 ns
newAbortControllerSignalOnAbort 106.49 ns/iter 9,390,164.9 (97.87 ns … 120.61 ns) 108.6 ns 114.24 ns 115.89 ns
newAbortControllerSignalAddEventListener 86.92 ns/iter 11,504,880.2 (81.88 ns … 103.15 ns) 90 ns 98.28 ns 99.55 ns
newAbortControllerSignalOnAbortNoListener 3.01 µs/iter 331,964.4 (2.97 µs … 3.1 µs) 3.06 µs 3.1 µs 3.1 µs
newAbortControllerSignalOnAbortAbort 3.26 µs/iter 306,662.6 (3.22 µs … 3.36 µs) 3.27 µs 3.36 µs 3.36 µs
```
Latest canary:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 72.86us 71.23us 4.47ms 99.05%
Req/Sec 64.66k 5.54k 72.48k 82.18%
1299015 requests in 10.10s, 185.83MB read
Requests/sec: 128616.02
Transfer/sec: 18.40MB
cpu: Apple M2 Pro
runtime: deno 1.43.0+bc4aa5f (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 1.25 µs/iter 800,005.2 (1.01 µs … 4.18 µs) 1.16 µs 4.18 µs 4.18 µs
newAbortController 18.56 ns/iter 53,868,204.3 (16.04 ns … 38.73 ns) 18.38 ns 26.1 ns 26.63 ns
newAbortControllerSignal 18.72 ns/iter 53,430,746.1 (16.13 ns … 36.71 ns) 18.71 ns 26.19 ns 26.98 ns
newAbortControllerSignalOnAbort 193.91 ns/iter 5,156,992.4 (184.25 ns … 211.41 ns) 194.96 ns 207.87 ns 209.4 ns
newAbortControllerSignalAddEventListener 171.45 ns/iter 5,832,569.2 (153 ns … 182.03 ns) 176.17 ns 180.75 ns 181.05 ns
newAbortControllerSignalOnAbortNoListener 3.07 µs/iter 326,263.3 (2.98 µs … 3.17 µs) 3.08 µs 3.17 µs 3.17 µs
newAbortControllerSignalOnAbortAbort 3.32 µs/iter 301,344.6 (3.29 µs … 3.4 µs) 3.33 µs 3.4 µs 3.4 µs
```
2024-04-25 14:52:24 -04:00
|
|
|
[abortReason] = undefined;
|
|
|
|
[abortAlgos] = null;
|
|
|
|
[dependent] = false;
|
|
|
|
[sourceSignals] = null;
|
|
|
|
[dependentSignals] = null;
|
|
|
|
[timerId] = null;
|
|
|
|
[webidl.brand] = webidl.brand;
|
|
|
|
|
2023-11-12 19:04:11 -05:00
|
|
|
static any(signals) {
|
2024-03-09 21:23:14 -05:00
|
|
|
const prefix = "Failed to execute 'AbortSignal.any'";
|
2023-11-12 19:04:11 -05:00
|
|
|
webidl.requiredArguments(arguments.length, 1, prefix);
|
|
|
|
return createDependentAbortSignal(signals, prefix);
|
|
|
|
}
|
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
static abort(reason = undefined) {
|
|
|
|
if (reason !== undefined) {
|
|
|
|
reason = webidl.converters.any(reason);
|
|
|
|
}
|
|
|
|
const signal = new AbortSignal(illegalConstructorKey);
|
|
|
|
signal[signalAbort](reason);
|
|
|
|
return signal;
|
|
|
|
}
|
2021-03-27 10:49:57 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
static timeout(millis) {
|
2024-03-09 21:23:14 -05:00
|
|
|
const prefix = "Failed to execute 'AbortSignal.timeout'";
|
2023-04-12 15:58:57 -04:00
|
|
|
webidl.requiredArguments(arguments.length, 1, prefix);
|
2023-05-01 06:47:13 -04:00
|
|
|
millis = webidl.converters["unsigned long long"](
|
|
|
|
millis,
|
|
|
|
prefix,
|
|
|
|
"Argument 1",
|
|
|
|
{
|
|
|
|
enforceRange: true,
|
|
|
|
},
|
|
|
|
);
|
2023-02-07 14:22:46 -05:00
|
|
|
|
|
|
|
const signal = new AbortSignal(illegalConstructorKey);
|
2024-06-18 17:47:05 -04:00
|
|
|
signal[timerId] = core.queueSystemTimer(
|
|
|
|
undefined,
|
|
|
|
false,
|
|
|
|
millis,
|
2023-02-07 14:22:46 -05:00
|
|
|
() => {
|
2024-06-18 17:47:05 -04:00
|
|
|
clearTimeout(signal[timerId]);
|
2023-02-07 14:22:46 -05:00
|
|
|
signal[timerId] = null;
|
|
|
|
signal[signalAbort](
|
|
|
|
new DOMException("Signal timed out.", "TimeoutError"),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
unrefTimer(signal[timerId]);
|
|
|
|
return signal;
|
|
|
|
}
|
2022-03-14 15:19:22 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
[add](algorithm) {
|
|
|
|
if (this.aborted) {
|
|
|
|
return;
|
2020-07-19 13:49:44 -04:00
|
|
|
}
|
2023-11-12 19:04:11 -05:00
|
|
|
this[abortAlgos] ??= new SafeSet();
|
2023-02-07 14:22:46 -05:00
|
|
|
SetPrototypeAdd(this[abortAlgos], algorithm);
|
|
|
|
}
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
[signalAbort](
|
|
|
|
reason = new DOMException("The signal has been aborted", "AbortError"),
|
|
|
|
) {
|
|
|
|
if (this.aborted) {
|
|
|
|
return;
|
2020-07-19 13:49:44 -04:00
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
this[abortReason] = reason;
|
2024-09-04 02:15:34 -04:00
|
|
|
|
|
|
|
const dependentSignalsToAbort = [];
|
|
|
|
if (this[dependentSignals] !== null) {
|
|
|
|
const dependentSignalArray = this[dependentSignals].toArray();
|
|
|
|
for (let i = 0; i < dependentSignalArray.length; ++i) {
|
|
|
|
const dependentSignal = dependentSignalArray[i];
|
|
|
|
if (dependentSignal[abortReason] === undefined) {
|
|
|
|
dependentSignal[abortReason] = this[abortReason];
|
|
|
|
ArrayPrototypePush(dependentSignalsToAbort, dependentSignal);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this[runAbortSteps]();
|
|
|
|
|
|
|
|
if (dependentSignalsToAbort.length !== 0) {
|
|
|
|
for (let i = 0; i < dependentSignalsToAbort.length; ++i) {
|
|
|
|
const dependentSignal = dependentSignalsToAbort[i];
|
|
|
|
dependentSignal[runAbortSteps]();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[runAbortSteps]() {
|
2023-08-08 06:05:42 -04:00
|
|
|
const algos = this[abortAlgos];
|
|
|
|
this[abortAlgos] = null;
|
|
|
|
|
|
|
|
if (algos !== null) {
|
|
|
|
for (const algorithm of new SafeSetIterator(algos)) {
|
|
|
|
algorithm();
|
|
|
|
}
|
|
|
|
}
|
2023-11-12 19:04:11 -05:00
|
|
|
|
2024-09-04 02:15:34 -04:00
|
|
|
if (listenerCount(this, "abort") > 0) {
|
|
|
|
const event = new Event("abort");
|
|
|
|
setIsTrusted(event, true);
|
|
|
|
super.dispatchEvent(event);
|
2023-11-12 19:04:11 -05:00
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
[remove](algorithm) {
|
|
|
|
this[abortAlgos] && SetPrototypeDelete(this[abortAlgos], algorithm);
|
|
|
|
}
|
2021-11-08 17:37:06 -05:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
constructor(key = null) {
|
2023-11-12 19:04:11 -05:00
|
|
|
if (key !== illegalConstructorKey) {
|
2024-12-02 22:30:39 -05:00
|
|
|
throw new TypeError("Illegal constructor");
|
2020-07-19 13:49:44 -04:00
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
super();
|
|
|
|
}
|
2021-12-10 09:12:38 -05:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
get aborted() {
|
|
|
|
webidl.assertBranded(this, AbortSignalPrototype);
|
|
|
|
return this[abortReason] !== undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
get reason() {
|
|
|
|
webidl.assertBranded(this, AbortSignalPrototype);
|
|
|
|
return this[abortReason];
|
|
|
|
}
|
|
|
|
|
|
|
|
throwIfAborted() {
|
|
|
|
webidl.assertBranded(this, AbortSignalPrototype);
|
|
|
|
if (this[abortReason] !== undefined) {
|
|
|
|
throw this[abortReason];
|
2021-12-10 09:12:38 -05:00
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
2022-03-14 15:19:22 -04:00
|
|
|
|
2023-06-26 09:10:27 -04:00
|
|
|
// `addEventListener` and `removeEventListener` have to be overridden in
|
2023-02-07 14:22:46 -05:00
|
|
|
// order to have the timer block the event loop while there are listeners.
|
|
|
|
// `[add]` and `[remove]` don't ref and unref the timer because they can
|
|
|
|
// only be used by Deno internals, which use it to essentially cancel async
|
|
|
|
// ops which would block the event loop.
|
perf(ext/http): recover memory for serve and optimize AbortController (#23559)
Max rps without a signal is unchanged, however we can drastically reduce
memory usage by not creating the signal until needed, and we can
optimize the rps in the case where the signal is created.
With a quick memory benchmark, it looks like this helps pretty
drastically with # of GCs when benchmarking w/wrk:
- 1.42.4: 1763
- canary: 1093
- this patch: 874
This branch:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 87.33us 439.95us 20.68ms 99.67%
Req/Sec 66.70k 6.39k 74.11k 83.66%
1340255 requests in 10.10s, 191.73MB read
Requests/sec: 132696.90
Transfer/sec: 18.98MB
cpu: Apple M2 Pro
runtime: deno 1.43.0 (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 986.5 ns/iter 1,013,682.6 (878.2 ns … 1.18 µs) 1.01 µs 1.18 µs 1.18 µs
newAbortController 18 ns/iter 55,541,104.1 (15.6 ns … 42.62 ns) 17.71 ns 25.05 ns 26.27 ns
newAbortControllerSignal 18.66 ns/iter 53,578,966.7 (16.49 ns … 32.16 ns) 18.71 ns 25.67 ns 26.39 ns
newAbortControllerSignalOnAbort 106.49 ns/iter 9,390,164.9 (97.87 ns … 120.61 ns) 108.6 ns 114.24 ns 115.89 ns
newAbortControllerSignalAddEventListener 86.92 ns/iter 11,504,880.2 (81.88 ns … 103.15 ns) 90 ns 98.28 ns 99.55 ns
newAbortControllerSignalOnAbortNoListener 3.01 µs/iter 331,964.4 (2.97 µs … 3.1 µs) 3.06 µs 3.1 µs 3.1 µs
newAbortControllerSignalOnAbortAbort 3.26 µs/iter 306,662.6 (3.22 µs … 3.36 µs) 3.27 µs 3.36 µs 3.36 µs
```
Latest canary:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 72.86us 71.23us 4.47ms 99.05%
Req/Sec 64.66k 5.54k 72.48k 82.18%
1299015 requests in 10.10s, 185.83MB read
Requests/sec: 128616.02
Transfer/sec: 18.40MB
cpu: Apple M2 Pro
runtime: deno 1.43.0+bc4aa5f (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 1.25 µs/iter 800,005.2 (1.01 µs … 4.18 µs) 1.16 µs 4.18 µs 4.18 µs
newAbortController 18.56 ns/iter 53,868,204.3 (16.04 ns … 38.73 ns) 18.38 ns 26.1 ns 26.63 ns
newAbortControllerSignal 18.72 ns/iter 53,430,746.1 (16.13 ns … 36.71 ns) 18.71 ns 26.19 ns 26.98 ns
newAbortControllerSignalOnAbort 193.91 ns/iter 5,156,992.4 (184.25 ns … 211.41 ns) 194.96 ns 207.87 ns 209.4 ns
newAbortControllerSignalAddEventListener 171.45 ns/iter 5,832,569.2 (153 ns … 182.03 ns) 176.17 ns 180.75 ns 181.05 ns
newAbortControllerSignalOnAbortNoListener 3.07 µs/iter 326,263.3 (2.98 µs … 3.17 µs) 3.08 µs 3.17 µs 3.17 µs
newAbortControllerSignalOnAbortAbort 3.32 µs/iter 301,344.6 (3.29 µs … 3.4 µs) 3.33 µs 3.4 µs 3.4 µs
```
2024-04-25 14:52:24 -04:00
|
|
|
addEventListener() {
|
|
|
|
FunctionPrototypeApply(super.addEventListener, this, arguments);
|
2023-11-12 19:04:11 -05:00
|
|
|
if (listenerCount(this, "abort") > 0) {
|
|
|
|
if (this[timerId] !== null) {
|
|
|
|
refTimer(this[timerId]);
|
|
|
|
} else if (this[sourceSignals] !== null) {
|
|
|
|
const sourceSignalArray = this[sourceSignals].toArray();
|
|
|
|
for (let i = 0; i < sourceSignalArray.length; ++i) {
|
|
|
|
const sourceSignal = sourceSignalArray[i];
|
|
|
|
if (sourceSignal[timerId] !== null) {
|
|
|
|
refTimer(sourceSignal[timerId]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-14 15:19:22 -04:00
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
2022-03-14 15:19:22 -04:00
|
|
|
|
perf(ext/http): recover memory for serve and optimize AbortController (#23559)
Max rps without a signal is unchanged, however we can drastically reduce
memory usage by not creating the signal until needed, and we can
optimize the rps in the case where the signal is created.
With a quick memory benchmark, it looks like this helps pretty
drastically with # of GCs when benchmarking w/wrk:
- 1.42.4: 1763
- canary: 1093
- this patch: 874
This branch:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 87.33us 439.95us 20.68ms 99.67%
Req/Sec 66.70k 6.39k 74.11k 83.66%
1340255 requests in 10.10s, 191.73MB read
Requests/sec: 132696.90
Transfer/sec: 18.98MB
cpu: Apple M2 Pro
runtime: deno 1.43.0 (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 986.5 ns/iter 1,013,682.6 (878.2 ns … 1.18 µs) 1.01 µs 1.18 µs 1.18 µs
newAbortController 18 ns/iter 55,541,104.1 (15.6 ns … 42.62 ns) 17.71 ns 25.05 ns 26.27 ns
newAbortControllerSignal 18.66 ns/iter 53,578,966.7 (16.49 ns … 32.16 ns) 18.71 ns 25.67 ns 26.39 ns
newAbortControllerSignalOnAbort 106.49 ns/iter 9,390,164.9 (97.87 ns … 120.61 ns) 108.6 ns 114.24 ns 115.89 ns
newAbortControllerSignalAddEventListener 86.92 ns/iter 11,504,880.2 (81.88 ns … 103.15 ns) 90 ns 98.28 ns 99.55 ns
newAbortControllerSignalOnAbortNoListener 3.01 µs/iter 331,964.4 (2.97 µs … 3.1 µs) 3.06 µs 3.1 µs 3.1 µs
newAbortControllerSignalOnAbortAbort 3.26 µs/iter 306,662.6 (3.22 µs … 3.36 µs) 3.27 µs 3.36 µs 3.36 µs
```
Latest canary:
```
Running 10s test @ http://localhost:8080/
2 threads and 10 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 72.86us 71.23us 4.47ms 99.05%
Req/Sec 64.66k 5.54k 72.48k 82.18%
1299015 requests in 10.10s, 185.83MB read
Requests/sec: 128616.02
Transfer/sec: 18.40MB
cpu: Apple M2 Pro
runtime: deno 1.43.0+bc4aa5f (aarch64-apple-darwin)
file:///Users/matt/Documents/scripts/bench_request.js
benchmark time (avg) iter/s (min … max) p75 p99 p995
----------------------------------------------------------------------------------------------- -----------------------------
newRequest 1.25 µs/iter 800,005.2 (1.01 µs … 4.18 µs) 1.16 µs 4.18 µs 4.18 µs
newAbortController 18.56 ns/iter 53,868,204.3 (16.04 ns … 38.73 ns) 18.38 ns 26.1 ns 26.63 ns
newAbortControllerSignal 18.72 ns/iter 53,430,746.1 (16.13 ns … 36.71 ns) 18.71 ns 26.19 ns 26.98 ns
newAbortControllerSignalOnAbort 193.91 ns/iter 5,156,992.4 (184.25 ns … 211.41 ns) 194.96 ns 207.87 ns 209.4 ns
newAbortControllerSignalAddEventListener 171.45 ns/iter 5,832,569.2 (153 ns … 182.03 ns) 176.17 ns 180.75 ns 181.05 ns
newAbortControllerSignalOnAbortNoListener 3.07 µs/iter 326,263.3 (2.98 µs … 3.17 µs) 3.08 µs 3.17 µs 3.17 µs
newAbortControllerSignalOnAbortAbort 3.32 µs/iter 301,344.6 (3.29 µs … 3.4 µs) 3.33 µs 3.4 µs 3.4 µs
```
2024-04-25 14:52:24 -04:00
|
|
|
removeEventListener() {
|
|
|
|
FunctionPrototypeApply(super.removeEventListener, this, arguments);
|
2023-11-12 19:04:11 -05:00
|
|
|
if (listenerCount(this, "abort") === 0) {
|
|
|
|
if (this[timerId] !== null) {
|
|
|
|
unrefTimer(this[timerId]);
|
|
|
|
} else if (this[sourceSignals] !== null) {
|
|
|
|
const sourceSignalArray = this[sourceSignals].toArray();
|
|
|
|
for (let i = 0; i < sourceSignalArray.length; ++i) {
|
|
|
|
const sourceSignal = sourceSignalArray[i];
|
|
|
|
if (sourceSignal[timerId] !== null) {
|
|
|
|
// Check that all dependent signals of the timer signal do not have listeners
|
|
|
|
if (
|
|
|
|
ArrayPrototypeEvery(
|
|
|
|
sourceSignal[dependentSignals].toArray(),
|
|
|
|
(dependentSignal) =>
|
|
|
|
dependentSignal === this ||
|
|
|
|
listenerCount(dependentSignal, "abort") === 0,
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
unrefTimer(sourceSignal[timerId]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-14 15:19:22 -04:00
|
|
|
}
|
2020-07-19 13:49:44 -04:00
|
|
|
}
|
2023-11-19 03:13:38 -05:00
|
|
|
|
|
|
|
[SymbolFor("Deno.privateCustomInspect")](inspect, inspectOptions) {
|
|
|
|
return inspect(
|
|
|
|
createFilteredInspectProxy({
|
|
|
|
object: this,
|
|
|
|
evaluate: ObjectPrototypeIsPrototypeOf(AbortSignalPrototype, this),
|
|
|
|
keys: [
|
|
|
|
"aborted",
|
|
|
|
"reason",
|
|
|
|
"onabort",
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
inspectOptions,
|
|
|
|
);
|
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
|
|
|
defineEventHandler(AbortSignal.prototype, "abort");
|
2021-06-07 04:04:10 -04:00
|
|
|
|
2023-10-09 23:01:01 -04:00
|
|
|
webidl.configureInterface(AbortSignal);
|
2023-02-07 14:22:46 -05:00
|
|
|
const AbortSignalPrototype = AbortSignal.prototype;
|
2021-06-07 04:04:10 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
class AbortController {
|
|
|
|
[signal] = new AbortSignal(illegalConstructorKey);
|
2021-11-08 17:37:06 -05:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
constructor() {
|
|
|
|
this[webidl.brand] = webidl.brand;
|
|
|
|
}
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
get signal() {
|
|
|
|
webidl.assertBranded(this, AbortControllerPrototype);
|
|
|
|
return this[signal];
|
|
|
|
}
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
abort(reason) {
|
|
|
|
webidl.assertBranded(this, AbortControllerPrototype);
|
|
|
|
this[signal][signalAbort](reason);
|
2020-07-19 13:49:44 -04:00
|
|
|
}
|
2023-11-19 03:13:38 -05:00
|
|
|
|
|
|
|
[SymbolFor("Deno.privateCustomInspect")](inspect, inspectOptions) {
|
|
|
|
return inspect(
|
|
|
|
createFilteredInspectProxy({
|
|
|
|
object: this,
|
|
|
|
evaluate: ObjectPrototypeIsPrototypeOf(AbortControllerPrototype, this),
|
|
|
|
keys: [
|
|
|
|
"signal",
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
inspectOptions,
|
|
|
|
);
|
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
2020-07-19 13:49:44 -04:00
|
|
|
|
2023-10-09 23:01:01 -04:00
|
|
|
webidl.configureInterface(AbortController);
|
2023-02-07 14:22:46 -05:00
|
|
|
const AbortControllerPrototype = AbortController.prototype;
|
2021-06-07 04:04:10 -04:00
|
|
|
|
2023-11-12 19:04:11 -05:00
|
|
|
webidl.converters.AbortSignal = webidl.createInterfaceConverter(
|
2023-02-07 14:22:46 -05:00
|
|
|
"AbortSignal",
|
|
|
|
AbortSignal.prototype,
|
|
|
|
);
|
2023-11-12 19:04:11 -05:00
|
|
|
webidl.converters["sequence<AbortSignal>"] = webidl.createSequenceConverter(
|
|
|
|
webidl.converters.AbortSignal,
|
|
|
|
);
|
2021-04-20 08:47:22 -04:00
|
|
|
|
2023-02-07 14:22:46 -05:00
|
|
|
function newSignal() {
|
|
|
|
return new AbortSignal(illegalConstructorKey);
|
|
|
|
}
|
2021-06-06 09:37:17 -04:00
|
|
|
|
2023-11-12 19:04:11 -05:00
|
|
|
function createDependentAbortSignal(signals, prefix) {
|
|
|
|
signals = webidl.converters["sequence<AbortSignal>"](
|
|
|
|
signals,
|
|
|
|
prefix,
|
|
|
|
"Argument 1",
|
|
|
|
);
|
|
|
|
|
|
|
|
const resultSignal = new AbortSignal(illegalConstructorKey);
|
|
|
|
for (let i = 0; i < signals.length; ++i) {
|
|
|
|
const signal = signals[i];
|
|
|
|
if (signal[abortReason] !== undefined) {
|
|
|
|
resultSignal[abortReason] = signal[abortReason];
|
|
|
|
return resultSignal;
|
|
|
|
}
|
2021-06-06 09:37:17 -04:00
|
|
|
}
|
2023-11-12 19:04:11 -05:00
|
|
|
|
|
|
|
resultSignal[dependent] = true;
|
|
|
|
resultSignal[sourceSignals] = new WeakRefSet();
|
|
|
|
for (let i = 0; i < signals.length; ++i) {
|
|
|
|
const signal = signals[i];
|
|
|
|
if (!signal[dependent]) {
|
|
|
|
signal[dependentSignals] ??= new WeakRefSet();
|
|
|
|
resultSignal[sourceSignals].add(signal);
|
|
|
|
signal[dependentSignals].add(resultSignal);
|
|
|
|
} else {
|
|
|
|
const sourceSignalArray = signal[sourceSignals].toArray();
|
|
|
|
for (let j = 0; j < sourceSignalArray.length; ++j) {
|
|
|
|
const sourceSignal = sourceSignalArray[j];
|
|
|
|
assert(sourceSignal[abortReason] === undefined);
|
|
|
|
assert(!sourceSignal[dependent]);
|
|
|
|
|
|
|
|
if (resultSignal[sourceSignals].has(sourceSignal)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
resultSignal[sourceSignals].add(sourceSignal);
|
|
|
|
sourceSignal[dependentSignals].add(resultSignal);
|
|
|
|
}
|
|
|
|
}
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
2023-11-12 19:04:11 -05:00
|
|
|
|
|
|
|
return resultSignal;
|
2023-02-07 14:22:46 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export {
|
|
|
|
AbortController,
|
|
|
|
AbortSignal,
|
|
|
|
AbortSignalPrototype,
|
|
|
|
add,
|
2023-11-12 19:04:11 -05:00
|
|
|
createDependentAbortSignal,
|
2023-02-07 14:22:46 -05:00
|
|
|
newSignal,
|
|
|
|
remove,
|
|
|
|
signalAbort,
|
2023-05-31 14:06:21 -04:00
|
|
|
timerId,
|
2023-02-07 14:22:46 -05:00
|
|
|
};
|