mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-09 16:50:43 +00:00
* refactor: extract filesystem safety primitives * refactor: use fs-safe for file access helpers * refactor: reuse fs-safe for media reads * refactor: use fs-safe for image reads * refactor: reuse fs-safe in qqbot media opener * refactor: reuse fs-safe for local media checks * refactor: consume cleaner fs-safe api * refactor: align fs-safe json option names * fix: preserve fs-safe migration contracts * refactor: use fs-safe primitive subpaths * refactor: use grouped fs-safe subpaths * refactor: align fs-safe api usage * refactor: adapt private state store api * chore: refresh proof gate * refactor: follow fs-safe json api split * refactor: follow reduced fs-safe surface * build: default fs-safe python helper off * fix: preserve fs-safe plugin sdk aliases * refactor: consolidate fs-safe usage * refactor: unify fs-safe store usage * refactor: trim fs-safe temp workspace usage * refactor: hide low-level fs-safe primitives * build: use published fs-safe package * fix: preserve outbound recovery durability after rebase * chore: refresh pr checks
93 lines
3.0 KiB
TypeScript
93 lines
3.0 KiB
TypeScript
import fsSync from "node:fs";
|
|
import fs from "node:fs/promises";
|
|
import path from "node:path";
|
|
import { describe, expect, it } from "vitest";
|
|
import { resolvePreferredOpenClawTmpDir } from "../infra/tmp-openclaw-dir.js";
|
|
import { buildRandomTempFilePath, withTempDownloadPath } from "./temp-path.js";
|
|
|
|
function expectPathInsideTmpRoot(resultPath: string) {
|
|
const tmpRoot = fsSync.realpathSync(resolvePreferredOpenClawTmpDir());
|
|
let resolved = path.resolve(resultPath);
|
|
try {
|
|
resolved = path.join(fsSync.realpathSync(path.dirname(resultPath)), path.basename(resultPath));
|
|
} catch {
|
|
// The temp parent is intentionally gone after withTempDownloadPath cleanup.
|
|
}
|
|
const rel = path.relative(tmpRoot, resolved);
|
|
expect(rel === ".." || rel.startsWith(`..${path.sep}`)).toBe(false);
|
|
expect(resultPath).not.toContain("..");
|
|
}
|
|
|
|
describe("buildRandomTempFilePath", () => {
|
|
it.each([
|
|
{
|
|
name: "builds deterministic paths when now/uuid are provided",
|
|
input: {
|
|
prefix: "line-media",
|
|
extension: ".jpg",
|
|
tmpDir: "/tmp",
|
|
now: 123,
|
|
uuid: "abc",
|
|
},
|
|
expectedPath: path.join("/tmp", "line-media-123-abc.jpg"),
|
|
expectedBasename: "line-media-123-abc.jpg",
|
|
verifyInsideTmpRoot: false,
|
|
},
|
|
{
|
|
name: "sanitizes prefix and extension to avoid path traversal segments",
|
|
input: {
|
|
prefix: "../../channels/../media",
|
|
extension: "/../.jpg",
|
|
now: 123,
|
|
uuid: "abc",
|
|
},
|
|
expectedBasename: "channels-media-123-abc.jpg",
|
|
verifyInsideTmpRoot: true,
|
|
},
|
|
])("$name", ({ input, expectedPath, expectedBasename, verifyInsideTmpRoot }) => {
|
|
const result = buildRandomTempFilePath(input);
|
|
if (expectedPath) {
|
|
expect(result).toBe(expectedPath);
|
|
}
|
|
expect(path.basename(result)).toBe(expectedBasename);
|
|
if (verifyInsideTmpRoot) {
|
|
expectPathInsideTmpRoot(result);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("withTempDownloadPath", () => {
|
|
it.each([
|
|
{
|
|
name: "creates a temp path under tmp dir and cleans up the temp directory",
|
|
input: { prefix: "line-media" },
|
|
expectCleanup: true,
|
|
expectedBasename: undefined,
|
|
},
|
|
{
|
|
name: "sanitizes prefix and fileName",
|
|
input: { prefix: "../../channels/../media", fileName: "../../evil.bin" },
|
|
expectCleanup: false,
|
|
expectedBasename: "evil.bin",
|
|
},
|
|
])("$name", async ({ input, expectCleanup, expectedBasename }) => {
|
|
let capturedPath = "";
|
|
await withTempDownloadPath(input, async (tmpPath) => {
|
|
capturedPath = tmpPath;
|
|
if (expectCleanup) {
|
|
await fs.writeFile(tmpPath, "ok");
|
|
}
|
|
});
|
|
|
|
expectPathInsideTmpRoot(capturedPath);
|
|
if (expectedBasename) {
|
|
expect(path.basename(capturedPath)).toBe(expectedBasename);
|
|
} else {
|
|
expect(capturedPath).toContain(path.join(resolvePreferredOpenClawTmpDir(), "line-media-"));
|
|
}
|
|
if (expectCleanup) {
|
|
await expect(fs.stat(capturedPath)).rejects.toMatchObject({ code: "ENOENT" });
|
|
}
|
|
});
|
|
});
|