feat(core-eslint): usecase-must-have-test-file rule

This commit is contained in:
2026-05-12 23:21:38 +02:00
parent b7bb37023f
commit d585b59590
2 changed files with 74 additions and 0 deletions

View File

@@ -0,0 +1,32 @@
import fs from "node:fs";
/** @type {import("eslint").Rule.RuleModule} */
export default {
meta: {
type: "problem",
docs: {
description:
"Every *.use-case.ts file must have a sibling *.use-case.test.ts (TDD discipline).",
},
schema: [],
messages: {
missingTestFile:
"Use case {{filename}} has no sibling test file at {{expected}}. Write the red test first.",
},
},
create(context) {
return {
Program(node) {
const filename = context.filename;
if (!filename.endsWith(".use-case.ts")) return;
const expected = filename.replace(/\.use-case\.ts$/, ".use-case.test.ts");
if (fs.existsSync(expected)) return;
context.report({
node,
messageId: "missingTestFile",
data: { filename: filename.split("/").pop(), expected: expected.split("/").pop() },
});
},
};
},
};

View File

@@ -0,0 +1,42 @@
import { describe, it } from "vitest";
import { RuleTester } from "eslint";
import path from "node:path";
import os from "node:os";
import fs from "node:fs";
import rule from "./usecase-must-have-test-file.js";
function makeUseCaseFixture({ withTest }) {
const dir = fs.mkdtempSync(path.join(os.tmpdir(), "umht-"));
const useCase = path.join(dir, "sign-in.use-case.ts");
fs.writeFileSync(useCase, `export const signInUseCase = () => async () => {};`);
if (withTest) {
fs.writeFileSync(path.join(dir, "sign-in.use-case.test.ts"), `import { it } from "vitest"; it("works", () => {});`);
}
return { useCase };
}
const tester = new RuleTester({ languageOptions: { ecmaVersion: "latest", sourceType: "module" } });
describe("usecase-must-have-test-file", () => {
it("passes when a sibling .test.ts exists", () => {
const { useCase } = makeUseCaseFixture({ withTest: true });
tester.run("usecase-must-have-test-file", rule, {
valid: [{ filename: useCase, code: fs.readFileSync(useCase, "utf8") }],
invalid: [],
});
});
it("fires when no sibling test file exists", () => {
const { useCase } = makeUseCaseFixture({ withTest: false });
tester.run("usecase-must-have-test-file", rule, {
valid: [],
invalid: [
{
filename: useCase,
code: fs.readFileSync(useCase, "utf8"),
errors: [{ messageId: "missingTestFile" }],
},
],
});
});
});