test: add executable path and backup summary coverage

This commit is contained in:
Peter Steinberger 2026-03-13 19:48:06 +00:00
parent c351b49aed
commit 549cb65ba4
2 changed files with 134 additions and 0 deletions

View File

@ -0,0 +1,84 @@
import { describe, expect, it } from "vitest";
import { formatBackupCreateSummary, type BackupCreateResult } from "./backup-create.js";
function makeResult(overrides: Partial<BackupCreateResult> = {}): BackupCreateResult {
return {
createdAt: "2026-01-01T00:00:00.000Z",
archiveRoot: "openclaw-backup-2026-01-01",
archivePath: "/tmp/openclaw-backup.tar.gz",
dryRun: false,
includeWorkspace: true,
onlyConfig: false,
verified: false,
assets: [],
skipped: [],
...overrides,
};
}
describe("formatBackupCreateSummary", () => {
it("formats created archives with included and skipped paths", () => {
const lines = formatBackupCreateSummary(
makeResult({
verified: true,
assets: [
{
kind: "state",
sourcePath: "/state",
archivePath: "archive/state",
displayPath: "~/.openclaw",
},
],
skipped: [
{
kind: "workspace",
sourcePath: "/workspace",
displayPath: "~/Projects/openclaw",
reason: "covered",
coveredBy: "~/.openclaw",
},
],
}),
);
expect(lines).toEqual([
"Backup archive: /tmp/openclaw-backup.tar.gz",
"Included 1 path:",
"- state: ~/.openclaw",
"Skipped 1 path:",
"- workspace: ~/Projects/openclaw (covered by ~/.openclaw)",
"Created /tmp/openclaw-backup.tar.gz",
"Archive verification: passed",
]);
});
it("formats dry runs and pluralized counts", () => {
const lines = formatBackupCreateSummary(
makeResult({
dryRun: true,
assets: [
{
kind: "config",
sourcePath: "/config",
archivePath: "archive/config",
displayPath: "~/.openclaw/config.json",
},
{
kind: "oauth",
sourcePath: "/oauth",
archivePath: "archive/oauth",
displayPath: "~/.openclaw/oauth",
},
],
}),
);
expect(lines).toEqual([
"Backup archive: /tmp/openclaw-backup.tar.gz",
"Included 2 paths:",
"- config: ~/.openclaw/config.json",
"- oauth: ~/.openclaw/oauth",
"Dry run only; archive was not written.",
]);
});
});

View File

@ -0,0 +1,50 @@
import fs from "node:fs/promises";
import os from "node:os";
import path from "node:path";
import { describe, expect, it } from "vitest";
import {
isExecutableFile,
resolveExecutableFromPathEnv,
resolveExecutablePath,
} from "./executable-path.js";
describe("executable path helpers", () => {
it("detects executable files and rejects directories or non-executables", async () => {
const base = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-exec-path-"));
const execPath = path.join(base, "tool");
const filePath = path.join(base, "plain.txt");
const dirPath = path.join(base, "dir");
await fs.writeFile(execPath, "#!/bin/sh\nexit 0\n", "utf8");
await fs.chmod(execPath, 0o755);
await fs.writeFile(filePath, "nope", "utf8");
await fs.mkdir(dirPath);
expect(isExecutableFile(execPath)).toBe(true);
expect(isExecutableFile(filePath)).toBe(false);
expect(isExecutableFile(dirPath)).toBe(false);
expect(isExecutableFile(path.join(base, "missing"))).toBe(false);
});
it("resolves executables from PATH entries and cwd-relative paths", async () => {
const base = await fs.mkdtemp(path.join(os.tmpdir(), "openclaw-exec-path-"));
const binDir = path.join(base, "bin");
const cwd = path.join(base, "cwd");
await fs.mkdir(binDir, { recursive: true });
await fs.mkdir(cwd, { recursive: true });
const pathTool = path.join(binDir, "runner");
const cwdTool = path.join(cwd, "local-tool");
await fs.writeFile(pathTool, "#!/bin/sh\nexit 0\n", "utf8");
await fs.writeFile(cwdTool, "#!/bin/sh\nexit 0\n", "utf8");
await fs.chmod(pathTool, 0o755);
await fs.chmod(cwdTool, 0o755);
expect(resolveExecutableFromPathEnv("runner", `${binDir}${path.delimiter}/usr/bin`)).toBe(
pathTool,
);
expect(resolveExecutableFromPathEnv("missing", binDir)).toBeUndefined();
expect(resolveExecutablePath("./local-tool", { cwd })).toBe(cwdTool);
expect(resolveExecutablePath("runner", { env: { PATH: binDir } })).toBe(pathTool);
expect(resolveExecutablePath("missing", { env: { PATH: binDir } })).toBeUndefined();
});
});