2024-03-16 22:29:24 +08:00
|
|
|
import "fake-indexeddb/auto";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { IndexedDBKvPrimitives } from "../data/indexeddb_kv_primitives.ts";
|
|
|
|
import { DenoKvPrimitives } from "../data/deno_kv_primitives.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { KvPrimitives } from "../data/kv_primitives.ts";
|
2024-07-30 23:24:17 +08:00
|
|
|
import { assertEquals, assertThrows } from "@std/assert";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { PrefixedKvPrimitives } from "../data/prefixed_kv_primitives.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { Query } from "../../plug-api/types.ts";
|
2024-08-02 22:47:36 +08:00
|
|
|
import { DataStore } from "$lib/data/datastore.ts";
|
2023-09-04 03:15:17 +08:00
|
|
|
|
|
|
|
async function test(db: KvPrimitives) {
|
2024-01-15 23:43:12 +08:00
|
|
|
const datastore = new DataStore(new PrefixedKvPrimitives(db, ["ds"]), {
|
2023-10-03 20:16:33 +08:00
|
|
|
count: (arr: any[]) => arr.length,
|
2024-02-03 02:19:07 +08:00
|
|
|
$query: (_query: Query) => {
|
|
|
|
return Promise.resolve([]);
|
|
|
|
},
|
2023-10-03 20:16:33 +08:00
|
|
|
});
|
|
|
|
await datastore.set(["user", "peter"], { name: "Peter" });
|
|
|
|
await datastore.set(["user", "hank"], { name: "Hank" });
|
|
|
|
let results = await datastore.query({
|
2023-09-04 03:15:17 +08:00
|
|
|
prefix: ["user"],
|
2023-10-03 20:16:33 +08:00
|
|
|
filter: ["=", ["attr", "name"], ["string", "Peter"]],
|
2023-09-04 03:15:17 +08:00
|
|
|
});
|
|
|
|
assertEquals(results, [{ key: ["user", "peter"], value: { name: "Peter" } }]);
|
2023-10-03 22:54:03 +08:00
|
|
|
|
|
|
|
assertEquals(
|
|
|
|
[{ key: ["user", "hank"], value: { name: "Hank" } }, {
|
|
|
|
key: ["user", "peter"],
|
|
|
|
value: { name: "Peter" },
|
|
|
|
}],
|
|
|
|
await datastore.query({
|
|
|
|
prefix: ["user"],
|
|
|
|
orderBy: [{ expr: ["attr", "name"], desc: false }],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
assertEquals(
|
|
|
|
[{ key: ["user", "peter"], value: { name: "Peter" } }, {
|
|
|
|
key: ["user", "hank"],
|
|
|
|
value: { name: "Hank" },
|
|
|
|
}],
|
|
|
|
await datastore.query({
|
|
|
|
prefix: ["user"],
|
|
|
|
orderBy: [{ expr: ["attr", "name"], desc: true }],
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
await datastore.batchSet<any>([
|
2023-09-04 03:15:17 +08:00
|
|
|
{ key: ["kv", "name"], value: "Zef" },
|
|
|
|
{ key: ["kv", "data"], value: new Uint8Array([1, 2, 3]) },
|
|
|
|
{
|
|
|
|
key: ["kv", "complicated"],
|
|
|
|
value: {
|
|
|
|
name: "Frank",
|
|
|
|
parents: ["John", "Jane"],
|
|
|
|
address: {
|
|
|
|
street: "123 Main St",
|
|
|
|
city: "San Francisco",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
]);
|
2023-10-03 20:16:33 +08:00
|
|
|
assertEquals(await datastore.get(["kv", "name"]), "Zef");
|
|
|
|
assertEquals(await datastore.get(["kv", "data"]), new Uint8Array([1, 2, 3]));
|
|
|
|
results = await datastore.query({
|
2023-09-04 03:15:17 +08:00
|
|
|
prefix: ["kv"],
|
2023-10-03 20:16:33 +08:00
|
|
|
filter: ["=~", ["attr", ""], ["regexp", "Z.f", "i"]],
|
2023-09-04 03:15:17 +08:00
|
|
|
});
|
|
|
|
assertEquals(results, [{ key: ["kv", "name"], value: "Zef" }]);
|
2023-10-03 20:16:33 +08:00
|
|
|
results = await datastore.query({
|
2023-09-04 03:15:17 +08:00
|
|
|
prefix: ["kv"],
|
2023-10-03 20:16:33 +08:00
|
|
|
filter: ["and", ["=", ["attr", "parents"], ["string", "John"]], [
|
2023-09-04 03:15:17 +08:00
|
|
|
"=",
|
2023-10-03 20:16:33 +08:00
|
|
|
["attr", ["attr", "address"], "city"],
|
|
|
|
["string", "San Francisco"],
|
2023-09-04 03:15:17 +08:00
|
|
|
]],
|
2023-10-03 20:16:33 +08:00
|
|
|
select: [
|
|
|
|
{ name: "parents" },
|
|
|
|
{
|
|
|
|
name: "name",
|
|
|
|
expr: ["+", ["attr", "name"], ["string", "!"]],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "parentCount",
|
|
|
|
expr: ["call", "count", [["attr", "parents"]]],
|
|
|
|
},
|
|
|
|
],
|
2023-09-07 18:33:25 +08:00
|
|
|
});
|
2023-10-03 20:16:33 +08:00
|
|
|
assertEquals(results.length, 1);
|
2023-09-07 18:33:25 +08:00
|
|
|
assertEquals(results[0], {
|
|
|
|
key: ["kv", "complicated"],
|
2023-10-03 20:16:33 +08:00
|
|
|
value: { name: "Frank!", parentCount: 2, parents: ["John", "Jane"] },
|
2023-09-04 03:15:17 +08:00
|
|
|
});
|
2024-02-03 02:19:07 +08:00
|
|
|
|
|
|
|
assertEquals(
|
|
|
|
await datastore.query({
|
|
|
|
prefix: ["kv"],
|
|
|
|
limit: ["number", 1],
|
|
|
|
select: [
|
|
|
|
{
|
|
|
|
name: "random",
|
|
|
|
expr: ["query", { querySource: "bla" }],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
[{ key: ["kv", "complicated"], value: { random: [] } }],
|
|
|
|
);
|
2024-07-30 03:21:16 +08:00
|
|
|
|
|
|
|
// Test object enrichment
|
2024-08-02 22:47:36 +08:00
|
|
|
datastore.objectDecorators = [{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
|
|
|
pageDecoration: {
|
|
|
|
prefix: ["string", "🧑 "],
|
|
|
|
},
|
|
|
|
list: ["array", [["number", 2]]],
|
|
|
|
},
|
|
|
|
}];
|
|
|
|
|
|
|
|
const simplePage = {
|
|
|
|
name: "My Page",
|
|
|
|
pageDecoration: {
|
|
|
|
hide: true,
|
|
|
|
},
|
|
|
|
list: [1],
|
|
|
|
};
|
|
|
|
const enrichedPage = datastore.enrichObject(simplePage);
|
|
|
|
assertEquals(enrichedPage.name, "My Page");
|
|
|
|
assertEquals(enrichedPage.pageDecoration.hide, true);
|
|
|
|
assertEquals(enrichedPage.pageDecoration.prefix, "🧑 ");
|
|
|
|
assertEquals(enrichedPage.list, [1, 2]);
|
|
|
|
// console.log("Enriched page", enrichedPage);
|
|
|
|
|
|
|
|
// More complicated case
|
|
|
|
datastore.objectDecorators = [
|
2024-07-30 03:21:16 +08:00
|
|
|
{ // fullName
|
|
|
|
where: ["=", ["attr", "tags"], ["string", "person"]],
|
|
|
|
attributes: {
|
|
|
|
fullName: ["+", ["+", ["attr", "firstName"], ["string", " "]], [
|
|
|
|
"attr",
|
|
|
|
"lastName",
|
|
|
|
]],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
where: ["=", ["attr", "tags"], ["string", "person"]],
|
|
|
|
attributes: {
|
2024-08-02 22:47:36 +08:00
|
|
|
pageDecoration: {
|
|
|
|
prefix: {
|
|
|
|
bla: {
|
|
|
|
doh: ["+", ["string", "🧑 "], ["attr", "fullName"]],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2024-07-30 03:21:16 +08:00
|
|
|
},
|
|
|
|
},
|
2024-07-30 17:46:42 +08:00
|
|
|
// Test extending existing array attributes
|
|
|
|
{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
2024-08-02 22:47:36 +08:00
|
|
|
listAttribute: ["array", [["string", "newValue1"]]],
|
|
|
|
nested: {
|
|
|
|
listAttribute1: ["array", [["string", "newValue 1"]]],
|
|
|
|
},
|
2024-07-30 17:46:42 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
2024-08-02 22:47:36 +08:00
|
|
|
listAttribute: ["array", [["string", "newValue2"]]],
|
|
|
|
nested: {
|
|
|
|
listAttribute1: ["array", [["string", "newValue 2"]]],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
|
|
|
nested: {
|
|
|
|
listAttribute1: ["array", [["string", "newValue 3"]]],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
|
|
|
nested: {
|
|
|
|
deeper: {
|
|
|
|
listAttribute2: ["array", [["string", "newValue 1"]]],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
|
|
|
nested: {
|
|
|
|
deeper: {
|
|
|
|
listAttribute2: ["array", [["string", "newValue 2"]]],
|
|
|
|
},
|
|
|
|
},
|
2024-07-30 17:46:42 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
// Test not being able to override existing attributes
|
|
|
|
{
|
|
|
|
where: ["boolean", true],
|
|
|
|
attributes: {
|
2024-08-02 22:47:36 +08:00
|
|
|
lastName: ["string", "Shouldn't be set"],
|
2024-07-30 17:46:42 +08:00
|
|
|
},
|
|
|
|
},
|
2024-07-30 03:21:16 +08:00
|
|
|
{
|
|
|
|
where: ["=", ["attr", "tags"], ["string", "person"]],
|
|
|
|
attributes: {
|
2024-08-02 22:47:36 +08:00
|
|
|
existingObjAttribute: {
|
|
|
|
another: ["string", "value"],
|
|
|
|
},
|
2024-07-30 03:21:16 +08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
2024-08-02 22:47:36 +08:00
|
|
|
let obj: Record<string, any> = {
|
2024-07-30 03:21:16 +08:00
|
|
|
firstName: "Pete",
|
|
|
|
lastName: "Smith",
|
2024-08-02 22:47:36 +08:00
|
|
|
pageDecoration: {},
|
2024-07-30 03:21:16 +08:00
|
|
|
existingObjAttribute: {
|
|
|
|
something: true,
|
|
|
|
},
|
|
|
|
tags: ["person"],
|
|
|
|
};
|
|
|
|
const pristineCopy = JSON.parse(JSON.stringify(obj));
|
|
|
|
|
2024-08-02 22:47:36 +08:00
|
|
|
obj = datastore.enrichObject(obj);
|
2024-07-30 17:46:42 +08:00
|
|
|
// console.log("Enrhiched", obj);
|
2024-07-30 03:21:16 +08:00
|
|
|
assertEquals(obj.fullName, "Pete Smith");
|
2024-07-30 17:46:42 +08:00
|
|
|
assertEquals(obj.lastName, "Smith");
|
2024-08-02 22:47:36 +08:00
|
|
|
assertEquals(obj.pageDecoration.prefix.bla.doh, "🧑 Pete Smith");
|
2024-07-30 03:21:16 +08:00
|
|
|
assertEquals(obj.existingObjAttribute.something, true);
|
|
|
|
assertEquals(obj.existingObjAttribute.another, "value");
|
2024-07-30 17:46:42 +08:00
|
|
|
assertEquals(obj.listAttribute, ["newValue1", "newValue2"]);
|
2024-08-02 22:47:36 +08:00
|
|
|
assertEquals(obj.nested.listAttribute1, [
|
|
|
|
"newValue 1",
|
|
|
|
"newValue 2",
|
|
|
|
"newValue 3",
|
|
|
|
]);
|
2024-07-30 03:21:16 +08:00
|
|
|
|
|
|
|
// And now let's clean it again
|
|
|
|
datastore.cleanEnrichedObject(obj);
|
|
|
|
|
|
|
|
assertEquals(obj, pristineCopy);
|
|
|
|
|
|
|
|
// Validate no async functions are called in the object enrichment
|
2024-08-02 22:47:36 +08:00
|
|
|
datastore.objectDecorators = [
|
2024-07-30 03:21:16 +08:00
|
|
|
{
|
|
|
|
where: ["call", "$query", []],
|
|
|
|
attributes: {},
|
|
|
|
},
|
|
|
|
];
|
2024-07-30 17:46:42 +08:00
|
|
|
|
|
|
|
assertThrows(() => {
|
|
|
|
datastore.enrichObject({});
|
|
|
|
});
|
2023-09-04 03:15:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Deno.test("Test Deno KV DataStore", async () => {
|
|
|
|
const tmpFile = await Deno.makeTempFile();
|
2023-10-03 20:16:33 +08:00
|
|
|
const db = new DenoKvPrimitives(await Deno.openKv(tmpFile));
|
2023-09-04 03:15:17 +08:00
|
|
|
await test(db);
|
|
|
|
db.close();
|
|
|
|
await Deno.remove(tmpFile);
|
|
|
|
});
|
|
|
|
|
|
|
|
Deno.test("Test IndexDB DataStore", {
|
|
|
|
sanitizeResources: false,
|
|
|
|
sanitizeOps: false,
|
|
|
|
}, async () => {
|
|
|
|
const db = new IndexedDBKvPrimitives("test");
|
|
|
|
await db.init();
|
|
|
|
await test(db);
|
|
|
|
db.close();
|
|
|
|
});
|