Files
openclaw/extensions/workboard/doctor-contract-api.ts
2026-05-31 07:17:57 +01:00

289 lines
9.2 KiB
TypeScript

import type {
PluginDoctorStateMigration,
PluginDoctorStateMigrationContext,
} from "openclaw/plugin-sdk/runtime-doctor";
import type {
PersistedWorkboardAttachment,
PersistedWorkboardBoard,
PersistedWorkboardCard,
PersistedWorkboardNotificationSubscription,
WorkboardKeyedStore,
} from "./src/persistence-types.js";
import { createWorkboardSqliteStores, resolveWorkboardSqlitePath } from "./src/sqlite-store.js";
const MAX_CARDS = 2000;
function migrationEnv(params: { env: NodeJS.ProcessEnv; stateDir: string }): NodeJS.ProcessEnv {
return { ...params.env, OPENCLAW_STATE_DIR: params.stateDir };
}
function openLegacyStore<T>(params: {
context: PluginDoctorStateMigrationContext;
env: NodeJS.ProcessEnv;
namespace: string;
maxEntries: number;
}): WorkboardKeyedStore<T> {
return params.context.openPluginStateKeyedStore<T>({
namespace: params.namespace,
maxEntries: params.maxEntries,
env: params.env,
});
}
function isPersistedCard(value: unknown): value is PersistedWorkboardCard {
return Boolean(
value && typeof value === "object" && (value as PersistedWorkboardCard).version === 1,
);
}
function isPersistedBoard(value: unknown): value is PersistedWorkboardBoard {
return Boolean(
value && typeof value === "object" && (value as PersistedWorkboardBoard).version === 1,
);
}
function isPersistedSubscription(
value: unknown,
): value is PersistedWorkboardNotificationSubscription {
return Boolean(
value &&
typeof value === "object" &&
(value as PersistedWorkboardNotificationSubscription).version === 1,
);
}
function isPersistedAttachment(value: unknown): value is PersistedWorkboardAttachment {
if (!value || typeof value !== "object") {
return false;
}
const candidate = value as {
version?: unknown;
attachment?: Partial<PersistedWorkboardAttachment["attachment"]>;
contentBase64?: unknown;
};
const attachment = candidate.attachment;
return (
candidate.version === 1 &&
attachment !== undefined &&
typeof attachment === "object" &&
typeof attachment.id === "string" &&
typeof attachment.cardId === "string" &&
typeof attachment.fileName === "string" &&
typeof attachment.byteSize === "number" &&
typeof attachment.createdAt === "number" &&
typeof candidate.contentBase64 === "string"
);
}
async function migrateNamespace<T>(params: {
label: string;
legacy: WorkboardKeyedStore<T>;
target: WorkboardKeyedStore<T>;
isValid: (value: unknown) => value is T;
}): Promise<{ imported: number; warnings: string[] }> {
const warnings: string[] = [];
let imported = 0;
for (const entry of await params.legacy.entries()) {
if (!params.isValid(entry.value)) {
warnings.push(`Skipped malformed legacy Workboard ${params.label} entry ${entry.key}`);
continue;
}
try {
const targetEntry = await params.target.lookup(entry.key);
if (targetEntry) {
if (JSON.stringify(targetEntry) === JSON.stringify(entry.value)) {
await params.legacy.delete(entry.key);
imported++;
continue;
}
warnings.push(
`Skipped legacy Workboard ${params.label} entry ${entry.key} because the SQLite target already exists`,
);
continue;
}
await params.target.register(entry.key, entry.value);
await params.legacy.delete(entry.key);
imported++;
} catch (err) {
warnings.push(
`Failed migrating legacy Workboard ${params.label} entry ${entry.key}: ${String(err)}`,
);
}
}
return { imported, warnings };
}
async function targetCardReferencesAttachment(
cards: WorkboardKeyedStore,
attachment: PersistedWorkboardAttachment,
): Promise<boolean> {
const card = await cards.lookup(attachment.attachment.cardId);
return Boolean(
card?.version === 1 &&
card.card.metadata?.attachments?.some(
(entry) =>
entry.id === attachment.attachment.id && entry.cardId === attachment.attachment.cardId,
),
);
}
async function migrateAttachments(params: {
legacy: WorkboardKeyedStore<PersistedWorkboardAttachment>;
cards: WorkboardKeyedStore;
target: WorkboardKeyedStore<PersistedWorkboardAttachment>;
}): Promise<{ imported: number; warnings: string[] }> {
const warnings: string[] = [];
let imported = 0;
for (const entry of await params.legacy.entries()) {
if (!isPersistedAttachment(entry.value)) {
warnings.push(`Skipped malformed legacy Workboard attachment entry ${entry.key}`);
continue;
}
if (!(await targetCardReferencesAttachment(params.cards, entry.value))) {
warnings.push(
`Skipped legacy Workboard attachment entry ${entry.key} because its owning card was not migrated or does not reference the attachment`,
);
continue;
}
const targetEntry = await params.target.lookup(entry.key);
if (targetEntry) {
if (JSON.stringify(targetEntry) === JSON.stringify(entry.value)) {
await params.legacy.delete(entry.key);
imported++;
continue;
}
warnings.push(
`Skipped legacy Workboard attachment entry ${entry.key} because the SQLite target already exists`,
);
continue;
}
try {
await params.target.register(entry.key, entry.value);
await params.legacy.delete(entry.key);
imported++;
} catch (err) {
warnings.push(
`Failed migrating legacy Workboard attachment entry ${entry.key}: ${String(err)}`,
);
}
}
return { imported, warnings };
}
export const stateMigrations: PluginDoctorStateMigration[] = [
{
id: "workboard-28-kv-to-sqlite",
label: "Workboard .28 plugin-state KV",
async detectLegacyState(params) {
const env = migrationEnv(params);
const cards = await openLegacyStore<PersistedWorkboardCard>({
context: params.context,
env,
namespace: "workboard.cards",
maxEntries: MAX_CARDS,
}).entries();
const boards = await openLegacyStore<PersistedWorkboardBoard>({
context: params.context,
env,
namespace: "workboard.boards",
maxEntries: 200,
}).entries();
const subscriptions = await openLegacyStore<PersistedWorkboardNotificationSubscription>({
context: params.context,
env,
namespace: "workboard.notify",
maxEntries: 2000,
}).entries();
const attachments = await openLegacyStore<PersistedWorkboardAttachment>({
context: params.context,
env,
namespace: "workboard.attachments",
maxEntries: MAX_CARDS * 21,
}).entries();
const count = cards.length + boards.length + subscriptions.length + attachments.length;
if (count === 0) {
return null;
}
return {
preview: [
`- Workboard: ${count} legacy .28 plugin-state KV ${count === 1 ? "entry" : "entries"}${resolveWorkboardSqlitePath(env)}`,
],
};
},
async migrateLegacyState(params) {
const env = migrationEnv(params);
const cards = openLegacyStore<PersistedWorkboardCard>({
context: params.context,
env,
namespace: "workboard.cards",
maxEntries: MAX_CARDS,
});
const boards = openLegacyStore<PersistedWorkboardBoard>({
context: params.context,
env,
namespace: "workboard.boards",
maxEntries: 200,
});
const subscriptions = openLegacyStore<PersistedWorkboardNotificationSubscription>({
context: params.context,
env,
namespace: "workboard.notify",
maxEntries: 2000,
});
const attachments = openLegacyStore<PersistedWorkboardAttachment>({
context: params.context,
env,
namespace: "workboard.attachments",
maxEntries: MAX_CARDS * 21,
});
const sqlite = createWorkboardSqliteStores({ env });
try {
const cardResult = await migrateNamespace({
label: "card",
legacy: cards,
target: sqlite.cards,
isValid: isPersistedCard,
});
const boardResult = await migrateNamespace({
label: "board",
legacy: boards,
target: sqlite.boards,
isValid: isPersistedBoard,
});
const subscriptionResult = await migrateNamespace({
label: "notification subscription",
legacy: subscriptions,
target: sqlite.subscriptions,
isValid: isPersistedSubscription,
});
const attachmentResult = await migrateAttachments({
legacy: attachments,
cards: sqlite.cards,
target: sqlite.attachments,
});
const imported =
cardResult.imported +
boardResult.imported +
subscriptionResult.imported +
attachmentResult.imported;
return {
changes:
imported > 0
? [
`Migrated ${imported} Workboard .28 plugin-state KV ${imported === 1 ? "entry" : "entries"} → relational SQLite`,
]
: [],
warnings: [
...cardResult.warnings,
...boardResult.warnings,
...subscriptionResult.warnings,
...attachmentResult.warnings,
],
};
} finally {
sqlite.close();
}
},
},
];