@@ -35,11 +35,12 @@ const {
35
35
isCancelled,
36
36
isAborted,
37
37
isErrorLike,
38
- fullyReadBody
38
+ fullyReadBody,
39
+ readableStreamClose
39
40
} = require ( './util' )
40
41
const { kState, kHeaders, kGuard, kRealm } = require ( './symbols' )
41
42
const assert = require ( 'assert' )
42
- const { safelyExtractBody, extractBody } = require ( './body' )
43
+ const { safelyExtractBody } = require ( './body' )
43
44
const {
44
45
redirectStatus,
45
46
nullBodyStatus,
@@ -54,9 +55,11 @@ const { Readable, pipeline } = require('stream')
54
55
const { isErrored, isReadable } = require ( '../core/util' )
55
56
const { dataURLProcessor, serializeAMimeType } = require ( './dataURL' )
56
57
const { TransformStream } = require ( 'stream/web' )
58
+ const { getGlobalDispatcher } = require ( '../../index' )
57
59
58
60
/** @type {import('buffer').resolveObjectURL } */
59
61
let resolveObjectURL
62
+ /** @type {globalThis['ReadableStream'] } */
60
63
let ReadableStream
61
64
62
65
const nodeVersion = process . versions . node . split ( '.' )
@@ -71,6 +74,12 @@ class Fetch extends EE {
71
74
this . connection = null
72
75
this . dump = false
73
76
this . state = 'ongoing'
77
+ // 2 terminated listeners get added per request,
78
+ // but only 1 gets removed. If there are 20 redirects,
79
+ // 21 listeners will be added.
80
+ // See https://github.com/nodejs/undici/issues/1711
81
+ // TODO (fix): Find and fix root cause for leaked listener.
82
+ this . setMaxListeners ( 21 )
74
83
}
75
84
76
85
terminate ( reason ) {
@@ -83,16 +92,30 @@ class Fetch extends EE {
83
92
this . emit ( 'terminated' , reason )
84
93
}
85
94
86
- abort ( ) {
95
+ // https://fetch.spec.whatwg.org/#fetch-controller-abort
96
+ abort ( error ) {
87
97
if ( this . state !== 'ongoing' ) {
88
98
return
89
99
}
90
100
91
- const reason = new DOMException ( 'The operation was aborted.' , 'AbortError' )
92
-
101
+ // 1. Set controller’s state to "aborted".
93
102
this . state = 'aborted'
94
- this . connection ?. destroy ( reason )
95
- this . emit ( 'terminated' , reason )
103
+
104
+ // 2. Let fallbackError be an "AbortError" DOMException.
105
+ // 3. Set error to fallbackError if it is not given.
106
+ if ( ! error ) {
107
+ error = new DOMException ( 'The operation was aborted.' , 'AbortError' )
108
+ }
109
+
110
+ // 4. Let serializedError be StructuredSerialize(error).
111
+ // If that threw an exception, catch it, and let
112
+ // serializedError be StructuredSerialize(fallbackError).
113
+
114
+ // 5. Set controller’s serialized abort reason to serializedError.
115
+ this . serializedAbortReason = error
116
+
117
+ this . connection ?. destroy ( error )
118
+ this . emit ( 'terminated' , error )
96
119
}
97
120
}
98
121
@@ -124,8 +147,9 @@ async function fetch (input, init = {}) {
124
147
125
148
// 4. If requestObject’s signal’s aborted flag is set, then:
126
149
if ( requestObject . signal . aborted ) {
127
- // 1. Abort fetch with p, request, and null.
128
- abortFetch ( p , request , null )
150
+ // 1. Abort the fetch() call with p, request, null, and
151
+ // requestObject’s signal’s abort reason.
152
+ abortFetch ( p , request , null , requestObject . signal . reason )
129
153
130
154
// 2. Return p.
131
155
return p . promise
@@ -159,8 +183,9 @@ async function fetch (input, init = {}) {
159
183
// 1. Set locallyAborted to true.
160
184
locallyAborted = true
161
185
162
- // 2. Abort fetch with p, request, and responseObject.
163
- abortFetch ( p , request , responseObject )
186
+ // 2. Abort the fetch() call with p, request, responseObject,
187
+ // and requestObject’s signal’s abort reason.
188
+ abortFetch ( p , request , responseObject , requestObject . signal . reason )
164
189
165
190
// 3. If controller is not null, then abort controller.
166
191
if ( controller != null ) {
@@ -185,10 +210,16 @@ async function fetch (input, init = {}) {
185
210
return
186
211
}
187
212
188
- // 2. If response’s aborted flag is set, then abort fetch with p,
189
- // request, and responseObject, and terminate these substeps.
213
+ // 2. If response’s aborted flag is set, then:
190
214
if ( response . aborted ) {
191
- abortFetch ( p , request , responseObject )
215
+ // 1. Let deserializedError be the result of deserialize a serialized
216
+ // abort reason given controller’s serialized abort reason and
217
+ // relevantRealm.
218
+
219
+ // 2. Abort the fetch() call with p, request, responseObject, and
220
+ // deserializedError.
221
+
222
+ abortFetch ( p , request , responseObject , controller . serializedAbortReason )
192
223
return
193
224
}
194
225
@@ -218,7 +249,7 @@ async function fetch (input, init = {}) {
218
249
request,
219
250
processResponseEndOfBody : handleFetchDone ,
220
251
processResponse,
221
- dispatcher : this // undici
252
+ dispatcher : init . dispatcher ?? getGlobalDispatcher ( ) // undici
222
253
} )
223
254
224
255
// 14. Return p.
@@ -296,14 +327,18 @@ function markResourceTiming (timingInfo, originalURL, initiatorType, globalThis,
296
327
}
297
328
298
329
// https://fetch.spec.whatwg.org/#abort-fetch
299
- function abortFetch ( p , request , responseObject ) {
300
- // 1. Let error be an "AbortError" DOMException.
301
- const error = new DOMException ( 'The operation was aborted.' , 'AbortError' )
330
+ function abortFetch ( p , request , responseObject , error ) {
331
+ // Note: AbortSignal.reason was added in node v17.2.0
332
+ // which would give us an undefined error to reject with.
333
+ // Remove this once node v16 is no longer supported.
334
+ if ( ! error ) {
335
+ error = new DOMException ( 'The operation was aborted.' , 'AbortError' )
336
+ }
302
337
303
- // 2 . Reject promise with error.
338
+ // 1 . Reject promise with error.
304
339
p . reject ( error )
305
340
306
- // 3 . If request’s body is not null and is readable, then cancel request’s
341
+ // 2 . If request’s body is not null and is readable, then cancel request’s
307
342
// body with error.
308
343
if ( request . body != null && isReadable ( request . body ?. stream ) ) {
309
344
request . body . stream . cancel ( error ) . catch ( ( err ) => {
@@ -315,15 +350,15 @@ function abortFetch (p, request, responseObject) {
315
350
} )
316
351
}
317
352
318
- // 4 . If responseObject is null, then return.
353
+ // 3 . If responseObject is null, then return.
319
354
if ( responseObject == null ) {
320
355
return
321
356
}
322
357
323
- // 5 . Let response be responseObject’s response.
358
+ // 4 . Let response be responseObject’s response.
324
359
const response = responseObject [ kState ]
325
360
326
- // 6 . If response’s body is not null and is readable, then error response’s
361
+ // 5 . If response’s body is not null and is readable, then error response’s
327
362
// body with error.
328
363
if ( response . body != null && isReadable ( response . body ?. stream ) ) {
329
364
response . body . stream . cancel ( error ) . catch ( ( err ) => {
@@ -399,8 +434,8 @@ function fetching ({
399
434
crossOriginIsolatedCapability
400
435
}
401
436
402
- // 7. If request’s body is a byte sequence, then set request’s body to the
403
- // first return value of safely extracting request’s body.
437
+ // 7. If request’s body is a byte sequence, then set request’s body to
438
+ // request’s body as a body.
404
439
// NOTE: Since fetching is only called from fetch, body should already be
405
440
// extracted.
406
441
assert ( ! request . body || request . body . stream )
@@ -730,8 +765,7 @@ async function mainFetch (fetchParams, recursive = false) {
730
765
return
731
766
}
732
767
733
- // 2. Set response’s body to the first return value of safely
734
- // extracting bytes.
768
+ // 2. Set response’s body to bytes as a body.
735
769
response . body = safelyExtractBody ( bytes ) [ 0 ]
736
770
737
771
// 3. Run fetch finale given fetchParams and response.
@@ -749,75 +783,73 @@ async function mainFetch (fetchParams, recursive = false) {
749
783
// https://fetch.spec.whatwg.org/#concept-scheme-fetch
750
784
// given a fetch params fetchParams
751
785
async function schemeFetch ( fetchParams ) {
752
- // let request be fetchParams’s request
786
+ // 1. If fetchParams is canceled, then return the appropriate network error for fetchParams.
787
+ if ( isCancelled ( fetchParams ) ) {
788
+ return makeAppropriateNetworkError ( fetchParams )
789
+ }
790
+
791
+ // 2. Let request be fetchParams’s request.
753
792
const { request } = fetchParams
754
793
755
- const {
756
- protocol : scheme ,
757
- pathname : path
758
- } = requestCurrentURL ( request )
794
+ const { protocol : scheme } = requestCurrentURL ( request )
759
795
760
- // switch on request’s current URL’s scheme, and run the associated steps:
796
+ // 3. Switch on request’s current URL’s scheme and run the associated steps:
761
797
switch ( scheme ) {
762
798
case 'about:' : {
763
799
// If request’s current URL’s path is the string "blank", then return a new response
764
800
// whose status message is `OK`, header list is « (`Content-Type`, `text/html;charset=utf-8`) »,
765
- // and body is the empty byte sequence.
766
- if ( path === 'blank' ) {
767
- const resp = makeResponse ( {
768
- statusText : 'OK' ,
769
- headersList : [
770
- [ 'content-type' , 'text/html;charset=utf-8' ]
771
- ]
772
- } )
773
-
774
- resp . urlList = [ new URL ( 'about:blank' ) ]
775
- return resp
776
- }
801
+ // and body is the empty byte sequence as a body.
777
802
778
803
// Otherwise, return a network error.
779
- return makeNetworkError ( 'invalid path called ' )
804
+ return makeNetworkError ( 'about scheme is not supported ' )
780
805
}
781
806
case 'blob:' : {
782
- resolveObjectURL = resolveObjectURL || require ( 'buffer' ) . resolveObjectURL
807
+ if ( ! resolveObjectURL ) {
808
+ resolveObjectURL = require ( 'buffer' ) . resolveObjectURL
809
+ }
783
810
784
- // 1. Run these steps, but abort when the ongoing fetch is terminated:
785
- // 1. Let blob be request’s current URL’s blob URL entry’s object.
786
- // https://w3c.github.io/FileAPI/#blob-url-entry
787
- // P.S. Thank God this method is available in node.
788
- const currentURL = requestCurrentURL ( request )
811
+ // 1. Let blobURLEntry be request’s current URL’s blob URL entry.
812
+ const blobURLEntry = requestCurrentURL ( request )
789
813
790
814
// https://github.com/web-platform-tests/wpt/blob/7b0ebaccc62b566a1965396e5be7bb2bc06f841f/FileAPI/url/resources/fetch-tests.js#L52-L56
791
815
// Buffer.resolveObjectURL does not ignore URL queries.
792
- if ( currentURL . search . length !== 0 ) {
816
+ if ( blobURLEntry . search . length !== 0 ) {
793
817
return makeNetworkError ( 'NetworkError when attempting to fetch resource.' )
794
818
}
795
819
796
- const blob = resolveObjectURL ( currentURL . toString ( ) )
820
+ const blobURLEntryObject = resolveObjectURL ( blobURLEntry . toString ( ) )
797
821
798
- // 2. If request’s method is not `GET` or blob is not a Blob object, then return a network error. [FILEAPI]
799
- if ( request . method !== 'GET' || ! isBlobLike ( blob ) ) {
822
+ // 2. If request’s method is not `GET`, blobURLEntry is null, or blobURLEntry’s
823
+ // object is not a Blob object, then return a network error.
824
+ if ( request . method !== 'GET' || ! isBlobLike ( blobURLEntryObject ) ) {
800
825
return makeNetworkError ( 'invalid method' )
801
826
}
802
827
803
- // 3. Let response be a new response whose status message is `OK` .
804
- const response = makeResponse ( { statusText : 'OK' , urlList : [ currentURL ] } )
828
+ // 3. Let bodyWithType be the result of safely extracting blobURLEntry’s object .
829
+ const bodyWithType = safelyExtractBody ( blobURLEntryObject )
805
830
806
- // 4. Append (`Content-Length`, blob’s size attribute value) to response ’s header list .
807
- response . headersList . set ( 'content-length' , ` ${ blob . size } ` )
831
+ // 4. Let body be bodyWithType ’s body .
832
+ const body = bodyWithType [ 0 ]
808
833
809
- // 5. Append (`Content-Type`, blob ’s type attribute value) to response’s header list .
810
- response . headersList . set ( 'content-type' , blob . type )
834
+ // 5. Let length be body ’s length, serialized and isomorphic encoded .
835
+ const length = ` ${ body . length } `
811
836
812
- // 6. Set response’s body to the result of performing the read operation on blob.
813
- // TODO (fix): This needs to read?
814
- response . body = extractBody ( blob ) [ 0 ]
837
+ // 6. Let type be bodyWithType’s type if it is non-null; otherwise the empty byte sequence.
838
+ const type = bodyWithType [ 1 ] ?? ''
815
839
816
- // 7. Return response.
817
- return response
840
+ // 7. Return a new response whose status message is `OK`, header list is
841
+ // « (`Content-Length`, length), (`Content-Type`, type) », and body is body.
842
+ const response = makeResponse ( {
843
+ statusText : 'OK' ,
844
+ headersList : [
845
+ [ 'content-length' , length ] ,
846
+ [ 'content-type' , type ]
847
+ ]
848
+ } )
818
849
819
- // 2. If aborted, then return the appropriate network error for fetchParams.
820
- // TODO
850
+ response . body = body
851
+
852
+ return response
821
853
}
822
854
case 'data:' : {
823
855
// 1. Let dataURLStruct be the result of running the
@@ -836,13 +868,13 @@ async function schemeFetch (fetchParams) {
836
868
837
869
// 4. Return a response whose status message is `OK`,
838
870
// header list is « (`Content-Type`, mimeType) »,
839
- // and body is dataURLStruct’s body.
871
+ // and body is dataURLStruct’s body as a body .
840
872
return makeResponse ( {
841
873
statusText : 'OK' ,
842
874
headersList : [
843
875
[ 'content-type' , mimeType ]
844
876
] ,
845
- body : extractBody ( dataURLStruct . body ) [ 0 ]
877
+ body : safelyExtractBody ( dataURLStruct . body ) [ 0 ]
846
878
} )
847
879
}
848
880
case 'file:' : {
@@ -930,6 +962,14 @@ async function fetchFinale (fetchParams, response) {
930
962
start ( ) { } ,
931
963
transform : identityTransformAlgorithm ,
932
964
flush : processResponseEndOfBody
965
+ } , {
966
+ size ( ) {
967
+ return 1
968
+ }
969
+ } , {
970
+ size ( ) {
971
+ return 1
972
+ }
933
973
} )
934
974
935
975
// 4. Set response’s body to the result of piping response’s body through transformStream.
@@ -1725,9 +1765,9 @@ async function httpNetworkFetch (
1725
1765
}
1726
1766
1727
1767
// 12. Let cancelAlgorithm be an algorithm that aborts fetchParams’s
1728
- // controller.
1729
- const cancelAlgorithm = ( ) => {
1730
- fetchParams . controller . abort ( )
1768
+ // controller with reason, given reason .
1769
+ const cancelAlgorithm = ( reason ) => {
1770
+ fetchParams . controller . abort ( reason )
1731
1771
}
1732
1772
1733
1773
// 13. Let highWaterMark be a non-negative, non-NaN number, chosen by
@@ -1758,7 +1798,12 @@ async function httpNetworkFetch (
1758
1798
await cancelAlgorithm ( reason )
1759
1799
}
1760
1800
} ,
1761
- { highWaterMark : 0 }
1801
+ {
1802
+ highWaterMark : 0 ,
1803
+ size ( ) {
1804
+ return 1
1805
+ }
1806
+ }
1762
1807
)
1763
1808
1764
1809
// 17. Run these steps, but abort when the ongoing fetch is terminated:
@@ -1814,14 +1859,7 @@ async function httpNetworkFetch (
1814
1859
// body is done normally and stream is readable, then close
1815
1860
// stream, finalize response for fetchParams and response, and
1816
1861
// abort these in-parallel steps.
1817
- try {
1818
- fetchParams . controller . controller . close ( )
1819
- } catch ( err ) {
1820
- // TODO (fix): How/Why can this happen? Do we have a bug?
1821
- if ( ! / C o n t r o l l e r i s a l r e a d y c l o s e d / . test ( err ) ) {
1822
- throw err
1823
- }
1824
- }
1862
+ readableStreamClose ( fetchParams . controller . controller )
1825
1863
1826
1864
finalizeResponse ( fetchParams , response )
1827
1865
@@ -1862,10 +1900,13 @@ async function httpNetworkFetch (
1862
1900
// 1. Set response’s aborted flag.
1863
1901
response . aborted = true
1864
1902
1865
- // 2. If stream is readable, error stream with an "AbortError" DOMException.
1903
+ // 2. If stream is readable, then error stream with the result of
1904
+ // deserialize a serialized abort reason given fetchParams’s
1905
+ // controller’s serialized abort reason and an
1906
+ // implementation-defined realm.
1866
1907
if ( isReadable ( stream ) ) {
1867
1908
fetchParams . controller . controller . error (
1868
- new DOMException ( 'The operation was aborted.' , 'AbortError' )
1909
+ fetchParams . controller . serializedAbortReason
1869
1910
)
1870
1911
}
1871
1912
} else {
0 commit comments