mirror of
https://github.com/openclaw/openclaw.git
synced 2026-04-20 21:51:28 +00:00
* Add embed rendering for Control UI assistant output * Add changelog entry for embed rendering * Harden canvas path resolution and stage isolation * Secure assistant media route and preserve UI avatar override * Fix chat media and history regressions * Harden embed iframe URL handling * Fix embed follow-up review regressions * Restore offloaded chat attachment persistence * Harden hook and media routing * Fix embed review follow-ups * feat(ui): add configurable embed sandbox mode * fix(gateway): harden assistant media and auth rotation * fix(gateway): restore websocket pairing handshake flows * fix(gateway): restore ws hello policy details * Restore dropped control UI shell wiring * Fix control UI reconnect cleanup regressions * fix(gateway): restore media root and auth getter compatibility * feat(ui): rename public canvas tag to embed * fix(ui): address remaining media and gateway review issues * fix(ui): address remaining embed and attachment review findings * fix(ui): restore stop control and tool card inputs * fix(ui): address history and attachment review findings * fix(ui): restore prompt contribution wiring * fix(ui): address latest history and directive reviews * fix(ui): forward password auth for assistant media * fix(ui): suppress silent transcript tokens with media * feat(ui): add granular embed sandbox modes * fix(ui): preserve relative media directives in history * docs(ui): document embed sandbox modes * fix(gateway): restrict canvas history hoisting to tool entries * fix(gateway): tighten embed follow-up review fixes * fix(ci): repair merged branch type drift * fix(prompt): restore stable runtime prompt rendering * fix(ui): harden local attachment preview checks * fix(prompt): restore channel-aware approval guidance * fix(gateway): enforce auth rotation and media cleanup * feat(ui): gate external embed urls behind config * fix(ci): repair rebased branch drift * fix(ci): resolve remaining branch check failures
348 lines
12 KiB
TypeScript
348 lines
12 KiB
TypeScript
import { randomUUID } from "node:crypto";
|
|
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
import { CANVAS_HOST_PATH } from "../canvas-host/a2ui.js";
|
|
import { resolveStateDir } from "../config/paths.js";
|
|
import { resolveUserPath } from "../utils.js";
|
|
|
|
export type CanvasDocumentKind = "html_bundle" | "url_embed" | "document" | "image" | "video_asset";
|
|
|
|
export type CanvasDocumentAsset = {
|
|
logicalPath: string;
|
|
sourcePath: string;
|
|
contentType?: string;
|
|
};
|
|
|
|
export type CanvasDocumentEntrypoint =
|
|
| { type: "html"; value: string }
|
|
| { type: "path"; value: string }
|
|
| { type: "url"; value: string };
|
|
|
|
export type CanvasDocumentCreateInput = {
|
|
id?: string;
|
|
kind: CanvasDocumentKind;
|
|
title?: string;
|
|
preferredHeight?: number;
|
|
entrypoint?: CanvasDocumentEntrypoint;
|
|
assets?: CanvasDocumentAsset[];
|
|
surface?: "assistant_message" | "tool_card" | "sidebar";
|
|
};
|
|
|
|
export type CanvasDocumentManifest = {
|
|
id: string;
|
|
kind: CanvasDocumentKind;
|
|
title?: string;
|
|
preferredHeight?: number;
|
|
createdAt: string;
|
|
entryUrl: string;
|
|
localEntrypoint?: string;
|
|
externalUrl?: string;
|
|
surface?: "assistant_message" | "tool_card" | "sidebar";
|
|
assets: Array<{
|
|
logicalPath: string;
|
|
contentType?: string;
|
|
}>;
|
|
};
|
|
|
|
export type CanvasDocumentResolvedAsset = {
|
|
logicalPath: string;
|
|
contentType?: string;
|
|
url: string;
|
|
localPath: string;
|
|
};
|
|
|
|
const CANVAS_DOCUMENTS_DIR_NAME = "documents";
|
|
|
|
function isPdfPathLike(value: string): boolean {
|
|
return /\.pdf(?:[?#].*)?$/i.test(value.trim());
|
|
}
|
|
|
|
function buildPdfWrapper(url: string): string {
|
|
const escaped = escapeHtml(url);
|
|
return `<!doctype html><html><body style="margin:0;background:#e5e7eb;"><object data="${escaped}" type="application/pdf" style="width:100%;height:100vh;border:0;"><iframe src="${escaped}" style="width:100%;height:100vh;border:0;"></iframe><p style="padding:16px;font:14px system-ui,sans-serif;">Unable to render PDF preview. <a href="${escaped}" target="_blank" rel="noopener noreferrer">Open PDF</a>.</p></object></body></html>`;
|
|
}
|
|
|
|
function escapeHtml(value: string): string {
|
|
return value
|
|
.replaceAll("&", "&")
|
|
.replaceAll("<", "<")
|
|
.replaceAll(">", ">")
|
|
.replaceAll('"', """)
|
|
.replaceAll("'", "'");
|
|
}
|
|
|
|
function normalizeLogicalPath(value: string): string {
|
|
const normalized = value.replaceAll("\\", "/").replace(/^\/+/, "");
|
|
const parts = normalized.split("/").filter(Boolean);
|
|
if (parts.length === 0 || parts.some((part) => part === "." || part === "..")) {
|
|
throw new Error("canvas document logicalPath invalid");
|
|
}
|
|
return parts.join("/");
|
|
}
|
|
|
|
function canvasDocumentId(): string {
|
|
return `cv_${randomUUID().replaceAll("-", "")}`;
|
|
}
|
|
|
|
function normalizeCanvasDocumentId(value: string): string {
|
|
const normalized = value.trim();
|
|
if (
|
|
!normalized ||
|
|
normalized === "." ||
|
|
normalized === ".." ||
|
|
!/^[A-Za-z0-9._-]+$/.test(normalized)
|
|
) {
|
|
throw new Error("canvas document id invalid");
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
export function resolveCanvasRootDir(rootDir?: string, stateDir = resolveStateDir()): string {
|
|
const resolved = rootDir?.trim() ? resolveUserPath(rootDir) : path.join(stateDir, "canvas");
|
|
return path.resolve(resolved);
|
|
}
|
|
|
|
export function resolveCanvasDocumentsDir(rootDir?: string, stateDir = resolveStateDir()): string {
|
|
return path.join(resolveCanvasRootDir(rootDir, stateDir), CANVAS_DOCUMENTS_DIR_NAME);
|
|
}
|
|
|
|
export function resolveCanvasDocumentDir(
|
|
documentId: string,
|
|
options?: { rootDir?: string; stateDir?: string },
|
|
): string {
|
|
return path.join(resolveCanvasDocumentsDir(options?.rootDir, options?.stateDir), documentId);
|
|
}
|
|
|
|
export function buildCanvasDocumentEntryUrl(documentId: string, entrypoint: string): string {
|
|
const normalizedEntrypoint = normalizeLogicalPath(entrypoint);
|
|
const encodedEntrypoint = normalizedEntrypoint
|
|
.split("/")
|
|
.map((segment) => encodeURIComponent(segment))
|
|
.join("/");
|
|
return `${CANVAS_HOST_PATH}/${CANVAS_DOCUMENTS_DIR_NAME}/${encodeURIComponent(documentId)}/${encodedEntrypoint}`;
|
|
}
|
|
|
|
export function buildCanvasDocumentAssetUrl(documentId: string, logicalPath: string): string {
|
|
return buildCanvasDocumentEntryUrl(documentId, logicalPath);
|
|
}
|
|
|
|
export function resolveCanvasHttpPathToLocalPath(
|
|
requestPath: string,
|
|
options?: { rootDir?: string; stateDir?: string },
|
|
): string | null {
|
|
const trimmed = requestPath.trim();
|
|
const prefix = `${CANVAS_HOST_PATH}/${CANVAS_DOCUMENTS_DIR_NAME}/`;
|
|
if (!trimmed.startsWith(prefix)) {
|
|
return null;
|
|
}
|
|
const pathWithoutQuery = trimmed.replace(/[?#].*$/, "");
|
|
const relative = pathWithoutQuery.slice(prefix.length);
|
|
const segments = relative
|
|
.split("/")
|
|
.map((segment) => {
|
|
try {
|
|
return decodeURIComponent(segment);
|
|
} catch {
|
|
return segment;
|
|
}
|
|
})
|
|
.filter(Boolean);
|
|
if (segments.length < 2) {
|
|
return null;
|
|
}
|
|
const [rawDocumentId, ...entrySegments] = segments;
|
|
try {
|
|
const documentId = normalizeCanvasDocumentId(rawDocumentId);
|
|
const normalizedEntrypoint = normalizeLogicalPath(entrySegments.join("/"));
|
|
const documentsDir = path.resolve(
|
|
resolveCanvasDocumentsDir(options?.rootDir, options?.stateDir),
|
|
);
|
|
const candidatePath = path.resolve(
|
|
resolveCanvasDocumentDir(documentId, options),
|
|
normalizedEntrypoint,
|
|
);
|
|
if (
|
|
!(candidatePath === documentsDir || candidatePath.startsWith(`${documentsDir}${path.sep}`))
|
|
) {
|
|
return null;
|
|
}
|
|
return candidatePath;
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
async function writeManifest(rootDir: string, manifest: CanvasDocumentManifest): Promise<void> {
|
|
await fs.writeFile(
|
|
path.join(rootDir, "manifest.json"),
|
|
`${JSON.stringify(manifest, null, 2)}\n`,
|
|
"utf8",
|
|
);
|
|
}
|
|
|
|
async function copyAssets(
|
|
rootDir: string,
|
|
assets: CanvasDocumentAsset[] | undefined,
|
|
workspaceDir: string,
|
|
): Promise<CanvasDocumentManifest["assets"]> {
|
|
const copied: CanvasDocumentManifest["assets"] = [];
|
|
for (const asset of assets ?? []) {
|
|
const logicalPath = normalizeLogicalPath(asset.logicalPath);
|
|
const sourcePath = asset.sourcePath.startsWith("~")
|
|
? resolveUserPath(asset.sourcePath)
|
|
: path.isAbsolute(asset.sourcePath)
|
|
? path.resolve(asset.sourcePath)
|
|
: path.resolve(workspaceDir, asset.sourcePath);
|
|
const destination = path.join(rootDir, logicalPath);
|
|
await fs.mkdir(path.dirname(destination), { recursive: true });
|
|
await fs.copyFile(sourcePath, destination);
|
|
copied.push({
|
|
logicalPath,
|
|
...(asset.contentType ? { contentType: asset.contentType } : {}),
|
|
});
|
|
}
|
|
return copied;
|
|
}
|
|
|
|
async function materializeEntrypoint(
|
|
rootDir: string,
|
|
input: CanvasDocumentCreateInput,
|
|
workspaceDir: string,
|
|
): Promise<Pick<CanvasDocumentManifest, "entryUrl" | "localEntrypoint" | "externalUrl">> {
|
|
const entrypoint = input.entrypoint;
|
|
if (!entrypoint) {
|
|
throw new Error("canvas document entrypoint required");
|
|
}
|
|
if (entrypoint.type === "html") {
|
|
const fileName = "index.html";
|
|
await fs.writeFile(path.join(rootDir, fileName), entrypoint.value, "utf8");
|
|
return {
|
|
localEntrypoint: fileName,
|
|
entryUrl: buildCanvasDocumentEntryUrl(path.basename(rootDir), fileName),
|
|
};
|
|
}
|
|
if (entrypoint.type === "url") {
|
|
if (input.kind === "document" && isPdfPathLike(entrypoint.value)) {
|
|
const fileName = "index.html";
|
|
await fs.writeFile(path.join(rootDir, fileName), buildPdfWrapper(entrypoint.value), "utf8");
|
|
return {
|
|
localEntrypoint: fileName,
|
|
externalUrl: entrypoint.value,
|
|
entryUrl: buildCanvasDocumentEntryUrl(path.basename(rootDir), fileName),
|
|
};
|
|
}
|
|
return {
|
|
externalUrl: entrypoint.value,
|
|
entryUrl: entrypoint.value,
|
|
};
|
|
}
|
|
|
|
const resolvedPath = entrypoint.value.startsWith("~")
|
|
? resolveUserPath(entrypoint.value)
|
|
: path.isAbsolute(entrypoint.value)
|
|
? path.resolve(entrypoint.value)
|
|
: path.resolve(workspaceDir, entrypoint.value);
|
|
|
|
if (input.kind === "image" || input.kind === "video_asset") {
|
|
const copiedName = path.basename(resolvedPath);
|
|
await fs.copyFile(resolvedPath, path.join(rootDir, copiedName));
|
|
const wrapper =
|
|
input.kind === "image"
|
|
? `<!doctype html><html><body style="margin:0;background:#0f172a;display:flex;align-items:center;justify-content:center;"><img src="${escapeHtml(copiedName)}" style="max-width:100%;max-height:100vh;object-fit:contain;" /></body></html>`
|
|
: `<!doctype html><html><body style="margin:0;background:#0f172a;"><video src="${escapeHtml(copiedName)}" controls autoplay style="width:100%;height:100vh;object-fit:contain;background:#000;"></video></body></html>`;
|
|
await fs.writeFile(path.join(rootDir, "index.html"), wrapper, "utf8");
|
|
return {
|
|
localEntrypoint: "index.html",
|
|
entryUrl: buildCanvasDocumentEntryUrl(path.basename(rootDir), "index.html"),
|
|
};
|
|
}
|
|
|
|
const fileName = path.basename(resolvedPath);
|
|
await fs.copyFile(resolvedPath, path.join(rootDir, fileName));
|
|
if (input.kind === "document" && isPdfPathLike(fileName)) {
|
|
await fs.writeFile(path.join(rootDir, "index.html"), buildPdfWrapper(fileName), "utf8");
|
|
return {
|
|
localEntrypoint: "index.html",
|
|
entryUrl: buildCanvasDocumentEntryUrl(path.basename(rootDir), "index.html"),
|
|
};
|
|
}
|
|
return {
|
|
localEntrypoint: fileName,
|
|
entryUrl: buildCanvasDocumentEntryUrl(path.basename(rootDir), fileName),
|
|
};
|
|
}
|
|
|
|
export async function createCanvasDocument(
|
|
input: CanvasDocumentCreateInput,
|
|
options?: { stateDir?: string; workspaceDir?: string; canvasRootDir?: string },
|
|
): Promise<CanvasDocumentManifest> {
|
|
const workspaceDir = options?.workspaceDir ?? process.cwd();
|
|
const id = input.id?.trim() ? normalizeCanvasDocumentId(input.id) : canvasDocumentId();
|
|
const rootDir = resolveCanvasDocumentDir(id, {
|
|
stateDir: options?.stateDir,
|
|
rootDir: options?.canvasRootDir,
|
|
});
|
|
await fs.rm(rootDir, { recursive: true, force: true }).catch(() => undefined);
|
|
await fs.mkdir(rootDir, { recursive: true });
|
|
const assets = await copyAssets(rootDir, input.assets, workspaceDir);
|
|
const entry = await materializeEntrypoint(rootDir, input, workspaceDir);
|
|
const manifest: CanvasDocumentManifest = {
|
|
id,
|
|
kind: input.kind,
|
|
...(input.title?.trim() ? { title: input.title.trim() } : {}),
|
|
...(typeof input.preferredHeight === "number"
|
|
? { preferredHeight: input.preferredHeight }
|
|
: {}),
|
|
...(input.surface ? { surface: input.surface } : {}),
|
|
createdAt: new Date().toISOString(),
|
|
entryUrl: entry.entryUrl,
|
|
...(entry.localEntrypoint ? { localEntrypoint: entry.localEntrypoint } : {}),
|
|
...(entry.externalUrl ? { externalUrl: entry.externalUrl } : {}),
|
|
assets,
|
|
};
|
|
await writeManifest(rootDir, manifest);
|
|
return manifest;
|
|
}
|
|
|
|
export async function loadCanvasDocumentManifest(
|
|
documentId: string,
|
|
options?: { stateDir?: string; canvasRootDir?: string },
|
|
): Promise<CanvasDocumentManifest | null> {
|
|
const id = normalizeCanvasDocumentId(documentId);
|
|
const manifestPath = path.join(
|
|
resolveCanvasDocumentDir(id, {
|
|
stateDir: options?.stateDir,
|
|
rootDir: options?.canvasRootDir,
|
|
}),
|
|
"manifest.json",
|
|
);
|
|
try {
|
|
const raw = await fs.readFile(manifestPath, "utf8");
|
|
const parsed = JSON.parse(raw);
|
|
return parsed && typeof parsed === "object" && !Array.isArray(parsed)
|
|
? (parsed as CanvasDocumentManifest)
|
|
: null;
|
|
} catch {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export function resolveCanvasDocumentAssets(
|
|
manifest: CanvasDocumentManifest,
|
|
options?: { baseUrl?: string; stateDir?: string; canvasRootDir?: string },
|
|
): CanvasDocumentResolvedAsset[] {
|
|
const baseUrl = options?.baseUrl?.trim().replace(/\/+$/, "");
|
|
const documentDir = resolveCanvasDocumentDir(manifest.id, {
|
|
stateDir: options?.stateDir,
|
|
rootDir: options?.canvasRootDir,
|
|
});
|
|
return manifest.assets.map((asset) => ({
|
|
logicalPath: asset.logicalPath,
|
|
...(asset.contentType ? { contentType: asset.contentType } : {}),
|
|
localPath: path.join(documentDir, asset.logicalPath),
|
|
url: baseUrl
|
|
? `${baseUrl}${buildCanvasDocumentAssetUrl(manifest.id, asset.logicalPath)}`
|
|
: buildCanvasDocumentAssetUrl(manifest.id, asset.logicalPath),
|
|
}));
|
|
}
|