Page MenuHomeSealhub

parse-body.test.ts
No OneTemporary

parse-body.test.ts

import assert from "assert";
import {
TestAppConstructor,
withRunningApp,
withStoppedApp,
} from "../test_utils/with-test-app.js";
import Field from "../chip-types/field.js";
import { Collection, FieldTypes } from "../main.js";
import asyncRequest from "../test_utils/async-request.js";
import parseBody from "./parse-body.js";
import JsonObject from "../app/base-chips/field-types/json-object.js";
import { TestApp } from "../test_utils/test-app.js";
function extend(t: TestAppConstructor<TestApp>) {
class ArrayOfObjects extends Field {
typeName = "array-of-objects";
async isProperValue(_: any, new_value: any) {
if (!Array.isArray(new_value)) {
return Field.invalid("It should be array of objects.");
}
for (const value of new_value) {
if (typeof value !== "object") {
return Field.invalid("One of array item isn't object.");
}
}
return Field.valid();
}
}
const complex_data = new (class ComplexData extends Collection {
name = "complex-data";
fields = {
body: new ArrayOfObjects(),
source: new FieldTypes.Image(),
};
})();
const any_json = new (class ComplexData extends Collection {
name = "any_json";
fields = {
body: new JsonObject(),
source: new FieldTypes.Image(),
};
})();
const strings = new (class Strings extends Collection {
name = "strings";
fields = {
title: new FieldTypes.Text(),
};
})();
return class extends t {
collections = {
...TestApp.BaseCollections,
strings,
"complex-data": complex_data,
any_json,
};
};
}
describe("parseBody", () => {
it("handles complex data sent as multipart/form-data", async () => {
await withRunningApp(extend, async ({ port }) => {
// PNG file is empty but it doesnt matter for the test
const form_data =
'------------------------------4ebf00fbcf09\r\nContent-Disposition: form-data; name="source"; filename="test.png"\r\nContent-Type: image/png\r\n\r\nPNG\r\n\r\n\r\n------------------------------4ebf00fbcf09\r\nContent-Disposition: form-data; name="body"; filename="blob"\r\nContent-Type: application/json\r\n\r\n[["Foo", {"Bar": "baz"}]]\r\n------------------------------4ebf00fbcf09--\r\n';
const options = {
hostname: "localhost",
port: port,
path: "/api/v1/collections/complex-data",
method: "POST",
headers: {
"Content-Type":
"multipart/form-data; boundary=----------------------------4ebf00fbcf09",
},
};
const response = (await asyncRequest(options, form_data)) as {
source: Record<string, unknown>;
body: Array<Record<string, unknown>>;
};
const { body, source } = response;
assert.strict.deepStrictEqual(body, [["Foo", { Bar: "baz" }]]);
assert.deepStrictEqual(source?.filename, "test.png");
});
});
it("includes the url query params as parts of body, if they don't overlap", async () => {
await withStoppedApp(extend, async (test) => {
test.app.HTTPServer.router.post(
"/echo-data",
parseBody(),
async (ctx) => {
ctx.body = ctx.$body;
}
);
await test.app.start();
const response = await test.rest_api.post(
"/echo-data?token=abc&json_data=ignore",
{
json_data: "123",
}
);
assert.strictEqual(response.json_data, "123");
assert.strictEqual(response.token, "abc");
await test.app.stop();
});
});
it("handles arrays data sent within multipart/form-data", async () => {
await withRunningApp(extend, async ({ port }) => {
// PNG file is empty but it doesnt matter for the test
const response = await fetch(
`http://localhost:${port}/api/v1/collections/any_json`,
{
credentials: "include",
headers: {
"Content-Type":
"multipart/form-data; boundary=209086842812694783493155782262",
},
body: `--209086842812694783493155782262\r\nContent-Disposition: form-data; name="body.component_args[table][rows][0][cells][0][color]"\r\n\r\nred\r\n--209086842812694783493155782262\r\nContent-Disposition: form-data; name="body.component_args[table][rows][0][cells][0][word]"\r\n\r\nbanana\r\n--209086842812694783493155782262\r\nContent-Disposition: form-data; name="body.component_args[table][rows][1][cells][0][image].old"\r\n\r\n{"persistent":true,"file_id":"4379b88e-d6f3-43d9-b87b-569d6179be61.data"}\r\n--209086842812694783493155782262\r\nContent-Disposition: form-data; name="body.component_args[table][rows][1][cells][0][color]"\r\n\r\ngreen\r\n--209086842812694783493155782262\r\nContent-Disposition: form-data; name="body.component_args[table][rows][2][cells][0][word]"\r\n\r\npineapple\r\n--209086842812694783493155782262\r\nContent-Disposition: form-data; name="body.component_args[table][rows][2][cells][0][image].old"\r\n\r\n{"persistent":true,"file_id":"2ee38767-af26-492f-997f-e3797917761e.data"}\r\n--209086842812694783493155782262--\r\n`,
method: "POST",
mode: "cors",
}
);
const { body } = (await response.json()) as any;
assert.strict.deepStrictEqual(body, {
component_args: {
table: {
rows: [
{
cells: [
{
color: "red",
word: "banana",
},
],
},
{
cells: [
{
color: "green",
image: {
old: '{"persistent":true,"file_id":"4379b88e-d6f3-43d9-b87b-569d6179be61.data"}',
},
},
],
},
{
cells: [
{
image: {
old: '{"persistent":true,"file_id":"2ee38767-af26-492f-997f-e3797917761e.data"}',
},
word: "pineapple",
},
],
},
],
},
},
});
});
});
});

File Metadata

Mime Type
text/x-java
Expires
Wed, Aug 13, 19:36 (6 h, 12 m)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
872432
Default Alt Text
parse-body.test.ts (5 KB)

Event Timeline