Skip to content

Commit 0d564ce

Browse files
committed
buffer: make Blob's slice method more spec-compliant
PR-URL: #37361 Fixes: #37335 Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com> Reviewed-By: James M Snell <jasnell@gmail.com>
1 parent daad7bb commit 0d564ce

14 files changed

+1355
-28
lines changed

lib/internal/blob.js

+30-14
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22

33
const {
44
ArrayFrom,
5+
MathMax,
6+
MathMin,
57
ObjectDefineProperty,
68
ObjectSetPrototypeOf,
79
PromiseResolve,
@@ -41,21 +43,21 @@ const {
4143
codes: {
4244
ERR_INVALID_ARG_TYPE,
4345
ERR_BUFFER_TOO_LARGE,
44-
ERR_OUT_OF_RANGE,
4546
}
4647
} = require('internal/errors');
4748

4849
const {
4950
validateObject,
5051
validateString,
51-
validateUint32,
5252
isUint32,
5353
} = require('internal/validators');
5454

5555
const kHandle = Symbol('kHandle');
5656
const kType = Symbol('kType');
5757
const kLength = Symbol('kLength');
5858

59+
const disallowedTypeCharacters = /[^\u{0020}-\u{007E}]/u;
60+
5961
let Buffer;
6062

6163
function lazyBuffer() {
@@ -139,7 +141,7 @@ class Blob extends JSTransferable {
139141
super();
140142
this[kHandle] = createBlob(sources_, length);
141143
this[kLength] = length;
142-
this[kType] = RegExpPrototypeTest(/[^\u{0020}-\u{007E}]/u, type) ?
144+
this[kType] = RegExpPrototypeTest(disallowedTypeCharacters, type) ?
143145
'' : StringPrototypeToLowerCase(type);
144146
}
145147

@@ -178,18 +180,32 @@ class Blob extends JSTransferable {
178180

179181
get size() { return this[kLength]; }
180182

181-
slice(start = 0, end = (this[kLength]), type = this[kType]) {
182-
validateUint32(start, 'start');
183-
if (end < 0) end = this[kLength] + end;
184-
validateUint32(end, 'end');
185-
validateString(type, 'type');
186-
if (end < start)
187-
throw new ERR_OUT_OF_RANGE('end', 'greater than start', end);
188-
if (end > this[kLength])
189-
throw new ERR_OUT_OF_RANGE('end', 'less than or equal to length', end);
183+
slice(start = 0, end = this[kLength], contentType = '') {
184+
if (start < 0) {
185+
start = MathMax(this[kLength] + start, 0);
186+
} else {
187+
start = MathMin(start, this[kLength]);
188+
}
189+
start |= 0;
190+
191+
if (end < 0) {
192+
end = MathMax(this[kLength] + end, 0);
193+
} else {
194+
end = MathMin(end, this[kLength]);
195+
}
196+
end |= 0;
197+
198+
contentType = `${contentType}`;
199+
if (RegExpPrototypeTest(disallowedTypeCharacters, contentType)) {
200+
contentType = '';
201+
} else {
202+
contentType = StringPrototypeToLowerCase(contentType);
203+
}
204+
205+
const span = MathMax(end - start, 0);
206+
190207
return new InternalBlob(
191-
this[kHandle].slice(start, end),
192-
end - start, type);
208+
this[kHandle].slice(start, start + span), span, contentType);
193209
}
194210

195211
async arrayBuffer() {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
// META: title=Blob constructor
2+
// META: script=../support/Blob.js
3+
'use strict';
4+
5+
var test_error = {
6+
name: "test",
7+
message: "test error",
8+
};
9+
10+
test(function() {
11+
var args = [
12+
document.createElement("div"),
13+
window,
14+
];
15+
args.forEach(function(arg) {
16+
assert_throws_js(TypeError, function() {
17+
new Blob(arg);
18+
}, "Should throw for argument " + format_value(arg) + ".");
19+
});
20+
}, "Passing platform objects for blobParts should throw a TypeError.");
21+
22+
test(function() {
23+
var element = document.createElement("div");
24+
element.appendChild(document.createElement("div"));
25+
element.appendChild(document.createElement("p"));
26+
var list = element.children;
27+
Object.defineProperty(list, "length", {
28+
get: function() { throw test_error; }
29+
});
30+
assert_throws_exactly(test_error, function() {
31+
new Blob(list);
32+
});
33+
}, "A platform object that supports indexed properties should be treated as a sequence for the blobParts argument (overwritten 'length'.)");
34+
35+
test_blob(function() {
36+
var select = document.createElement("select");
37+
select.appendChild(document.createElement("option"));
38+
return new Blob(select);
39+
}, {
40+
expected: "[object HTMLOptionElement]",
41+
type: "",
42+
desc: "Passing an platform object that supports indexed properties as the blobParts array should work (select)."
43+
});
44+
45+
test_blob(function() {
46+
var elm = document.createElement("div");
47+
elm.setAttribute("foo", "bar");
48+
return new Blob(elm.attributes);
49+
}, {
50+
expected: "[object Attr]",
51+
type: "",
52+
desc: "Passing an platform object that supports indexed properties as the blobParts array should work (attributes)."
53+
});

0 commit comments

Comments
 (0)