Spaces:
Runtime error
Runtime error
| // https://github.com/Ethan-Arrowood/undici-fetch | |
| const { | |
| Response, | |
| makeNetworkError, | |
| makeAppropriateNetworkError, | |
| filterResponse, | |
| makeResponse | |
| } = require('./response') | |
| const { Headers } = require('./headers') | |
| const { Request, makeRequest } = require('./request') | |
| const zlib = require('zlib') | |
| const { | |
| bytesMatch, | |
| makePolicyContainer, | |
| clonePolicyContainer, | |
| requestBadPort, | |
| TAOCheck, | |
| appendRequestOriginHeader, | |
| responseLocationURL, | |
| requestCurrentURL, | |
| setRequestReferrerPolicyOnRedirect, | |
| tryUpgradeRequestToAPotentiallyTrustworthyURL, | |
| createOpaqueTimingInfo, | |
| appendFetchMetadata, | |
| corsCheck, | |
| crossOriginResourcePolicyCheck, | |
| determineRequestsReferrer, | |
| coarsenedSharedCurrentTime, | |
| createDeferredPromise, | |
| isBlobLike, | |
| sameOrigin, | |
| isCancelled, | |
| isAborted, | |
| isErrorLike, | |
| fullyReadBody, | |
| readableStreamClose, | |
| isomorphicEncode, | |
| urlIsLocal, | |
| urlIsHttpHttpsScheme, | |
| urlHasHttpsScheme | |
| } = require('./util') | |
| const { kState, kHeaders, kGuard, kRealm } = require('./symbols') | |
| const assert = require('assert') | |
| const { safelyExtractBody } = require('./body') | |
| const { | |
| redirectStatusSet, | |
| nullBodyStatus, | |
| safeMethodsSet, | |
| requestBodyHeader, | |
| subresourceSet, | |
| DOMException | |
| } = require('./constants') | |
| const { kHeadersList } = require('../core/symbols') | |
| const EE = require('events') | |
| const { Readable, pipeline } = require('stream') | |
| const { addAbortListener, isErrored, isReadable, nodeMajor, nodeMinor } = require('../core/util') | |
| const { dataURLProcessor, serializeAMimeType } = require('./dataURL') | |
| const { TransformStream } = require('stream/web') | |
| const { getGlobalDispatcher } = require('../global') | |
| const { webidl } = require('./webidl') | |
| const { STATUS_CODES } = require('http') | |
| const GET_OR_HEAD = ['GET', 'HEAD'] | |
| /** @type {import('buffer').resolveObjectURL} */ | |
| let resolveObjectURL | |
| let ReadableStream = globalThis.ReadableStream | |
| class Fetch extends EE { | |
| constructor (dispatcher) { | |
| super() | |
| this.dispatcher = dispatcher | |
| this.connection = null | |
| this.dump = false | |
| this.state = 'ongoing' | |
| // 2 terminated listeners get added per request, | |
| // but only 1 gets removed. If there are 20 redirects, | |
| // 21 listeners will be added. | |
| // See https://github.com/nodejs/undici/issues/1711 | |
| // TODO (fix): Find and fix root cause for leaked listener. | |
| this.setMaxListeners(21) | |
| } | |
| terminate (reason) { | |
| if (this.state !== 'ongoing') { | |
| return | |
| } | |
| this.state = 'terminated' | |
| this.connection?.destroy(reason) | |
| this.emit('terminated', reason) | |
| } | |
| // https://fetch.spec.whatwg.org/#fetch-controller-abort | |
| abort (error) { | |
| if (this.state !== 'ongoing') { | |
| return | |
| } | |
| // 1. Set controller’s state to "aborted". | |
| this.state = 'aborted' | |
| // 2. Let fallbackError be an "AbortError" DOMException. | |
| // 3. Set error to fallbackError if it is not given. | |
| if (!error) { | |
| error = new DOMException('The operation was aborted.', 'AbortError') | |
| } | |
| // 4. Let serializedError be StructuredSerialize(error). | |
| // If that threw an exception, catch it, and let | |
| // serializedError be StructuredSerialize(fallbackError). | |
| // 5. Set controller’s serialized abort reason to serializedError. | |
| this.serializedAbortReason = error | |
| this.connection?.destroy(error) | |
| this.emit('terminated', error) | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#fetch-method | |
| function fetch (input, init = {}) { | |
| webidl.argumentLengthCheck(arguments, 1, { header: 'globalThis.fetch' }) | |
| // 1. Let p be a new promise. | |
| const p = createDeferredPromise() | |
| // 2. Let requestObject be the result of invoking the initial value of | |
| // Request as constructor with input and init as arguments. If this throws | |
| // an exception, reject p with it and return p. | |
| let requestObject | |
| try { | |
| requestObject = new Request(input, init) | |
| } catch (e) { | |
| p.reject(e) | |
| return p.promise | |
| } | |
| // 3. Let request be requestObject’s request. | |
| const request = requestObject[kState] | |
| // 4. If requestObject’s signal’s aborted flag is set, then: | |
| if (requestObject.signal.aborted) { | |
| // 1. Abort the fetch() call with p, request, null, and | |
| // requestObject’s signal’s abort reason. | |
| abortFetch(p, request, null, requestObject.signal.reason) | |
| // 2. Return p. | |
| return p.promise | |
| } | |
| // 5. Let globalObject be request’s client’s global object. | |
| const globalObject = request.client.globalObject | |
| // 6. If globalObject is a ServiceWorkerGlobalScope object, then set | |
| // request’s service-workers mode to "none". | |
| if (globalObject?.constructor?.name === 'ServiceWorkerGlobalScope') { | |
| request.serviceWorkers = 'none' | |
| } | |
| // 7. Let responseObject be null. | |
| let responseObject = null | |
| // 8. Let relevantRealm be this’s relevant Realm. | |
| const relevantRealm = null | |
| // 9. Let locallyAborted be false. | |
| let locallyAborted = false | |
| // 10. Let controller be null. | |
| let controller = null | |
| // 11. Add the following abort steps to requestObject’s signal: | |
| addAbortListener( | |
| requestObject.signal, | |
| () => { | |
| // 1. Set locallyAborted to true. | |
| locallyAborted = true | |
| // 2. Assert: controller is non-null. | |
| assert(controller != null) | |
| // 3. Abort controller with requestObject’s signal’s abort reason. | |
| controller.abort(requestObject.signal.reason) | |
| // 4. Abort the fetch() call with p, request, responseObject, | |
| // and requestObject’s signal’s abort reason. | |
| abortFetch(p, request, responseObject, requestObject.signal.reason) | |
| } | |
| ) | |
| // 12. Let handleFetchDone given response response be to finalize and | |
| // report timing with response, globalObject, and "fetch". | |
| const handleFetchDone = (response) => | |
| finalizeAndReportTiming(response, 'fetch') | |
| // 13. Set controller to the result of calling fetch given request, | |
| // with processResponseEndOfBody set to handleFetchDone, and processResponse | |
| // given response being these substeps: | |
| const processResponse = (response) => { | |
| // 1. If locallyAborted is true, terminate these substeps. | |
| if (locallyAborted) { | |
| return Promise.resolve() | |
| } | |
| // 2. If response’s aborted flag is set, then: | |
| if (response.aborted) { | |
| // 1. Let deserializedError be the result of deserialize a serialized | |
| // abort reason given controller’s serialized abort reason and | |
| // relevantRealm. | |
| // 2. Abort the fetch() call with p, request, responseObject, and | |
| // deserializedError. | |
| abortFetch(p, request, responseObject, controller.serializedAbortReason) | |
| return Promise.resolve() | |
| } | |
| // 3. If response is a network error, then reject p with a TypeError | |
| // and terminate these substeps. | |
| if (response.type === 'error') { | |
| p.reject( | |
| Object.assign(new TypeError('fetch failed'), { cause: response.error }) | |
| ) | |
| return Promise.resolve() | |
| } | |
| // 4. Set responseObject to the result of creating a Response object, | |
| // given response, "immutable", and relevantRealm. | |
| responseObject = new Response() | |
| responseObject[kState] = response | |
| responseObject[kRealm] = relevantRealm | |
| responseObject[kHeaders][kHeadersList] = response.headersList | |
| responseObject[kHeaders][kGuard] = 'immutable' | |
| responseObject[kHeaders][kRealm] = relevantRealm | |
| // 5. Resolve p with responseObject. | |
| p.resolve(responseObject) | |
| } | |
| controller = fetching({ | |
| request, | |
| processResponseEndOfBody: handleFetchDone, | |
| processResponse, | |
| dispatcher: init.dispatcher ?? getGlobalDispatcher() // undici | |
| }) | |
| // 14. Return p. | |
| return p.promise | |
| } | |
| // https://fetch.spec.whatwg.org/#finalize-and-report-timing | |
| function finalizeAndReportTiming (response, initiatorType = 'other') { | |
| // 1. If response is an aborted network error, then return. | |
| if (response.type === 'error' && response.aborted) { | |
| return | |
| } | |
| // 2. If response’s URL list is null or empty, then return. | |
| if (!response.urlList?.length) { | |
| return | |
| } | |
| // 3. Let originalURL be response’s URL list[0]. | |
| const originalURL = response.urlList[0] | |
| // 4. Let timingInfo be response’s timing info. | |
| let timingInfo = response.timingInfo | |
| // 5. Let cacheState be response’s cache state. | |
| let cacheState = response.cacheState | |
| // 6. If originalURL’s scheme is not an HTTP(S) scheme, then return. | |
| if (!urlIsHttpHttpsScheme(originalURL)) { | |
| return | |
| } | |
| // 7. If timingInfo is null, then return. | |
| if (timingInfo === null) { | |
| return | |
| } | |
| // 8. If response’s timing allow passed flag is not set, then: | |
| if (!timingInfo.timingAllowPassed) { | |
| // 1. Set timingInfo to a the result of creating an opaque timing info for timingInfo. | |
| timingInfo = createOpaqueTimingInfo({ | |
| startTime: timingInfo.startTime | |
| }) | |
| // 2. Set cacheState to the empty string. | |
| cacheState = '' | |
| } | |
| // 9. Set timingInfo’s end time to the coarsened shared current time | |
| // given global’s relevant settings object’s cross-origin isolated | |
| // capability. | |
| // TODO: given global’s relevant settings object’s cross-origin isolated | |
| // capability? | |
| timingInfo.endTime = coarsenedSharedCurrentTime() | |
| // 10. Set response’s timing info to timingInfo. | |
| response.timingInfo = timingInfo | |
| // 11. Mark resource timing for timingInfo, originalURL, initiatorType, | |
| // global, and cacheState. | |
| markResourceTiming( | |
| timingInfo, | |
| originalURL, | |
| initiatorType, | |
| globalThis, | |
| cacheState | |
| ) | |
| } | |
| // https://w3c.github.io/resource-timing/#dfn-mark-resource-timing | |
| function markResourceTiming (timingInfo, originalURL, initiatorType, globalThis, cacheState) { | |
| if (nodeMajor > 18 || (nodeMajor === 18 && nodeMinor >= 2)) { | |
| performance.markResourceTiming(timingInfo, originalURL.href, initiatorType, globalThis, cacheState) | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#abort-fetch | |
| function abortFetch (p, request, responseObject, error) { | |
| // Note: AbortSignal.reason was added in node v17.2.0 | |
| // which would give us an undefined error to reject with. | |
| // Remove this once node v16 is no longer supported. | |
| if (!error) { | |
| error = new DOMException('The operation was aborted.', 'AbortError') | |
| } | |
| // 1. Reject promise with error. | |
| p.reject(error) | |
| // 2. If request’s body is not null and is readable, then cancel request’s | |
| // body with error. | |
| if (request.body != null && isReadable(request.body?.stream)) { | |
| request.body.stream.cancel(error).catch((err) => { | |
| if (err.code === 'ERR_INVALID_STATE') { | |
| // Node bug? | |
| return | |
| } | |
| throw err | |
| }) | |
| } | |
| // 3. If responseObject is null, then return. | |
| if (responseObject == null) { | |
| return | |
| } | |
| // 4. Let response be responseObject’s response. | |
| const response = responseObject[kState] | |
| // 5. If response’s body is not null and is readable, then error response’s | |
| // body with error. | |
| if (response.body != null && isReadable(response.body?.stream)) { | |
| response.body.stream.cancel(error).catch((err) => { | |
| if (err.code === 'ERR_INVALID_STATE') { | |
| // Node bug? | |
| return | |
| } | |
| throw err | |
| }) | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#fetching | |
| function fetching ({ | |
| request, | |
| processRequestBodyChunkLength, | |
| processRequestEndOfBody, | |
| processResponse, | |
| processResponseEndOfBody, | |
| processResponseConsumeBody, | |
| useParallelQueue = false, | |
| dispatcher // undici | |
| }) { | |
| // 1. Let taskDestination be null. | |
| let taskDestination = null | |
| // 2. Let crossOriginIsolatedCapability be false. | |
| let crossOriginIsolatedCapability = false | |
| // 3. If request’s client is non-null, then: | |
| if (request.client != null) { | |
| // 1. Set taskDestination to request’s client’s global object. | |
| taskDestination = request.client.globalObject | |
| // 2. Set crossOriginIsolatedCapability to request’s client’s cross-origin | |
| // isolated capability. | |
| crossOriginIsolatedCapability = | |
| request.client.crossOriginIsolatedCapability | |
| } | |
| // 4. If useParallelQueue is true, then set taskDestination to the result of | |
| // starting a new parallel queue. | |
| // TODO | |
| // 5. Let timingInfo be a new fetch timing info whose start time and | |
| // post-redirect start time are the coarsened shared current time given | |
| // crossOriginIsolatedCapability. | |
| const currenTime = coarsenedSharedCurrentTime(crossOriginIsolatedCapability) | |
| const timingInfo = createOpaqueTimingInfo({ | |
| startTime: currenTime | |
| }) | |
| // 6. Let fetchParams be a new fetch params whose | |
| // request is request, | |
| // timing info is timingInfo, | |
| // process request body chunk length is processRequestBodyChunkLength, | |
| // process request end-of-body is processRequestEndOfBody, | |
| // process response is processResponse, | |
| // process response consume body is processResponseConsumeBody, | |
| // process response end-of-body is processResponseEndOfBody, | |
| // task destination is taskDestination, | |
| // and cross-origin isolated capability is crossOriginIsolatedCapability. | |
| const fetchParams = { | |
| controller: new Fetch(dispatcher), | |
| request, | |
| timingInfo, | |
| processRequestBodyChunkLength, | |
| processRequestEndOfBody, | |
| processResponse, | |
| processResponseConsumeBody, | |
| processResponseEndOfBody, | |
| taskDestination, | |
| crossOriginIsolatedCapability | |
| } | |
| // 7. If request’s body is a byte sequence, then set request’s body to | |
| // request’s body as a body. | |
| // NOTE: Since fetching is only called from fetch, body should already be | |
| // extracted. | |
| assert(!request.body || request.body.stream) | |
| // 8. If request’s window is "client", then set request’s window to request’s | |
| // client, if request’s client’s global object is a Window object; otherwise | |
| // "no-window". | |
| if (request.window === 'client') { | |
| // TODO: What if request.client is null? | |
| request.window = | |
| request.client?.globalObject?.constructor?.name === 'Window' | |
| ? request.client | |
| : 'no-window' | |
| } | |
| // 9. If request’s origin is "client", then set request’s origin to request’s | |
| // client’s origin. | |
| if (request.origin === 'client') { | |
| // TODO: What if request.client is null? | |
| request.origin = request.client?.origin | |
| } | |
| // 10. If all of the following conditions are true: | |
| // TODO | |
| // 11. If request’s policy container is "client", then: | |
| if (request.policyContainer === 'client') { | |
| // 1. If request’s client is non-null, then set request’s policy | |
| // container to a clone of request’s client’s policy container. [HTML] | |
| if (request.client != null) { | |
| request.policyContainer = clonePolicyContainer( | |
| request.client.policyContainer | |
| ) | |
| } else { | |
| // 2. Otherwise, set request’s policy container to a new policy | |
| // container. | |
| request.policyContainer = makePolicyContainer() | |
| } | |
| } | |
| // 12. If request’s header list does not contain `Accept`, then: | |
| if (!request.headersList.contains('accept')) { | |
| // 1. Let value be `*/*`. | |
| const value = '*/*' | |
| // 2. A user agent should set value to the first matching statement, if | |
| // any, switching on request’s destination: | |
| // "document" | |
| // "frame" | |
| // "iframe" | |
| // `text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8` | |
| // "image" | |
| // `image/png,image/svg+xml,image/*;q=0.8,*/*;q=0.5` | |
| // "style" | |
| // `text/css,*/*;q=0.1` | |
| // TODO | |
| // 3. Append `Accept`/value to request’s header list. | |
| request.headersList.append('accept', value) | |
| } | |
| // 13. If request’s header list does not contain `Accept-Language`, then | |
| // user agents should append `Accept-Language`/an appropriate value to | |
| // request’s header list. | |
| if (!request.headersList.contains('accept-language')) { | |
| request.headersList.append('accept-language', '*') | |
| } | |
| // 14. If request’s priority is null, then use request’s initiator and | |
| // destination appropriately in setting request’s priority to a | |
| // user-agent-defined object. | |
| if (request.priority === null) { | |
| // TODO | |
| } | |
| // 15. If request is a subresource request, then: | |
| if (subresourceSet.has(request.destination)) { | |
| // TODO | |
| } | |
| // 16. Run main fetch given fetchParams. | |
| mainFetch(fetchParams) | |
| .catch(err => { | |
| fetchParams.controller.terminate(err) | |
| }) | |
| // 17. Return fetchParam's controller | |
| return fetchParams.controller | |
| } | |
| // https://fetch.spec.whatwg.org/#concept-main-fetch | |
| async function mainFetch (fetchParams, recursive = false) { | |
| // 1. Let request be fetchParams’s request. | |
| const request = fetchParams.request | |
| // 2. Let response be null. | |
| let response = null | |
| // 3. If request’s local-URLs-only flag is set and request’s current URL is | |
| // not local, then set response to a network error. | |
| if (request.localURLsOnly && !urlIsLocal(requestCurrentURL(request))) { | |
| response = makeNetworkError('local URLs only') | |
| } | |
| // 4. Run report Content Security Policy violations for request. | |
| // TODO | |
| // 5. Upgrade request to a potentially trustworthy URL, if appropriate. | |
| tryUpgradeRequestToAPotentiallyTrustworthyURL(request) | |
| // 6. If should request be blocked due to a bad port, should fetching request | |
| // be blocked as mixed content, or should request be blocked by Content | |
| // Security Policy returns blocked, then set response to a network error. | |
| if (requestBadPort(request) === 'blocked') { | |
| response = makeNetworkError('bad port') | |
| } | |
| // TODO: should fetching request be blocked as mixed content? | |
| // TODO: should request be blocked by Content Security Policy? | |
| // 7. If request’s referrer policy is the empty string, then set request’s | |
| // referrer policy to request’s policy container’s referrer policy. | |
| if (request.referrerPolicy === '') { | |
| request.referrerPolicy = request.policyContainer.referrerPolicy | |
| } | |
| // 8. If request’s referrer is not "no-referrer", then set request’s | |
| // referrer to the result of invoking determine request’s referrer. | |
| if (request.referrer !== 'no-referrer') { | |
| request.referrer = determineRequestsReferrer(request) | |
| } | |
| // 9. Set request’s current URL’s scheme to "https" if all of the following | |
| // conditions are true: | |
| // - request’s current URL’s scheme is "http" | |
| // - request’s current URL’s host is a domain | |
| // - Matching request’s current URL’s host per Known HSTS Host Domain Name | |
| // Matching results in either a superdomain match with an asserted | |
| // includeSubDomains directive or a congruent match (with or without an | |
| // asserted includeSubDomains directive). [HSTS] | |
| // TODO | |
| // 10. If recursive is false, then run the remaining steps in parallel. | |
| // TODO | |
| // 11. If response is null, then set response to the result of running | |
| // the steps corresponding to the first matching statement: | |
| if (response === null) { | |
| response = await (async () => { | |
| const currentURL = requestCurrentURL(request) | |
| if ( | |
| // - request’s current URL’s origin is same origin with request’s origin, | |
| // and request’s response tainting is "basic" | |
| (sameOrigin(currentURL, request.url) && request.responseTainting === 'basic') || | |
| // request’s current URL’s scheme is "data" | |
| (currentURL.protocol === 'data:') || | |
| // - request’s mode is "navigate" or "websocket" | |
| (request.mode === 'navigate' || request.mode === 'websocket') | |
| ) { | |
| // 1. Set request’s response tainting to "basic". | |
| request.responseTainting = 'basic' | |
| // 2. Return the result of running scheme fetch given fetchParams. | |
| return await schemeFetch(fetchParams) | |
| } | |
| // request’s mode is "same-origin" | |
| if (request.mode === 'same-origin') { | |
| // 1. Return a network error. | |
| return makeNetworkError('request mode cannot be "same-origin"') | |
| } | |
| // request’s mode is "no-cors" | |
| if (request.mode === 'no-cors') { | |
| // 1. If request’s redirect mode is not "follow", then return a network | |
| // error. | |
| if (request.redirect !== 'follow') { | |
| return makeNetworkError( | |
| 'redirect mode cannot be "follow" for "no-cors" request' | |
| ) | |
| } | |
| // 2. Set request’s response tainting to "opaque". | |
| request.responseTainting = 'opaque' | |
| // 3. Return the result of running scheme fetch given fetchParams. | |
| return await schemeFetch(fetchParams) | |
| } | |
| // request’s current URL’s scheme is not an HTTP(S) scheme | |
| if (!urlIsHttpHttpsScheme(requestCurrentURL(request))) { | |
| // Return a network error. | |
| return makeNetworkError('URL scheme must be a HTTP(S) scheme') | |
| } | |
| // - request’s use-CORS-preflight flag is set | |
| // - request’s unsafe-request flag is set and either request’s method is | |
| // not a CORS-safelisted method or CORS-unsafe request-header names with | |
| // request’s header list is not empty | |
| // 1. Set request’s response tainting to "cors". | |
| // 2. Let corsWithPreflightResponse be the result of running HTTP fetch | |
| // given fetchParams and true. | |
| // 3. If corsWithPreflightResponse is a network error, then clear cache | |
| // entries using request. | |
| // 4. Return corsWithPreflightResponse. | |
| // TODO | |
| // Otherwise | |
| // 1. Set request’s response tainting to "cors". | |
| request.responseTainting = 'cors' | |
| // 2. Return the result of running HTTP fetch given fetchParams. | |
| return await httpFetch(fetchParams) | |
| })() | |
| } | |
| // 12. If recursive is true, then return response. | |
| if (recursive) { | |
| return response | |
| } | |
| // 13. If response is not a network error and response is not a filtered | |
| // response, then: | |
| if (response.status !== 0 && !response.internalResponse) { | |
| // If request’s response tainting is "cors", then: | |
| if (request.responseTainting === 'cors') { | |
| // 1. Let headerNames be the result of extracting header list values | |
| // given `Access-Control-Expose-Headers` and response’s header list. | |
| // TODO | |
| // 2. If request’s credentials mode is not "include" and headerNames | |
| // contains `*`, then set response’s CORS-exposed header-name list to | |
| // all unique header names in response’s header list. | |
| // TODO | |
| // 3. Otherwise, if headerNames is not null or failure, then set | |
| // response’s CORS-exposed header-name list to headerNames. | |
| // TODO | |
| } | |
| // Set response to the following filtered response with response as its | |
| // internal response, depending on request’s response tainting: | |
| if (request.responseTainting === 'basic') { | |
| response = filterResponse(response, 'basic') | |
| } else if (request.responseTainting === 'cors') { | |
| response = filterResponse(response, 'cors') | |
| } else if (request.responseTainting === 'opaque') { | |
| response = filterResponse(response, 'opaque') | |
| } else { | |
| assert(false) | |
| } | |
| } | |
| // 14. Let internalResponse be response, if response is a network error, | |
| // and response’s internal response otherwise. | |
| let internalResponse = | |
| response.status === 0 ? response : response.internalResponse | |
| // 15. If internalResponse’s URL list is empty, then set it to a clone of | |
| // request’s URL list. | |
| if (internalResponse.urlList.length === 0) { | |
| internalResponse.urlList.push(...request.urlList) | |
| } | |
| // 16. If request’s timing allow failed flag is unset, then set | |
| // internalResponse’s timing allow passed flag. | |
| if (!request.timingAllowFailed) { | |
| response.timingAllowPassed = true | |
| } | |
| // 17. If response is not a network error and any of the following returns | |
| // blocked | |
| // - should internalResponse to request be blocked as mixed content | |
| // - should internalResponse to request be blocked by Content Security Policy | |
| // - should internalResponse to request be blocked due to its MIME type | |
| // - should internalResponse to request be blocked due to nosniff | |
| // TODO | |
| // 18. If response’s type is "opaque", internalResponse’s status is 206, | |
| // internalResponse’s range-requested flag is set, and request’s header | |
| // list does not contain `Range`, then set response and internalResponse | |
| // to a network error. | |
| if ( | |
| response.type === 'opaque' && | |
| internalResponse.status === 206 && | |
| internalResponse.rangeRequested && | |
| !request.headers.contains('range') | |
| ) { | |
| response = internalResponse = makeNetworkError() | |
| } | |
| // 19. If response is not a network error and either request’s method is | |
| // `HEAD` or `CONNECT`, or internalResponse’s status is a null body status, | |
| // set internalResponse’s body to null and disregard any enqueuing toward | |
| // it (if any). | |
| if ( | |
| response.status !== 0 && | |
| (request.method === 'HEAD' || | |
| request.method === 'CONNECT' || | |
| nullBodyStatus.includes(internalResponse.status)) | |
| ) { | |
| internalResponse.body = null | |
| fetchParams.controller.dump = true | |
| } | |
| // 20. If request’s integrity metadata is not the empty string, then: | |
| if (request.integrity) { | |
| // 1. Let processBodyError be this step: run fetch finale given fetchParams | |
| // and a network error. | |
| const processBodyError = (reason) => | |
| fetchFinale(fetchParams, makeNetworkError(reason)) | |
| // 2. If request’s response tainting is "opaque", or response’s body is null, | |
| // then run processBodyError and abort these steps. | |
| if (request.responseTainting === 'opaque' || response.body == null) { | |
| processBodyError(response.error) | |
| return | |
| } | |
| // 3. Let processBody given bytes be these steps: | |
| const processBody = (bytes) => { | |
| // 1. If bytes do not match request’s integrity metadata, | |
| // then run processBodyError and abort these steps. [SRI] | |
| if (!bytesMatch(bytes, request.integrity)) { | |
| processBodyError('integrity mismatch') | |
| return | |
| } | |
| // 2. Set response’s body to bytes as a body. | |
| response.body = safelyExtractBody(bytes)[0] | |
| // 3. Run fetch finale given fetchParams and response. | |
| fetchFinale(fetchParams, response) | |
| } | |
| // 4. Fully read response’s body given processBody and processBodyError. | |
| await fullyReadBody(response.body, processBody, processBodyError) | |
| } else { | |
| // 21. Otherwise, run fetch finale given fetchParams and response. | |
| fetchFinale(fetchParams, response) | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#concept-scheme-fetch | |
| // given a fetch params fetchParams | |
| function schemeFetch (fetchParams) { | |
| // Note: since the connection is destroyed on redirect, which sets fetchParams to a | |
| // cancelled state, we do not want this condition to trigger *unless* there have been | |
| // no redirects. See https://github.com/nodejs/undici/issues/1776 | |
| // 1. If fetchParams is canceled, then return the appropriate network error for fetchParams. | |
| if (isCancelled(fetchParams) && fetchParams.request.redirectCount === 0) { | |
| return Promise.resolve(makeAppropriateNetworkError(fetchParams)) | |
| } | |
| // 2. Let request be fetchParams’s request. | |
| const { request } = fetchParams | |
| const { protocol: scheme } = requestCurrentURL(request) | |
| // 3. Switch on request’s current URL’s scheme and run the associated steps: | |
| switch (scheme) { | |
| case 'about:': { | |
| // If request’s current URL’s path is the string "blank", then return a new response | |
| // whose status message is `OK`, header list is « (`Content-Type`, `text/html;charset=utf-8`) », | |
| // and body is the empty byte sequence as a body. | |
| // Otherwise, return a network error. | |
| return Promise.resolve(makeNetworkError('about scheme is not supported')) | |
| } | |
| case 'blob:': { | |
| if (!resolveObjectURL) { | |
| resolveObjectURL = require('buffer').resolveObjectURL | |
| } | |
| // 1. Let blobURLEntry be request’s current URL’s blob URL entry. | |
| const blobURLEntry = requestCurrentURL(request) | |
| // https://github.com/web-platform-tests/wpt/blob/7b0ebaccc62b566a1965396e5be7bb2bc06f841f/FileAPI/url/resources/fetch-tests.js#L52-L56 | |
| // Buffer.resolveObjectURL does not ignore URL queries. | |
| if (blobURLEntry.search.length !== 0) { | |
| return Promise.resolve(makeNetworkError('NetworkError when attempting to fetch resource.')) | |
| } | |
| const blobURLEntryObject = resolveObjectURL(blobURLEntry.toString()) | |
| // 2. If request’s method is not `GET`, blobURLEntry is null, or blobURLEntry’s | |
| // object is not a Blob object, then return a network error. | |
| if (request.method !== 'GET' || !isBlobLike(blobURLEntryObject)) { | |
| return Promise.resolve(makeNetworkError('invalid method')) | |
| } | |
| // 3. Let bodyWithType be the result of safely extracting blobURLEntry’s object. | |
| const bodyWithType = safelyExtractBody(blobURLEntryObject) | |
| // 4. Let body be bodyWithType’s body. | |
| const body = bodyWithType[0] | |
| // 5. Let length be body’s length, serialized and isomorphic encoded. | |
| const length = isomorphicEncode(`${body.length}`) | |
| // 6. Let type be bodyWithType’s type if it is non-null; otherwise the empty byte sequence. | |
| const type = bodyWithType[1] ?? '' | |
| // 7. Return a new response whose status message is `OK`, header list is | |
| // « (`Content-Length`, length), (`Content-Type`, type) », and body is body. | |
| const response = makeResponse({ | |
| statusText: 'OK', | |
| headersList: [ | |
| ['content-length', { name: 'Content-Length', value: length }], | |
| ['content-type', { name: 'Content-Type', value: type }] | |
| ] | |
| }) | |
| response.body = body | |
| return Promise.resolve(response) | |
| } | |
| case 'data:': { | |
| // 1. Let dataURLStruct be the result of running the | |
| // data: URL processor on request’s current URL. | |
| const currentURL = requestCurrentURL(request) | |
| const dataURLStruct = dataURLProcessor(currentURL) | |
| // 2. If dataURLStruct is failure, then return a | |
| // network error. | |
| if (dataURLStruct === 'failure') { | |
| return Promise.resolve(makeNetworkError('failed to fetch the data URL')) | |
| } | |
| // 3. Let mimeType be dataURLStruct’s MIME type, serialized. | |
| const mimeType = serializeAMimeType(dataURLStruct.mimeType) | |
| // 4. Return a response whose status message is `OK`, | |
| // header list is « (`Content-Type`, mimeType) », | |
| // and body is dataURLStruct’s body as a body. | |
| return Promise.resolve(makeResponse({ | |
| statusText: 'OK', | |
| headersList: [ | |
| ['content-type', { name: 'Content-Type', value: mimeType }] | |
| ], | |
| body: safelyExtractBody(dataURLStruct.body)[0] | |
| })) | |
| } | |
| case 'file:': { | |
| // For now, unfortunate as it is, file URLs are left as an exercise for the reader. | |
| // When in doubt, return a network error. | |
| return Promise.resolve(makeNetworkError('not implemented... yet...')) | |
| } | |
| case 'http:': | |
| case 'https:': { | |
| // Return the result of running HTTP fetch given fetchParams. | |
| return httpFetch(fetchParams) | |
| .catch((err) => makeNetworkError(err)) | |
| } | |
| default: { | |
| return Promise.resolve(makeNetworkError('unknown scheme')) | |
| } | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#finalize-response | |
| function finalizeResponse (fetchParams, response) { | |
| // 1. Set fetchParams’s request’s done flag. | |
| fetchParams.request.done = true | |
| // 2, If fetchParams’s process response done is not null, then queue a fetch | |
| // task to run fetchParams’s process response done given response, with | |
| // fetchParams’s task destination. | |
| if (fetchParams.processResponseDone != null) { | |
| queueMicrotask(() => fetchParams.processResponseDone(response)) | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#fetch-finale | |
| function fetchFinale (fetchParams, response) { | |
| // 1. If response is a network error, then: | |
| if (response.type === 'error') { | |
| // 1. Set response’s URL list to « fetchParams’s request’s URL list[0] ». | |
| response.urlList = [fetchParams.request.urlList[0]] | |
| // 2. Set response’s timing info to the result of creating an opaque timing | |
| // info for fetchParams’s timing info. | |
| response.timingInfo = createOpaqueTimingInfo({ | |
| startTime: fetchParams.timingInfo.startTime | |
| }) | |
| } | |
| // 2. Let processResponseEndOfBody be the following steps: | |
| const processResponseEndOfBody = () => { | |
| // 1. Set fetchParams’s request’s done flag. | |
| fetchParams.request.done = true | |
| // If fetchParams’s process response end-of-body is not null, | |
| // then queue a fetch task to run fetchParams’s process response | |
| // end-of-body given response with fetchParams’s task destination. | |
| if (fetchParams.processResponseEndOfBody != null) { | |
| queueMicrotask(() => fetchParams.processResponseEndOfBody(response)) | |
| } | |
| } | |
| // 3. If fetchParams’s process response is non-null, then queue a fetch task | |
| // to run fetchParams’s process response given response, with fetchParams’s | |
| // task destination. | |
| if (fetchParams.processResponse != null) { | |
| queueMicrotask(() => fetchParams.processResponse(response)) | |
| } | |
| // 4. If response’s body is null, then run processResponseEndOfBody. | |
| if (response.body == null) { | |
| processResponseEndOfBody() | |
| } else { | |
| // 5. Otherwise: | |
| // 1. Let transformStream be a new a TransformStream. | |
| // 2. Let identityTransformAlgorithm be an algorithm which, given chunk, | |
| // enqueues chunk in transformStream. | |
| const identityTransformAlgorithm = (chunk, controller) => { | |
| controller.enqueue(chunk) | |
| } | |
| // 3. Set up transformStream with transformAlgorithm set to identityTransformAlgorithm | |
| // and flushAlgorithm set to processResponseEndOfBody. | |
| const transformStream = new TransformStream({ | |
| start () {}, | |
| transform: identityTransformAlgorithm, | |
| flush: processResponseEndOfBody | |
| }, { | |
| size () { | |
| return 1 | |
| } | |
| }, { | |
| size () { | |
| return 1 | |
| } | |
| }) | |
| // 4. Set response’s body to the result of piping response’s body through transformStream. | |
| response.body = { stream: response.body.stream.pipeThrough(transformStream) } | |
| } | |
| // 6. If fetchParams’s process response consume body is non-null, then: | |
| if (fetchParams.processResponseConsumeBody != null) { | |
| // 1. Let processBody given nullOrBytes be this step: run fetchParams’s | |
| // process response consume body given response and nullOrBytes. | |
| const processBody = (nullOrBytes) => fetchParams.processResponseConsumeBody(response, nullOrBytes) | |
| // 2. Let processBodyError be this step: run fetchParams’s process | |
| // response consume body given response and failure. | |
| const processBodyError = (failure) => fetchParams.processResponseConsumeBody(response, failure) | |
| // 3. If response’s body is null, then queue a fetch task to run processBody | |
| // given null, with fetchParams’s task destination. | |
| if (response.body == null) { | |
| queueMicrotask(() => processBody(null)) | |
| } else { | |
| // 4. Otherwise, fully read response’s body given processBody, processBodyError, | |
| // and fetchParams’s task destination. | |
| return fullyReadBody(response.body, processBody, processBodyError) | |
| } | |
| return Promise.resolve() | |
| } | |
| } | |
| // https://fetch.spec.whatwg.org/#http-fetch | |
| async function httpFetch (fetchParams) { | |
| // 1. Let request be fetchParams’s request. | |
| const request = fetchParams.request | |
| // 2. Let response be null. | |
| let response = null | |
| // 3. Let actualResponse be null. | |
| let actualResponse = null | |
| // 4. Let timingInfo be fetchParams’s timing info. | |
| const timingInfo = fetchParams.timingInfo | |
| // 5. If request’s service-workers mode is "all", then: | |
| if (request.serviceWorkers === 'all') { | |
| // TODO | |
| } | |
| // 6. If response is null, then: | |
| if (response === null) { | |
| // 1. If makeCORSPreflight is true and one of these conditions is true: | |
| // TODO | |
| // 2. If request’s redirect mode is "follow", then set request’s | |
| // service-workers mode to "none". | |
| if (request.redirect === 'follow') { | |
| request.serviceWorkers = 'none' | |
| } | |
| // 3. Set response and actualResponse to the result of running | |
| // HTTP-network-or-cache fetch given fetchParams. | |
| actualResponse = response = await httpNetworkOrCacheFetch(fetchParams) | |
| // 4. If request’s response tainting is "cors" and a CORS check | |
| // for request and response returns failure, then return a network error. | |
| if ( | |
| request.responseTainting === 'cors' && | |
| corsCheck(request, response) === 'failure' | |
| ) { | |
| return makeNetworkError('cors failure') | |
| } | |
| // 5. If the TAO check for request and response returns failure, then set | |
| // request’s timing allow failed flag. | |
| if (TAOCheck(request, response) === 'failure') { | |
| request.timingAllowFailed = true | |
| } | |
| } | |
| // 7. If either request’s response tainting or response’s type | |
| // is "opaque", and the cross-origin resource policy check with | |
| // request’s origin, request’s client, request’s destination, | |
| // and actualResponse returns blocked, then return a network error. | |
| if ( | |
| (request.responseTainting === 'opaque' || response.type === 'opaque') && | |
| crossOriginResourcePolicyCheck( | |
| request.origin, | |
| request.client, | |
| request.destination, | |
| actualResponse | |
| ) === 'blocked' | |
| ) { | |
| return makeNetworkError('blocked') | |
| } | |
| // 8. If actualResponse’s status is a redirect status, then: | |
| if (redirectStatusSet.has(actualResponse.status)) { | |
| // 1. If actualResponse’s status is not 303, request’s body is not null, | |
| // and the connection uses HTTP/2, then user agents may, and are even | |
| // encouraged to, transmit an RST_STREAM frame. | |
| // See, https://github.com/whatwg/fetch/issues/1288 | |
| if (request.redirect !== 'manual') { | |
| fetchParams.controller.connection.destroy() | |
| } | |
| // 2. Switch on request’s redirect mode: | |
| if (request.redirect === 'error') { | |
| // Set response to a network error. | |
| response = makeNetworkError('unexpected redirect') | |
| } else if (request.redirect === 'manual') { | |
| // Set response to an opaque-redirect filtered response whose internal | |
| // response is actualResponse. | |
| // NOTE(spec): On the web this would return an `opaqueredirect` response, | |
| // but that doesn't make sense server side. | |
| // See https://github.com/nodejs/undici/issues/1193. | |
| response = actualResponse | |
| } else if (request.redirect === 'follow') { | |
| // Set response to the result of running HTTP-redirect fetch given | |
| // fetchParams and response. | |
| response = await httpRedirectFetch(fetchParams, response) | |
| } else { | |
| assert(false) | |
| } | |
| } | |
| // 9. Set response’s timing info to timingInfo. | |
| response.timingInfo = timingInfo | |
| // 10. Return response. | |
| return response | |
| } | |
| // https://fetch.spec.whatwg.org/#http-redirect-fetch | |
| function httpRedirectFetch (fetchParams, response) { | |
| // 1. Let request be fetchParams’s request. | |
| const request = fetchParams.request | |
| // 2. Let actualResponse be response, if response is not a filtered response, | |
| // and response’s internal response otherwise. | |
| const actualResponse = response.internalResponse | |
| ? response.internalResponse | |
| : response | |
| // 3. Let locationURL be actualResponse’s location URL given request’s current | |
| // URL’s fragment. | |
| let locationURL | |
| try { | |
| locationURL = responseLocationURL( | |
| actualResponse, | |
| requestCurrentURL(request).hash | |
| ) | |
| // 4. If locationURL is null, then return response. | |
| if (locationURL == null) { | |
| return response | |
| } | |
| } catch (err) { | |
| // 5. If locationURL is failure, then return a network error. | |
| return Promise.resolve(makeNetworkError(err)) | |
| } | |
| // 6. If locationURL’s scheme is not an HTTP(S) scheme, then return a network | |
| // error. | |
| if (!urlIsHttpHttpsScheme(locationURL)) { | |
| return Promise.resolve(makeNetworkError('URL scheme must be a HTTP(S) scheme')) | |
| } | |
| // 7. If request’s redirect count is 20, then return a network error. | |
| if (request.redirectCount === 20) { | |
| return Promise.resolve(makeNetworkError('redirect count exceeded')) | |
| } | |
| // 8. Increase request’s redirect count by 1. | |
| request.redirectCount += 1 | |
| // 9. If request’s mode is "cors", locationURL includes credentials, and | |
| // request’s origin is not same origin with locationURL’s origin, then return | |
| // a network error. | |
| if ( | |
| request.mode === 'cors' && | |
| (locationURL.username || locationURL.password) && | |
| !sameOrigin(request, locationURL) | |
| ) { | |
| return Promise.resolve(makeNetworkError('cross origin not allowed for request mode "cors"')) | |
| } | |
| // 10. If request’s response tainting is "cors" and locationURL includes | |
| // credentials, then return a network error. | |
| if ( | |
| request.responseTainting === 'cors' && | |
| (locationURL.username || locationURL.password) | |
| ) { | |
| return Promise.resolve(makeNetworkError( | |
| 'URL cannot contain credentials for request mode "cors"' | |
| )) | |
| } | |
| // 11. If actualResponse’s status is not 303, request’s body is non-null, | |
| // and request’s body’s source is null, then return a network error. | |
| if ( | |
| actualResponse.status !== 303 && | |
| request.body != null && | |
| request.body.source == null | |
| ) { | |
| return Promise.resolve(makeNetworkError()) | |
| } | |
| // 12. If one of the following is true | |
| // - actualResponse’s status is 301 or 302 and request’s method is `POST` | |
| // - actualResponse’s status is 303 and request’s method is not `GET` or `HEAD` | |
| if ( | |
| ([301, 302].includes(actualResponse.status) && request.method === 'POST') || | |
| (actualResponse.status === 303 && | |
| !GET_OR_HEAD.includes(request.method)) | |
| ) { | |
| // then: | |
| // 1. Set request’s method to `GET` and request’s body to null. | |
| request.method = 'GET' | |
| request.body = null | |
| // 2. For each headerName of request-body-header name, delete headerName from | |
| // request’s header list. | |
| for (const headerName of requestBodyHeader) { | |
| request.headersList.delete(headerName) | |
| } | |
| } | |
| // 13. If request’s current URL’s origin is not same origin with locationURL’s | |
| // origin, then for each headerName of CORS non-wildcard request-header name, | |
| // delete headerName from request’s header list. | |
| if (!sameOrigin(requestCurrentURL(request), locationURL)) { | |
| // https://fetch.spec.whatwg.org/#cors-non-wildcard-request-header-name | |
| request.headersList.delete('authorization') | |
| // "Cookie" and "Host" are forbidden request-headers, which undici doesn't implement. | |
| request.headersList.delete('cookie') | |
| request.headersList.delete('host') | |
| } | |
| // 14. If request’s body is non-null, then set request’s body to the first return | |
| // value of safely extracting request’s body’s source. | |
| if (request.body != null) { | |
| assert(request.body.source != null) | |
| request.body = safelyExtractBody(request.body.source)[0] | |
| } | |
| // 15. Let timingInfo be fetchParams’s timing info. | |
| const timingInfo = fetchParams.timingInfo | |
| // 16. Set timingInfo’s redirect end time and post-redirect start time to the | |
| // coarsened shared current time given fetchParams’s cross-origin isolated | |
| // capability. | |
| timingInfo.redirectEndTime = timingInfo.postRedirectStartTime = | |
| coarsenedSharedCurrentTime(fetchParams.crossOriginIsolatedCapability) | |
| // 17. If timingInfo’s redirect start time is 0, then set timingInfo’s | |
| // redirect start time to timingInfo’s start time. | |
| if (timingInfo.redirectStartTime === 0) { | |
| timingInfo.redirectStartTime = timingInfo.startTime | |
| } | |
| // 18. Append locationURL to request’s URL list. | |
| request.urlList.push(locationURL) | |
| // 19. Invoke set request’s referrer policy on redirect on request and | |
| // actualResponse. | |
| setRequestReferrerPolicyOnRedirect(request, actualResponse) | |
| // 20. Return the result of running main fetch given fetchParams and true. | |
| return mainFetch(fetchParams, true) | |
| } | |
| // https://fetch.spec.whatwg.org/#http-network-or-cache-fetch | |
| async function httpNetworkOrCacheFetch ( | |
| fetchParams, | |
| isAuthenticationFetch = false, | |
| isNewConnectionFetch = false | |
| ) { | |
| // 1. Let request be fetchParams’s request. | |
| const request = fetchParams.request | |
| // 2. Let httpFetchParams be null. | |
| let httpFetchParams = null | |
| // 3. Let httpRequest be null. | |
| let httpRequest = null | |
| // 4. Let response be null. | |
| let response = null | |
| // 5. Let storedResponse be null. | |
| // TODO: cache | |
| // 6. Let httpCache be null. | |
| const httpCache = null | |
| // 7. Let the revalidatingFlag be unset. | |
| const revalidatingFlag = false | |
| // 8. Run these steps, but abort when the ongoing fetch is terminated: | |
| // 1. If request’s window is "no-window" and request’s redirect mode is | |
| // "error", then set httpFetchParams to fetchParams and httpRequest to | |
| // request. | |
| if (request.window === 'no-window' && request.redirect === 'error') { | |
| httpFetchParams = fetchParams | |
| httpRequest = request | |
| } else { | |
| // Otherwise: | |
| // 1. Set httpRequest to a clone of request. | |
| httpRequest = makeRequest(request) | |
| // 2. Set httpFetchParams to a copy of fetchParams. | |
| httpFetchParams = { ...fetchParams } | |
| // 3. Set httpFetchParams’s request to httpRequest. | |
| httpFetchParams.request = httpRequest | |
| } | |
| // 3. Let includeCredentials be true if one of | |
| const includeCredentials = | |
| request.credentials === 'include' || | |
| (request.credentials === 'same-origin' && | |
| request.responseTainting === 'basic') | |
| // 4. Let contentLength be httpRequest’s body’s length, if httpRequest’s | |
| // body is non-null; otherwise null. | |
| const contentLength = httpRequest.body ? httpRequest.body.length : null | |
| // 5. Let contentLengthHeaderValue be null. | |
| let contentLengthHeaderValue = null | |
| // 6. If httpRequest’s body is null and httpRequest’s method is `POST` or | |
| // `PUT`, then set contentLengthHeaderValue to `0`. | |
| if ( | |
| httpRequest.body == null && | |
| ['POST', 'PUT'].includes(httpRequest.method) | |
| ) { | |
| contentLengthHeaderValue = '0' | |
| } | |
| // 7. If contentLength is non-null, then set contentLengthHeaderValue to | |
| // contentLength, serialized and isomorphic encoded. | |
| if (contentLength != null) { | |
| contentLengthHeaderValue = isomorphicEncode(`${contentLength}`) | |
| } | |
| // 8. If contentLengthHeaderValue is non-null, then append | |
| // `Content-Length`/contentLengthHeaderValue to httpRequest’s header | |
| // list. | |
| if (contentLengthHeaderValue != null) { | |
| httpRequest.headersList.append('content-length', contentLengthHeaderValue) | |
| } | |
| // 9. If contentLengthHeaderValue is non-null, then append (`Content-Length`, | |
| // contentLengthHeaderValue) to httpRequest’s header list. | |
| // 10. If contentLength is non-null and httpRequest’s keepalive is true, | |
| // then: | |
| if (contentLength != null && httpRequest.keepalive) { | |
| // NOTE: keepalive is a noop outside of browser context. | |
| } | |
| // 11. If httpRequest’s referrer is a URL, then append | |
| // `Referer`/httpRequest’s referrer, serialized and isomorphic encoded, | |
| // to httpRequest’s header list. | |
| if (httpRequest.referrer instanceof URL) { | |
| httpRequest.headersList.append('referer', isomorphicEncode(httpRequest.referrer.href)) | |
| } | |
| // 12. Append a request `Origin` header for httpRequest. | |
| appendRequestOriginHeader(httpRequest) | |
| // 13. Append the Fetch metadata headers for httpRequest. [FETCH-METADATA] | |
| appendFetchMetadata(httpRequest) | |
| // 14. If httpRequest’s header list does not contain `User-Agent`, then | |
| // user agents should append `User-Agent`/default `User-Agent` value to | |
| // httpRequest’s header list. | |
| if (!httpRequest.headersList.contains('user-agent')) { | |
| httpRequest.headersList.append('user-agent', typeof esbuildDetection === 'undefined' ? 'undici' : 'node') | |
| } | |
| // 15. If httpRequest’s cache mode is "default" and httpRequest’s header | |
| // list contains `If-Modified-Since`, `If-None-Match`, | |
| // `If-Unmodified-Since`, `If-Match`, or `If-Range`, then set | |
| // httpRequest’s cache mode to "no-store". | |
| if ( | |
| httpRequest.cache === 'default' && | |
| (httpRequest.headersList.contains('if-modified-since') || | |
| httpRequest.headersList.contains('if-none-match') || | |
| httpRequest.headersList.contains('if-unmodified-since') || | |
| httpRequest.headersList.contains('if-match') || | |
| httpRequest.headersList.contains('if-range')) | |
| ) { | |
| httpRequest.cache = 'no-store' | |
| } | |
| // 16. If httpRequest’s cache mode is "no-cache", httpRequest’s prevent | |
| // no-cache cache-control header modification flag is unset, and | |
| // httpRequest’s header list does not contain `Cache-Control`, then append | |
| // `Cache-Control`/`max-age=0` to httpRequest’s header list. | |
| if ( | |
| httpRequest.cache === 'no-cache' && | |
| !httpRequest.preventNoCacheCacheControlHeaderModification && | |
| !httpRequest.headersList.contains('cache-control') | |
| ) { | |
| httpRequest.headersList.append('cache-control', 'max-age=0') | |
| } | |
| // 17. If httpRequest’s cache mode is "no-store" or "reload", then: | |
| if (httpRequest.cache === 'no-store' || httpRequest.cache === 'reload') { | |
| // 1. If httpRequest’s header list does not contain `Pragma`, then append | |
| // `Pragma`/`no-cache` to httpRequest’s header list. | |
| if (!httpRequest.headersList.contains('pragma')) { | |
| httpRequest.headersList.append('pragma', 'no-cache') | |
| } | |
| // 2. If httpRequest’s header list does not contain `Cache-Control`, | |
| // then append `Cache-Control`/`no-cache` to httpRequest’s header list. | |
| if (!httpRequest.headersList.contains('cache-control')) { | |
| httpRequest.headersList.append('cache-control', 'no-cache') | |
| } | |
| } | |
| // 18. If httpRequest’s header list contains `Range`, then append | |
| // `Accept-Encoding`/`identity` to httpRequest’s header list. | |
| if (httpRequest.headersList.contains('range')) { | |
| httpRequest.headersList.append('accept-encoding', 'identity') | |
| } | |
| // 19. Modify httpRequest’s header list per HTTP. Do not append a given | |
| // header if httpRequest’s header list contains that header’s name. | |
| // TODO: https://github.com/whatwg/fetch/issues/1285#issuecomment-896560129 | |
| if (!httpRequest.headersList.contains('accept-encoding')) { | |
| if (urlHasHttpsScheme(requestCurrentURL(httpRequest))) { | |
| httpRequest.headersList.append('accept-encoding', 'br, gzip, deflate') | |
| } else { | |
| httpRequest.headersList.append('accept-encoding', 'gzip, deflate') | |
| } | |
| } | |
| httpRequest.headersList.delete('host') | |
| // 20. If includeCredentials is true, then: | |
| if (includeCredentials) { | |
| // 1. If the user agent is not configured to block cookies for httpRequest | |
| // (see section 7 of [COOKIES]), then: | |
| // TODO: credentials | |
| // 2. If httpRequest’s header list does not contain `Authorization`, then: | |
| // TODO: credentials | |
| } | |
| // 21. If there’s a proxy-authentication entry, use it as appropriate. | |
| // TODO: proxy-authentication | |
| // 22. Set httpCache to the result of determining the HTTP cache | |
| // partition, given httpRequest. | |
| // TODO: cache | |
| // 23. If httpCache is null, then set httpRequest’s cache mode to | |
| // "no-store". | |
| if (httpCache == null) { | |
| httpRequest.cache = 'no-store' | |
| } | |
| // 24. If httpRequest’s cache mode is neither "no-store" nor "reload", | |
| // then: | |
| if (httpRequest.mode !== 'no-store' && httpRequest.mode !== 'reload') { | |
| // TODO: cache | |
| } | |
| // 9. If aborted, then return the appropriate network error for fetchParams. | |
| // TODO | |
| // 10. If response is null, then: | |
| if (response == null) { | |
| // 1. If httpRequest’s cache mode is "only-if-cached", then return a | |
| // network error. | |
| if (httpRequest.mode === 'only-if-cached') { | |
| return makeNetworkError('only if cached') | |
| } | |
| // 2. Let forwardResponse be the result of running HTTP-network fetch | |
| // given httpFetchParams, includeCredentials, and isNewConnectionFetch. | |
| const forwardResponse = await httpNetworkFetch( | |
| httpFetchParams, | |
| includeCredentials, | |
| isNewConnectionFetch | |
| ) | |
| // 3. If httpRequest’s method is unsafe and forwardResponse’s status is | |
| // in the range 200 to 399, inclusive, invalidate appropriate stored | |
| // responses in httpCache, as per the "Invalidation" chapter of HTTP | |
| // Caching, and set storedResponse to null. [HTTP-CACHING] | |
| if ( | |
| !safeMethodsSet.has(httpRequest.method) && | |
| forwardResponse.status >= 200 && | |
| forwardResponse.status <= 399 | |
| ) { | |
| // TODO: cache | |
| } | |
| // 4. If the revalidatingFlag is set and forwardResponse’s status is 304, | |
| // then: | |
| if (revalidatingFlag && forwardResponse.status === 304) { | |
| // TODO: cache | |
| } | |
| // 5. If response is null, then: | |
| if (response == null) { | |
| // 1. Set response to forwardResponse. | |
| response = forwardResponse | |
| // 2. Store httpRequest and forwardResponse in httpCache, as per the | |
| // "Storing Responses in Caches" chapter of HTTP Caching. [HTTP-CACHING] | |
| // TODO: cache | |
| } | |
| } | |
| // 11. Set response’s URL list to a clone of httpRequest’s URL list. | |
| response.urlList = [...httpRequest.urlList] | |
| // 12. If httpRequest’s header list contains `Range`, then set response’s | |
| // range-requested flag. | |
| if (httpRequest.headersList.contains('range')) { | |
| response.rangeRequested = true | |
| } | |
| // 13. Set response’s request-includes-credentials to includeCredentials. | |
| response.requestIncludesCredentials = includeCredentials | |
| // 14. If response’s status is 401, httpRequest’s response tainting is not | |
| // "cors", includeCredentials is true, and request’s window is an environment | |
| // settings object, then: | |
| // TODO | |
| // 15. If response’s status is 407, then: | |
| if (response.status === 407) { | |
| // 1. If request’s window is "no-window", then return a network error. | |
| if (request.window === 'no-window') { | |
| return makeNetworkError() | |
| } | |
| // 2. ??? | |
| // 3. If fetchParams is canceled, then return the appropriate network error for fetchParams. | |
| if (isCancelled(fetchParams)) { | |
| return makeAppropriateNetworkError(fetchParams) | |
| } | |
| // 4. Prompt the end user as appropriate in request’s window and store | |
| // the result as a proxy-authentication entry. [HTTP-AUTH] | |
| // TODO: Invoke some kind of callback? | |
| // 5. Set response to the result of running HTTP-network-or-cache fetch given | |
| // fetchParams. | |
| // TODO | |
| return makeNetworkError('proxy authentication required') | |
| } | |
| // 16. If all of the following are true | |
| if ( | |
| // response’s status is 421 | |
| response.status === 421 && | |
| // isNewConnectionFetch is false | |
| !isNewConnectionFetch && | |
| // request’s body is null, or request’s body is non-null and request’s body’s source is non-null | |
| (request.body == null || request.body.source != null) | |
| ) { | |
| // then: | |
| // 1. If fetchParams is canceled, then return the appropriate network error for fetchParams. | |
| if (isCancelled(fetchParams)) { | |
| return makeAppropriateNetworkError(fetchParams) | |
| } | |
| // 2. Set response to the result of running HTTP-network-or-cache | |
| // fetch given fetchParams, isAuthenticationFetch, and true. | |
| // TODO (spec): The spec doesn't specify this but we need to cancel | |
| // the active response before we can start a new one. | |
| // https://github.com/whatwg/fetch/issues/1293 | |
| fetchParams.controller.connection.destroy() | |
| response = await httpNetworkOrCacheFetch( | |
| fetchParams, | |
| isAuthenticationFetch, | |
| true | |
| ) | |
| } | |
| // 17. If isAuthenticationFetch is true, then create an authentication entry | |
| if (isAuthenticationFetch) { | |
| // TODO | |
| } | |
| // 18. Return response. | |
| return response | |
| } | |
| // https://fetch.spec.whatwg.org/#http-network-fetch | |
| async function httpNetworkFetch ( | |
| fetchParams, | |
| includeCredentials = false, | |
| forceNewConnection = false | |
| ) { | |
| assert(!fetchParams.controller.connection || fetchParams.controller.connection.destroyed) | |
| fetchParams.controller.connection = { | |
| abort: null, | |
| destroyed: false, | |
| destroy (err) { | |
| if (!this.destroyed) { | |
| this.destroyed = true | |
| this.abort?.(err ?? new DOMException('The operation was aborted.', 'AbortError')) | |
| } | |
| } | |
| } | |
| // 1. Let request be fetchParams’s request. | |
| const request = fetchParams.request | |
| // 2. Let response be null. | |
| let response = null | |
| // 3. Let timingInfo be fetchParams’s timing info. | |
| const timingInfo = fetchParams.timingInfo | |
| // 4. Let httpCache be the result of determining the HTTP cache partition, | |
| // given request. | |
| // TODO: cache | |
| const httpCache = null | |
| // 5. If httpCache is null, then set request’s cache mode to "no-store". | |
| if (httpCache == null) { | |
| request.cache = 'no-store' | |
| } | |
| // 6. Let networkPartitionKey be the result of determining the network | |
| // partition key given request. | |
| // TODO | |
| // 7. Let newConnection be "yes" if forceNewConnection is true; otherwise | |
| // "no". | |
| const newConnection = forceNewConnection ? 'yes' : 'no' // eslint-disable-line no-unused-vars | |
| // 8. Switch on request’s mode: | |
| if (request.mode === 'websocket') { | |
| // Let connection be the result of obtaining a WebSocket connection, | |
| // given request’s current URL. | |
| // TODO | |
| } else { | |
| // Let connection be the result of obtaining a connection, given | |
| // networkPartitionKey, request’s current URL’s origin, | |
| // includeCredentials, and forceNewConnection. | |
| // TODO | |
| } | |
| // 9. Run these steps, but abort when the ongoing fetch is terminated: | |
| // 1. If connection is failure, then return a network error. | |
| // 2. Set timingInfo’s final connection timing info to the result of | |
| // calling clamp and coarsen connection timing info with connection’s | |
| // timing info, timingInfo’s post-redirect start time, and fetchParams’s | |
| // cross-origin isolated capability. | |
| // 3. If connection is not an HTTP/2 connection, request’s body is non-null, | |
| // and request’s body’s source is null, then append (`Transfer-Encoding`, | |
| // `chunked`) to request’s header list. | |
| // 4. Set timingInfo’s final network-request start time to the coarsened | |
| // shared current time given fetchParams’s cross-origin isolated | |
| // capability. | |
| // 5. Set response to the result of making an HTTP request over connection | |
| // using request with the following caveats: | |
| // - Follow the relevant requirements from HTTP. [HTTP] [HTTP-SEMANTICS] | |
| // [HTTP-COND] [HTTP-CACHING] [HTTP-AUTH] | |
| // - If request’s body is non-null, and request’s body’s source is null, | |
| // then the user agent may have a buffer of up to 64 kibibytes and store | |
| // a part of request’s body in that buffer. If the user agent reads from | |
| // request’s body beyond that buffer’s size and the user agent needs to | |
| // resend request, then instead return a network error. | |
| // - Set timingInfo’s final network-response start time to the coarsened | |
| // shared current time given fetchParams’s cross-origin isolated capability, | |
| // immediately after the user agent’s HTTP parser receives the first byte | |
| // of the response (e.g., frame header bytes for HTTP/2 or response status | |
| // line for HTTP/1.x). | |
| // - Wait until all the headers are transmitted. | |
| // - Any responses whose status is in the range 100 to 199, inclusive, | |
| // and is not 101, are to be ignored, except for the purposes of setting | |
| // timingInfo’s final network-response start time above. | |
| // - If request’s header list contains `Transfer-Encoding`/`chunked` and | |
| // response is transferred via HTTP/1.0 or older, then return a network | |
| // error. | |
| // - If the HTTP request results in a TLS client certificate dialog, then: | |
| // 1. If request’s window is an environment settings object, make the | |
| // dialog available in request’s window. | |
| // 2. Otherwise, return a network error. | |
| // To transmit request’s body body, run these steps: | |
| let requestBody = null | |
| // 1. If body is null and fetchParams’s process request end-of-body is | |
| // non-null, then queue a fetch task given fetchParams’s process request | |
| // end-of-body and fetchParams’s task destination. | |
| if (request.body == null && fetchParams.processRequestEndOfBody) { | |
| queueMicrotask(() => fetchParams.processRequestEndOfBody()) | |
| } else if (request.body != null) { | |
| // 2. Otherwise, if body is non-null: | |
| // 1. Let processBodyChunk given bytes be these steps: | |
| const processBodyChunk = async function * (bytes) { | |
| // 1. If the ongoing fetch is terminated, then abort these steps. | |
| if (isCancelled(fetchParams)) { | |
| return | |
| } | |
| // 2. Run this step in parallel: transmit bytes. | |
| yield bytes | |
| // 3. If fetchParams’s process request body is non-null, then run | |
| // fetchParams’s process request body given bytes’s length. | |
| fetchParams.processRequestBodyChunkLength?.(bytes.byteLength) | |
| } | |
| // 2. Let processEndOfBody be these steps: | |
| const processEndOfBody = () => { | |
| // 1. If fetchParams is canceled, then abort these steps. | |
| if (isCancelled(fetchParams)) { | |
| return | |
| } | |
| // 2. If fetchParams’s process request end-of-body is non-null, | |
| // then run fetchParams’s process request end-of-body. | |
| if (fetchParams.processRequestEndOfBody) { | |
| fetchParams.processRequestEndOfBody() | |
| } | |
| } | |
| // 3. Let processBodyError given e be these steps: | |
| const processBodyError = (e) => { | |
| // 1. If fetchParams is canceled, then abort these steps. | |
| if (isCancelled(fetchParams)) { | |
| return | |
| } | |
| // 2. If e is an "AbortError" DOMException, then abort fetchParams’s controller. | |
| if (e.name === 'AbortError') { | |
| fetchParams.controller.abort() | |
| } else { | |
| fetchParams.controller.terminate(e) | |
| } | |
| } | |
| // 4. Incrementally read request’s body given processBodyChunk, processEndOfBody, | |
| // processBodyError, and fetchParams’s task destination. | |
| requestBody = (async function * () { | |
| try { | |
| for await (const bytes of request.body.stream) { | |
| yield * processBodyChunk(bytes) | |
| } | |
| processEndOfBody() | |
| } catch (err) { | |
| processBodyError(err) | |
| } | |
| })() | |
| } | |
| try { | |
| // socket is only provided for websockets | |
| const { body, status, statusText, headersList, socket } = await dispatch({ body: requestBody }) | |
| if (socket) { | |
| response = makeResponse({ status, statusText, headersList, socket }) | |
| } else { | |
| const iterator = body[Symbol.asyncIterator]() | |
| fetchParams.controller.next = () => iterator.next() | |
| response = makeResponse({ status, statusText, headersList }) | |
| } | |
| } catch (err) { | |
| // 10. If aborted, then: | |
| if (err.name === 'AbortError') { | |
| // 1. If connection uses HTTP/2, then transmit an RST_STREAM frame. | |
| fetchParams.controller.connection.destroy() | |
| // 2. Return the appropriate network error for fetchParams. | |
| return makeAppropriateNetworkError(fetchParams, err) | |
| } | |
| return makeNetworkError(err) | |
| } | |
| // 11. Let pullAlgorithm be an action that resumes the ongoing fetch | |
| // if it is suspended. | |
| const pullAlgorithm = () => { | |
| fetchParams.controller.resume() | |
| } | |
| // 12. Let cancelAlgorithm be an algorithm that aborts fetchParams’s | |
| // controller with reason, given reason. | |
| const cancelAlgorithm = (reason) => { | |
| fetchParams.controller.abort(reason) | |
| } | |
| // 13. Let highWaterMark be a non-negative, non-NaN number, chosen by | |
| // the user agent. | |
| // TODO | |
| // 14. Let sizeAlgorithm be an algorithm that accepts a chunk object | |
| // and returns a non-negative, non-NaN, non-infinite number, chosen by the user agent. | |
| // TODO | |
| // 15. Let stream be a new ReadableStream. | |
| // 16. Set up stream with pullAlgorithm set to pullAlgorithm, | |
| // cancelAlgorithm set to cancelAlgorithm, highWaterMark set to | |
| // highWaterMark, and sizeAlgorithm set to sizeAlgorithm. | |
| if (!ReadableStream) { | |
| ReadableStream = require('stream/web').ReadableStream | |
| } | |
| const stream = new ReadableStream( | |
| { | |
| async start (controller) { | |
| fetchParams.controller.controller = controller | |
| }, | |
| async pull (controller) { | |
| await pullAlgorithm(controller) | |
| }, | |
| async cancel (reason) { | |
| await cancelAlgorithm(reason) | |
| } | |
| }, | |
| { | |
| highWaterMark: 0, | |
| size () { | |
| return 1 | |
| } | |
| } | |
| ) | |
| // 17. Run these steps, but abort when the ongoing fetch is terminated: | |
| // 1. Set response’s body to a new body whose stream is stream. | |
| response.body = { stream } | |
| // 2. If response is not a network error and request’s cache mode is | |
| // not "no-store", then update response in httpCache for request. | |
| // TODO | |
| // 3. If includeCredentials is true and the user agent is not configured | |
| // to block cookies for request (see section 7 of [COOKIES]), then run the | |
| // "set-cookie-string" parsing algorithm (see section 5.2 of [COOKIES]) on | |
| // the value of each header whose name is a byte-case-insensitive match for | |
| // `Set-Cookie` in response’s header list, if any, and request’s current URL. | |
| // TODO | |
| // 18. If aborted, then: | |
| // TODO | |
| // 19. Run these steps in parallel: | |
| // 1. Run these steps, but abort when fetchParams is canceled: | |
| fetchParams.controller.on('terminated', onAborted) | |
| fetchParams.controller.resume = async () => { | |
| // 1. While true | |
| while (true) { | |
| // 1-3. See onData... | |
| // 4. Set bytes to the result of handling content codings given | |
| // codings and bytes. | |
| let bytes | |
| let isFailure | |
| try { | |
| const { done, value } = await fetchParams.controller.next() | |
| if (isAborted(fetchParams)) { | |
| break | |
| } | |
| bytes = done ? undefined : value | |
| } catch (err) { | |
| if (fetchParams.controller.ended && !timingInfo.encodedBodySize) { | |
| // zlib doesn't like empty streams. | |
| bytes = undefined | |
| } else { | |
| bytes = err | |
| // err may be propagated from the result of calling readablestream.cancel, | |
| // which might not be an error. https://github.com/nodejs/undici/issues/2009 | |
| isFailure = true | |
| } | |
| } | |
| if (bytes === undefined) { | |
| // 2. Otherwise, if the bytes transmission for response’s message | |
| // body is done normally and stream is readable, then close | |
| // stream, finalize response for fetchParams and response, and | |
| // abort these in-parallel steps. | |
| readableStreamClose(fetchParams.controller.controller) | |
| finalizeResponse(fetchParams, response) | |
| return | |
| } | |
| // 5. Increase timingInfo’s decoded body size by bytes’s length. | |
| timingInfo.decodedBodySize += bytes?.byteLength ?? 0 | |
| // 6. If bytes is failure, then terminate fetchParams’s controller. | |
| if (isFailure) { | |
| fetchParams.controller.terminate(bytes) | |
| return | |
| } | |
| // 7. Enqueue a Uint8Array wrapping an ArrayBuffer containing bytes | |
| // into stream. | |
| fetchParams.controller.controller.enqueue(new Uint8Array(bytes)) | |
| // 8. If stream is errored, then terminate the ongoing fetch. | |
| if (isErrored(stream)) { | |
| fetchParams.controller.terminate() | |
| return | |
| } | |
| // 9. If stream doesn’t need more data ask the user agent to suspend | |
| // the ongoing fetch. | |
| if (!fetchParams.controller.controller.desiredSize) { | |
| return | |
| } | |
| } | |
| } | |
| // 2. If aborted, then: | |
| function onAborted (reason) { | |
| // 2. If fetchParams is aborted, then: | |
| if (isAborted(fetchParams)) { | |
| // 1. Set response’s aborted flag. | |
| response.aborted = true | |
| // 2. If stream is readable, then error stream with the result of | |
| // deserialize a serialized abort reason given fetchParams’s | |
| // controller’s serialized abort reason and an | |
| // implementation-defined realm. | |
| if (isReadable(stream)) { | |
| fetchParams.controller.controller.error( | |
| fetchParams.controller.serializedAbortReason | |
| ) | |
| } | |
| } else { | |
| // 3. Otherwise, if stream is readable, error stream with a TypeError. | |
| if (isReadable(stream)) { | |
| fetchParams.controller.controller.error(new TypeError('terminated', { | |
| cause: isErrorLike(reason) ? reason : undefined | |
| })) | |
| } | |
| } | |
| // 4. If connection uses HTTP/2, then transmit an RST_STREAM frame. | |
| // 5. Otherwise, the user agent should close connection unless it would be bad for performance to do so. | |
| fetchParams.controller.connection.destroy() | |
| } | |
| // 20. Return response. | |
| return response | |
| async function dispatch ({ body }) { | |
| const url = requestCurrentURL(request) | |
| /** @type {import('../..').Agent} */ | |
| const agent = fetchParams.controller.dispatcher | |
| return new Promise((resolve, reject) => agent.dispatch( | |
| { | |
| path: url.pathname + url.search, | |
| origin: url.origin, | |
| method: request.method, | |
| body: fetchParams.controller.dispatcher.isMockActive ? request.body && request.body.source : body, | |
| headers: request.headersList.entries, | |
| maxRedirections: 0, | |
| upgrade: request.mode === 'websocket' ? 'websocket' : undefined | |
| }, | |
| { | |
| body: null, | |
| abort: null, | |
| onConnect (abort) { | |
| // TODO (fix): Do we need connection here? | |
| const { connection } = fetchParams.controller | |
| if (connection.destroyed) { | |
| abort(new DOMException('The operation was aborted.', 'AbortError')) | |
| } else { | |
| fetchParams.controller.on('terminated', abort) | |
| this.abort = connection.abort = abort | |
| } | |
| }, | |
| onHeaders (status, headersList, resume, statusText) { | |
| if (status < 200) { | |
| return | |
| } | |
| let codings = [] | |
| let location = '' | |
| const headers = new Headers() | |
| // For H2, the headers are a plain JS object | |
| // We distinguish between them and iterate accordingly | |
| if (Array.isArray(headersList)) { | |
| for (let n = 0; n < headersList.length; n += 2) { | |
| const key = headersList[n + 0].toString('latin1') | |
| const val = headersList[n + 1].toString('latin1') | |
| if (key.toLowerCase() === 'content-encoding') { | |
| // https://www.rfc-editor.org/rfc/rfc7231#section-3.1.2.1 | |
| // "All content-coding values are case-insensitive..." | |
| codings = val.toLowerCase().split(',').map((x) => x.trim()) | |
| } else if (key.toLowerCase() === 'location') { | |
| location = val | |
| } | |
| headers.append(key, val) | |
| } | |
| } else { | |
| const keys = Object.keys(headersList) | |
| for (const key of keys) { | |
| const val = headersList[key] | |
| if (key.toLowerCase() === 'content-encoding') { | |
| // https://www.rfc-editor.org/rfc/rfc7231#section-3.1.2.1 | |
| // "All content-coding values are case-insensitive..." | |
| codings = val.toLowerCase().split(',').map((x) => x.trim()).reverse() | |
| } else if (key.toLowerCase() === 'location') { | |
| location = val | |
| } | |
| headers.append(key, val) | |
| } | |
| } | |
| this.body = new Readable({ read: resume }) | |
| const decoders = [] | |
| const willFollow = request.redirect === 'follow' && | |
| location && | |
| redirectStatusSet.has(status) | |
| // https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Content-Encoding | |
| if (request.method !== 'HEAD' && request.method !== 'CONNECT' && !nullBodyStatus.includes(status) && !willFollow) { | |
| for (const coding of codings) { | |
| // https://www.rfc-editor.org/rfc/rfc9112.html#section-7.2 | |
| if (coding === 'x-gzip' || coding === 'gzip') { | |
| decoders.push(zlib.createGunzip({ | |
| // Be less strict when decoding compressed responses, since sometimes | |
| // servers send slightly invalid responses that are still accepted | |
| // by common browsers. | |
| // Always using Z_SYNC_FLUSH is what cURL does. | |
| flush: zlib.constants.Z_SYNC_FLUSH, | |
| finishFlush: zlib.constants.Z_SYNC_FLUSH | |
| })) | |
| } else if (coding === 'deflate') { | |
| decoders.push(zlib.createInflate()) | |
| } else if (coding === 'br') { | |
| decoders.push(zlib.createBrotliDecompress()) | |
| } else { | |
| decoders.length = 0 | |
| break | |
| } | |
| } | |
| } | |
| resolve({ | |
| status, | |
| statusText, | |
| headersList: headers[kHeadersList], | |
| body: decoders.length | |
| ? pipeline(this.body, ...decoders, () => { }) | |
| : this.body.on('error', () => {}) | |
| }) | |
| return true | |
| }, | |
| onData (chunk) { | |
| if (fetchParams.controller.dump) { | |
| return | |
| } | |
| // 1. If one or more bytes have been transmitted from response’s | |
| // message body, then: | |
| // 1. Let bytes be the transmitted bytes. | |
| const bytes = chunk | |
| // 2. Let codings be the result of extracting header list values | |
| // given `Content-Encoding` and response’s header list. | |
| // See pullAlgorithm. | |
| // 3. Increase timingInfo’s encoded body size by bytes’s length. | |
| timingInfo.encodedBodySize += bytes.byteLength | |
| // 4. See pullAlgorithm... | |
| return this.body.push(bytes) | |
| }, | |
| onComplete () { | |
| if (this.abort) { | |
| fetchParams.controller.off('terminated', this.abort) | |
| } | |
| fetchParams.controller.ended = true | |
| this.body.push(null) | |
| }, | |
| onError (error) { | |
| if (this.abort) { | |
| fetchParams.controller.off('terminated', this.abort) | |
| } | |
| this.body?.destroy(error) | |
| fetchParams.controller.terminate(error) | |
| reject(error) | |
| }, | |
| onUpgrade (status, headersList, socket) { | |
| if (status !== 101) { | |
| return | |
| } | |
| const headers = new Headers() | |
| for (let n = 0; n < headersList.length; n += 2) { | |
| const key = headersList[n + 0].toString('latin1') | |
| const val = headersList[n + 1].toString('latin1') | |
| headers.append(key, val) | |
| } | |
| resolve({ | |
| status, | |
| statusText: STATUS_CODES[status], | |
| headersList: headers[kHeadersList], | |
| socket | |
| }) | |
| return true | |
| } | |
| } | |
| )) | |
| } | |
| } | |
| module.exports = { | |
| fetch, | |
| Fetch, | |
| fetching, | |
| finalizeAndReportTiming | |
| } | |