Spaces:
Runtime error
Runtime error
/** | |
* Copyright (c) Meta Platforms, Inc. and affiliates. | |
* | |
* Licensed under the Apache License, Version 2.0 (the "License"); | |
* you may not use this file except in compliance with the License. | |
* You may obtain a copy of the License at | |
* | |
* http://www.apache.org/licenses/LICENSE-2.0 | |
* | |
* Unless required by applicable law or agreed to in writing, software | |
* distributed under the License is distributed on an "AS IS" BASIS, | |
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
* See the License for the specific language governing permissions and | |
* limitations under the License. | |
*/ | |
const decoder = new TextDecoder(); | |
const encoder = new TextEncoder(); | |
const blankLine = encoder.encode('\r\n'); | |
const STATE_BOUNDARY = 0; | |
const STATE_HEADERS = 1; | |
const STATE_BODY = 2; | |
/** | |
* Compares two Uint8Array objects for equality. | |
* @param {Uint8Array} a | |
* @param {Uint8Array} b | |
* @return {bool} | |
*/ | |
function compareArrays(a: Uint8Array, b: Uint8Array): boolean { | |
if (a.length != b.length) { | |
return false; | |
} | |
for (let i = 0; i < a.length; i++) { | |
if (a[i] != b[i]) { | |
return false; | |
} | |
} | |
return true; | |
} | |
/** | |
* Parses a Content-Type into a multipart boundary. | |
* @param {string} contentType | |
* @return {Uint8Array} boundary line, including preceding -- and trailing \r\n | |
*/ | |
function getBoundary(contentType: string): Uint8Array | null { | |
// Expects the form "multipart/...; boundary=...". | |
// This is not a full MIME media type parser but should be good enough. | |
const MULTIPART_TYPE = 'multipart/'; | |
const BOUNDARY_PARAM = '; boundary='; | |
if (!contentType.startsWith(MULTIPART_TYPE)) { | |
return null; | |
} | |
const i = contentType.indexOf(BOUNDARY_PARAM, MULTIPART_TYPE.length); | |
if (i == -1) { | |
return null; | |
} | |
const suffix = contentType.substring(i + BOUNDARY_PARAM.length); | |
return encoder.encode('--' + suffix + '\r\n'); | |
} | |
/** | |
* Creates a multipart stream. | |
* @param {string} contentType A Content-Type header. | |
* @param {ReadableStream} body The body of a HTTP response. | |
* @return {ReadableStream} a stream of {headers: Headers, body: Uint8Array} | |
* objects. | |
*/ | |
export default function multipartStream( | |
contentType: string, | |
body: ReadableStream, | |
): ReadableStream { | |
const reader = body.getReader(); | |
return new ReadableStream({ | |
async start(controller) { | |
// Define the boundary. | |
const boundary = getBoundary(contentType); | |
if (boundary === null) { | |
controller.error( | |
new Error( | |
'Invalid content type for multipart stream: ' + contentType, | |
), | |
); | |
return; | |
} | |
let pos = 0; | |
let buf = new Uint8Array(); // buf.slice(pos) has unprocessed data. | |
let state = STATE_BOUNDARY; | |
let headers: Headers | null = null; // non-null in STATE_HEADERS and STATE_BODY. | |
let contentLength: number | null = null; // non-null in STATE_BODY. | |
/** | |
* Consumes all complete data in buf or raises an Error. | |
* May leave incomplete data at buf.slice(pos). | |
*/ | |
function processBuf() { | |
// The while(true) condition is reqired | |
// eslint-disable-next-line no-constant-condition | |
while (true) { | |
if (boundary === null) { | |
controller.error( | |
new Error( | |
'Invalid content type for multipart stream: ' + contentType, | |
), | |
); | |
return; | |
} | |
switch (state) { | |
case STATE_BOUNDARY: | |
// Read blank lines (if any) then boundary. | |
while ( | |
buf.length >= pos + blankLine.length && | |
compareArrays(buf.slice(pos, pos + blankLine.length), blankLine) | |
) { | |
pos += blankLine.length; | |
} | |
// Check that it starts with a boundary. | |
if (buf.length < pos + boundary.length) { | |
return; | |
} | |
if ( | |
!compareArrays(buf.slice(pos, pos + boundary.length), boundary) | |
) { | |
throw new Error('bad part boundary'); | |
} | |
pos += boundary.length; | |
state = STATE_HEADERS; | |
headers = new Headers(); | |
break; | |
case STATE_HEADERS: { | |
const cr = buf.indexOf('\r'.charCodeAt(0), pos); | |
if (cr == -1 || buf.length == cr + 1) { | |
return; | |
} | |
if (buf[cr + 1] != '\n'.charCodeAt(0)) { | |
throw new Error('bad part header line (CR without NL)'); | |
} | |
const line = decoder.decode(buf.slice(pos, cr)); | |
pos = cr + 2; | |
if (line == '') { | |
const rawContentLength = headers?.get('Content-Length'); | |
if (rawContentLength == null) { | |
throw new Error('missing/invalid part Content-Length'); | |
} | |
contentLength = parseInt(rawContentLength, 10); | |
if (isNaN(contentLength)) { | |
throw new Error('missing/invalid part Content-Length'); | |
} | |
state = STATE_BODY; | |
break; | |
} | |
const colon = line.indexOf(':'); | |
const name = line.substring(0, colon); | |
if (colon == line.length || line[colon + 1] != ' ') { | |
throw new Error('bad part header line (no ": ")'); | |
} | |
const value = line.substring(colon + 2); | |
headers?.append(name, value); | |
break; | |
} | |
case STATE_BODY: { | |
if (contentLength === null) { | |
throw new Error('content length not set'); | |
} | |
if (buf.length < pos + contentLength) { | |
return; | |
} | |
const body = buf.slice(pos, pos + contentLength); | |
pos += contentLength; | |
controller.enqueue({ | |
headers: headers, | |
body: body, | |
}); | |
headers = null; | |
contentLength = null; | |
state = STATE_BOUNDARY; | |
break; | |
} | |
} | |
} | |
} | |
// The while(true) condition is required | |
// eslint-disable-next-line no-constant-condition | |
while (true) { | |
const {done, value} = await reader.read(); | |
const buffered = buf.length - pos; | |
if (done) { | |
if (state != STATE_BOUNDARY || buffered > 0) { | |
throw Error('multipart stream ended mid-part'); | |
} | |
controller.close(); | |
return; | |
} | |
// Update buf.slice(pos) to include the new data from value. | |
if (buffered == 0) { | |
buf = value; | |
} else { | |
const newLen = buffered + value.length; | |
const newBuf = new Uint8Array(newLen); | |
newBuf.set(buf.slice(pos), 0); | |
newBuf.set(value, buffered); | |
buf = newBuf; | |
} | |
pos = 0; | |
processBuf(); | |
} | |
}, | |
cancel(reason) { | |
return body.cancel(reason); | |
}, | |
}); | |
} | |