mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-25 11:39:34 +00:00
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 head132f14a381. - Required merge gates passed before the squash merge. Prepared head SHA:132f14a381Review: 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>
247 lines
8.3 KiB
TypeScript
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");
|
|
});
|
|
});
|