2024-03-16 22:29:24 +08:00
|
|
|
import YAML from "js-yaml";
|
2024-02-09 04:00:45 +08:00
|
|
|
import { INDEX_TEMPLATE, SETTINGS_TEMPLATE } from "./PAGE_TEMPLATES.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { SpacePrimitives } from "./spaces/space_primitives.ts";
|
2024-07-06 21:07:40 +08:00
|
|
|
import { cleanupJSON } from "../plug-api/lib/json.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { BuiltinSettings } from "$type/settings.ts";
|
|
|
|
import type { DataStore, ObjectEnricher } from "$lib/data/datastore.ts";
|
2024-07-30 03:21:16 +08:00
|
|
|
import { parseExpression } from "$common/expression_parser.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { QueryExpression } from "$sb/types.ts";
|
2022-08-02 18:43:39 +08:00
|
|
|
|
2024-06-05 02:27:10 +08:00
|
|
|
const yamlSettingsRegex = /^(```+|~~~+)ya?ml\r?\n([\S\s]+?)\1/m;
|
2022-08-02 18:43:39 +08:00
|
|
|
|
2024-01-05 03:08:12 +08:00
|
|
|
/**
|
|
|
|
* Parses YAML settings from a Markdown string.
|
|
|
|
* @param settingsMarkdown - The Markdown string containing the YAML settings.
|
|
|
|
* @returns An object representing the parsed YAML settings.
|
|
|
|
*/
|
2022-08-02 18:43:39 +08:00
|
|
|
export function parseYamlSettings(settingsMarkdown: string): {
|
|
|
|
[key: string]: any;
|
|
|
|
} {
|
|
|
|
const match = yamlSettingsRegex.exec(settingsMarkdown);
|
|
|
|
if (!match) {
|
|
|
|
return {};
|
|
|
|
}
|
2024-02-23 16:03:14 +08:00
|
|
|
const yaml = match[2]; // The first group captures the code fence to look for same terminator
|
2022-12-15 20:23:49 +08:00
|
|
|
try {
|
2023-11-29 23:51:28 +08:00
|
|
|
return YAML.load(yaml) as {
|
2022-12-15 20:23:49 +08:00
|
|
|
[key: string]: any;
|
|
|
|
};
|
|
|
|
} catch (e: any) {
|
|
|
|
console.error("Error parsing SETTINGS as YAML", e.message);
|
|
|
|
return {};
|
|
|
|
}
|
2022-08-02 18:43:39 +08:00
|
|
|
}
|
2023-01-08 19:24:12 +08:00
|
|
|
|
2024-01-25 18:42:36 +08:00
|
|
|
export const defaultSettings: BuiltinSettings = {
|
|
|
|
indexPage: "index",
|
|
|
|
hideSyncButton: false,
|
2024-03-28 04:21:12 +08:00
|
|
|
maximumAttachmentSize: 10, // MiB
|
2024-05-28 02:33:41 +08:00
|
|
|
defaultLinkStyle: "wikilink", // wikilink [[]] or markdown []()
|
2024-01-25 18:42:36 +08:00
|
|
|
actionButtons: [
|
|
|
|
{
|
|
|
|
icon: "Home",
|
|
|
|
description: "Go to the index page",
|
|
|
|
command: "Navigate: Home",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: "Book",
|
|
|
|
description: `Open page`,
|
|
|
|
command: "Navigate: Page Picker",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
icon: "Terminal",
|
|
|
|
description: `Run command`,
|
|
|
|
command: "Open Command Palette",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
|
2024-01-05 03:08:12 +08:00
|
|
|
/**
|
|
|
|
* Ensures that the settings and index page exist in the given space.
|
|
|
|
* If they don't exist, default settings and index page will be created.
|
|
|
|
* @param space - The SpacePrimitives object representing the space.
|
|
|
|
* @returns A promise that resolves to the built-in settings.
|
|
|
|
*/
|
2024-01-25 18:42:36 +08:00
|
|
|
export async function ensureAndLoadSettingsAndIndex(
|
2023-05-24 02:53:53 +08:00
|
|
|
space: SpacePrimitives,
|
2023-12-18 23:54:55 +08:00
|
|
|
): Promise<BuiltinSettings> {
|
2023-05-29 15:53:49 +08:00
|
|
|
let settingsText: string | undefined;
|
2023-01-08 19:24:12 +08:00
|
|
|
try {
|
2023-05-29 15:53:49 +08:00
|
|
|
settingsText = new TextDecoder().decode(
|
|
|
|
(await space.readFile("SETTINGS.md")).data,
|
|
|
|
);
|
2023-06-14 02:47:05 +08:00
|
|
|
} catch (e: any) {
|
|
|
|
if (e.message === "Not found") {
|
2023-12-17 18:46:18 +08:00
|
|
|
console.log("No settings found, creating default settings");
|
2023-06-14 02:47:05 +08:00
|
|
|
await space.writeFile(
|
|
|
|
"SETTINGS.md",
|
|
|
|
new TextEncoder().encode(SETTINGS_TEMPLATE),
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
console.error("Error reading settings", e.message);
|
2023-12-18 23:54:55 +08:00
|
|
|
console.warn("Falling back to default settings");
|
2024-01-25 18:42:36 +08:00
|
|
|
return defaultSettings;
|
2023-06-14 02:47:05 +08:00
|
|
|
}
|
2023-05-29 15:53:49 +08:00
|
|
|
settingsText = SETTINGS_TEMPLATE;
|
2023-06-14 02:47:05 +08:00
|
|
|
// Ok, then let's also check the index page
|
2023-05-24 02:53:53 +08:00
|
|
|
try {
|
|
|
|
await space.getFileMeta("index.md");
|
2023-12-17 18:46:18 +08:00
|
|
|
} catch (e: any) {
|
|
|
|
console.log(
|
|
|
|
"No index page found, creating default index page",
|
|
|
|
e.message,
|
|
|
|
);
|
2023-05-24 02:53:53 +08:00
|
|
|
await space.writeFile(
|
|
|
|
"index.md",
|
2024-01-21 02:16:07 +08:00
|
|
|
new TextEncoder().encode(INDEX_TEMPLATE),
|
2023-05-24 02:53:53 +08:00
|
|
|
);
|
|
|
|
}
|
2023-01-08 19:24:12 +08:00
|
|
|
}
|
2023-05-29 15:53:49 +08:00
|
|
|
|
2023-12-18 23:54:55 +08:00
|
|
|
const settings: any = parseYamlSettings(settingsText);
|
2024-07-06 21:07:40 +08:00
|
|
|
cleanupJSON(settings);
|
2024-01-25 18:42:36 +08:00
|
|
|
return { ...defaultSettings, ...settings };
|
2023-01-08 19:24:12 +08:00
|
|
|
}
|
2024-07-30 03:21:16 +08:00
|
|
|
|
|
|
|
export function updateObjectDecorators(
|
|
|
|
settings: BuiltinSettings,
|
|
|
|
ds: DataStore,
|
|
|
|
) {
|
|
|
|
if (settings.objectDecorators) {
|
|
|
|
// Reload object decorators
|
|
|
|
const newDecorators: ObjectEnricher[] = [];
|
|
|
|
for (
|
|
|
|
const decorator of settings.objectDecorators
|
|
|
|
) {
|
|
|
|
try {
|
|
|
|
const parsedWhere = parseExpression(decorator.where);
|
|
|
|
const parsedDynamicAttributes: Record<string, QueryExpression> = {};
|
|
|
|
for (const [key, value] of Object.entries(decorator.attributes)) {
|
|
|
|
parsedDynamicAttributes[key] = parseExpression(value);
|
|
|
|
}
|
|
|
|
newDecorators.push({
|
|
|
|
where: parsedWhere,
|
|
|
|
attributes: parsedDynamicAttributes,
|
|
|
|
});
|
|
|
|
} catch (e: any) {
|
|
|
|
console.error(
|
|
|
|
"Error parsing object decorator",
|
|
|
|
decorator,
|
|
|
|
"got error",
|
|
|
|
e.message,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
console.info(`Loaded ${newDecorators.length} object decorators`);
|
|
|
|
ds.objectEnrichers = newDecorators;
|
|
|
|
}
|
|
|
|
}
|