mirror of
https://github.com/openclaw/openclaw.git
synced 2026-03-13 19:10:39 +00:00
112 lines
4.1 KiB
TypeScript
112 lines
4.1 KiB
TypeScript
import fs from "node:fs/promises";
|
|
import os from "node:os";
|
|
import path from "node:path";
|
|
import { afterAll, beforeAll, describe, expect, it } from "vitest";
|
|
import { loadExtraBootstrapFiles, loadExtraBootstrapFilesWithDiagnostics } from "./workspace.js";
|
|
|
|
describe("loadExtraBootstrapFiles", () => {
|
|
let fixtureRoot = "";
|
|
let fixtureCount = 0;
|
|
|
|
const createWorkspaceDir = async (prefix: string) => {
|
|
const dir = path.join(fixtureRoot, `${prefix}-${fixtureCount++}`);
|
|
await fs.mkdir(dir, { recursive: true });
|
|
return dir;
|
|
};
|
|
|
|
beforeAll(async () => {
|
|
fixtureRoot = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-extra-bootstrap-"));
|
|
});
|
|
|
|
afterAll(async () => {
|
|
if (fixtureRoot) {
|
|
await fs.rm(fixtureRoot, { recursive: true, force: true });
|
|
}
|
|
});
|
|
|
|
it("loads recognized bootstrap files from glob patterns", async () => {
|
|
const workspaceDir = await createWorkspaceDir("glob");
|
|
const packageDir = path.join(workspaceDir, "packages", "core");
|
|
await fs.mkdir(packageDir, { recursive: true });
|
|
await fs.writeFile(path.join(packageDir, "TOOLS.md"), "tools", "utf-8");
|
|
await fs.writeFile(path.join(packageDir, "README.md"), "not bootstrap", "utf-8");
|
|
|
|
const files = await loadExtraBootstrapFiles(workspaceDir, ["packages/*/*"]);
|
|
|
|
expect(files).toHaveLength(1);
|
|
expect(files[0]?.name).toBe("TOOLS.md");
|
|
expect(files[0]?.content).toBe("tools");
|
|
});
|
|
|
|
it("keeps path-traversal attempts outside workspace excluded", async () => {
|
|
const rootDir = await createWorkspaceDir("root");
|
|
const workspaceDir = path.join(rootDir, "workspace");
|
|
const outsideDir = path.join(rootDir, "outside");
|
|
await fs.mkdir(workspaceDir, { recursive: true });
|
|
await fs.mkdir(outsideDir, { recursive: true });
|
|
await fs.writeFile(path.join(outsideDir, "AGENTS.md"), "outside", "utf-8");
|
|
|
|
const files = await loadExtraBootstrapFiles(workspaceDir, ["../outside/AGENTS.md"]);
|
|
|
|
expect(files).toHaveLength(0);
|
|
});
|
|
|
|
it("supports symlinked workspace roots with realpath checks", async () => {
|
|
if (process.platform === "win32") {
|
|
return;
|
|
}
|
|
|
|
const rootDir = await createWorkspaceDir("symlink");
|
|
const realWorkspace = path.join(rootDir, "real-workspace");
|
|
const linkedWorkspace = path.join(rootDir, "linked-workspace");
|
|
await fs.mkdir(realWorkspace, { recursive: true });
|
|
await fs.writeFile(path.join(realWorkspace, "AGENTS.md"), "linked agents", "utf-8");
|
|
await fs.symlink(realWorkspace, linkedWorkspace, "dir");
|
|
|
|
const files = await loadExtraBootstrapFiles(linkedWorkspace, ["AGENTS.md"]);
|
|
|
|
expect(files).toHaveLength(1);
|
|
expect(files[0]?.name).toBe("AGENTS.md");
|
|
expect(files[0]?.content).toBe("linked agents");
|
|
});
|
|
|
|
it("rejects hardlinked aliases to files outside workspace", async () => {
|
|
if (process.platform === "win32") {
|
|
return;
|
|
}
|
|
|
|
const rootDir = await createWorkspaceDir("hardlink");
|
|
const workspaceDir = path.join(rootDir, "workspace");
|
|
const outsideDir = path.join(rootDir, "outside");
|
|
await fs.mkdir(workspaceDir, { recursive: true });
|
|
await fs.mkdir(outsideDir, { recursive: true });
|
|
const outsideFile = path.join(outsideDir, "AGENTS.md");
|
|
const linkedFile = path.join(workspaceDir, "AGENTS.md");
|
|
await fs.writeFile(outsideFile, "outside", "utf-8");
|
|
try {
|
|
await fs.link(outsideFile, linkedFile);
|
|
} catch (err) {
|
|
if ((err as NodeJS.ErrnoException).code === "EXDEV") {
|
|
return;
|
|
}
|
|
throw err;
|
|
}
|
|
|
|
const files = await loadExtraBootstrapFiles(workspaceDir, ["AGENTS.md"]);
|
|
expect(files).toHaveLength(0);
|
|
});
|
|
|
|
it("skips oversized bootstrap files and reports diagnostics", async () => {
|
|
const workspaceDir = await createWorkspaceDir("oversized");
|
|
const payload = "x".repeat(2 * 1024 * 1024 + 1);
|
|
await fs.writeFile(path.join(workspaceDir, "AGENTS.md"), payload, "utf-8");
|
|
|
|
const { files, diagnostics } = await loadExtraBootstrapFilesWithDiagnostics(workspaceDir, [
|
|
"AGENTS.md",
|
|
]);
|
|
|
|
expect(files).toHaveLength(0);
|
|
expect(diagnostics.some((d) => d.reason === "security")).toBe(true);
|
|
});
|
|
});
|