Files
community-rule/tests/unit/methodRecommendations.test.ts
T
2026-04-20 12:41:10 -06:00

161 lines
4.4 KiB
TypeScript

import { beforeEach, describe, expect, it, vi } from "vitest";
const findManyMock = vi.fn();
vi.mock("../../lib/server/env", () => ({
isDatabaseConfigured: () => true,
}));
vi.mock("../../lib/server/db", () => ({
prisma: {
methodFacet: {
findMany: (...args: unknown[]) => findManyMock(...args),
},
},
}));
import {
listMethodRecommendations,
scoreTemplatesByFacets,
} from "../../lib/server/methodRecommendations";
beforeEach(() => {
findManyMock.mockReset();
});
describe("listMethodRecommendations (CR-88 §9.2)", () => {
it("returns empty rankings when no facets are requested", async () => {
const result = await listMethodRecommendations({
section: "communication",
facets: {},
});
expect(result).toEqual({ rankedSlugs: [], matchesBySlug: {} });
expect(findManyMock).not.toHaveBeenCalled();
});
it("scores methods by counting matched (group, value) pairs", async () => {
findManyMock.mockResolvedValueOnce([
{ slug: "loomio", group: "size", value: "thirteenToOneHundred" },
{ slug: "loomio", group: "orgType", value: "workersCoop" },
{ slug: "in-person", group: "size", value: "thirteenToOneHundred" },
]);
const result = await listMethodRecommendations({
section: "communication",
facets: {
size: ["thirteenToOneHundred"],
orgType: ["workersCoop"],
},
});
expect(result).toEqual({
rankedSlugs: ["loomio", "in-person"],
matchesBySlug: {
loomio: {
score: 2,
matchedFacets: ["size:thirteenToOneHundred", "orgType:workersCoop"],
},
"in-person": {
score: 1,
matchedFacets: ["size:thirteenToOneHundred"],
},
},
});
});
it("returns null on query failure (caller falls back to authoring order)", async () => {
findManyMock.mockRejectedValueOnce(new Error("db down"));
const result = await listMethodRecommendations({
section: "communication",
facets: { size: ["oneMember"] },
});
expect(result).toBeNull();
});
it("dedupes (group, value) so the same row never double-counts", async () => {
findManyMock.mockResolvedValueOnce([
{ slug: "loomio", group: "size", value: "twoToFive" },
{ slug: "loomio", group: "size", value: "twoToFive" },
]);
const result = await listMethodRecommendations({
section: "communication",
facets: { size: ["twoToFive"] },
});
expect(result?.matchesBySlug["loomio"]?.score).toBe(1);
});
});
describe("scoreTemplatesByFacets (CR-88 §9.1)", () => {
it("aggregates per-method matches per template", async () => {
findManyMock.mockResolvedValueOnce([
{
section: "communication",
slug: "loomio",
group: "size",
value: "twoToFive",
},
{
section: "decisionApproaches",
slug: "consensus-decision-making",
group: "orgType",
value: "workersCoop",
},
]);
const result = await scoreTemplatesByFacets({
facets: {
size: ["twoToFive"],
orgType: ["workersCoop"],
},
templateMethods: [
{
templateSlug: "consensus",
methods: [
{ section: "communication", slug: "loomio" },
{
section: "decisionApproaches",
slug: "consensus-decision-making",
},
],
},
{
templateSlug: "monarch",
methods: [
{
section: "decisionApproaches",
slug: "benevolent-dictator",
},
],
},
],
});
expect(result).toEqual([
{
templateSlug: "consensus",
score: 2,
matchedFacets: [
"communication:loomio:size:twoToFive",
"decisionApproaches:consensus-decision-making:orgType:workersCoop",
],
},
{ templateSlug: "monarch", score: 0, matchedFacets: [] },
]);
});
it("returns 0-score entries for every template when facets are empty", async () => {
const result = await scoreTemplatesByFacets({
facets: {},
templateMethods: [
{ templateSlug: "consensus", methods: [] },
{ templateSlug: "monarch", methods: [] },
],
});
expect(result).toEqual([
{ templateSlug: "consensus", score: 0, matchedFacets: [] },
{ templateSlug: "monarch", score: 0, matchedFacets: [] },
]);
expect(findManyMock).not.toHaveBeenCalled();
});
});