2024-08-02 22:47:36 +08:00
|
|
|
import YAML from "js-yaml";
|
|
|
|
import { INDEX_TEMPLATE, SETTINGS_TEMPLATE } from "./PAGE_TEMPLATES.ts";
|
|
|
|
import type { SpacePrimitives } from "./spaces/space_primitives.ts";
|
|
|
|
import { cleanupJSON } from "../plug-api/lib/json.ts";
|
2024-08-15 22:39:06 +08:00
|
|
|
import {
|
|
|
|
type Config,
|
|
|
|
defaultConfig,
|
|
|
|
type DynamicAttributeDefinitionConfig,
|
2024-08-02 22:47:36 +08:00
|
|
|
} from "../type/config.ts";
|
|
|
|
import type {
|
|
|
|
DataStore,
|
|
|
|
DynamicAttributeDefinitions,
|
|
|
|
ObjectDecorators,
|
|
|
|
} from "$lib/data/datastore.ts";
|
|
|
|
import { parseExpression } from "$common/expression_parser.ts";
|
|
|
|
import type { System } from "$lib/plugos/system.ts";
|
|
|
|
import type { ConfigObject } from "../plugs/index/config.ts";
|
2024-08-07 02:11:38 +08:00
|
|
|
import { deepObjectMerge } from "@silverbulletmd/silverbullet/lib/json";
|
2024-08-02 22:47:36 +08:00
|
|
|
|
|
|
|
const yamlConfigRegex = /^(```+|~~~+)(ya?ml|space-config)\r?\n([\S\s]+?)\1/m;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Parses YAML config from a Markdown string.
|
|
|
|
* @param configMarkdown - The Markdown string containing the YAML config.
|
|
|
|
* @returns An object representing the parsed YAML config.
|
|
|
|
*/
|
|
|
|
export function parseYamlConfig(configMarkdown: string): {
|
|
|
|
[key: string]: any;
|
|
|
|
} {
|
|
|
|
const match = yamlConfigRegex.exec(configMarkdown);
|
|
|
|
if (!match) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
const yaml = match[3];
|
|
|
|
try {
|
|
|
|
return YAML.load(yaml) as {
|
|
|
|
[key: string]: any;
|
|
|
|
};
|
|
|
|
} catch (e: any) {
|
|
|
|
console.error("Error parsing SETTINGS as YAML", e.message);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads space-configs from a system using the `index` plug.
|
|
|
|
* @param system - The system object
|
|
|
|
* @returns A promise that resolves to merged configs from the system.
|
|
|
|
*/
|
|
|
|
async function loadConfigsFromSystem(
|
|
|
|
system: System<any>,
|
|
|
|
): Promise<Config> {
|
|
|
|
if (!system.loadedPlugs.has("index")) {
|
|
|
|
console.warn("Index plug not loaded yet, falling back to default config");
|
|
|
|
return defaultConfig;
|
|
|
|
}
|
2024-08-24 18:35:09 +08:00
|
|
|
// We'll start witht the default config
|
|
|
|
let fullConfig: any = { ...defaultConfig };
|
|
|
|
// Then merge in all plug-based configs
|
|
|
|
for (const plugDef of system.loadedPlugs.values()) {
|
|
|
|
if (plugDef.manifest?.config) {
|
|
|
|
const plugConfig = cleanupJSON(plugDef.manifest.config);
|
|
|
|
fullConfig = deepObjectMerge(fullConfig, plugConfig);
|
|
|
|
}
|
|
|
|
}
|
2024-08-02 22:47:36 +08:00
|
|
|
// Query all space-configs
|
|
|
|
const allConfigs: ConfigObject[] = await system.invokeFunction(
|
|
|
|
"index.queryObjects",
|
|
|
|
["space-config", {}],
|
|
|
|
);
|
|
|
|
// Now let's intelligently merge them
|
|
|
|
for (const config of allConfigs) {
|
2024-08-15 22:39:06 +08:00
|
|
|
let configObject = { [config.key]: config.value };
|
|
|
|
configObject = cleanupJSON(configObject);
|
|
|
|
fullConfig = deepObjectMerge(fullConfig, configObject);
|
2024-08-02 22:47:36 +08:00
|
|
|
}
|
|
|
|
// And clean up the JSON (expand .-separated paths, convert dates to strings)
|
2024-08-04 02:59:53 +08:00
|
|
|
return cleanupJSON(fullConfig);
|
2024-08-02 22:47:36 +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.
|
|
|
|
*/
|
|
|
|
export async function ensureAndLoadSettingsAndIndex(
|
|
|
|
space: SpacePrimitives,
|
|
|
|
system?: System<any>,
|
|
|
|
): Promise<Config> {
|
|
|
|
let configText: string | undefined;
|
2024-08-15 22:39:06 +08:00
|
|
|
let config = defaultConfig;
|
2024-08-02 22:47:36 +08:00
|
|
|
|
|
|
|
try {
|
|
|
|
configText = new TextDecoder().decode(
|
|
|
|
(await space.readFile("SETTINGS.md")).data,
|
|
|
|
);
|
|
|
|
} catch (e: any) {
|
|
|
|
if (e.message === "Not found") {
|
|
|
|
console.log("No SETTINGS found, creating default SETTINGS");
|
|
|
|
await space.writeFile(
|
|
|
|
"SETTINGS.md",
|
|
|
|
new TextEncoder().encode(SETTINGS_TEMPLATE),
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
console.error("Error reading SETTINGS", e.message);
|
|
|
|
console.warn("Falling back to default SETTINGS");
|
|
|
|
return defaultConfig;
|
|
|
|
}
|
|
|
|
configText = SETTINGS_TEMPLATE;
|
|
|
|
// Ok, then let's also check the index page
|
|
|
|
try {
|
|
|
|
await space.getFileMeta("index.md");
|
|
|
|
} catch (e: any) {
|
|
|
|
console.log(
|
|
|
|
"No index page found, creating default index page",
|
|
|
|
e.message,
|
|
|
|
);
|
|
|
|
// This should trigger indexing of the configs too
|
|
|
|
await space.writeFile(
|
|
|
|
"index.md",
|
|
|
|
new TextEncoder().encode(INDEX_TEMPLATE),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (system) {
|
|
|
|
// If we're NOT in SB_SYNC_ONLY, we can load settings from the index (ideal case)
|
2024-08-15 22:39:06 +08:00
|
|
|
config = await loadConfigsFromSystem(system);
|
2024-08-02 22:47:36 +08:00
|
|
|
} else {
|
|
|
|
// If we are in SB_SYNC_ONLY, this is best effort, and we can only support settings in the SETTINGS.md file
|
2024-08-15 22:39:06 +08:00
|
|
|
config = parseYamlConfig(configText) as Config;
|
2024-08-02 22:47:36 +08:00
|
|
|
config = cleanupJSON(config);
|
|
|
|
config = { ...defaultConfig, ...config };
|
|
|
|
// console.log("Loaded settings from SETTINGS.md", config);
|
|
|
|
}
|
2024-08-15 22:39:06 +08:00
|
|
|
// Some essential sanity checking
|
|
|
|
if (typeof config.indexPage !== "string") {
|
|
|
|
console.warn("indexPage is not a string, falling back to default");
|
|
|
|
config.indexPage = "index";
|
|
|
|
}
|
|
|
|
return config;
|
2024-08-02 22:47:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function updateObjectDecorators(
|
|
|
|
config: Config,
|
|
|
|
ds: DataStore,
|
|
|
|
) {
|
|
|
|
if (config.objectDecorators) {
|
|
|
|
// Reload object decorators
|
|
|
|
const newDecorators: ObjectDecorators[] = [];
|
|
|
|
for (
|
|
|
|
const decorator of config.objectDecorators
|
|
|
|
) {
|
|
|
|
try {
|
|
|
|
newDecorators.push({
|
|
|
|
where: parseExpression(decorator.where),
|
|
|
|
attributes: objectAttributeToExpressions(decorator.attributes),
|
|
|
|
});
|
|
|
|
} catch (e: any) {
|
|
|
|
console.error(
|
|
|
|
"Error parsing object decorator",
|
|
|
|
decorator,
|
|
|
|
"got error",
|
|
|
|
e.message,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// console.info(`Loaded ${newDecorators.length} object decorators`);
|
|
|
|
ds.objectDecorators = newDecorators;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function objectAttributeToExpressions(
|
|
|
|
dynamicAttributes: DynamicAttributeDefinitionConfig,
|
|
|
|
): DynamicAttributeDefinitions {
|
|
|
|
const result: DynamicAttributeDefinitions = {};
|
|
|
|
for (const [key, value] of Object.entries(dynamicAttributes)) {
|
|
|
|
if (typeof value === "string") {
|
|
|
|
result[key] = parseExpression(value);
|
|
|
|
} else {
|
|
|
|
result[key] = objectAttributeToExpressions(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|