Files
openclaw/test/scripts/report-test-temp-creations.test.ts
Mason Huang 06431fd99b test: add temp directory helper guidance (#87298)
Summary:
- Merged test: add temp directory helper guidance after ClawSweeper review.

Automerge notes:
- PR branch already contained follow-up commit before automerge: fix(scripts): honor temp report failure mode
- PR branch already contained follow-up commit before automerge: fix(scripts): reduce temp report noise
- PR branch already contained follow-up commit before automerge: fix(scripts): cover test support temp reports
- PR branch already contained follow-up commit before automerge: fix(scripts): report temp use in test helpers
- PR branch already contained follow-up commit before automerge: fix(scripts): broaden temp report test surface
- PR branch already contained follow-up commit before automerge: fix(scripts): cover nested test temp reports

Validation:
- ClawSweeper review passed for head 132f14a381.
- Required merge gates passed before the squash merge.

Prepared head SHA: 132f14a381
Review: https://github.com/openclaw/openclaw/pull/87298#issuecomment-4704338581

Co-authored-by: masonxhuang <masonxhuang@tencent.com>
Co-authored-by: Mason Huang <masonxhuang@tencent.com>
Co-authored-by: clawsweeper <274271284+clawsweeper[bot]@users.noreply.github.com>
Co-authored-by: clawsweeper[bot] <274271284+clawsweeper[bot]@users.noreply.github.com>
Approved-by: hxy91819
Co-authored-by: hxy91819 <8814856+hxy91819@users.noreply.github.com>
2026-06-15 03:51:43 +00:00

247 lines
8.3 KiB
TypeScript

import { execFileSync, spawnSync } from "node:child_process";
import fs from "node:fs";
import path from "node:path";
import { afterEach, describe, expect, it } from "vitest";
import {
collectTempCreationFindingsFromDiff,
formatGithubWarning,
} from "../../scripts/report-test-temp-creations.mjs";
import { createTempDirTracker } from "../helpers/temp-dir.js";
const repoRoot = process.cwd();
const tempDirs = createTempDirTracker();
const nestedGitEnvKeys = [
"GIT_ALTERNATE_OBJECT_DIRECTORIES",
"GIT_DIR",
"GIT_INDEX_FILE",
"GIT_OBJECT_DIRECTORY",
"GIT_QUARANTINE_PATH",
"GIT_WORK_TREE",
] as const;
function createNestedGitEnv(): NodeJS.ProcessEnv {
const env = {
...process.env,
GIT_CONFIG_NOSYSTEM: "1",
GIT_TERMINAL_PROMPT: "0",
};
for (const key of nestedGitEnvKeys) {
delete env[key];
}
return env;
}
afterEach(() => {
tempDirs.cleanup();
});
describe("report-test-temp-creations", () => {
it("keeps a non-executed warning fixture for changed-gate proof", () => {
// openclaw-temp-dir: allow test fixture for the temp warning report
const warningFixture = 'fs.mkdtempSync("openclaw-warning-fixture-")';
expect(warningFixture).toContain("mkdtempSync");
});
it("reports added bare temp creation lines using changed-lane test path scope", () => {
const bareTempSource = [
"const tempRoot = fs.",
"mkdtemp",
'Sync(path.join(os.tmpdir(), "case-"));',
].join("");
const mkdtempSource = ["const tempRoot = fs.", "mkdtemp", 'Sync("case-");'].join("");
const diff = [
"diff --git a/src/example.test.ts b/src/example.test.ts",
"--- a/src/example.test.ts",
"+++ b/src/example.test.ts",
"@@ -10,0 +11,3 @@",
`+${bareTempSource}`,
'+const helperRoot = makeTempDir(tempDirs, "case-");',
"+console.log(tempRoot, helperRoot);",
"diff --git a/src/example.ts b/src/example.ts",
"--- a/src/example.ts",
"+++ b/src/example.ts",
"@@ -4,0 +5,1 @@",
`+${["const productionTemp = fs.", "mkdtemp", 'Sync("case-");'].join("")}`,
"diff --git a/test/helper.test-support.mjs b/test/helper.test-support.mjs",
"--- a/test/helper.test-support.mjs",
"+++ b/test/helper.test-support.mjs",
"@@ -1,0 +2,1 @@",
`+${mkdtempSource}`,
"diff --git a/test/helpers/temp-fixture.ts b/test/helpers/temp-fixture.ts",
"--- a/test/helpers/temp-fixture.ts",
"+++ b/test/helpers/temp-fixture.ts",
"@@ -1,0 +2,1 @@",
`+${mkdtempSource}`,
"diff --git a/test/helpers/temp-dir.ts b/test/helpers/temp-dir.ts",
"--- a/test/helpers/temp-dir.ts",
"+++ b/test/helpers/temp-dir.ts",
"@@ -1,0 +2,1 @@",
`+${mkdtempSource}`,
"diff --git a/packages/foo/__tests__/helper.ts b/packages/foo/__tests__/helper.ts",
"--- a/packages/foo/__tests__/helper.ts",
"+++ b/packages/foo/__tests__/helper.ts",
"@@ -1,0 +2,1 @@",
`+${mkdtempSource}`,
"diff --git a/extensions/discord/src/monitor/message-handler.test-helpers.ts b/extensions/discord/src/monitor/message-handler.test-helpers.ts",
"--- a/extensions/discord/src/monitor/message-handler.test-helpers.ts",
"+++ b/extensions/discord/src/monitor/message-handler.test-helpers.ts",
"@@ -1,0 +2,1 @@",
`+${mkdtempSource}`,
].join("\n");
expect(collectTempCreationFindingsFromDiff(diff)).toEqual([
{
file: "src/example.test.ts",
line: 11,
reason: "new mkdtemp temp directory creation",
source: bareTempSource,
},
{
file: "test/helper.test-support.mjs",
line: 2,
reason: "new mkdtemp temp directory creation",
source: mkdtempSource,
},
{
file: "test/helpers/temp-fixture.ts",
line: 2,
reason: "new mkdtemp temp directory creation",
source: mkdtempSource,
},
{
file: "packages/foo/__tests__/helper.ts",
line: 2,
reason: "new mkdtemp temp directory creation",
source: mkdtempSource,
},
]);
});
it("honors explicit allow comments with reasons", () => {
const mkdtempCall = ["fs.", "mkdtemp", 'Sync("case-")'].join("");
const tmpDirCall = ["tmp.", "dir", 'Sync({ prefix: "case-" })'].join("");
const allowedSource = `const allowed = ${mkdtempCall};`;
const inlineAllowedSource = `const inlineAllowed = ${tmpDirCall}; // openclaw-temp-dir: allow verifies tmp API behavior`;
const blockedSource = `const blocked = ${mkdtempCall};`;
const stringMarkerSource = `const stringMarker = ${mkdtempCall}; const note = "openclaw-temp-dir: allow quoted text";`;
const emptyReasonSource = `const emptyReason = ${mkdtempCall};`;
const diff = [
"diff --git a/test/helpers/raw-temp.test.ts b/test/helpers/raw-temp.test.ts",
"--- a/test/helpers/raw-temp.test.ts",
"+++ b/test/helpers/raw-temp.test.ts",
"@@ -1,0 +2,5 @@",
"+// openclaw-temp-dir: allow verifies raw fs cleanup behavior",
`+${allowedSource}`,
`+${inlineAllowedSource}`,
`+${blockedSource}`,
`+${stringMarkerSource}`,
"diff --git a/test/helpers/empty-allow.test.ts b/test/helpers/empty-allow.test.ts",
"--- a/test/helpers/empty-allow.test.ts",
"+++ b/test/helpers/empty-allow.test.ts",
"@@ -1,0 +2,2 @@",
"+// openclaw-temp-dir: allow",
`+${emptyReasonSource}`,
].join("\n");
expect(collectTempCreationFindingsFromDiff(diff)).toEqual([
{
file: "test/helpers/raw-temp.test.ts",
line: 5,
reason: "new mkdtemp temp directory creation",
source: blockedSource,
},
{
file: "test/helpers/raw-temp.test.ts",
line: 6,
reason: "new mkdtemp temp directory creation",
source: stringMarkerSource,
},
{
file: "test/helpers/empty-allow.test.ts",
line: 3,
reason: "new mkdtemp temp directory creation",
source: emptyReasonSource,
},
]);
});
it("prints help with usage, outputs, and examples", () => {
const output = execFileSync(
process.execPath,
[path.join(repoRoot, "scripts", "report-test-temp-creations.mjs"), "--help"],
{
cwd: repoRoot,
encoding: "utf8",
},
);
expect(output).toContain("Usage: node scripts/report-test-temp-creations.mjs");
expect(output).toContain("Outputs:");
expect(output).toContain("--no-merge-base");
expect(output).toContain("Examples:");
});
it("formats GitHub warning annotations for CI report mode", () => {
expect(
formatGithubWarning({
file: "test/helpers/temp,fixture.ts",
line: 12,
reason: "new mkdtemp temp directory creation",
// openclaw-temp-dir: allow test fixture for GitHub warning formatting
source: "const tempRoot = fs.mkdtempSync();",
}),
).toBe(
"::warning file=test/helpers/temp%2Cfixture.ts,line=12::new mkdtemp temp directory creation: prefer test/helpers/temp-dir.ts for new test-owned temp directories.",
);
});
it("exits non-zero for staged findings when requested", () => {
const root = tempDirs.make("openclaw-temp-report-");
const env = createNestedGitEnv();
execFileSync("git", ["init", "-q", "--initial-branch=main"], { cwd: root, env });
fs.mkdirSync(path.join(root, "test", "helpers"), { recursive: true });
fs.writeFileSync(path.join(root, "test", "helpers", "case.ts"), "const value = 1;\n", "utf8");
execFileSync("git", ["add", "test/helpers/case.ts"], { cwd: root, env });
execFileSync(
"git",
[
"-c",
"user.email=test@example.com",
"-c",
"user.name=Test User",
"commit",
"-q",
"-m",
"initial",
],
{ cwd: root, env },
);
const source = [
"const tempRoot = fs.",
"mkdtemp",
'Sync(path.join(os.tmpdir(), "case-"));\n',
].join("");
fs.appendFileSync(path.join(root, "test", "helpers", "case.ts"), source, "utf8");
execFileSync("git", ["add", "test/helpers/case.ts"], { cwd: root, env });
const result = spawnSync(
process.execPath,
[
path.join(repoRoot, "scripts", "report-test-temp-creations.mjs"),
"--staged",
"--fail-on-findings",
],
{
cwd: root,
encoding: "utf8",
env,
},
);
expect(result.status).toBe(1);
expect(result.stderr).toContain("test/helpers/case.ts");
});
});