Implement CLI store using Deno store

pull/503/head
Zef Hemel 2023-08-04 21:17:36 +02:00
parent ba8625cbf1
commit bc561eb723
5 changed files with 69 additions and 82 deletions

View File

@ -11,7 +11,7 @@ import { AssetBundle } from "../plugos/asset_bundle/bundle.ts";
import { createSandbox } from "../plugos/environments/deno_sandbox.ts"; import { createSandbox } from "../plugos/environments/deno_sandbox.ts";
import { CronHook } from "../plugos/hooks/cron.ts"; import { CronHook } from "../plugos/hooks/cron.ts";
import { EventHook } from "../plugos/hooks/event.ts"; import { EventHook } from "../plugos/hooks/event.ts";
import { JSONKVStore } from "../plugos/lib/kv_store.json_file.ts"; import { DenoKVStore } from "../plugos/lib/kv_store.deno_kv.ts";
import assetSyscalls from "../plugos/syscalls/asset.ts"; import assetSyscalls from "../plugos/syscalls/asset.ts";
import { eventSyscalls } from "../plugos/syscalls/event.ts"; import { eventSyscalls } from "../plugos/syscalls/event.ts";
import { sandboxFetchSyscalls } from "../plugos/syscalls/fetch.ts"; import { sandboxFetchSyscalls } from "../plugos/syscalls/fetch.ts";
@ -44,7 +44,10 @@ export async function runPlug(
const cronHook = new CronHook(system); const cronHook = new CronHook(system);
system.addHook(cronHook); system.addHook(cronHook);
const pageIndexCalls = pageIndexSyscalls("run.db"); const kvStore = new DenoKVStore();
await kvStore.init("run.db");
const pageIndexCalls = pageIndexSyscalls(kvStore);
// TODO: Add endpoint // TODO: Add endpoint
@ -61,7 +64,6 @@ export async function runPlug(
), ),
pageIndexCalls, pageIndexCalls,
); );
const kvStore = new JSONKVStore();
const space = new Space(spacePrimitives, kvStore); const space = new Space(spacePrimitives, kvStore);
// Add syscalls // Add syscalls
@ -118,7 +120,7 @@ export async function runPlug(
const result = await plug.invoke(funcName, args); const result = await plug.invoke(funcName, args);
await system.unloadAll(); await system.unloadAll();
await pageIndexCalls["index.close"]({} as any); await kvStore.delete();
return result; return result;
} }

View File

@ -1,9 +1,12 @@
import { DenoKVStore } from "../../plugos/lib/kv_store.deno_kv.ts";
import { assertEquals } from "../../test_deps.ts"; import { assertEquals } from "../../test_deps.ts";
import { pageIndexSyscalls } from "./index.ts"; import { pageIndexSyscalls } from "./index.ts";
Deno.test("Test KV index", async () => { Deno.test("Test KV index", async () => {
const ctx: any = {}; const ctx: any = {};
const calls = pageIndexSyscalls(); const kv = new DenoKVStore();
await kv.init("test.db");
const calls = pageIndexSyscalls(kv);
await calls["index.set"](ctx, "page", "test", "value"); await calls["index.set"](ctx, "page", "test", "value");
assertEquals(await calls["index.get"](ctx, "page", "test"), "value"); assertEquals(await calls["index.get"](ctx, "page", "test"), "value");
await calls["index.delete"](ctx, "page", "test"); await calls["index.delete"](ctx, "page", "test");
@ -24,11 +27,12 @@ Deno.test("Test KV index", async () => {
}, { key: "random", value: "value3" }]); }, { key: "random", value: "value3" }]);
let results = await calls["index.queryPrefix"](ctx, "attr:"); let results = await calls["index.queryPrefix"](ctx, "attr:");
assertEquals(results.length, 4); assertEquals(results.length, 4);
console.log("here");
await calls["index.clearPageIndexForPage"](ctx, "page"); await calls["index.clearPageIndexForPage"](ctx, "page");
results = await calls["index.queryPrefix"](ctx, "attr:"); results = await calls["index.queryPrefix"](ctx, "attr:");
assertEquals(results.length, 2); assertEquals(results.length, 2);
await calls["index.clearPageIndex"](ctx); await calls["index.clearPageIndex"](ctx);
results = await calls["index.queryPrefix"](ctx, ""); results = await calls["index.queryPrefix"](ctx, "");
assertEquals(results.length, 0); assertEquals(results.length, 0);
await calls["index.close"](ctx); await kv.delete();
}); });

