guardia-messenger/node_modules/metro/src/Server/MultipartResponse.js
DESKTOP-TKLFCPRython f29f525c77 refactor: 101.79.17.164 → zioinfo.co.kr 전체 도메인 변환 + Manager UI 배포
- 37개 파일 IP → zioinfo.co.kr 치환 (소스/매뉴얼/설정/하네스)
- Manager DrConsole/NetworkConsole/CsapConsole 빌드 + /var/www/manager/ 배포
- 테스트: Manager HTTP 200, ITSM 신규 API 7개 전체 200

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-31 10:09:17 +09:00

65 lines
1.5 KiB
JavaScript

"use strict";
const accepts = require("accepts");
const CRLF = "\r\n";
const BOUNDARY = "3beqjf3apnqeu3h5jqorms4i";
class MultipartResponse {
static wrapIfSupported(req, res) {
if (accepts(req).types().includes("multipart/mixed")) {
return new MultipartResponse(res);
}
return res;
}
static serializeHeaders(headers) {
return Object.keys(headers)
.map((key) => `${key}: ${headers[key]}`)
.join(CRLF);
}
constructor(res) {
this.res = res;
this.headers = {};
res.writeHead(200, {
"Content-Type": `multipart/mixed; boundary="${BOUNDARY}"`,
});
res.write(
"If you are seeing this, your client does not support multipart response"
);
}
writeChunk(headers, data, isLast = false) {
if (this.res.finished) {
return;
}
this.res.write(`${CRLF}--${BOUNDARY}${CRLF}`);
if (headers) {
this.res.write(MultipartResponse.serializeHeaders(headers) + CRLF + CRLF);
}
if (data != null) {
this.res.write(data);
}
if (isLast) {
this.res.write(`${CRLF}--${BOUNDARY}--${CRLF}`);
}
}
writeHead(status, headers) {
this.setHeader("X-Http-Status", status);
if (!headers) {
return;
}
for (const key in headers) {
this.setHeader(key, headers[key]);
}
}
setHeader(name, value) {
this.headers[name] = value;
}
end(data) {
this.writeChunk(this.headers, data, true);
this.res.end();
}
once(name, fn) {
this.res.once(name, fn);
return this;
}
}
module.exports = MultipartResponse;