From b6aa794e2db44f20adc4c9cc99009e72048f4fed Mon Sep 17 00:00:00 2001 From: Robert Nagy Date: Sun, 7 Apr 2024 10:59:06 +0200 Subject: [PATCH] fix: signal handling (#3053) --- lib/api/abort-signal.js | 5 ++++- lib/api/api-connect.js | 10 +++++++--- lib/api/api-pipeline.js | 10 ++++++---- lib/api/api-request.js | 13 +++++++------ lib/api/api-stream.js | 14 +++++++------- lib/api/api-upgrade.js | 9 ++++++--- lib/mock/mock-utils.js | 10 +++++----- 7 files changed, 42 insertions(+), 29 deletions(-) diff --git a/lib/api/abort-signal.js b/lib/api/abort-signal.js index 3c15202ff44..2afe747d3a1 100644 --- a/lib/api/abort-signal.js +++ b/lib/api/abort-signal.js @@ -8,11 +8,14 @@ function abort (self) { if (self.abort) { self.abort(self[kSignal]?.reason) } else { - self.onError(self[kSignal]?.reason ?? new RequestAbortedError()) + self.reason = self[kSignal]?.reason ?? new RequestAbortedError() } + removeSignal(self) } function addSignal (self, signal) { + self.reason = null + self[kSignal] = null self[kListener] = null diff --git a/lib/api/api-connect.js b/lib/api/api-connect.js index d19e67f69a2..f9dbbf64fe8 100644 --- a/lib/api/api-connect.js +++ b/lib/api/api-connect.js @@ -1,7 +1,8 @@ 'use strict' +const assert = require('node:assert') const { AsyncResource } = require('node:async_hooks') -const { InvalidArgumentError, RequestAbortedError, SocketError } = require('../core/errors') +const { InvalidArgumentError, SocketError } = require('../core/errors') const util = require('../core/util') const { addSignal, removeSignal } = require('./abort-signal') @@ -32,10 +33,13 @@ class ConnectHandler extends AsyncResource { } onConnect (abort, context) { - if (!this.callback) { - throw new RequestAbortedError() + if (this.reason) { + abort(this.reason) + return } + assert(this.callback) + this.abort = abort this.context = context } diff --git a/lib/api/api-pipeline.js b/lib/api/api-pipeline.js index f5112415bf7..e64b3294965 100644 --- a/lib/api/api-pipeline.js +++ b/lib/api/api-pipeline.js @@ -147,12 +147,14 @@ class PipelineHandler extends AsyncResource { onConnect (abort, context) { const { ret, res } = this - assert(!res, 'pipeline cannot be retried') - - if (ret.destroyed) { - throw new RequestAbortedError() + if (this.reason) { + abort(this.reason) + return } + assert(!res, 'pipeline cannot be retried') + assert(!ret.destroyed) + this.abort = abort this.context = context } diff --git a/lib/api/api-request.js b/lib/api/api-request.js index 92c8c84228e..e5d598aa6dd 100644 --- a/lib/api/api-request.js +++ b/lib/api/api-request.js @@ -1,10 +1,8 @@ 'use strict' +const assert = require('node:assert') const { Readable } = require('./readable') -const { - InvalidArgumentError, - RequestAbortedError -} = require('../core/errors') +const { InvalidArgumentError } = require('../core/errors') const util = require('../core/util') const { getResolveErrorBodyCallback } = require('./util') const { AsyncResource } = require('node:async_hooks') @@ -69,10 +67,13 @@ class RequestHandler extends AsyncResource { } onConnect (abort, context) { - if (!this.callback) { - throw new RequestAbortedError() + if (this.reason) { + abort(this.reason) + return } + assert(this.callback) + this.abort = abort this.context = context } diff --git a/lib/api/api-stream.js b/lib/api/api-stream.js index 0998c9f2da1..fba2266dd7d 100644 --- a/lib/api/api-stream.js +++ b/lib/api/api-stream.js @@ -1,11 +1,8 @@ 'use strict' +const assert = require('node:assert') const { finished, PassThrough } = require('node:stream') -const { - InvalidArgumentError, - InvalidReturnValueError, - RequestAbortedError -} = require('../core/errors') +const { InvalidArgumentError, InvalidReturnValueError } = require('../core/errors') const util = require('../core/util') const { getResolveErrorBodyCallback } = require('./util') const { AsyncResource } = require('node:async_hooks') @@ -70,10 +67,13 @@ class StreamHandler extends AsyncResource { } onConnect (abort, context) { - if (!this.callback) { - throw new RequestAbortedError() + if (this.reason) { + abort(this.reason) + return } + assert(this.callback) + this.abort = abort this.context = context } diff --git a/lib/api/api-upgrade.js b/lib/api/api-upgrade.js index bed946fdca9..019fe1efa2d 100644 --- a/lib/api/api-upgrade.js +++ b/lib/api/api-upgrade.js @@ -1,6 +1,6 @@ 'use strict' -const { InvalidArgumentError, RequestAbortedError, SocketError } = require('../core/errors') +const { InvalidArgumentError, SocketError } = require('../core/errors') const { AsyncResource } = require('node:async_hooks') const util = require('../core/util') const { addSignal, removeSignal } = require('./abort-signal') @@ -34,10 +34,13 @@ class UpgradeHandler extends AsyncResource { } onConnect (abort, context) { - if (!this.callback) { - throw new RequestAbortedError() + if (this.reason) { + abort(this.reason) + return } + assert(this.callback) + this.abort = abort this.context = null } diff --git a/lib/mock/mock-utils.js b/lib/mock/mock-utils.js index c8c0ed1eef1..f3c284d7891 100644 --- a/lib/mock/mock-utils.js +++ b/lib/mock/mock-utils.js @@ -8,7 +8,7 @@ const { kOrigin, kGetNetConnect } = require('./mock-symbols') -const { buildURL, nop } = require('../core/util') +const { buildURL } = require('../core/util') const { STATUS_CODES } = require('node:http') const { types: { @@ -285,10 +285,10 @@ function mockDispatch (opts, handler) { const responseHeaders = generateKeyValues(headers) const responseTrailers = generateKeyValues(trailers) - handler.abort = nop - handler.onHeaders(statusCode, responseHeaders, resume, getStatusText(statusCode)) - handler.onData(Buffer.from(responseData)) - handler.onComplete(responseTrailers) + handler.onConnect?.(err => handler.onError(err), null) + handler.onHeaders?.(statusCode, responseHeaders, resume, getStatusText(statusCode)) + handler.onData?.(Buffer.from(responseData)) + handler.onComplete?.(responseTrailers) deleteMockDispatch(mockDispatches, key) }