vskill 0.2.26 → 0.2.28

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (77) hide show
  1. package/dist/commands/__tests__/eval-router.test.d.ts +1 -0
  2. package/dist/commands/__tests__/eval-router.test.js +60 -0
  3. package/dist/commands/__tests__/eval-router.test.js.map +1 -0
  4. package/dist/commands/add.js +113 -19
  5. package/dist/commands/add.js.map +1 -1
  6. package/dist/commands/eval/__tests__/coverage.test.d.ts +1 -0
  7. package/dist/commands/eval/__tests__/coverage.test.js +122 -0
  8. package/dist/commands/eval/__tests__/coverage.test.js.map +1 -0
  9. package/dist/commands/eval/__tests__/generate-all.test.d.ts +1 -0
  10. package/dist/commands/eval/__tests__/generate-all.test.js +133 -0
  11. package/dist/commands/eval/__tests__/generate-all.test.js.map +1 -0
  12. package/dist/commands/eval/__tests__/init.test.d.ts +1 -0
  13. package/dist/commands/eval/__tests__/init.test.js +116 -0
  14. package/dist/commands/eval/__tests__/init.test.js.map +1 -0
  15. package/dist/commands/eval/__tests__/run.test.d.ts +1 -0
  16. package/dist/commands/eval/__tests__/run.test.js +149 -0
  17. package/dist/commands/eval/__tests__/run.test.js.map +1 -0
  18. package/dist/commands/eval/coverage.d.ts +1 -0
  19. package/dist/commands/eval/coverage.js +79 -0
  20. package/dist/commands/eval/coverage.js.map +1 -0
  21. package/dist/commands/eval/generate-all.d.ts +1 -0
  22. package/dist/commands/eval/generate-all.js +64 -0
  23. package/dist/commands/eval/generate-all.js.map +1 -0
  24. package/dist/commands/eval/init.d.ts +1 -0
  25. package/dist/commands/eval/init.js +38 -0
  26. package/dist/commands/eval/init.js.map +1 -0
  27. package/dist/commands/eval/run.d.ts +1 -0
  28. package/dist/commands/eval/run.js +107 -0
  29. package/dist/commands/eval/run.js.map +1 -0
  30. package/dist/commands/eval.d.ts +4 -0
  31. package/dist/commands/eval.js +48 -0
  32. package/dist/commands/eval.js.map +1 -0
  33. package/dist/eval/__tests__/benchmark.test.d.ts +1 -0
  34. package/dist/eval/__tests__/benchmark.test.js +65 -0
  35. package/dist/eval/__tests__/benchmark.test.js.map +1 -0
  36. package/dist/eval/__tests__/judge.test.d.ts +1 -0
  37. package/dist/eval/__tests__/judge.test.js +45 -0
  38. package/dist/eval/__tests__/judge.test.js.map +1 -0
  39. package/dist/eval/__tests__/llm.test.d.ts +1 -0
  40. package/dist/eval/__tests__/llm.test.js +85 -0
  41. package/dist/eval/__tests__/llm.test.js.map +1 -0
  42. package/dist/eval/__tests__/prompt-builder.test.d.ts +1 -0
  43. package/dist/eval/__tests__/prompt-builder.test.js +72 -0
  44. package/dist/eval/__tests__/prompt-builder.test.js.map +1 -0
  45. package/dist/eval/__tests__/schema.test.d.ts +1 -0
  46. package/dist/eval/__tests__/schema.test.js +209 -0
  47. package/dist/eval/__tests__/schema.test.js.map +1 -0
  48. package/dist/eval/__tests__/skill-scanner.test.d.ts +1 -0
  49. package/dist/eval/__tests__/skill-scanner.test.js +78 -0
  50. package/dist/eval/__tests__/skill-scanner.test.js.map +1 -0
  51. package/dist/eval/benchmark.d.ts +22 -0
  52. package/dist/eval/benchmark.js +24 -0
  53. package/dist/eval/benchmark.js.map +1 -0
  54. package/dist/eval/judge.d.ts +9 -0
  55. package/dist/eval/judge.js +40 -0
  56. package/dist/eval/judge.js.map +1 -0
  57. package/dist/eval/llm.d.ts +5 -0
  58. package/dist/eval/llm.js +34 -0
  59. package/dist/eval/llm.js.map +1 -0
  60. package/dist/eval/prompt-builder.d.ts +3 -0
  61. package/dist/eval/prompt-builder.js +155 -0
  62. package/dist/eval/prompt-builder.js.map +1 -0
  63. package/dist/eval/schema.d.ts +26 -0
  64. package/dist/eval/schema.js +128 -0
  65. package/dist/eval/schema.js.map +1 -0
  66. package/dist/eval/skill-scanner.d.ts +8 -0
  67. package/dist/eval/skill-scanner.js +44 -0
  68. package/dist/eval/skill-scanner.js.map +1 -0
  69. package/dist/index.js +9 -0
  70. package/dist/index.js.map +1 -1
  71. package/dist/marketplace/index.d.ts +2 -2
  72. package/dist/marketplace/index.js +1 -1
  73. package/dist/marketplace/index.js.map +1 -1
  74. package/dist/marketplace/marketplace.d.ts +13 -0
  75. package/dist/marketplace/marketplace.js +35 -0
  76. package/dist/marketplace/marketplace.js.map +1 -1
  77. package/package.json +2 -1
