@@ -24,7 +24,10 @@ const {
24
24
getDataObject,
25
25
} = internalBinding ( 'blob' ) ;
26
26
27
- const { TextDecoder } = require ( 'internal/encoding' ) ;
27
+ const {
28
+ TextDecoder,
29
+ TextEncoder,
30
+ } = require ( 'internal/encoding' ) ;
28
31
29
32
const {
30
33
makeTransferable,
@@ -48,6 +51,7 @@ const {
48
51
AbortError,
49
52
codes : {
50
53
ERR_INVALID_ARG_TYPE ,
54
+ ERR_INVALID_ARG_VALUE ,
51
55
ERR_INVALID_THIS ,
52
56
ERR_BUFFER_TOO_LARGE ,
53
57
}
@@ -65,10 +69,11 @@ const kArrayBufferPromise = Symbol('kArrayBufferPromise');
65
69
66
70
const disallowedTypeCharacters = / [ ^ \u{0020} - \u{007E} ] / u;
67
71
68
- let Buffer ;
69
72
let ReadableStream ;
70
73
let URL ;
74
+ let EOL ;
71
75
76
+ const enc = new TextEncoder ( ) ;
72
77
73
78
// Yes, lazy loading is annoying but because of circular
74
79
// references between the url, internal/blob, and buffer
@@ -79,29 +84,35 @@ function lazyURL(id) {
79
84
return new URL ( id ) ;
80
85
}
81
86
82
- function lazyBuffer ( ) {
83
- Buffer ??= require ( 'buffer' ) . Buffer ;
84
- return Buffer ;
85
- }
86
-
87
87
function lazyReadableStream ( options ) {
88
88
ReadableStream ??=
89
89
require ( 'internal/webstreams/readablestream' ) . ReadableStream ;
90
90
return new ReadableStream ( options ) ;
91
91
}
92
92
93
+ // TODO(@jasnell): This is annoying but this has to be lazy because
94
+ // requiring the 'os' module too early causes building Node.js to
95
+ // fail with an unknown reference failure.
96
+ function lazyEOL ( ) {
97
+ EOL ??= require ( 'os' ) . EOL ;
98
+ return EOL ;
99
+ }
100
+
93
101
function isBlob ( object ) {
94
102
return object ?. [ kHandle ] !== undefined ;
95
103
}
96
104
97
- function getSource ( source , encoding ) {
105
+ function getSource ( source , endings ) {
98
106
if ( isBlob ( source ) )
99
107
return [ source . size , source [ kHandle ] ] ;
100
108
101
109
if ( isAnyArrayBuffer ( source ) ) {
102
110
source = new Uint8Array ( source ) ;
103
111
} else if ( ! isArrayBufferView ( source ) ) {
104
- source = lazyBuffer ( ) . from ( `${ source } ` , encoding ) ;
112
+ source = `${ source } ` ;
113
+ if ( endings === 'native' )
114
+ source = source . replaceAll ( / ( \n | \r \n ) / g, lazyEOL ( ) ) ;
115
+ source = enc . encode ( source ) ;
105
116
}
106
117
107
118
// We copy into a new Uint8Array because the underlying
@@ -113,6 +124,16 @@ function getSource(source, encoding) {
113
124
}
114
125
115
126
class Blob {
127
+ /**
128
+ * @typedef {string|ArrayBuffer|ArrayBufferView|Blob } SourcePart
129
+ *
130
+ * @param {SourcePart[] } [sources]
131
+ * @param {{
132
+ * endings? : string,
133
+ * type? : string,
134
+ * }} options
135
+ * @returns
136
+ */
116
137
constructor ( sources = [ ] , options = { } ) {
117
138
emitExperimentalWarning ( 'buffer.Blob' ) ;
118
139
if ( sources === null ||
@@ -121,12 +142,18 @@ class Blob {
121
142
throw new ERR_INVALID_ARG_TYPE ( 'sources' , 'Iterable' , sources ) ;
122
143
}
123
144
validateObject ( options , 'options' ) ;
124
- const { encoding = 'utf8' } = options ;
125
- let { type = '' } = options ;
145
+ let {
146
+ type = '' ,
147
+ endings = 'transparent' ,
148
+ } = options ;
149
+
150
+ endings = `${ endings } ` ;
151
+ if ( endings !== 'transparent' || endings !== 'native' )
152
+ throw new ERR_INVALID_ARG_VALUE ( 'options.endings' , endings ) ;
126
153
127
154
let length = 0 ;
128
155
const sources_ = ArrayFrom ( sources , ( source ) => {
129
- const { 0 : len , 1 : src } = getSource ( source , encoding ) ;
156
+ const { 0 : len , 1 : src } = getSource ( source , endings ) ;
130
157
length += len ;
131
158
return src ;
132
159
} ) ;
0 commit comments