UNPKG

happy-dom

Version:

Happy DOM is a JavaScript implementation of a web browser without its graphical user interface. It includes many web standards from WHATWG DOM and HTML.

98 lines 3.87 kB
import { ReadableStream } from 'stream/web'; import * as PropertySymbol from '../../PropertySymbol.js'; import MultipartReader from './MultipartReader.js'; import DOMException from '../../exception/DOMException.js'; import DOMExceptionNameEnum from '../../exception/DOMExceptionNameEnum.js'; import { Buffer } from 'buffer'; /** * Multipart form data factory. * * Based on: * https://github.com/node-fetch/node-fetch/blob/main/src/utils/multipart-parser.js (MIT) */ export default class MultipartFormDataParser { /** * Returns form data. * * @param body Body. * @param contentType Content type header value. * @returns Form data. */ static async streamToFormData(body, contentType) { if (!/multipart/i.test(contentType)) { throw new DOMException(`Failed to build FormData object: The "content-type" header isn't of type "multipart/form-data".`, DOMExceptionNameEnum.invalidStateError); } const match = contentType.match(/boundary=(?:"([^"]+)"|([^;]+))/i); if (!match) { throw new DOMException(`Failed to build FormData object: The "content-type" header doesn't contain any multipart boundary.`, DOMExceptionNameEnum.invalidStateError); } const bodyReader = body.getReader(); const reader = new MultipartReader(match[1] || match[2]); const chunks = []; let buffer; const bytes = 0; let readResult = await bodyReader.read(); while (!readResult.done) { reader.write(readResult.value); readResult = await bodyReader.read(); } try { buffer = typeof chunks[0] === 'string' ? Buffer.from(chunks.join('')) : Buffer.concat(chunks, bytes); } catch (error) { throw new DOMException(`Could not create Buffer from response body. Error: ${error.message}.`, DOMExceptionNameEnum.invalidStateError); } return { formData: reader.end(), buffer }; } /** * Converts a FormData object to a ReadableStream. * * @param formData FormData. * @returns Stream and type. */ static formDataToStream(formData) { const boundary = '----HappyDOMFormDataBoundary' + Math.random().toString(36); const chunks = []; const prefix = `--${boundary}\r\nContent-Disposition: form-data; name="`; for (const [name, value] of formData) { if (typeof value === 'string') { chunks.push(Buffer.from(`${prefix}${this.escapeName(name)}"\r\n\r\n${value.replace(/\r(?!\n)|(?<!\r)\n/g, '\r\n')}\r\n`)); } else { chunks.push(Buffer.from(`${prefix}${this.escapeName(name)}"; filename="${this.escapeName(value.name, true)}"\r\nContent-Type: ${value.type || 'application/octet-stream'}\r\n\r\n`)); chunks.push(value[PropertySymbol.buffer]); chunks.push(Buffer.from('\r\n')); } } const buffer = Buffer.concat(chunks); return { contentType: `multipart/form-data; boundary=${boundary}`, contentLength: buffer.length, buffer, stream: new ReadableStream({ start(controller) { controller.enqueue(buffer); controller.close(); } }) }; } /** * Escapes a form data entry name. * * @param name Name. * @param filename Whether it is a filename. * @returns Escaped name. */ static escapeName(name, filename = false) { return (filename ? name : name.replace(/\r?\n|\r/g, '\r\n')) .replace(/\n/g, '%0A') .replace(/\r/g, '%0D') .replace(/"/g, '%22'); } } //# sourceMappingURL=MultipartFormDataParser.js.map