@@ -0,0 +1,4 @@
1
+ export declare function evalCommand(subcommand: string, target?: string, opts?: {
2
+ force?: boolean;
3
+ root?: string;
4
+ }): Promise<void>;
@@ -0,0 +1,48 @@
1
+ // ---------------------------------------------------------------------------
2
+ // vskill eval -- subcommand router
3
+ // ---------------------------------------------------------------------------
4
+ import { join, resolve } from "node:path";
5
+ import { red, dim } from "../utils/output.js";
6
+ export async function evalCommand(subcommand, target, opts = {}) {
7
+ const root = opts.root ? resolve(opts.root) : resolve("plugins");
8
+ switch (subcommand) {
9
+ case "init": {
10
+ if (!target) {
11
+ console.error(red("Usage: vskill eval init <plugin>/<skill>"));
12
+ process.exit(1);
13
+ }
14
+ const skillDir = resolveSkillDir(root, target);
15
+ const { runEvalInit } = await import("./eval/init.js");
16
+ return runEvalInit(skillDir, !!opts.force);
17
+ }
18
+ case "run": {
19
+ if (!target) {
20
+ console.error(red("Usage: vskill eval run <plugin>/<skill>"));
21
+ process.exit(1);
22
+ }
23
+ const skillDir = resolveSkillDir(root, target);
24
+ const { runEvalRun } = await import("./eval/run.js");
25
+ return runEvalRun(skillDir);
26
+ }
27
+ case "coverage": {
28
+ const { runEvalCoverage } = await import("./eval/coverage.js");
29
+ return runEvalCoverage(root);
30
+ }
31
+ case "generate-all": {
32
+ const { runEvalGenerateAll } = await import("./eval/generate-all.js");
33
+ return runEvalGenerateAll(root, !!opts.force);
34
+ }
35
+ default:
36
+ console.error(red(`Unknown subcommand: "${subcommand}"\n`) +
37
+ dim("Available: init, run, coverage, generate-all"));
38
+ }
39
+ }
40
+ function resolveSkillDir(root, target) {
41
+ const parts = target.split("/");
42
+ if (parts.length !== 2) {
43
+ console.error(red(`Invalid target "${target}". Expected format: <plugin>/<skill>`));
44
+ process.exit(1);
45
+ }
46
+ return join(root, parts[0], "skills", parts[1]);
47
+ }
48
+ //# sourceMappingURL=eval.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eval.js","sourceRoot":"","sources":["../../src/commands/eval.ts"],"names":[],"mappings":"AAAA,8EAA8E;AAC9E,mCAAmC;AACnC,8EAA8E;AAE9E,OAAO,EAAE,IAAI,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,GAAG,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AAE9C,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,UAAkB,EAClB,MAAe,EACf,OAA2C,EAAE;IAE7C,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;IAEjE,QAAQ,UAAU,EAAE,CAAC;QACnB,KAAK,MAAM,CAAC,CAAC,CAAC;YACZ,IAAI,CAAC,MAAM,EAAE,CAAC;gBACZ,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,0CAA0C,CAAC,CAAC,CAAC;gBAC/D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;YACD,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;YAC/C,MAAM,EAAE,WAAW,EAAE,GAAG,MAAM,MAAM,CAAC,gBAAgB,CAAC,CAAC;YACvD,OAAO,WAAW,CAAC,QAAQ,EAAE,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC7C,CAAC;QAED,KAAK,KAAK,CAAC,CAAC,CAAC;YACX,IAAI,CAAC,MAAM,EAAE,CAAC;gBACZ,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAC,CAAC;gBAC9D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;YACD,MAAM,QAAQ,GAAG,eAAe,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;YAC/C,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,MAAM,CAAC,eAAe,CAAC,CAAC;YACrD,OAAO,UAAU,CAAC,QAAQ,CAAC,CAAC;QAC9B,CAAC;QAED,KAAK,UAAU,CAAC,CAAC,CAAC;YAChB,MAAM,EAAE,eAAe,EAAE,GAAG,MAAM,MAAM,CAAC,oBAAoB,CAAC,CAAC;YAC/D,OAAO,eAAe,CAAC,IAAI,CAAC,CAAC;QAC/B,CAAC;QAED,KAAK,cAAc,CAAC,CAAC,CAAC;YACpB,MAAM,EAAE,kBAAkB,EAAE,GAAG,MAAM,MAAM,CAAC,wBAAwB,CAAC,CAAC;YACtE,OAAO,kBAAkB,CAAC,IAAI,EAAE,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChD,CAAC;QAED;YACE,OAAO,CAAC,KAAK,CACX,GAAG,CAAC,wBAAwB,UAAU,KAAK,CAAC;gBAC1C,GAAG,CAAC,8CAA8C,CAAC,CACtD,CAAC;IACN,CAAC;AACH,CAAC;AAED,SAAS,eAAe,CAAC,IAAY,EAAE,MAAc;IACnD,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAChC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACvB,OAAO,CAAC,KAAK,CACX,GAAG,CAAC,mBAAmB,MAAM,sCAAsC,CAAC,CACrE,CAAC;QACF,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IACD,OAAO,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;AAClD,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,65 @@
1
+ import { describe, it, expect, beforeEach, afterEach } from "vitest";
2
+ import { mkdirSync, writeFileSync, rmSync } from "node:fs";
3
+ import { join } from "node:path";
4
+ import { tmpdir } from "node:os";
5
+ import { writeBenchmark, readBenchmark } from "../benchmark.js";
6
+ // ---------------------------------------------------------------------------
7
+ // Helpers
8
+ // ---------------------------------------------------------------------------
9
+ let testDir;
10
+ const SAMPLE_BENCHMARK = {
11
+ timestamp: "2026-03-01T00:00:00.000Z",
12
+ model: "claude-sonnet-4-20250514",
13
+ skill_name: "test-skill",
14
+ cases: [
15
+ {
16
+ eval_id: 1,
17
+ eval_name: "Basic test",
18
+ status: "pass",
19
+ error_message: null,
20
+ pass_rate: 1.0,
21
+ assertions: [
22
+ {
23
+ id: "a1",
24
+ text: "Check result",
25
+ pass: true,
26
+ reasoning: "Looks good",
27
+ },
28
+ ],
29
+ },
30
+ ],
31
+ };
32
+ // ---------------------------------------------------------------------------
33
+ // Tests
34
+ // ---------------------------------------------------------------------------
35
+ describe("benchmark", () => {
36
+ beforeEach(() => {
37
+ testDir = join(tmpdir(), `vskill-bench-${Date.now()}`);
38
+ mkdirSync(join(testDir, "evals"), { recursive: true });
39
+ });
40
+ afterEach(() => {
41
+ rmSync(testDir, { recursive: true, force: true });
42
+ });
43
+ it("writes benchmark.json with all required fields", async () => {
44
+ await writeBenchmark(testDir, SAMPLE_BENCHMARK);
45
+ const result = await readBenchmark(testDir);
46
+ expect(result).not.toBeNull();
47
+ expect(result.timestamp).toBe("2026-03-01T00:00:00.000Z");
48
+ expect(result.model).toBe("claude-sonnet-4-20250514");
49
+ expect(result.skill_name).toBe("test-skill");
50
+ expect(result.cases).toHaveLength(1);
51
+ expect(result.cases[0].assertions).toHaveLength(1);
52
+ });
53
+ it("reads benchmark.json and returns typed result", async () => {
54
+ writeFileSync(join(testDir, "evals", "benchmark.json"), JSON.stringify(SAMPLE_BENCHMARK));
55
+ const result = await readBenchmark(testDir);
56
+ expect(result.skill_name).toBe("test-skill");
57
+ expect(result.cases[0].pass_rate).toBe(1.0);
58
+ });
59
+ it("returns null for missing benchmark.json", async () => {
60
+ rmSync(join(testDir, "evals"), { recursive: true, force: true });
61
+ const result = await readBenchmark(testDir);
62
+ expect(result).toBeNull();
63
+ });
64
+ });
65
+ //# sourceMappingURL=benchmark.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"benchmark.test.js","sourceRoot":"","sources":["../../../src/eval/__tests__/benchmark.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,45 @@
1
+ import { describe, it, expect, vi } from "vitest";
2
+ import { judgeAssertion } from "../judge.js";
3
+ // ---------------------------------------------------------------------------
4
+ // Helpers
5
+ // ---------------------------------------------------------------------------
6
+ function mockClient(response) {
7
+ return { generate: vi.fn().mockResolvedValue(response), model: "test-model" };
8
+ }
9
+ const ASSERTION = {
10
+ id: "assert-1",
11
+ text: "Output mentions a file path",
12
+ type: "boolean",
13
+ };
14
+ // ---------------------------------------------------------------------------
15
+ // Tests
16
+ // ---------------------------------------------------------------------------
17
+ describe("judgeAssertion", () => {
18
+ it("returns pass result when LLM judge says pass", async () => {
19
+ const client = mockClient(JSON.stringify({ pass: true, reasoning: "output contains file path" }));
20
+ const result = await judgeAssertion("The report has been saved to reports/q1.csv", ASSERTION, client);
21
+ expect(result.pass).toBe(true);
22
+ expect(result.reasoning).toBe("output contains file path");
23
+ expect(result.id).toBe("assert-1");
24
+ expect(result.text).toBe("Output mentions a file path");
25
+ });
26
+ it("returns fail result when LLM judge says fail", async () => {
27
+ const client = mockClient(JSON.stringify({
28
+ pass: false,
29
+ reasoning: "no file path found in output",
30
+ }));
31
+ const result = await judgeAssertion("Hello world", ASSERTION, client);
32
+ expect(result.pass).toBe(false);
33
+ expect(result.reasoning).toBe("no file path found in output");
34
+ });
35
+ it("throws on malformed judge response", async () => {
36
+ const client = mockClient("This is not JSON");
37
+ await expect(judgeAssertion("some output", ASSERTION, client)).rejects.toThrow(/invalid judge output/i);
38
+ });
39
+ it("handles JSON wrapped in code fence", async () => {
40
+ const client = mockClient('```json\n{"pass": true, "reasoning": "looks good"}\n```');
41
+ const result = await judgeAssertion("some output", ASSERTION, client);
42
+ expect(result.pass).toBe(true);
43
+ });
44
+ });
45
+ //# sourceMappingURL=judge.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"judge.test.js","sourceRoot":"","sources":["../../../src/eval/__tests__/judge.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAc,MAAM,QAAQ,CAAC;AAG9D,OAAO,EAAE,cAAc,EAAE,MAAM,aAAa,CAAC;AAE7C,8EAA8E;AAC9E,UAAU;AACV,8EAA8E;AAE9E,SAAS,UAAU,CAAC,QAAgB;IAClC,OAAO,EAAE,QAAQ,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC;AAChF,CAAC;AAED,MAAM,SAAS,GAAc;IAC3B,EAAE,EAAE,UAAU;IACd,IAAI,EAAE,6BAA6B;IACnC,IAAI,EAAE,SAAS;CAChB,CAAC;AAEF,8EAA8E;AAC9E,QAAQ;AACR,8EAA8E;AAE9E,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;IAC9B,EAAE,CAAC,8CAA8C,EAAE,KAAK,IAAI,EAAE;QAC5D,MAAM,MAAM,GAAG,UAAU,CACvB,IAAI,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,2BAA2B,EAAE,CAAC,CACvE,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,cAAc,CACjC,6CAA6C,EAC7C,SAAS,EACT,MAAM,CACP,CAAC;QAEF,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC/B,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;QAC3D,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACnC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8CAA8C,EAAE,KAAK,IAAI,EAAE;QAC5D,MAAM,MAAM,GAAG,UAAU,CACvB,IAAI,CAAC,SAAS,CAAC;YACb,IAAI,EAAE,KAAK;YACX,SAAS,EAAE,8BAA8B;SAC1C,CAAC,CACH,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,cAAc,CAAC,aAAa,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC;QAEtE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAChC,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;IAChE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,MAAM,GAAG,UAAU,CAAC,kBAAkB,CAAC,CAAC;QAE9C,MAAM,MAAM,CACV,cAAc,CAAC,aAAa,EAAE,SAAS,EAAE,MAAM,CAAC,CACjD,CAAC,OAAO,CAAC,OAAO,CAAC,uBAAuB,CAAC,CAAC;IAC7C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oCAAoC,EAAE,KAAK,IAAI,EAAE;QAClD,MAAM,MAAM,GAAG,UAAU,CACvB,yDAAyD,CAC1D,CAAC;QAEF,MAAM,MAAM,GAAG,MAAM,cAAc,CAAC,aAAa,EAAE,SAAS,EAAE,MAAM,CAAC,CAAC;QACtE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,85 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
2
+ // ---------------------------------------------------------------------------
3
+ // Mocks
4
+ // ---------------------------------------------------------------------------
5
+ const mockCreate = vi.hoisted(() => vi.fn());
6
+ vi.mock("@anthropic-ai/sdk", () => ({
7
+ default: class MockAnthropic {
8
+ messages = { create: mockCreate };
9
+ },
10
+ }));
11
+ // ---------------------------------------------------------------------------
12
+ // Import module under test AFTER mocks
13
+ // ---------------------------------------------------------------------------
14
+ const { createLlmClient } = await import("../llm.js");
15
+ // ---------------------------------------------------------------------------
16
+ // Tests
17
+ // ---------------------------------------------------------------------------
18
+ describe("createLlmClient", () => {
19
+ const origEnv = { ...process.env };
20
+ beforeEach(() => {
21
+ vi.resetAllMocks();
22
+ process.env.ANTHROPIC_API_KEY = "test-key";
23
+ delete process.env.VSKILL_EVAL_MODEL;
24
+ });
25
+ afterEach(() => {
26
+ process.env = { ...origEnv };
27
+ });
28
+ it("returns text content on successful generate call", async () => {
29
+ mockCreate.mockResolvedValue({
30
+ content: [{ type: "text", text: "Generated response" }],
31
+ });
32
+ const client = createLlmClient();
33
+ const result = await client.generate("system prompt", "user prompt");
34
+ expect(result).toBe("Generated response");
35
+ expect(mockCreate).toHaveBeenCalledOnce();
36
+ });
37
+ it("uses default model claude-sonnet-4-20250514 when env not set", async () => {
38
+ mockCreate.mockResolvedValue({
39
+ content: [{ type: "text", text: "ok" }],
40
+ });
41
+ const client = createLlmClient();
42
+ await client.generate("sys", "usr");
43
+ expect(mockCreate).toHaveBeenCalledWith(expect.objectContaining({ model: "claude-sonnet-4-20250514" }), expect.anything());
44
+ });
45
+ it("uses custom model from VSKILL_EVAL_MODEL env var", async () => {
46
+ process.env.VSKILL_EVAL_MODEL = "claude-opus-4-20250514";
47
+ mockCreate.mockResolvedValue({
48
+ content: [{ type: "text", text: "ok" }],
49
+ });
50
+ const client = createLlmClient();
51
+ await client.generate("sys", "usr");
52
+ expect(mockCreate).toHaveBeenCalledWith(expect.objectContaining({ model: "claude-opus-4-20250514" }), expect.anything());
53
+ });
54
+ it("propagates network error from SDK", async () => {
55
+ mockCreate.mockRejectedValue(new Error("Connection timeout"));
56
+ const client = createLlmClient();
57
+ await expect(client.generate("sys", "usr")).rejects.toThrow("Connection timeout");
58
+ });
59
+ it("passes system and user prompts correctly", async () => {
60
+ mockCreate.mockResolvedValue({
61
+ content: [{ type: "text", text: "ok" }],
62
+ });
63
+ const client = createLlmClient();
64
+ await client.generate("my system prompt", "my user prompt");
65
+ expect(mockCreate).toHaveBeenCalledWith(expect.objectContaining({
66
+ system: "my system prompt",
67
+ messages: [{ role: "user", content: "my user prompt" }],
68
+ max_tokens: 4096,
69
+ }), expect.anything());
70
+ });
71
+ it("throws when ANTHROPIC_API_KEY is not set", () => {
72
+ delete process.env.ANTHROPIC_API_KEY;
73
+ expect(() => createLlmClient()).toThrow("ANTHROPIC_API_KEY is not set");
74
+ });
75
+ it("exposes model name on the client", () => {
76
+ const client = createLlmClient();
77
+ expect(client.model).toBe("claude-sonnet-4-20250514");
78
+ });
79
+ it("exposes custom model name when VSKILL_EVAL_MODEL is set", () => {
80
+ process.env.VSKILL_EVAL_MODEL = "claude-opus-4-20250514";
81
+ const client = createLlmClient();
82
+ expect(client.model).toBe("claude-opus-4-20250514");
83
+ });
84
+ });
85
+ //# sourceMappingURL=llm.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"llm.test.js","sourceRoot":"","sources":["../../../src/eval/__tests__/llm.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,72 @@
1
+ import { describe, it, expect } from "vitest";
2
+ import { buildEvalInitPrompt, parseGeneratedEvals, } from "../prompt-builder.js";
3
+ // ---------------------------------------------------------------------------
4
+ // buildEvalInitPrompt
5
+ // ---------------------------------------------------------------------------
6
+ describe("buildEvalInitPrompt", () => {
7
+ const skillContent = "# My Skill\nThis skill does amazing things.";
8
+ it("includes skill content in the prompt", () => {
9
+ const prompt = buildEvalInitPrompt(skillContent);
10
+ expect(prompt).toContain(skillContent);
11
+ });
12
+ it("includes schema reference fields", () => {
13
+ const prompt = buildEvalInitPrompt(skillContent);
14
+ expect(prompt).toContain("skill_name");
15
+ expect(prompt).toContain("assertions");
16
+ expect(prompt).toContain("expected_output");
17
+ });
18
+ it("includes social-media-posting example", () => {
19
+ const prompt = buildEvalInitPrompt(skillContent);
20
+ expect(prompt).toContain("social-media-posting");
21
+ });
22
+ it("includes best practices section", () => {
23
+ const prompt = buildEvalInitPrompt(skillContent);
24
+ expect(prompt).toContain("Best Practices");
25
+ expect(prompt).toContain("objectively verifiable");
26
+ });
27
+ });
28
+ // ---------------------------------------------------------------------------
29
+ // parseGeneratedEvals
30
+ // ---------------------------------------------------------------------------
31
+ describe("parseGeneratedEvals", () => {
32
+ it("extracts JSON from markdown code fence", () => {
33
+ const raw = `Here is the evals.json:
34
+
35
+ \`\`\`json
36
+ {
37
+ "skill_name": "test-skill",
38
+ "evals": [
39
+ {
40
+ "id": 1,
41
+ "name": "Basic test",
42
+ "prompt": "Test prompt",
43
+ "expected_output": "Expected output",
44
+ "files": [],
45
+ "assertions": [
46
+ { "id": "a1", "text": "Check result", "type": "boolean" }
47
+ ]
48
+ }
49
+ ]
50
+ }
51
+ \`\`\`
52
+
53
+ That's the evals file.`;
54
+ const result = parseGeneratedEvals(raw);
55
+ expect(result.skill_name).toBe("test-skill");
56
+ expect(result.evals).toHaveLength(1);
57
+ expect(result.evals[0].assertions).toHaveLength(1);
58
+ });
59
+ it("throws when no code block is found", () => {
60
+ const raw = "Just some text without any JSON code block.";
61
+ expect(() => parseGeneratedEvals(raw)).toThrow(/code block/i);
62
+ });
63
+ it("throws when JSON inside fence is invalid", () => {
64
+ const raw = "```json\n{ invalid json }\n```";
65
+ expect(() => parseGeneratedEvals(raw)).toThrow();
66
+ });
67
+ it("validates extracted JSON against schema", () => {
68
+ const raw = '```json\n{ "skill_name": "test" }\n```';
69
+ expect(() => parseGeneratedEvals(raw)).toThrow(); // missing evals array
70
+ });
71
+ });
72
+ //# sourceMappingURL=prompt-builder.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompt-builder.test.js","sourceRoot":"","sources":["../../../src/eval/__tests__/prompt-builder.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,209 @@
1
+ import { describe, it, expect, vi, beforeEach } from "vitest";
2
+ // ---------------------------------------------------------------------------
3
+ // Mocks
4
+ // ---------------------------------------------------------------------------
5
+ const mocks = vi.hoisted(() => ({
6
+ readFileSync: vi.fn(),
7
+ existsSync: vi.fn(),
8
+ }));
9
+ vi.mock("node:fs", () => ({
10
+ readFileSync: mocks.readFileSync,
11
+ existsSync: mocks.existsSync,
12
+ }));
13
+ // ---------------------------------------------------------------------------
14
+ // Import module under test AFTER mocks
15
+ // ---------------------------------------------------------------------------
16
+ const { loadAndValidateEvals, EvalValidationError } = await import("../schema.js");
17
+ // ---------------------------------------------------------------------------
18
+ // Fixtures
19
+ // ---------------------------------------------------------------------------
20
+ const VALID_EVALS = {
21
+ skill_name: "social-media-posting",
22
+ evals: [
23
+ {
24
+ id: 1,
25
+ name: "Basic post generation",
26
+ prompt: "Create a social media post about AI",
27
+ expected_output: "A well-crafted social media post",
28
+ files: [],
29
+ assertions: [
30
+ { id: "assert-1", text: "Output mentions AI", type: "boolean" },
31
+ { id: "assert-2", text: "Output is under 280 chars", type: "boolean" },
32
+ ],
33
+ },
34
+ ],
35
+ };
36
+ // ---------------------------------------------------------------------------
37
+ // Tests
38
+ // ---------------------------------------------------------------------------
39
+ describe("loadAndValidateEvals", () => {
40
+ beforeEach(() => {
41
+ vi.resetAllMocks();
42
+ });
43
+ it("accepts valid evals.json and returns typed EvalsFile", () => {
44
+ mocks.existsSync.mockReturnValue(true);
45
+ mocks.readFileSync.mockReturnValue(JSON.stringify(VALID_EVALS));
46
+ const result = loadAndValidateEvals("/skills/my-skill");
47
+ expect(result.skill_name).toBe("social-media-posting");
48
+ expect(result.evals).toHaveLength(1);
49
+ expect(result.evals[0].assertions).toHaveLength(2);
50
+ });
51
+ it("throws for missing skill_name", () => {
52
+ const invalid = { ...VALID_EVALS, skill_name: undefined };
53
+ mocks.existsSync.mockReturnValue(true);
54
+ mocks.readFileSync.mockReturnValue(JSON.stringify(invalid));
55
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
56
+ try {
57
+ loadAndValidateEvals("/skills/my-skill");
58
+ }
59
+ catch (e) {
60
+ expect(e.errors.some((err) => err.path === "skill_name")).toBe(true);
61
+ }
62
+ });
63
+ it("throws for missing evals array", () => {
64
+ const invalid = { skill_name: "test" };
65
+ mocks.existsSync.mockReturnValue(true);
66
+ mocks.readFileSync.mockReturnValue(JSON.stringify(invalid));
67
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
68
+ try {
69
+ loadAndValidateEvals("/skills/my-skill");
70
+ }
71
+ catch (e) {
72
+ expect(e.errors.some((err) => err.path === "evals")).toBe(true);
73
+ }
74
+ });
75
+ it("throws for missing prompt in eval case", () => {
76
+ const invalid = {
77
+ skill_name: "test",
78
+ evals: [
79
+ {
80
+ id: 1,
81
+ name: "Test",
82
+ expected_output: "output",
83
+ assertions: [{ id: "a1", text: "ok", type: "boolean" }],
84
+ },
85
+ ],
86
+ };
87
+ mocks.existsSync.mockReturnValue(true);
88
+ mocks.readFileSync.mockReturnValue(JSON.stringify(invalid));
89
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
90
+ try {
91
+ loadAndValidateEvals("/skills/my-skill");
92
+ }
93
+ catch (e) {
94
+ expect(e.errors.some((err) => err.path === "evals[0].prompt")).toBe(true);
95
+ }
96
+ });
97
+ it("throws for missing assertions", () => {
98
+ const invalid = {
99
+ skill_name: "test",
100
+ evals: [
101
+ {
102
+ id: 1,
103
+ name: "Test",
104
+ prompt: "do something",
105
+ expected_output: "output",
106
+ },
107
+ ],
108
+ };
109
+ mocks.existsSync.mockReturnValue(true);
110
+ mocks.readFileSync.mockReturnValue(JSON.stringify(invalid));
111
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
112
+ try {
113
+ loadAndValidateEvals("/skills/my-skill");
114
+ }
115
+ catch (e) {
116
+ expect(e.errors.some((err) => err.path === "evals[0].assertions")).toBe(true);
117
+ }
118
+ });
119
+ it("throws for duplicate assertion IDs within a case", () => {
120
+ const invalid = {
121
+ skill_name: "test",
122
+ evals: [
123
+ {
124
+ id: 1,
125
+ name: "Test",
126
+ prompt: "do something",
127
+ expected_output: "output",
128
+ assertions: [
129
+ { id: "assert-1", text: "first", type: "boolean" },
130
+ { id: "assert-1", text: "duplicate", type: "boolean" },
131
+ ],
132
+ },
133
+ ],
134
+ };
135
+ mocks.existsSync.mockReturnValue(true);
136
+ mocks.readFileSync.mockReturnValue(JSON.stringify(invalid));
137
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
138
+ try {
139
+ loadAndValidateEvals("/skills/my-skill");
140
+ }
141
+ catch (e) {
142
+ expect(e.errors.some((err) => err.message.includes("assert-1"))).toBe(true);
143
+ }
144
+ });
145
+ it("throws for invalid JSON with file path info", () => {
146
+ mocks.existsSync.mockReturnValue(true);
147
+ mocks.readFileSync.mockReturnValue("{ broken json");
148
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
149
+ try {
150
+ loadAndValidateEvals("/skills/my-skill");
151
+ }
152
+ catch (e) {
153
+ expect(e.errors[0].path).toContain("evals.json");
154
+ expect(e.errors[0].message).toMatch(/parse|JSON/i);
155
+ }
156
+ });
157
+ it("throws for empty assertions array", () => {
158
+ const invalid = {
159
+ skill_name: "test",
160
+ evals: [
161
+ {
162
+ id: 1,
163
+ name: "Test",
164
+ prompt: "do something",
165
+ expected_output: "output",
166
+ assertions: [],
167
+ },
168
+ ],
169
+ };
170
+ mocks.existsSync.mockReturnValue(true);
171
+ mocks.readFileSync.mockReturnValue(JSON.stringify(invalid));
172
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
173
+ try {
174
+ loadAndValidateEvals("/skills/my-skill");
175
+ }
176
+ catch (e) {
177
+ expect(e.errors.some((err) => err.message.includes("at least 1"))).toBe(true);
178
+ }
179
+ });
180
+ it("throws when evals.json does not exist", () => {
181
+ mocks.existsSync.mockReturnValue(false);
182
+ expect(() => loadAndValidateEvals("/skills/my-skill")).toThrow(EvalValidationError);
183
+ try {
184
+ loadAndValidateEvals("/skills/my-skill");
185
+ }
186
+ catch (e) {
187
+ expect(e.errors[0].message).toMatch(/not found|No evals\.json/i);
188
+ }
189
+ });
190
+ it("accepts evals with optional files field defaulting to empty array", () => {
191
+ const valid = {
192
+ skill_name: "test",
193
+ evals: [
194
+ {
195
+ id: 1,
196
+ name: "Test",
197
+ prompt: "do something",
198
+ expected_output: "output",
199
+ assertions: [{ id: "a1", text: "check", type: "boolean" }],
200
+ },
201
+ ],
202
+ };
203
+ mocks.existsSync.mockReturnValue(true);
204
+ mocks.readFileSync.mockReturnValue(JSON.stringify(valid));
205
+ const result = loadAndValidateEvals("/skills/my-skill");
206
+ expect(result.evals[0].files).toEqual([]);
207
+ });
208
+ });
209
+ //# sourceMappingURL=schema.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"schema.test.js","sourceRoot":"","sources":["../../../src/eval/__tests__/schema.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};