feat: add HTTP response and parsing helpers for form-based UX flows

This commit is contained in:
Codex
2026-02-18 12:54:05 +00:00
parent 4e443c3507
commit 8be203fb91
2 changed files with 134 additions and 0 deletions

103
src/lib/http.js Normal file
View File

@@ -0,0 +1,103 @@
"use strict";
function json(status, data, extraHeaders) {
return {
status,
headers: {
"content-type": "application/json; charset=utf-8",
...(extraHeaders || {}),
},
body: JSON.stringify(data),
};
}
function html(status, markup, extraHeaders) {
return {
status,
headers: {
"content-type": "text/html; charset=utf-8",
...(extraHeaders || {}),
},
body: markup,
};
}
function text(status, body, contentType, extraHeaders) {
return {
status,
headers: {
"content-type": contentType || "text/plain; charset=utf-8",
...(extraHeaders || {}),
},
body,
};
}
function redirect(location, extraHeaders) {
return {
status: 303,
headers: {
location,
...(extraHeaders || {}),
},
body: "",
};
}
function parseJSON(rawBody) {
if (!rawBody) {
return {};
}
try {
return JSON.parse(rawBody);
} catch {
throw new Error("invalid_json");
}
}
function parseFormUrlEncoded(rawBody) {
if (!rawBody) {
return {};
}
return String(rawBody)
.split("&")
.filter(Boolean)
.reduce((acc, part) => {
const idx = part.indexOf("=");
if (idx === -1) {
const key = decodeURIComponent(part.replaceAll("+", " "));
acc[key] = "";
return acc;
}
const key = decodeURIComponent(part.slice(0, idx).replaceAll("+", " "));
const value = decodeURIComponent(part.slice(idx + 1).replaceAll("+", " "));
acc[key] = value;
return acc;
}, {});
}
function withQuery(path, queryObj) {
const entries = Object.entries(queryObj || {}).filter(([, v]) => v !== undefined && v !== null && v !== "");
if (entries.length === 0) {
return path;
}
const query = entries
.map(([k, v]) => `${encodeURIComponent(k)}=${encodeURIComponent(String(v))}`)
.join("&");
return `${path}?${query}`;
}
module.exports = {
json,
html,
text,
redirect,
parseJSON,
parseFormUrlEncoded,
withQuery,
};

31
test/http.test.js Normal file
View File

@@ -0,0 +1,31 @@
"use strict";
const test = require("node:test");
const assert = require("node:assert/strict");
const {
redirect,
parseJSON,
parseFormUrlEncoded,
withQuery,
} = require("../src/lib/http");
test("redirect returns 303 and location", () => {
const response = redirect("/app");
assert.equal(response.status, 303);
assert.equal(response.headers.location, "/app");
});
test("parseJSON throws on invalid payload", () => {
assert.throws(() => parseJSON("{invalid}"), /invalid_json/);
});
test("parseFormUrlEncoded parses plus and percent encoding", () => {
const body = "title=Hello+World&body=Line%201%0ALine%202";
const parsed = parseFormUrlEncoded(body);
assert.equal(parsed.title, "Hello World");
assert.equal(parsed.body, "Line 1\nLine 2");
});
test("withQuery appends query params", () => {
assert.equal(withQuery("/login", { returnTo: "/audio/1", flash: "done" }), "/login?returnTo=%2Faudio%2F1&flash=done");
});