import FormData from '../../form-data/FormData.js'; import Stream from 'stream'; import MultipartReader from './MultipartReader.js'; import DOMException from '../../exception/DOMException.js'; import DOMExceptionNameEnum from '../../exception/DOMExceptionNameEnum.js'; /** * 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. */ public static async streamToFormData( body: Stream.Readable, contentType: string ): Promise { 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 reader = new MultipartReader(match[1] || match[2]); for await (const chunk of body) { reader.write(chunk); } return reader.end(); } /** * Converts a FormData object to a ReadableStream. * * @param formData FormData. * @returns Stream and type. */ public static formDataToStream(formData: FormData): { contentType: string; contentLength: number; buffer: Buffer; stream: Stream.Readable; } { const boundary = '----HappyDOMFormDataBoundary' + Math.random().toString(36); const chunks: Buffer[] = []; 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)|(?