Materialized query parser + data indexing and querying
parent
b3c3302970
commit
a24eaaf4b4
|
@ -11,7 +11,7 @@ syntax:
|
||||||
regex: "@[A-Za-z\\.]+"
|
regex: "@[A-Za-z\\.]+"
|
||||||
styles:
|
styles:
|
||||||
color: blue
|
color: blue
|
||||||
URL:
|
NakedURL:
|
||||||
firstCharacters:
|
firstCharacters:
|
||||||
- "h"
|
- "h"
|
||||||
regex: "https?:\\/\\/[-a-zA-Z0-9@:%._\\+~#=]{1,256}([-a-zA-Z0-9()@:%_\\+.~#?&=\\/]*)"
|
regex: "https?:\\/\\/[-a-zA-Z0-9@:%._\\+~#=]{1,256}([-a-zA-Z0-9()@:%_\\+.~#?&=\\/]*)"
|
||||||
|
|
|
@ -3,11 +3,14 @@ import { whiteOutQueries } from "../query/materialized_queries";
|
||||||
|
|
||||||
import { batchSet } from "plugos-silverbullet-syscall/index";
|
import { batchSet } from "plugos-silverbullet-syscall/index";
|
||||||
import { parseMarkdown } from "plugos-silverbullet-syscall/markdown";
|
import { parseMarkdown } from "plugos-silverbullet-syscall/markdown";
|
||||||
import { collectNodesMatching, ParseTree, renderToText } from "../../common/tree";
|
import { collectNodesOfType, ParseTree, renderToText } from "../../common/tree";
|
||||||
|
|
||||||
type Item = {
|
export type Item = {
|
||||||
item: string;
|
name: string;
|
||||||
nested?: string;
|
nested?: string;
|
||||||
|
// Not stored in DB
|
||||||
|
page?: string;
|
||||||
|
pos?: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export async function indexItems({ name, text }: IndexEvent) {
|
export async function indexItems({ name, text }: IndexEvent) {
|
||||||
|
@ -17,7 +20,7 @@ export async function indexItems({ name, text }: IndexEvent) {
|
||||||
console.log("Indexing items", name);
|
console.log("Indexing items", name);
|
||||||
let mdTree = await parseMarkdown(text);
|
let mdTree = await parseMarkdown(text);
|
||||||
|
|
||||||
let coll = collectNodesMatching(mdTree, (n) => n.type === "ListItem");
|
let coll = collectNodesOfType(mdTree, "ListItem");
|
||||||
|
|
||||||
coll.forEach((n) => {
|
coll.forEach((n) => {
|
||||||
if (!n.children) {
|
if (!n.children) {
|
||||||
|
@ -34,7 +37,7 @@ export async function indexItems({ name, text }: IndexEvent) {
|
||||||
}
|
}
|
||||||
let item = textNodes.map(renderToText).join("").trim();
|
let item = textNodes.map(renderToText).join("").trim();
|
||||||
let value: Item = {
|
let value: Item = {
|
||||||
item,
|
name: item,
|
||||||
};
|
};
|
||||||
if (nested) {
|
if (nested) {
|
||||||
value.nested = nested;
|
value.nested = nested;
|
||||||
|
|
|
@ -20,6 +20,7 @@ async function actionClickOrActionEnter(mdTree: ParseTree | null) {
|
||||||
await navigateTo(pageLink, +pos);
|
await navigateTo(pageLink, +pos);
|
||||||
break;
|
break;
|
||||||
case "URL":
|
case "URL":
|
||||||
|
case "NakedURL":
|
||||||
await openUrl(mdTree.children![0].text!);
|
await openUrl(mdTree.children![0].text!);
|
||||||
break;
|
break;
|
||||||
case "Link":
|
case "Link":
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -0,0 +1,54 @@
|
||||||
|
// Index key space:
|
||||||
|
// data:page@pos
|
||||||
|
|
||||||
|
import { IndexEvent } from "../../webapp/app_event";
|
||||||
|
import { whiteOutQueries } from "./materialized_queries";
|
||||||
|
import { batchSet } from "plugos-silverbullet-syscall";
|
||||||
|
import { parseMarkdown } from "plugos-silverbullet-syscall/markdown";
|
||||||
|
import { collectNodesOfType, findNodeOfType } from "../../common/tree";
|
||||||
|
import YAML from "yaml";
|
||||||
|
|
||||||
|
export async function indexData({ name, text }: IndexEvent) {
|
||||||
|
let e;
|
||||||
|
text = whiteOutQueries(text);
|
||||||
|
console.log("Now data indexing", name);
|
||||||
|
console.log("Indexing items", name);
|
||||||
|
let mdTree = await parseMarkdown(text);
|
||||||
|
|
||||||
|
let dataObjects: { key: string; value: Object }[] = [];
|
||||||
|
|
||||||
|
collectNodesOfType(mdTree, "FencedCode").forEach((t) => {
|
||||||
|
let codeInfoNode = findNodeOfType(t, "CodeInfo");
|
||||||
|
if (!codeInfoNode) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (codeInfoNode.children![0].text !== "data") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let codeTextNode = findNodeOfType(t, "CodeText");
|
||||||
|
if (!codeTextNode) {
|
||||||
|
// Honestly, this shouldn't happen
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let codeText = codeTextNode.children![0].text!;
|
||||||
|
try {
|
||||||
|
// We support multiple YAML documents in one block
|
||||||
|
for (let doc of YAML.parseAllDocuments(codeText)) {
|
||||||
|
if (!doc.contents) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
console.log(doc.contents.toJSON());
|
||||||
|
dataObjects.push({
|
||||||
|
key: `data:${name}@${t.from! + doc.range[0]}`,
|
||||||
|
value: doc.contents.toJSON(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// console.log("Parsed data", parsedData);
|
||||||
|
} catch (e) {
|
||||||
|
console.error("Could not parse data", codeText, "error:", e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
console.log("Found", dataObjects, "data objects");
|
||||||
|
await batchSet(name, dataObjects);
|
||||||
|
}
|
|
@ -24,13 +24,13 @@ test("Test parser", () => {
|
||||||
value: "{{today}}",
|
value: "{{today}}",
|
||||||
});
|
});
|
||||||
|
|
||||||
let parsedQuery2 = parseQuery(`page where name like "interview/%"`);
|
let parsedQuery2 = parseQuery(`page where name =~ /interview\\/.*/"`);
|
||||||
expect(parsedQuery2.table).toBe("page");
|
expect(parsedQuery2.table).toBe("page");
|
||||||
expect(parsedQuery2.filter.length).toBe(1);
|
expect(parsedQuery2.filter.length).toBe(1);
|
||||||
expect(parsedQuery2.filter[0]).toStrictEqual({
|
expect(parsedQuery2.filter[0]).toStrictEqual({
|
||||||
op: "like",
|
op: "=~",
|
||||||
prop: "name",
|
prop: "name",
|
||||||
value: "interview/%",
|
value: /interview\/.*/,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -42,29 +42,36 @@ test("Test performing the queries", () => {
|
||||||
{ name: "Angie", age: 28 },
|
{ name: "Angie", age: 28 },
|
||||||
];
|
];
|
||||||
|
|
||||||
expect(applyQuery(`page where name like "interview/%"`, data)).toStrictEqual([
|
|
||||||
{ name: "interview/My Interview", lastModified: 1 },
|
|
||||||
{ name: "interview/My Interview 2", lastModified: 2 },
|
|
||||||
]);
|
|
||||||
expect(
|
expect(
|
||||||
applyQuery(`page where name like "interview/%" order by lastModified`, data)
|
applyQuery(parseQuery(`page where name =~ /interview\\/.*/`), data)
|
||||||
).toStrictEqual([
|
).toStrictEqual([
|
||||||
{ name: "interview/My Interview", lastModified: 1 },
|
{ name: "interview/My Interview", lastModified: 1 },
|
||||||
{ name: "interview/My Interview 2", lastModified: 2 },
|
{ name: "interview/My Interview 2", lastModified: 2 },
|
||||||
]);
|
]);
|
||||||
expect(
|
expect(
|
||||||
applyQuery(
|
applyQuery(
|
||||||
`page where name like "interview/%" order by lastModified desc`,
|
parseQuery(`page where name =~ /interview\\/.*/ order by lastModified`),
|
||||||
|
data
|
||||||
|
)
|
||||||
|
).toStrictEqual([
|
||||||
|
{ name: "interview/My Interview", lastModified: 1 },
|
||||||
|
{ name: "interview/My Interview 2", lastModified: 2 },
|
||||||
|
]);
|
||||||
|
expect(
|
||||||
|
applyQuery(
|
||||||
|
parseQuery(
|
||||||
|
`page where name =~ /interview\\/.*/ order by lastModified desc`
|
||||||
|
),
|
||||||
data
|
data
|
||||||
)
|
)
|
||||||
).toStrictEqual([
|
).toStrictEqual([
|
||||||
{ name: "interview/My Interview 2", lastModified: 2 },
|
{ name: "interview/My Interview 2", lastModified: 2 },
|
||||||
{ name: "interview/My Interview", lastModified: 1 },
|
{ name: "interview/My Interview", lastModified: 1 },
|
||||||
]);
|
]);
|
||||||
expect(applyQuery(`page where age > 30`, data)).toStrictEqual([
|
expect(applyQuery(parseQuery(`page where age > 30`), data)).toStrictEqual([
|
||||||
{ name: "Pete", age: 38 },
|
{ name: "Pete", age: 38 },
|
||||||
]);
|
]);
|
||||||
expect(applyQuery(`page where age > 28 and age < 38`, data)).toStrictEqual(
|
expect(
|
||||||
[]
|
applyQuery(parseQuery(`page where age > 28 and age < 38`), data)
|
||||||
);
|
).toStrictEqual([]);
|
||||||
});
|
});
|
||||||
|
|
|
@ -31,6 +31,8 @@ export function parseQuery(query: string): ParsedQuery {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// console.log("Parsed", JSON.stringify(n, null, 2));
|
||||||
|
|
||||||
let queryNode = n.children![0];
|
let queryNode = n.children![0];
|
||||||
let parsedQuery: ParsedQuery = {
|
let parsedQuery: ParsedQuery = {
|
||||||
table: queryNode.children![0].children![0].text!,
|
table: queryNode.children![0].children![0].text!,
|
||||||
|
@ -64,6 +66,10 @@ export function parseQuery(query: string): ParsedQuery {
|
||||||
case "Name":
|
case "Name":
|
||||||
val = valNode.children![0].text!;
|
val = valNode.children![0].text!;
|
||||||
break;
|
break;
|
||||||
|
case "Regex":
|
||||||
|
val = valNode.children![0].text!;
|
||||||
|
val = new RegExp(val.substring(1, val.length - 1));
|
||||||
|
break;
|
||||||
case "String":
|
case "String":
|
||||||
val = valNode.children![0].text!;
|
val = valNode.children![0].text!;
|
||||||
val = val.substring(1, val.length - 1);
|
val = val.substring(1, val.length - 1);
|
||||||
|
@ -80,55 +86,53 @@ export function parseQuery(query: string): ParsedQuery {
|
||||||
return parsedQuery;
|
return parsedQuery;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function applyQuery(query: string, records: any[]): any {
|
export function applyQuery<T>(parsedQuery: ParsedQuery, records: T[]): T[] {
|
||||||
const parsedQuery = parseQuery(query);
|
|
||||||
|
|
||||||
let resultRecords: any[] = [];
|
let resultRecords: any[] = [];
|
||||||
if (parsedQuery.filter.length === 0) {
|
if (parsedQuery.filter.length === 0) {
|
||||||
resultRecords = records.slice();
|
resultRecords = records.slice();
|
||||||
} else {
|
} else {
|
||||||
recordLoop: for (let record of records) {
|
recordLoop: for (let record of records) {
|
||||||
|
const recordAny: any = record;
|
||||||
for (let { op, prop, value } of parsedQuery.filter) {
|
for (let { op, prop, value } of parsedQuery.filter) {
|
||||||
switch (op) {
|
switch (op) {
|
||||||
case "=":
|
case "=":
|
||||||
if (!(record[prop] === value)) {
|
if (!(recordAny[prop] === value)) {
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "!=":
|
case "!=":
|
||||||
if (!(record[prop] !== value)) {
|
if (!(recordAny[prop] !== value)) {
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "<":
|
case "<":
|
||||||
if (!(record[prop] < value)) {
|
if (!(recordAny[prop] < value)) {
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "<=":
|
case "<=":
|
||||||
if (!(record[prop] <= value)) {
|
if (!(recordAny[prop] <= value)) {
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ">":
|
case ">":
|
||||||
if (!(record[prop] > value)) {
|
if (!(recordAny[prop] > value)) {
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ">=":
|
case ">=":
|
||||||
if (!(record[prop] >= value)) {
|
if (!(recordAny[prop] >= value)) {
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "like":
|
case "=~":
|
||||||
let re = new RegExp(value.replaceAll("%", ".*"));
|
if (!value.exec(recordAny[prop])) {
|
||||||
if (!re.exec(record[prop])) {
|
|
||||||
continue recordLoop;
|
continue recordLoop;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
resultRecords.push(record);
|
resultRecords.push(recordAny);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Now the sorting
|
// Now the sorting
|
||||||
|
|
|
@ -4,15 +4,17 @@ import { listPages, readPage, writePage } from "plugos-silverbullet-syscall/spac
|
||||||
import { invokeFunction } from "plugos-silverbullet-syscall/system";
|
import { invokeFunction } from "plugos-silverbullet-syscall/system";
|
||||||
import { scanPrefixGlobal } from "plugos-silverbullet-syscall";
|
import { scanPrefixGlobal } from "plugos-silverbullet-syscall";
|
||||||
import { niceDate } from "../core/dates";
|
import { niceDate } from "../core/dates";
|
||||||
|
import { applyQuery, parseQuery } from "./engine";
|
||||||
|
import { PageMeta } from "../../common/types";
|
||||||
|
import type { Task } from "../tasks/task";
|
||||||
|
import { Item } from "../core/item";
|
||||||
|
import YAML from "yaml";
|
||||||
|
|
||||||
export const queryRegex =
|
export const queryRegex =
|
||||||
/(<!--\s*#query\s+(?<table>\w+)\s*(filter\s+["'“”‘’](?<filter>[^"'“”‘’]+)["'“”‘’])?\s*\s*(order by\s+(?<orderBy>\w+)(?<orderDesc>\s+desc)?)?(group by\s+(?<groupBy>\w+))?\s*(limit\s+(?<limit>\d+))?\s*-->)(.+?)(<!--\s*#end\s*-->)/gs;
|
/(<!--\s*#query\s+(.+?)-->)(.+?)(<!--\s*#end\s*-->)/gs;
|
||||||
|
|
||||||
export const newQueryRegex =
|
|
||||||
/<!--\s*#query\s+(.+?)(?=\s*-->)-->(.+?)<!--\s*#end\s*-->/gs;
|
|
||||||
|
|
||||||
export function whiteOutQueries(text: string): string {
|
export function whiteOutQueries(text: string): string {
|
||||||
return text.replaceAll(newQueryRegex, (match) =>
|
return text.replaceAll(queryRegex, (match) =>
|
||||||
new Array(match.length + 1).join(" ")
|
new Array(match.length + 1).join(" ")
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -59,87 +61,86 @@ function replaceTemplateVars(s: string): string {
|
||||||
export async function updateMaterializedQueriesOnPage(pageName: string) {
|
export async function updateMaterializedQueriesOnPage(pageName: string) {
|
||||||
let { text } = await readPage(pageName);
|
let { text } = await readPage(pageName);
|
||||||
|
|
||||||
text = await replaceAsync(text, queryRegex, async (match, ...args) => {
|
text = await replaceAsync(
|
||||||
let { table, filter, groupBy, limit, orderBy, orderDesc } =
|
text,
|
||||||
args[args.length - 1];
|
queryRegex,
|
||||||
const startQuery = args[0];
|
async (fullMatch, startQuery, query, body, endQuery) => {
|
||||||
const endQuery = args[args.length - 4];
|
let parsedQuery = parseQuery(replaceTemplateVars(query));
|
||||||
let results = [];
|
|
||||||
filter = filter && replaceTemplateVars(filter);
|
|
||||||
switch (table) {
|
|
||||||
case "page":
|
|
||||||
let pages = await listPages();
|
|
||||||
if (orderBy) {
|
|
||||||
pages = pages.sort((a: any, b: any) => {
|
|
||||||
if (a[orderBy] === b[orderBy]) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (a[orderBy] < b[orderBy]) {
|
console.log("Parsed query", parsedQuery);
|
||||||
return !!orderDesc ? 1 : -1;
|
|
||||||
} else {
|
switch (parsedQuery.table) {
|
||||||
return !!orderDesc ? -1 : 1;
|
case "page":
|
||||||
}
|
let allPages = await listPages();
|
||||||
});
|
let markdownPages = applyQuery(parsedQuery, allPages).map(
|
||||||
}
|
(pageMeta: PageMeta) => `* [[${pageMeta.name}]]`
|
||||||
let matchCount = 0;
|
);
|
||||||
for (let pageMeta of pages) {
|
return `${startQuery}\n${markdownPages.join("\n")}\n${endQuery}`;
|
||||||
if (!filter || (filter && pageMeta.name.includes(filter))) {
|
case "task":
|
||||||
matchCount++;
|
let allTasks: Task[] = [];
|
||||||
results.push(`* [[${pageMeta.name}]]`);
|
for (let { key, page, value } of await scanPrefixGlobal("task:")) {
|
||||||
if (limit && matchCount === +limit) {
|
let [, pos] = key.split(":");
|
||||||
break;
|
allTasks.push({
|
||||||
}
|
...value,
|
||||||
|
page: page,
|
||||||
|
pos: pos,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
let markdownTasks = applyQuery(parsedQuery, allTasks).map(
|
||||||
return `${startQuery}\n${results.join("\n")}\n${endQuery}`;
|
(t) =>
|
||||||
case "task":
|
`* [${t.done ? "x" : " "}] [[${t.page}@${t.pos}]] ${t.name}` +
|
||||||
for (let {
|
(t.nested ? "\n " + t.nested : "")
|
||||||
key,
|
);
|
||||||
page,
|
return `${startQuery}\n${markdownTasks.join("\n")}\n${endQuery}`;
|
||||||
value: { task, complete, nested },
|
case "link":
|
||||||
} of await scanPrefixGlobal("task:")) {
|
let uniqueLinks = new Set<string>();
|
||||||
let [, pos] = key.split(":");
|
for (let { value: name } of await scanPrefixGlobal(
|
||||||
if (!filter || (filter && task.includes(filter))) {
|
`pl:${pageName}:`
|
||||||
results.push(
|
)) {
|
||||||
`* [${complete ? "x" : " "}] [[${page}@${pos}]] ${task}` +
|
|
||||||
(nested ? "\n " + nested : "")
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return `${startQuery}\n${results.sort().join("\n")}\n${endQuery}`;
|
|
||||||
case "link":
|
|
||||||
let uniqueLinks = new Set<string>();
|
|
||||||
for (let { key, page, value: name } of await scanPrefixGlobal(
|
|
||||||
`pl:${pageName}:`
|
|
||||||
)) {
|
|
||||||
let [, pos] = key.split(":");
|
|
||||||
if (!filter || (filter && name.includes(filter))) {
|
|
||||||
uniqueLinks.add(name);
|
uniqueLinks.add(name);
|
||||||
}
|
}
|
||||||
}
|
let markdownLinks = applyQuery(
|
||||||
for (const uniqueResult of uniqueLinks) {
|
parsedQuery,
|
||||||
results.push(`* [[${uniqueResult}]]`);
|
[...uniqueLinks].map((l) => ({ name: l }))
|
||||||
}
|
).map((pageMeta) => `* [[${pageMeta.name}]]`);
|
||||||
return `${startQuery}\n${results.sort().join("\n")}\n${endQuery}`;
|
return `${startQuery}\n${markdownLinks.join("\n")}\n${endQuery}`;
|
||||||
case "item":
|
case "item":
|
||||||
for (let {
|
let allItems: Item[] = [];
|
||||||
key,
|
for (let { key, page, value } of await scanPrefixGlobal("it:")) {
|
||||||
page,
|
let [, pos] = key.split("@");
|
||||||
value: { item, nested },
|
allItems.push({
|
||||||
} of await scanPrefixGlobal("it:")) {
|
...value,
|
||||||
let [, pos] = key.split(":");
|
page: page,
|
||||||
if (!filter || (filter && item.includes(filter))) {
|
pos: +pos,
|
||||||
results.push(
|
});
|
||||||
`* [[${page}@${pos}]] ${item}` + (nested ? "\n " + nested : "")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
let markdownItems = applyQuery(parsedQuery, allItems).map(
|
||||||
return `${startQuery}\n${results.sort().join("\n")}\n${endQuery}`;
|
(item) =>
|
||||||
default:
|
`* [[${item.page}@${item.pos}]] ${item.name}` +
|
||||||
return match;
|
(item.nested ? "\n " + item.nested : "")
|
||||||
|
);
|
||||||
|
return `${startQuery}\n${markdownItems.join("\n")}\n${endQuery}`;
|
||||||
|
case "data":
|
||||||
|
let allData: Object[] = [];
|
||||||
|
for (let { key, page, value } of await scanPrefixGlobal("data:")) {
|
||||||
|
let [, pos] = key.split("@");
|
||||||
|
allData.push({
|
||||||
|
...value,
|
||||||
|
page: page,
|
||||||
|
pos: +pos,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
let markdownData = applyQuery(parsedQuery, allData).map((item) =>
|
||||||
|
YAML.stringify(item)
|
||||||
|
);
|
||||||
|
return `${startQuery}\n\`\`\`data\n${markdownData.join(
|
||||||
|
"---\n"
|
||||||
|
)}\`\`\`\n${endQuery}`;
|
||||||
|
default:
|
||||||
|
return fullMatch;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
);
|
||||||
// console.log("New text", text);
|
// console.log("New text", text);
|
||||||
await writePage(pageName, text);
|
await writePage(pageName, text);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,17 @@
|
||||||
|
// This file was generated by lezer-generator. You probably shouldn't edit it.
|
||||||
|
import { LRParser } from "@lezer/lr";
|
||||||
|
|
||||||
|
export const parser = LRParser.deserialize({
|
||||||
|
version: 13,
|
||||||
|
states: "$UOVQPOOO[QQO'#C^QOQPOOOjQPO'#C`OoQQO'#CiOtQPO'#CkOOQO'#Cl'#ClOyQQO,58xO!XQPO'#CcO!pQQO'#CaOOQO'#Ca'#CaOOQO,58z,58zO#RQPO,59TOOQO,59V,59VOOQO-E6j-E6jO#WQQO,58}OjQPO,58|O#iQQO1G.oOOQO'#Cg'#CgOOQO'#Cd'#CdOOQO1G.i1G.iOOQO1G.h1G.hOOQO'#Cj'#CjOOQO7+$Z7+$Z",
|
||||||
|
stateData: "#}~OcOS~ORPO~OdROoSOsTOaQX~ORWO~Op[O~OX]O~OdROoSOsTOaQa~Oe_Oh_Oi_Oj_Ok_Ol_Om_O~On`OaTXdTXoTXsTX~ORaO~OXcOYcO[cOfbOgbO~OqfOrfOa]id]io]is]i~O",
|
||||||
|
goto: "!UaPPbPeilouPPxPe{e!ORQOTUPVRZRRYRQXRRe`Rd_Rc_RgaQVPR^V",
|
||||||
|
nodeNames: "⚠ Program Query Name WhereClause LogicalExpr AndExpr FilterExpr Value Number String Bool Regex OrderClause Order LimitClause",
|
||||||
|
maxTerm: 35,
|
||||||
|
skippedNodes: [0],
|
||||||
|
repeatNodeCount: 1,
|
||||||
|
tokenData: "3X~RsX^#`pq#`qr$Trs$`!P!Q$z!Q![%q!^!_%y!_!`&W!`!a&e!c!}&r#T#U&}#U#V(t#V#W&r#W#X)d#X#Y&r#Y#Z*v#Z#`&r#`#a,h#a#c&r#c#d.]#d#h&r#h#i0Q#i#k&r#k#l1d#l#o&r#y#z#`$f$g#`#BY#BZ#`$IS$I_#`$Ip$Iq$`$Iq$Ir$`$I|$JO#`$JT$JU#`$KV$KW#`&FU&FV#`~#eYc~X^#`pq#`#y#z#`$f$g#`#BY#BZ#`$IS$I_#`$I|$JO#`$JT$JU#`$KV$KW#`&FU&FV#`~$WP!_!`$Z~$`Oj~~$cUOr$`rs$us$Ip$`$Ip$Iq$u$Iq$Ir$u$Ir~$`~$zOY~~%PV[~OY$zZ]$z^!P$z!P!Q%f!Q#O$z#O#P%k#P~$z~%kO[~~%nPO~$z~%vPX~!Q![%q~&OPe~!_!`&R~&WOh~~&]Pi~#r#s&`~&eOm~~&jPl~!_!`&m~&rOk~P&wQRP!c!}&r#T#o&rR'SURP!c!}&r#T#b&r#b#c'f#c#g&r#g#h(U#h#o&rR'kSRP!c!}&r#T#W&r#W#X'w#X#o&rR(OQnQRP!c!}&r#T#o&rR(ZSRP!c!}&r#T#V&r#V#W(g#W#o&rR(nQrQRP!c!}&r#T#o&rR(ySRP!c!}&r#T#m&r#m#n)V#n#o&rR)^QpQRP!c!}&r#T#o&rR)iSRP!c!}&r#T#X&r#X#Y)u#Y#o&rR)zSRP!c!}&r#T#g&r#g#h*W#h#o&rR*]SRP!c!}&r#T#V&r#V#W*i#W#o&rR*pQqQRP!c!}&r#T#o&rR*{RRP!c!}&r#T#U+U#U#o&rR+ZSRP!c!}&r#T#`&r#`#a+g#a#o&rR+lSRP!c!}&r#T#g&r#g#h+x#h#o&rR+}SRP!c!}&r#T#X&r#X#Y,Z#Y#o&rR,bQgQRP!c!}&r#T#o&rR,mSRP!c!}&r#T#]&r#]#^,y#^#o&rR-OSRP!c!}&r#T#a&r#a#b-[#b#o&rR-aSRP!c!}&r#T#]&r#]#^-m#^#o&rR-rSRP!c!}&r#T#h&r#h#i.O#i#o&rR.VQsQRP!c!}&r#T#o&rR.bSRP!c!}&r#T#f&r#f#g.n#g#o&rR.sSRP!c!}&r#T#W&r#W#X/P#X#o&rR/USRP!c!}&r#T#X&r#X#Y/b#Y#o&rR/gSRP!c!}&r#T#f&r#f#g/s#g#o&rR/zQoQRP!c!}&r#T#o&rR0VSRP!c!}&r#T#f&r#f#g0c#g#o&rR0hSRP!c!}&r#T#i&r#i#j0t#j#o&rR0ySRP!c!}&r#T#X&r#X#Y1V#Y#o&rR1^QfQRP!c!}&r#T#o&rR1iSRP!c!}&r#T#[&r#[#]1u#]#o&rR1zSRP!c!}&r#T#X&r#X#Y2W#Y#o&rR2]SRP!c!}&r#T#f&r#f#g2i#g#o&rR2nSRP!c!}&r#T#X&r#X#Y2z#Y#o&rR3RQdQRP!c!}&r#T#o&r",
|
||||||
|
tokenizers: [0, 1],
|
||||||
|
topRules: {"Program":[0,1]},
|
||||||
|
tokenPrec: 0
|
||||||
|
})
|
|
@ -0,0 +1,17 @@
|
||||||
|
// This file was generated by lezer-generator. You probably shouldn't edit it.
|
||||||
|
export const
|
||||||
|
Program = 1,
|
||||||
|
Query = 2,
|
||||||
|
Name = 3,
|
||||||
|
WhereClause = 4,
|
||||||
|
LogicalExpr = 5,
|
||||||
|
AndExpr = 6,
|
||||||
|
FilterExpr = 7,
|
||||||
|
Value = 8,
|
||||||
|
Number = 9,
|
||||||
|
String = 10,
|
||||||
|
Bool = 11,
|
||||||
|
Regex = 12,
|
||||||
|
OrderClause = 13,
|
||||||
|
Order = 14,
|
||||||
|
LimitClause = 15
|
|
@ -14,7 +14,7 @@ Order {
|
||||||
"desc" | "asc"
|
"desc" | "asc"
|
||||||
}
|
}
|
||||||
|
|
||||||
Value { Number | String | Bool }
|
Value { Number | String | Bool | Regex }
|
||||||
|
|
||||||
LogicalExpr { AndExpr | FilterExpr }
|
LogicalExpr { AndExpr | FilterExpr }
|
||||||
|
|
||||||
|
@ -27,7 +27,7 @@ FilterExpr {
|
||||||
| Name "!=" Value
|
| Name "!=" Value
|
||||||
| Name ">=" Value
|
| Name ">=" Value
|
||||||
| Name ">" Value
|
| Name ">" Value
|
||||||
| Name "like" Value
|
| Name "=~" Value
|
||||||
}
|
}
|
||||||
|
|
||||||
@skip { space }
|
@skip { space }
|
||||||
|
@ -39,6 +39,10 @@ Bool {
|
||||||
@tokens {
|
@tokens {
|
||||||
space { std.whitespace+ }
|
space { std.whitespace+ }
|
||||||
Name { std.asciiLetter+ }
|
Name { std.asciiLetter+ }
|
||||||
String { "\"" ![\"]* "\"" }
|
String {
|
||||||
|
("\"" | "“" | "”") ![\"”“]* ("\"" | "“" | "”")
|
||||||
|
}
|
||||||
|
Regex { "/" ( ![/\\\n\r] | "\\" _ )* "/"? }
|
||||||
|
|
||||||
Number { std.digit+ }
|
Number { std.digit+ }
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,3 +5,8 @@ functions:
|
||||||
path: ./materialized_queries.ts:updateMaterializedQueriesCommand
|
path: ./materialized_queries.ts:updateMaterializedQueriesCommand
|
||||||
command:
|
command:
|
||||||
name: "Materialized Queries: Update"
|
name: "Materialized Queries: Update"
|
||||||
|
key: "Alt-q"
|
||||||
|
indexData:
|
||||||
|
path: ./data.ts:indexData
|
||||||
|
events:
|
||||||
|
- page:index
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { whiteOutQueries } from "../query/materialized_queries";
|
||||||
import { batchSet } from "plugos-silverbullet-syscall/index";
|
import { batchSet } from "plugos-silverbullet-syscall/index";
|
||||||
import { readPage, writePage } from "plugos-silverbullet-syscall/space";
|
import { readPage, writePage } from "plugos-silverbullet-syscall/space";
|
||||||
import { parseMarkdown } from "plugos-silverbullet-syscall/markdown";
|
import { parseMarkdown } from "plugos-silverbullet-syscall/markdown";
|
||||||
import { dispatch, getText } from "plugos-silverbullet-syscall/editor";
|
import { dispatch, getCurrentPage, getText } from "plugos-silverbullet-syscall/editor";
|
||||||
import {
|
import {
|
||||||
addParentPointers,
|
addParentPointers,
|
||||||
collectNodesMatching,
|
collectNodesMatching,
|
||||||
|
@ -14,12 +14,14 @@ import {
|
||||||
renderToText
|
renderToText
|
||||||
} from "../../common/tree";
|
} from "../../common/tree";
|
||||||
|
|
||||||
type Task = {
|
export type Task = {
|
||||||
task: string;
|
name: string;
|
||||||
complete: boolean;
|
done: boolean;
|
||||||
deadline?: string;
|
deadline?: string;
|
||||||
pos?: number;
|
|
||||||
nested?: string;
|
nested?: string;
|
||||||
|
// Not saved in DB, just added when pulled out (from key)
|
||||||
|
pos?: number;
|
||||||
|
page?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export async function indexTasks({ name, text }: IndexEvent) {
|
export async function indexTasks({ name, text }: IndexEvent) {
|
||||||
|
@ -32,8 +34,8 @@ export async function indexTasks({ name, text }: IndexEvent) {
|
||||||
let task = n.children!.slice(1).map(renderToText).join("").trim();
|
let task = n.children!.slice(1).map(renderToText).join("").trim();
|
||||||
let complete = n.children![0].children![0].text! !== "[ ]";
|
let complete = n.children![0].children![0].text! !== "[ ]";
|
||||||
let value: Task = {
|
let value: Task = {
|
||||||
task,
|
name: task,
|
||||||
complete,
|
done: complete,
|
||||||
};
|
};
|
||||||
|
|
||||||
let deadlineNodes = collectNodesOfType(n, "DeadlineDate");
|
let deadlineNodes = collectNodesOfType(n, "DeadlineDate");
|
||||||
|
@ -62,6 +64,7 @@ export async function taskToggle(event: ClickEvent) {
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function taskToggleAtPos(pos: number) {
|
export async function taskToggleAtPos(pos: number) {
|
||||||
|
let currentpage = await getCurrentPage();
|
||||||
let text = await getText();
|
let text = await getText();
|
||||||
let mdTree = await parseMarkdown(text);
|
let mdTree = await parseMarkdown(text);
|
||||||
addParentPointers(mdTree);
|
addParentPointers(mdTree);
|
||||||
|
@ -91,8 +94,9 @@ export async function taskToggleAtPos(pos: number) {
|
||||||
let ref = wikiLink.children![0].text!;
|
let ref = wikiLink.children![0].text!;
|
||||||
if (ref.includes("@")) {
|
if (ref.includes("@")) {
|
||||||
let [page, pos] = ref.split("@");
|
let [page, pos] = ref.split("@");
|
||||||
let pageData = await readPage(page);
|
if (page !== currentpage) {
|
||||||
let text = pageData.text;
|
text = (await readPage(page)).text;
|
||||||
|
}
|
||||||
|
|
||||||
let referenceMdTree = await parseMarkdown(text);
|
let referenceMdTree = await parseMarkdown(text);
|
||||||
// Adding +1 to immediately hit the task marker
|
// Adding +1 to immediately hit the task marker
|
||||||
|
|
|
@ -32,7 +32,7 @@ export function mdExtensionSyntaxConfig({
|
||||||
}
|
}
|
||||||
return cx.addElement(cx.elt(nodeType, pos, pos + match[0].length));
|
return cx.addElement(cx.elt(nodeType, pos, pos + match[0].length));
|
||||||
},
|
},
|
||||||
after: "Emphasis",
|
// after: "Emphasis",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
|
|
|
@ -78,6 +78,7 @@ export default function buildMarkdown(mdExtensions: MDExt[]): Language {
|
||||||
codeParser: getCodeParser([
|
codeParser: getCodeParser([
|
||||||
LanguageDescription.of({
|
LanguageDescription.of({
|
||||||
name: "yaml",
|
name: "yaml",
|
||||||
|
alias: ["meta", "data"],
|
||||||
support: new LanguageSupport(StreamLanguage.define(yaml)),
|
support: new LanguageSupport(StreamLanguage.define(yaml)),
|
||||||
}),
|
}),
|
||||||
LanguageDescription.of({
|
LanguageDescription.of({
|
||||||
|
|
|
@ -63,7 +63,7 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.line-fenced-code {
|
.line-fenced-code {
|
||||||
background-color: #efefef;
|
background-color: rgba(72, 72, 72, 0.1);
|
||||||
}
|
}
|
||||||
|
|
||||||
.meta {
|
.meta {
|
||||||
|
@ -99,6 +99,10 @@
|
||||||
text-decoration: underline;
|
text-decoration: underline;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.atom {
|
||||||
|
color: darkred;
|
||||||
|
}
|
||||||
|
|
||||||
.wiki-link-page {
|
.wiki-link-page {
|
||||||
color: #0330cb;
|
color: #0330cb;
|
||||||
background-color: rgba(77,141,255,0.07);
|
background-color: rgba(77,141,255,0.07);
|
||||||
|
@ -111,13 +115,7 @@
|
||||||
//background-color: rgba(77,141,255,0.07);
|
//background-color: rgba(77,141,255,0.07);
|
||||||
}
|
}
|
||||||
|
|
||||||
.mention {
|
|
||||||
color: #0330cb;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tag {
|
|
||||||
color: #8d8d8d;
|
|
||||||
}
|
|
||||||
|
|
||||||
.code {
|
.code {
|
||||||
background-color: #efefef;
|
background-color: #efefef;
|
||||||
|
|
Loading…
Reference in New Issue