View File

@ -1,13 +1,6 @@
/// <reference lib="deno.unstable" /> import { KVStore } from "../../plugos/lib/kv_store.ts";
import type { SysCallMapping } from "../../plugos/system.ts"; import type { SysCallMapping } from "../../plugos/system.ts";
type Item = {
page: string;
key: string;
value: any;
};
export type KV = { export type KV = {
key: string; key: string;
value: any; value: any;
@ -17,57 +10,50 @@ export type KV = {
// ["index", page, key] -> value // ["index", page, key] -> value
// ["indexByKey", key, page] -> value // ["indexByKey", key, page] -> value
const sep = "!";
/** /**
* Implements the index syscalls using Deno's KV store. * Implements the index syscalls using Deno's KV store.
* @param dbFile * @param dbFile
* @returns * @returns
*/ */
export function pageIndexSyscalls(dbFile?: string): SysCallMapping { export function pageIndexSyscalls(kv: KVStore): SysCallMapping {
const kv = Deno.openKv(dbFile);
const apiObj: SysCallMapping = { const apiObj: SysCallMapping = {
"index.set": async (_ctx, page: string, key: string, value: any) => { "index.set": (_ctx, page: string, key: string, value: any) => {
const res = await (await kv).atomic() return kv.batchSet(
.set(["index", page, key], value) [{
.set(["indexByKey", key, page], value) key: `index${sep}${page}${sep}${key}`,
.commit(); value,
if (!res.ok) { }, {
throw res; key: `indexByKey${sep}${key}${sep}${page}`,
} value,
}],
);
}, },
"index.batchSet": async (_ctx, page: string, kvs: KV[]) => { "index.batchSet": async (_ctx, page: string, kvs: KV[]) => {
// await items.bulkPut(kvs);
for (const { key, value } of kvs) { for (const { key, value } of kvs) {
await apiObj["index.set"](_ctx, page, key, value); await apiObj["index.set"](_ctx, page, key, value);
} }
}, },
"index.delete": async (_ctx, page: string, key: string) => { "index.delete": (_ctx, page: string, key: string) => {
const res = await (await kv).atomic() console.log("delete", page, key);
.delete(["index", page, key]) return kv.batchDelete([
.delete(["indexByKey", key, page]) `index${sep}${page}${sep}${key}`,
.commit(); `indexByKey${sep}${key}${sep}${page}`,
if (!res.ok) { ]);
throw res;
}
}, },
"index.get": async (_ctx, page: string, key: string) => { "index.get": (_ctx, page: string, key: string) => {
return (await (await kv).get(["index", page, key])).value; return kv.get(`index${sep}${page}${sep}${key}`);
}, },
"index.queryPrefix": async (_ctx, prefix: string) => { "index.queryPrefix": async (_ctx, prefix: string) => {
const results: { key: string; page: string; value: any }[] = []; const results: { key: string; page: string; value: any }[] = [];
for await ( for (
const result of (await kv).list({ const result of await kv.queryPrefix(`indexByKey!${prefix}`)
start: ["indexByKey", prefix],
end: [
"indexByKey",
prefix.slice(0, -1) +
// This is a hack to get the next character in the ASCII table (e.g. "a" -> "b")
String.fromCharCode(prefix.charCodeAt(prefix.length - 1) + 1),
],
})
) { ) {
const [_ns, key, page] = result.key.split(sep);
results.push({ results.push({
key: result.key[1] as string, key,
page: result.key[2] as string, page,
value: result.value, value: result.value,
}); });
} }
@ -77,27 +63,22 @@ export function pageIndexSyscalls(dbFile?: string): SysCallMapping {
await apiObj["index.deletePrefixForPage"](ctx, page, ""); await apiObj["index.deletePrefixForPage"](ctx, page, "");
}, },
"index.deletePrefixForPage": async (_ctx, page: string, prefix: string) => { "index.deletePrefixForPage": async (_ctx, page: string, prefix: string) => {
for await ( for (
const result of (await kv).list({ const result of await kv.queryPrefix(
start: ["index", page, prefix], `index${sep}${page}${sep}${prefix}`,
end: ["index", page, prefix + "~"], )
})
) { ) {
await apiObj["index.delete"](_ctx, page, result.key[2]); console.log("GOt back this key to delete", result.key);
const [_ns, page, key] = result.key.split(sep);
await apiObj["index.delete"](_ctx, page, key);
} }
}, },
"index.clearPageIndex": async (ctx) => { "index.clearPageIndex": async (ctx) => {
for await ( for (const result of await kv.queryPrefix(`index${sep}`)) {
const result of (await kv).list({ const [_ns, page, key] = result.key.split(sep);
prefix: ["index"], await apiObj["index.delete"](ctx, page, key);
})
) {
await apiObj["index.delete"](ctx, result.key[1], result.key[2]);
} }
}, },
"index.close": async () => {
(await kv).close();
},
}; };
return apiObj; return apiObj;
} }

View File

@ -30,10 +30,10 @@ Depending on where these attributes appear, they attach to different things. For
Example query: Example query:
<!-- #query page where name = "Attributes" --> <!-- #query page where name = "Attributes" select name, pageAttribute -->
|name |lastModified |contentType |size|perm|pageAttribute| |name |pageAttribute|
|----------|-------------|-------------|----|--|-----| |----------|-----|
|Attributes|1691165890257|text/markdown|1609|rw|hello| |Attributes|hello|
<!-- /query --> <!-- /query -->
This attaches an attribute to an item: This attaches an attribute to an item:
@ -45,7 +45,7 @@ Example query:
<!-- #query item where page = "Attributes" and itemAttribute = "hello" --> <!-- #query item where page = "Attributes" and itemAttribute = "hello" -->
|name|itemAttribute|page |pos | |name|itemAttribute|page |pos |
|----|-----|----------|----| |----|-----|----------|----|
|Item|hello|Attributes|1079| |Item|hello|Attributes|1106|
<!-- /query --> <!-- /query -->
This attaches an attribute to a task: This attaches an attribute to a task:
@ -57,5 +57,5 @@ Example query:
<!-- #query task where page = "Attributes" and taskAttribute = "hello" --> <!-- #query task where page = "Attributes" and taskAttribute = "hello" -->
|name|done |taskAttribute|page |pos | |name|done |taskAttribute|page |pos |
|----|-----|-----|----------|----| |----|-----|-----|----------|----|
|Task|false|hello|Attributes|1355| |Task|false|hello|Attributes|1382|
<!-- /query --> <!-- /query -->

View File

@ -20,21 +20,21 @@ Hadnt we mentioned [[Markdown]] yet? Yeah, thats the markup language you
You will notice this whole page section is wrapped in a strange type of block. This is a SilverBullet specific feature called a [[🔌 Directive]] (in this case `#use`). There are various types of directives, and while were not keeping score, likely the coolest ones are [[🔌 Directive/Query|queries]] — so you should definitely look into those. You will notice this whole page section is wrapped in a strange type of block. This is a SilverBullet specific feature called a [[🔌 Directive]] (in this case `#use`). There are various types of directives, and while were not keeping score, likely the coolest ones are [[🔌 Directive/Query|queries]] — so you should definitely look into those.
Dont believe me, check this out, heres a list of (max 10) pages in your space ordered by last modified date, it updates (somewhat) dynamically 🤯. Create some new pages and come back here to see that it works: Dont believe me, check this out, heres a list of (max 10) pages in your space ordered by name, it updates (somewhat) dynamically 🤯. Create some new pages and come back here to see that it works:
<!-- #query page select name order by lastModified desc limit 10 --> <!-- #query page select name order by name limit 10 -->
|name | |name |
|------------------| |---------------|
|🔌 Directive/Query| |API |
|Attributes | |Attributes |
|Getting Started | |Authelia |
|🔌 Core/Tags | |Authentication |
|🔌 Github | |CHANGELOG |
|🔌 Mattermost | |Cloud Links |
|🔌 Git | |Deployments |
|🔌 Ghost | |Federation |
|🔌 Share | |Frontmatter |
|Install | |Getting Started|
<!-- /query --> <!-- /query -->
That said, the directive used wrapping this page section is `#use` which uses the content of another page as a template and inlines it. Directives recalculate their bodies in two scenarios: That said, the directive used wrapping this page section is `#use` which uses the content of another page as a template and inlines it. Directives recalculate their bodies in two scenarios: