@nocobase/plugin-idp-oauth 2.1.0-alpha.17 → 2.1.0-alpha.18
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build.config.ts +1 -1
- package/dist/externalVersion.js +4 -4
- package/dist/node_modules/light-my-request/package.json +1 -1
- package/dist/node_modules/undici/LICENSE +21 -0
- package/dist/node_modules/undici/README.md +741 -0
- package/dist/node_modules/undici/docs/docs/api/Agent.md +84 -0
- package/dist/node_modules/undici/docs/docs/api/BalancedPool.md +99 -0
- package/dist/node_modules/undici/docs/docs/api/CacheStorage.md +30 -0
- package/dist/node_modules/undici/docs/docs/api/CacheStore.md +164 -0
- package/dist/node_modules/undici/docs/docs/api/Client.md +285 -0
- package/dist/node_modules/undici/docs/docs/api/ClientStats.md +27 -0
- package/dist/node_modules/undici/docs/docs/api/Connector.md +115 -0
- package/dist/node_modules/undici/docs/docs/api/ContentType.md +57 -0
- package/dist/node_modules/undici/docs/docs/api/Cookies.md +101 -0
- package/dist/node_modules/undici/docs/docs/api/Debug.md +62 -0
- package/dist/node_modules/undici/docs/docs/api/DiagnosticsChannel.md +315 -0
- package/dist/node_modules/undici/docs/docs/api/Dispatcher.md +1392 -0
- package/dist/node_modules/undici/docs/docs/api/EnvHttpProxyAgent.md +159 -0
- package/dist/node_modules/undici/docs/docs/api/Errors.md +49 -0
- package/dist/node_modules/undici/docs/docs/api/EventSource.md +45 -0
- package/dist/node_modules/undici/docs/docs/api/Fetch.md +60 -0
- package/dist/node_modules/undici/docs/docs/api/GlobalInstallation.md +139 -0
- package/dist/node_modules/undici/docs/docs/api/H2CClient.md +263 -0
- package/dist/node_modules/undici/docs/docs/api/MockAgent.md +603 -0
- package/dist/node_modules/undici/docs/docs/api/MockCallHistory.md +197 -0
- package/dist/node_modules/undici/docs/docs/api/MockCallHistoryLog.md +43 -0
- package/dist/node_modules/undici/docs/docs/api/MockClient.md +81 -0
- package/dist/node_modules/undici/docs/docs/api/MockErrors.md +12 -0
- package/dist/node_modules/undici/docs/docs/api/MockPool.md +555 -0
- package/dist/node_modules/undici/docs/docs/api/Pool.md +84 -0
- package/dist/node_modules/undici/docs/docs/api/PoolStats.md +35 -0
- package/dist/node_modules/undici/docs/docs/api/ProxyAgent.md +229 -0
- package/dist/node_modules/undici/docs/docs/api/RedirectHandler.md +93 -0
- package/dist/node_modules/undici/docs/docs/api/RetryAgent.md +50 -0
- package/dist/node_modules/undici/docs/docs/api/RetryHandler.md +118 -0
- package/dist/node_modules/undici/docs/docs/api/RoundRobinPool.md +145 -0
- package/dist/node_modules/undici/docs/docs/api/SnapshotAgent.md +616 -0
- package/dist/node_modules/undici/docs/docs/api/Socks5ProxyAgent.md +274 -0
- package/dist/node_modules/undici/docs/docs/api/Util.md +25 -0
- package/dist/node_modules/undici/docs/docs/api/WebSocket.md +141 -0
- package/dist/node_modules/undici/docs/docs/api/api-lifecycle.md +91 -0
- package/dist/node_modules/undici/docs/docs/best-practices/client-certificate.md +64 -0
- package/dist/node_modules/undici/docs/docs/best-practices/crawling.md +58 -0
- package/dist/node_modules/undici/docs/docs/best-practices/mocking-request.md +190 -0
- package/dist/node_modules/undici/docs/docs/best-practices/proxy.md +127 -0
- package/dist/node_modules/undici/docs/docs/best-practices/undici-vs-builtin-fetch.md +224 -0
- package/dist/node_modules/undici/docs/docs/best-practices/writing-tests.md +63 -0
- package/dist/node_modules/undici/index-fetch.js +65 -0
- package/dist/node_modules/undici/index.d.ts +3 -0
- package/dist/node_modules/undici/index.js +234 -0
- package/dist/node_modules/undici/lib/api/abort-signal.js +59 -0
- package/dist/node_modules/undici/lib/api/api-connect.js +110 -0
- package/dist/node_modules/undici/lib/api/api-pipeline.js +252 -0
- package/dist/node_modules/undici/lib/api/api-request.js +214 -0
- package/dist/node_modules/undici/lib/api/api-stream.js +209 -0
- package/dist/node_modules/undici/lib/api/api-upgrade.js +111 -0
- package/dist/node_modules/undici/lib/api/index.js +7 -0
- package/dist/node_modules/undici/lib/api/readable.js +580 -0
- package/dist/node_modules/undici/lib/cache/memory-cache-store.js +234 -0
- package/dist/node_modules/undici/lib/cache/sqlite-cache-store.js +461 -0
- package/dist/node_modules/undici/lib/core/connect.js +137 -0
- package/dist/node_modules/undici/lib/core/constants.js +143 -0
- package/dist/node_modules/undici/lib/core/diagnostics.js +227 -0
- package/dist/node_modules/undici/lib/core/errors.js +477 -0
- package/dist/node_modules/undici/lib/core/request.js +438 -0
- package/dist/node_modules/undici/lib/core/socks5-client.js +407 -0
- package/dist/node_modules/undici/lib/core/socks5-utils.js +203 -0
- package/dist/node_modules/undici/lib/core/symbols.js +75 -0
- package/dist/node_modules/undici/lib/core/tree.js +160 -0
- package/dist/node_modules/undici/lib/core/util.js +992 -0
- package/dist/node_modules/undici/lib/dispatcher/agent.js +158 -0
- package/dist/node_modules/undici/lib/dispatcher/balanced-pool.js +219 -0
- package/dist/node_modules/undici/lib/dispatcher/client-h1.js +1610 -0
- package/dist/node_modules/undici/lib/dispatcher/client-h2.js +995 -0
- package/dist/node_modules/undici/lib/dispatcher/client.js +659 -0
- package/dist/node_modules/undici/lib/dispatcher/dispatcher-base.js +165 -0
- package/dist/node_modules/undici/lib/dispatcher/dispatcher.js +48 -0
- package/dist/node_modules/undici/lib/dispatcher/env-http-proxy-agent.js +146 -0
- package/dist/node_modules/undici/lib/dispatcher/fixed-queue.js +135 -0
- package/dist/node_modules/undici/lib/dispatcher/h2c-client.js +51 -0
- package/dist/node_modules/undici/lib/dispatcher/pool-base.js +214 -0
- package/dist/node_modules/undici/lib/dispatcher/pool.js +118 -0
- package/dist/node_modules/undici/lib/dispatcher/proxy-agent.js +318 -0
- package/dist/node_modules/undici/lib/dispatcher/retry-agent.js +35 -0
- package/dist/node_modules/undici/lib/dispatcher/round-robin-pool.js +137 -0
- package/dist/node_modules/undici/lib/dispatcher/socks5-proxy-agent.js +249 -0
- package/dist/node_modules/undici/lib/encoding/index.js +33 -0
- package/dist/node_modules/undici/lib/global.js +50 -0
- package/dist/node_modules/undici/lib/handler/cache-handler.js +578 -0
- package/dist/node_modules/undici/lib/handler/cache-revalidation-handler.js +124 -0
- package/dist/node_modules/undici/lib/handler/decorator-handler.js +67 -0
- package/dist/node_modules/undici/lib/handler/deduplication-handler.js +460 -0
- package/dist/node_modules/undici/lib/handler/redirect-handler.js +238 -0
- package/dist/node_modules/undici/lib/handler/retry-handler.js +394 -0
- package/dist/node_modules/undici/lib/handler/unwrap-handler.js +100 -0
- package/dist/node_modules/undici/lib/handler/wrap-handler.js +105 -0
- package/dist/node_modules/undici/lib/interceptor/cache.js +495 -0
- package/dist/node_modules/undici/lib/interceptor/decompress.js +259 -0
- package/dist/node_modules/undici/lib/interceptor/deduplicate.js +117 -0
- package/dist/node_modules/undici/lib/interceptor/dns.js +571 -0
- package/dist/node_modules/undici/lib/interceptor/dump.js +112 -0
- package/dist/node_modules/undici/lib/interceptor/redirect.js +21 -0
- package/dist/node_modules/undici/lib/interceptor/response-error.js +95 -0
- package/dist/node_modules/undici/lib/interceptor/retry.js +19 -0
- package/dist/node_modules/undici/lib/llhttp/.gitkeep +0 -0
- package/dist/node_modules/undici/lib/llhttp/constants.d.ts +195 -0
- package/dist/node_modules/undici/lib/llhttp/constants.js +531 -0
- package/dist/node_modules/undici/lib/llhttp/llhttp-wasm.js +15 -0
- package/dist/node_modules/undici/lib/llhttp/llhttp_simd-wasm.js +15 -0
- package/dist/node_modules/undici/lib/llhttp/utils.d.ts +2 -0
- package/dist/node_modules/undici/lib/llhttp/utils.js +12 -0
- package/dist/node_modules/undici/lib/mock/mock-agent.js +232 -0
- package/dist/node_modules/undici/lib/mock/mock-call-history.js +248 -0
- package/dist/node_modules/undici/lib/mock/mock-client.js +68 -0
- package/dist/node_modules/undici/lib/mock/mock-errors.js +29 -0
- package/dist/node_modules/undici/lib/mock/mock-interceptor.js +209 -0
- package/dist/node_modules/undici/lib/mock/mock-pool.js +68 -0
- package/dist/node_modules/undici/lib/mock/mock-symbols.js +32 -0
- package/dist/node_modules/undici/lib/mock/mock-utils.js +486 -0
- package/dist/node_modules/undici/lib/mock/pending-interceptors-formatter.js +43 -0
- package/dist/node_modules/undici/lib/mock/snapshot-agent.js +353 -0
- package/dist/node_modules/undici/lib/mock/snapshot-recorder.js +588 -0
- package/dist/node_modules/undici/lib/mock/snapshot-utils.js +158 -0
- package/dist/node_modules/undici/lib/util/cache.js +407 -0
- package/dist/node_modules/undici/lib/util/date.js +653 -0
- package/dist/node_modules/undici/lib/util/promise.js +28 -0
- package/dist/node_modules/undici/lib/util/runtime-features.js +124 -0
- package/dist/node_modules/undici/lib/util/stats.js +32 -0
- package/dist/node_modules/undici/lib/util/timers.js +425 -0
- package/dist/node_modules/undici/lib/web/cache/cache.js +864 -0
- package/dist/node_modules/undici/lib/web/cache/cachestorage.js +152 -0
- package/dist/node_modules/undici/lib/web/cache/util.js +45 -0
- package/dist/node_modules/undici/lib/web/cookies/constants.js +12 -0
- package/dist/node_modules/undici/lib/web/cookies/index.js +199 -0
- package/dist/node_modules/undici/lib/web/cookies/parse.js +322 -0
- package/dist/node_modules/undici/lib/web/cookies/util.js +282 -0
- package/dist/node_modules/undici/lib/web/eventsource/eventsource-stream.js +399 -0
- package/dist/node_modules/undici/lib/web/eventsource/eventsource.js +501 -0
- package/dist/node_modules/undici/lib/web/eventsource/util.js +29 -0
- package/dist/node_modules/undici/lib/web/fetch/LICENSE +21 -0
- package/dist/node_modules/undici/lib/web/fetch/body.js +509 -0
- package/dist/node_modules/undici/lib/web/fetch/constants.js +131 -0
- package/dist/node_modules/undici/lib/web/fetch/data-url.js +596 -0
- package/dist/node_modules/undici/lib/web/fetch/formdata-parser.js +575 -0
- package/dist/node_modules/undici/lib/web/fetch/formdata.js +259 -0
- package/dist/node_modules/undici/lib/web/fetch/global.js +40 -0
- package/dist/node_modules/undici/lib/web/fetch/headers.js +719 -0
- package/dist/node_modules/undici/lib/web/fetch/index.js +2397 -0
- package/dist/node_modules/undici/lib/web/fetch/request.js +1115 -0
- package/dist/node_modules/undici/lib/web/fetch/response.js +641 -0
- package/dist/node_modules/undici/lib/web/fetch/util.js +1520 -0
- package/dist/node_modules/undici/lib/web/infra/index.js +229 -0
- package/dist/node_modules/undici/lib/web/subresource-integrity/Readme.md +9 -0
- package/dist/node_modules/undici/lib/web/subresource-integrity/subresource-integrity.js +307 -0
- package/dist/node_modules/undici/lib/web/webidl/index.js +1006 -0
- package/dist/node_modules/undici/lib/web/websocket/connection.js +329 -0
- package/dist/node_modules/undici/lib/web/websocket/constants.js +126 -0
- package/dist/node_modules/undici/lib/web/websocket/events.js +331 -0
- package/dist/node_modules/undici/lib/web/websocket/frame.js +133 -0
- package/dist/node_modules/undici/lib/web/websocket/permessage-deflate.js +118 -0
- package/dist/node_modules/undici/lib/web/websocket/receiver.js +450 -0
- package/dist/node_modules/undici/lib/web/websocket/sender.js +109 -0
- package/dist/node_modules/undici/lib/web/websocket/stream/websocketerror.js +104 -0
- package/dist/node_modules/undici/lib/web/websocket/stream/websocketstream.js +497 -0
- package/dist/node_modules/undici/lib/web/websocket/util.js +347 -0
- package/dist/node_modules/undici/lib/web/websocket/websocket.js +751 -0
- package/dist/node_modules/undici/package.json +152 -0
- package/dist/node_modules/undici/scripts/strip-comments.js +10 -0
- package/dist/node_modules/undici/types/README.md +6 -0
- package/dist/node_modules/undici/types/agent.d.ts +32 -0
- package/dist/node_modules/undici/types/api.d.ts +43 -0
- package/dist/node_modules/undici/types/balanced-pool.d.ts +30 -0
- package/dist/node_modules/undici/types/cache-interceptor.d.ts +179 -0
- package/dist/node_modules/undici/types/cache.d.ts +36 -0
- package/dist/node_modules/undici/types/client-stats.d.ts +15 -0
- package/dist/node_modules/undici/types/client.d.ts +123 -0
- package/dist/node_modules/undici/types/connector.d.ts +36 -0
- package/dist/node_modules/undici/types/content-type.d.ts +21 -0
- package/dist/node_modules/undici/types/cookies.d.ts +30 -0
- package/dist/node_modules/undici/types/diagnostics-channel.d.ts +74 -0
- package/dist/node_modules/undici/types/dispatcher.d.ts +273 -0
- package/dist/node_modules/undici/types/env-http-proxy-agent.d.ts +22 -0
- package/dist/node_modules/undici/types/errors.d.ts +177 -0
- package/dist/node_modules/undici/types/eventsource.d.ts +66 -0
- package/dist/node_modules/undici/types/fetch.d.ts +231 -0
- package/dist/node_modules/undici/types/formdata.d.ts +114 -0
- package/dist/node_modules/undici/types/global-dispatcher.d.ts +9 -0
- package/dist/node_modules/undici/types/global-origin.d.ts +7 -0
- package/dist/node_modules/undici/types/h2c-client.d.ts +73 -0
- package/dist/node_modules/undici/types/handlers.d.ts +14 -0
- package/dist/node_modules/undici/types/header.d.ts +160 -0
- package/dist/node_modules/undici/types/index.d.ts +91 -0
- package/dist/node_modules/undici/types/interceptors.d.ts +80 -0
- package/dist/node_modules/undici/types/mock-agent.d.ts +68 -0
- package/dist/node_modules/undici/types/mock-call-history.d.ts +111 -0
- package/dist/node_modules/undici/types/mock-client.d.ts +27 -0
- package/dist/node_modules/undici/types/mock-errors.d.ts +12 -0
- package/dist/node_modules/undici/types/mock-interceptor.d.ts +94 -0
- package/dist/node_modules/undici/types/mock-pool.d.ts +27 -0
- package/dist/node_modules/undici/types/patch.d.ts +29 -0
- package/dist/node_modules/undici/types/pool-stats.d.ts +19 -0
- package/dist/node_modules/undici/types/pool.d.ts +41 -0
- package/dist/node_modules/undici/types/proxy-agent.d.ts +29 -0
- package/dist/node_modules/undici/types/readable.d.ts +68 -0
- package/dist/node_modules/undici/types/retry-agent.d.ts +8 -0
- package/dist/node_modules/undici/types/retry-handler.d.ts +125 -0
- package/dist/node_modules/undici/types/round-robin-pool.d.ts +41 -0
- package/dist/node_modules/undici/types/snapshot-agent.d.ts +109 -0
- package/dist/node_modules/undici/types/socks5-proxy-agent.d.ts +25 -0
- package/dist/node_modules/undici/types/util.d.ts +18 -0
- package/dist/node_modules/undici/types/utility.d.ts +7 -0
- package/dist/node_modules/undici/types/webidl.d.ts +347 -0
- package/dist/node_modules/undici/types/websocket.d.ts +188 -0
- package/package.json +2 -2
|
@@ -0,0 +1,259 @@
|
|
|
1
|
+
'use strict'
|
|
2
|
+
|
|
3
|
+
const { createInflate, createGunzip, createBrotliDecompress, createZstdDecompress } = require('node:zlib')
|
|
4
|
+
const { pipeline } = require('node:stream')
|
|
5
|
+
const DecoratorHandler = require('../handler/decorator-handler')
|
|
6
|
+
const { runtimeFeatures } = require('../util/runtime-features')
|
|
7
|
+
|
|
8
|
+
/** @typedef {import('node:stream').Transform} Transform */
|
|
9
|
+
/** @typedef {import('node:stream').Transform} Controller */
|
|
10
|
+
/** @typedef {Transform&import('node:zlib').Zlib} DecompressorStream */
|
|
11
|
+
|
|
12
|
+
/** @type {Record<string, () => DecompressorStream>} */
|
|
13
|
+
const supportedEncodings = {
|
|
14
|
+
gzip: createGunzip,
|
|
15
|
+
'x-gzip': createGunzip,
|
|
16
|
+
br: createBrotliDecompress,
|
|
17
|
+
deflate: createInflate,
|
|
18
|
+
compress: createInflate,
|
|
19
|
+
'x-compress': createInflate,
|
|
20
|
+
...(runtimeFeatures.has('zstd') ? { zstd: createZstdDecompress } : {})
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const defaultSkipStatusCodes = /** @type {const} */ ([204, 304])
|
|
24
|
+
|
|
25
|
+
let warningEmitted = /** @type {boolean} */ (false)
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* @typedef {Object} DecompressHandlerOptions
|
|
29
|
+
* @property {number[]|Readonly<number[]>} [skipStatusCodes=[204, 304]] - List of status codes to skip decompression for
|
|
30
|
+
* @property {boolean} [skipErrorResponses] - Whether to skip decompression for error responses (status codes >= 400)
|
|
31
|
+
*/
|
|
32
|
+
|
|
33
|
+
class DecompressHandler extends DecoratorHandler {
|
|
34
|
+
/** @type {Transform[]} */
|
|
35
|
+
#decompressors = []
|
|
36
|
+
/** @type {Readonly<number[]>} */
|
|
37
|
+
#skipStatusCodes
|
|
38
|
+
/** @type {boolean} */
|
|
39
|
+
#skipErrorResponses
|
|
40
|
+
|
|
41
|
+
constructor (handler, { skipStatusCodes = defaultSkipStatusCodes, skipErrorResponses = true } = {}) {
|
|
42
|
+
super(handler)
|
|
43
|
+
this.#skipStatusCodes = skipStatusCodes
|
|
44
|
+
this.#skipErrorResponses = skipErrorResponses
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
/**
|
|
48
|
+
* Determines if decompression should be skipped based on encoding and status code
|
|
49
|
+
* @param {string} contentEncoding - Content-Encoding header value
|
|
50
|
+
* @param {number} statusCode - HTTP status code of the response
|
|
51
|
+
* @returns {boolean} - True if decompression should be skipped
|
|
52
|
+
*/
|
|
53
|
+
#shouldSkipDecompression (contentEncoding, statusCode) {
|
|
54
|
+
if (!contentEncoding || statusCode < 200) return true
|
|
55
|
+
if (this.#skipStatusCodes.includes(statusCode)) return true
|
|
56
|
+
if (this.#skipErrorResponses && statusCode >= 400) return true
|
|
57
|
+
return false
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
/**
|
|
61
|
+
* Creates a chain of decompressors for multiple content encodings
|
|
62
|
+
*
|
|
63
|
+
* @param {string} encodings - Comma-separated list of content encodings
|
|
64
|
+
* @returns {Array<DecompressorStream>} - Array of decompressor streams
|
|
65
|
+
* @throws {Error} - If the number of content-encodings exceeds the maximum allowed
|
|
66
|
+
*/
|
|
67
|
+
#createDecompressionChain (encodings) {
|
|
68
|
+
const parts = encodings.split(',')
|
|
69
|
+
|
|
70
|
+
// Limit the number of content-encodings to prevent resource exhaustion.
|
|
71
|
+
// CVE fix similar to urllib3 (GHSA-gm62-xv2j-4w53) and curl (CVE-2022-32206).
|
|
72
|
+
const maxContentEncodings = 5
|
|
73
|
+
if (parts.length > maxContentEncodings) {
|
|
74
|
+
throw new Error(`too many content-encodings in response: ${parts.length}, maximum allowed is ${maxContentEncodings}`)
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/** @type {DecompressorStream[]} */
|
|
78
|
+
const decompressors = []
|
|
79
|
+
|
|
80
|
+
for (let i = parts.length - 1; i >= 0; i--) {
|
|
81
|
+
const encoding = parts[i].trim()
|
|
82
|
+
if (!encoding) continue
|
|
83
|
+
|
|
84
|
+
if (!supportedEncodings[encoding]) {
|
|
85
|
+
decompressors.length = 0 // Clear if unsupported encoding
|
|
86
|
+
return decompressors // Unsupported encoding
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
decompressors.push(supportedEncodings[encoding]())
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return decompressors
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Sets up event handlers for a decompressor stream using readable events
|
|
97
|
+
* @param {DecompressorStream} decompressor - The decompressor stream
|
|
98
|
+
* @param {Controller} controller - The controller to coordinate with
|
|
99
|
+
* @returns {void}
|
|
100
|
+
*/
|
|
101
|
+
#setupDecompressorEvents (decompressor, controller) {
|
|
102
|
+
decompressor.on('readable', () => {
|
|
103
|
+
let chunk
|
|
104
|
+
while ((chunk = decompressor.read()) !== null) {
|
|
105
|
+
const result = super.onResponseData(controller, chunk)
|
|
106
|
+
if (result === false) {
|
|
107
|
+
break
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
})
|
|
111
|
+
|
|
112
|
+
decompressor.on('error', (error) => {
|
|
113
|
+
super.onResponseError(controller, error)
|
|
114
|
+
})
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
/**
|
|
118
|
+
* Sets up event handling for a single decompressor
|
|
119
|
+
* @param {Controller} controller - The controller to handle events
|
|
120
|
+
* @returns {void}
|
|
121
|
+
*/
|
|
122
|
+
#setupSingleDecompressor (controller) {
|
|
123
|
+
const decompressor = this.#decompressors[0]
|
|
124
|
+
this.#setupDecompressorEvents(decompressor, controller)
|
|
125
|
+
|
|
126
|
+
decompressor.on('end', () => {
|
|
127
|
+
super.onResponseEnd(controller, {})
|
|
128
|
+
})
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Sets up event handling for multiple chained decompressors using pipeline
|
|
133
|
+
* @param {Controller} controller - The controller to handle events
|
|
134
|
+
* @returns {void}
|
|
135
|
+
*/
|
|
136
|
+
#setupMultipleDecompressors (controller) {
|
|
137
|
+
const lastDecompressor = this.#decompressors[this.#decompressors.length - 1]
|
|
138
|
+
this.#setupDecompressorEvents(lastDecompressor, controller)
|
|
139
|
+
|
|
140
|
+
pipeline(this.#decompressors, (err) => {
|
|
141
|
+
if (err) {
|
|
142
|
+
super.onResponseError(controller, err)
|
|
143
|
+
return
|
|
144
|
+
}
|
|
145
|
+
super.onResponseEnd(controller, {})
|
|
146
|
+
})
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
/**
|
|
150
|
+
* Cleans up decompressor references to prevent memory leaks
|
|
151
|
+
* @returns {void}
|
|
152
|
+
*/
|
|
153
|
+
#cleanupDecompressors () {
|
|
154
|
+
this.#decompressors.length = 0
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
/**
|
|
158
|
+
* @param {Controller} controller
|
|
159
|
+
* @param {number} statusCode
|
|
160
|
+
* @param {Record<string, string | string[] | undefined>} headers
|
|
161
|
+
* @param {string} statusMessage
|
|
162
|
+
* @returns {void}
|
|
163
|
+
*/
|
|
164
|
+
onResponseStart (controller, statusCode, headers, statusMessage) {
|
|
165
|
+
const contentEncoding = headers['content-encoding']
|
|
166
|
+
|
|
167
|
+
// If content encoding is not supported or status code is in skip list
|
|
168
|
+
if (this.#shouldSkipDecompression(contentEncoding, statusCode)) {
|
|
169
|
+
return super.onResponseStart(controller, statusCode, headers, statusMessage)
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
const decompressors = this.#createDecompressionChain(contentEncoding.toLowerCase())
|
|
173
|
+
|
|
174
|
+
if (decompressors.length === 0) {
|
|
175
|
+
this.#cleanupDecompressors()
|
|
176
|
+
return super.onResponseStart(controller, statusCode, headers, statusMessage)
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
this.#decompressors = decompressors
|
|
180
|
+
|
|
181
|
+
// Remove compression headers since we're decompressing
|
|
182
|
+
const { 'content-encoding': _, 'content-length': __, ...newHeaders } = headers
|
|
183
|
+
|
|
184
|
+
if (this.#decompressors.length === 1) {
|
|
185
|
+
this.#setupSingleDecompressor(controller)
|
|
186
|
+
} else {
|
|
187
|
+
this.#setupMultipleDecompressors(controller)
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
return super.onResponseStart(controller, statusCode, newHeaders, statusMessage)
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
/**
|
|
194
|
+
* @param {Controller} controller
|
|
195
|
+
* @param {Buffer} chunk
|
|
196
|
+
* @returns {void}
|
|
197
|
+
*/
|
|
198
|
+
onResponseData (controller, chunk) {
|
|
199
|
+
if (this.#decompressors.length > 0) {
|
|
200
|
+
this.#decompressors[0].write(chunk)
|
|
201
|
+
return
|
|
202
|
+
}
|
|
203
|
+
super.onResponseData(controller, chunk)
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
/**
|
|
207
|
+
* @param {Controller} controller
|
|
208
|
+
* @param {Record<string, string | string[]> | undefined} trailers
|
|
209
|
+
* @returns {void}
|
|
210
|
+
*/
|
|
211
|
+
onResponseEnd (controller, trailers) {
|
|
212
|
+
if (this.#decompressors.length > 0) {
|
|
213
|
+
this.#decompressors[0].end()
|
|
214
|
+
this.#cleanupDecompressors()
|
|
215
|
+
return
|
|
216
|
+
}
|
|
217
|
+
super.onResponseEnd(controller, trailers)
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
/**
|
|
221
|
+
* @param {Controller} controller
|
|
222
|
+
* @param {Error} err
|
|
223
|
+
* @returns {void}
|
|
224
|
+
*/
|
|
225
|
+
onResponseError (controller, err) {
|
|
226
|
+
if (this.#decompressors.length > 0) {
|
|
227
|
+
for (const decompressor of this.#decompressors) {
|
|
228
|
+
decompressor.destroy(err)
|
|
229
|
+
}
|
|
230
|
+
this.#cleanupDecompressors()
|
|
231
|
+
}
|
|
232
|
+
super.onResponseError(controller, err)
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
/**
|
|
237
|
+
* Creates a decompression interceptor for HTTP responses
|
|
238
|
+
* @param {DecompressHandlerOptions} [options] - Options for the interceptor
|
|
239
|
+
* @returns {Function} - Interceptor function
|
|
240
|
+
*/
|
|
241
|
+
function createDecompressInterceptor (options = {}) {
|
|
242
|
+
// Emit experimental warning only once
|
|
243
|
+
if (!warningEmitted) {
|
|
244
|
+
process.emitWarning(
|
|
245
|
+
'DecompressInterceptor is experimental and subject to change',
|
|
246
|
+
'ExperimentalWarning'
|
|
247
|
+
)
|
|
248
|
+
warningEmitted = true
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
return (dispatch) => {
|
|
252
|
+
return (opts, handler) => {
|
|
253
|
+
const decompressHandler = new DecompressHandler(handler, options)
|
|
254
|
+
return dispatch(opts, decompressHandler)
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
module.exports = createDecompressInterceptor
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
'use strict'
|
|
2
|
+
|
|
3
|
+
const diagnosticsChannel = require('node:diagnostics_channel')
|
|
4
|
+
const util = require('../core/util')
|
|
5
|
+
const DeduplicationHandler = require('../handler/deduplication-handler')
|
|
6
|
+
const { normalizeHeaders, makeCacheKey, makeDeduplicationKey } = require('../util/cache.js')
|
|
7
|
+
|
|
8
|
+
const pendingRequestsChannel = diagnosticsChannel.channel('undici:request:pending-requests')
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* @param {import('../../types/interceptors.d.ts').default.DeduplicateInterceptorOpts} [opts]
|
|
12
|
+
* @returns {import('../../types/dispatcher.d.ts').default.DispatcherComposeInterceptor}
|
|
13
|
+
*/
|
|
14
|
+
module.exports = (opts = {}) => {
|
|
15
|
+
const {
|
|
16
|
+
methods = ['GET'],
|
|
17
|
+
skipHeaderNames = [],
|
|
18
|
+
excludeHeaderNames = [],
|
|
19
|
+
maxBufferSize = 5 * 1024 * 1024
|
|
20
|
+
} = opts
|
|
21
|
+
|
|
22
|
+
if (typeof opts !== 'object' || opts === null) {
|
|
23
|
+
throw new TypeError(`expected type of opts to be an Object, got ${opts === null ? 'null' : typeof opts}`)
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
if (!Array.isArray(methods)) {
|
|
27
|
+
throw new TypeError(`expected opts.methods to be an array, got ${typeof methods}`)
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
for (const method of methods) {
|
|
31
|
+
if (!util.safeHTTPMethods.includes(method)) {
|
|
32
|
+
throw new TypeError(`expected opts.methods to only contain safe HTTP methods, got ${method}`)
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
if (!Array.isArray(skipHeaderNames)) {
|
|
37
|
+
throw new TypeError(`expected opts.skipHeaderNames to be an array, got ${typeof skipHeaderNames}`)
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
if (!Array.isArray(excludeHeaderNames)) {
|
|
41
|
+
throw new TypeError(`expected opts.excludeHeaderNames to be an array, got ${typeof excludeHeaderNames}`)
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
if (!Number.isFinite(maxBufferSize) || maxBufferSize <= 0) {
|
|
45
|
+
throw new TypeError(`expected opts.maxBufferSize to be a positive finite number, got ${maxBufferSize}`)
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// Convert to lowercase Set for case-insensitive header matching
|
|
49
|
+
const skipHeaderNamesSet = new Set(skipHeaderNames.map(name => name.toLowerCase()))
|
|
50
|
+
|
|
51
|
+
// Convert to lowercase Set for case-insensitive header exclusion from deduplication key
|
|
52
|
+
const excludeHeaderNamesSet = new Set(excludeHeaderNames.map(name => name.toLowerCase()))
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Map of pending requests for deduplication
|
|
56
|
+
* @type {Map<string, DeduplicationHandler>}
|
|
57
|
+
*/
|
|
58
|
+
const pendingRequests = new Map()
|
|
59
|
+
|
|
60
|
+
return dispatch => {
|
|
61
|
+
return (opts, handler) => {
|
|
62
|
+
if (!opts.origin || methods.includes(opts.method) === false) {
|
|
63
|
+
return dispatch(opts, handler)
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
opts = {
|
|
67
|
+
...opts,
|
|
68
|
+
headers: normalizeHeaders(opts)
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// Skip deduplication if request contains any of the specified headers
|
|
72
|
+
if (skipHeaderNamesSet.size > 0) {
|
|
73
|
+
for (const headerName of Object.keys(opts.headers)) {
|
|
74
|
+
if (skipHeaderNamesSet.has(headerName.toLowerCase())) {
|
|
75
|
+
return dispatch(opts, handler)
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
const cacheKey = makeCacheKey(opts)
|
|
81
|
+
const dedupeKey = makeDeduplicationKey(cacheKey, excludeHeaderNamesSet)
|
|
82
|
+
|
|
83
|
+
// Check if there's already a pending request for this key
|
|
84
|
+
const pendingHandler = pendingRequests.get(dedupeKey)
|
|
85
|
+
if (pendingHandler) {
|
|
86
|
+
// Add this handler to the waiting list when safe.
|
|
87
|
+
// If body streaming has already started, this request must be sent independently.
|
|
88
|
+
if (pendingHandler.addWaitingHandler(handler)) {
|
|
89
|
+
return true
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return dispatch(opts, handler)
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Create a new deduplication handler
|
|
96
|
+
const deduplicationHandler = new DeduplicationHandler(
|
|
97
|
+
handler,
|
|
98
|
+
() => {
|
|
99
|
+
// Clean up when request completes
|
|
100
|
+
pendingRequests.delete(dedupeKey)
|
|
101
|
+
if (pendingRequestsChannel.hasSubscribers) {
|
|
102
|
+
pendingRequestsChannel.publish({ size: pendingRequests.size, key: dedupeKey, type: 'removed' })
|
|
103
|
+
}
|
|
104
|
+
},
|
|
105
|
+
maxBufferSize
|
|
106
|
+
)
|
|
107
|
+
|
|
108
|
+
// Register the pending request
|
|
109
|
+
pendingRequests.set(dedupeKey, deduplicationHandler)
|
|
110
|
+
if (pendingRequestsChannel.hasSubscribers) {
|
|
111
|
+
pendingRequestsChannel.publish({ size: pendingRequests.size, key: dedupeKey, type: 'added' })
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
return dispatch(opts, deduplicationHandler)
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
}
|