@lssm/lib.testing 0.2.0 → 0.2.2
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.
- package/package.json +4 -4
- package/dist/adapters/jest-adapter.d.ts +0 -13
- package/dist/adapters/jest-adapter.d.ts.map +0 -1
- package/dist/adapters/jest-adapter.js +0 -14
- package/dist/adapters/jest-adapter.js.map +0 -1
- package/dist/adapters/vitest-adapter.d.ts +0 -13
- package/dist/adapters/vitest-adapter.d.ts.map +0 -1
- package/dist/adapters/vitest-adapter.js +0 -15
- package/dist/adapters/vitest-adapter.js.map +0 -1
- package/dist/generator/assertion-builder.d.ts +0 -12
- package/dist/generator/assertion-builder.d.ts.map +0 -1
- package/dist/generator/assertion-builder.js +0 -3
- package/dist/generator/assertion-builder.js.map +0 -1
- package/dist/generator/golden-test-generator.d.ts +0 -27
- package/dist/generator/golden-test-generator.d.ts.map +0 -1
- package/dist/generator/golden-test-generator.js +0 -2
- package/dist/generator/golden-test-generator.js.map +0 -1
- package/dist/index.d.ts +0 -7
- package/dist/index.js +0 -1
- package/dist/recorder/traffic-recorder.d.ts +0 -40
- package/dist/recorder/traffic-recorder.d.ts.map +0 -1
- package/dist/recorder/traffic-recorder.js +0 -2
- package/dist/recorder/traffic-recorder.js.map +0 -1
- package/dist/types.d.ts +0 -43
- package/dist/types.d.ts.map +0 -1
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@lssm/lib.testing",
|
|
3
|
-
"version": "0.2.
|
|
3
|
+
"version": "0.2.2",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"main": "./dist/index.js",
|
|
6
6
|
"module": "./dist/index.js",
|
|
@@ -21,11 +21,11 @@
|
|
|
21
21
|
"test": "vitest run"
|
|
22
22
|
},
|
|
23
23
|
"dependencies": {
|
|
24
|
-
"@lssm/lib.contracts": "^1.9.
|
|
24
|
+
"@lssm/lib.contracts": "^1.9.2"
|
|
25
25
|
},
|
|
26
26
|
"devDependencies": {
|
|
27
|
-
"@lssm/tool.tsdown": "
|
|
28
|
-
"@lssm/tool.typescript": "
|
|
27
|
+
"@lssm/tool.tsdown": "*",
|
|
28
|
+
"@lssm/tool.typescript": "*",
|
|
29
29
|
"tsdown": "^0.16.6",
|
|
30
30
|
"typescript": "^5.9.3",
|
|
31
31
|
"vitest": "^1.6.0"
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
import { GoldenTestCase } from "../types.js";
|
|
2
|
-
|
|
3
|
-
//#region src/adapters/jest-adapter.d.ts
|
|
4
|
-
interface JestAdapterOptions {
|
|
5
|
-
suiteName: string;
|
|
6
|
-
cases: GoldenTestCase[];
|
|
7
|
-
runnerImport: string;
|
|
8
|
-
runnerFunction: string;
|
|
9
|
-
}
|
|
10
|
-
declare function generateJestSuite(options: JestAdapterOptions): string;
|
|
11
|
-
//#endregion
|
|
12
|
-
export { JestAdapterOptions, generateJestSuite };
|
|
13
|
-
//# sourceMappingURL=jest-adapter.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"jest-adapter.d.ts","names":[],"sources":["../../src/adapters/jest-adapter.ts"],"sourcesContent":[],"mappings":";;;UAGiB,kBAAA;;EAAA,KAAA,EAER,cAF0B,EAAA;EAOnB,YAAA,EAAA,MAAiB;;;iBAAjB,iBAAA,UAA2B"}
|
|
@@ -1,14 +0,0 @@
|
|
|
1
|
-
import{serialize as e}from"../generator/assertion-builder.js";function t(t){let n=t.cases.map(n=>{let r=e(n.input),i=e(n.metadata??{}),a=`const result = await ${t.runnerFunction}(input${n.id}, metadata${n.id});
|
|
2
|
-
expect(result).toEqual(${e(n.expectedOutput??null)});`,o=`await expect(${t.runnerFunction}(input${n.id}, metadata${n.id})).rejects.toMatchObject(${e(n.expectedError??{message:`expected failure`})});`;return`
|
|
3
|
-
test('${n.name}', async () => {
|
|
4
|
-
const input${n.id} = ${r};
|
|
5
|
-
const metadata${n.id} = ${i};
|
|
6
|
-
${n.success?a:o}
|
|
7
|
-
});`}).join(`
|
|
8
|
-
`);return`
|
|
9
|
-
import { ${t.runnerFunction} } from '${t.runnerImport}';
|
|
10
|
-
|
|
11
|
-
describe('${t.suiteName}', () => {${n}
|
|
12
|
-
});
|
|
13
|
-
`.trim()}export{t as generateJestSuite};
|
|
14
|
-
//# sourceMappingURL=jest-adapter.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"jest-adapter.js","names":[],"sources":["../../src/adapters/jest-adapter.ts"],"sourcesContent":["import type { GoldenTestCase } from '../types';\nimport { serialize } from '../generator/assertion-builder';\n\nexport interface JestAdapterOptions {\n suiteName: string;\n cases: GoldenTestCase[];\n runnerImport: string;\n runnerFunction: string;\n}\n\nexport function generateJestSuite(options: JestAdapterOptions) {\n const caseBlocks = options.cases\n .map((testCase) => {\n const inputConst = serialize(testCase.input);\n const metadataConst = serialize(testCase.metadata ?? {});\n const successBlock = `const result = await ${options.runnerFunction}(input${testCase.id}, metadata${testCase.id});\n expect(result).toEqual(${serialize(testCase.expectedOutput ?? null)});`;\n const failureBlock = `await expect(${options.runnerFunction}(input${testCase.id}, metadata${testCase.id})).rejects.toMatchObject(${serialize(\n testCase.expectedError ?? { message: 'expected failure' }\n )});`;\n\n return `\n test('${testCase.name}', async () => {\n const input${testCase.id} = ${inputConst};\n const metadata${testCase.id} = ${metadataConst};\n ${testCase.success ? successBlock : failureBlock}\n });`;\n })\n .join('\\n');\n\n return `\nimport { ${options.runnerFunction} } from '${options.runnerImport}';\n\ndescribe('${options.suiteName}', () => {${caseBlocks}\n});\n`.trim();\n}\n"],"mappings":"8DAUA,SAAgB,EAAkB,EAA6B,CAC7D,IAAM,EAAa,EAAQ,MACxB,IAAK,GAAa,CACjB,IAAM,EAAa,EAAU,EAAS,MAAM,CACtC,EAAgB,EAAU,EAAS,UAAY,EAAE,CAAC,CAClD,EAAe,wBAAwB,EAAQ,eAAe,QAAQ,EAAS,GAAG,YAAY,EAAS,GAAG;6BACzF,EAAU,EAAS,gBAAkB,KAAK,CAAC,IAC5D,EAAe,gBAAgB,EAAQ,eAAe,QAAQ,EAAS,GAAG,YAAY,EAAS,GAAG,2BAA2B,EACjI,EAAS,eAAiB,CAAE,QAAS,mBAAoB,CAC1D,CAAC,IAEF,MAAO;UACH,EAAS,KAAK;iBACP,EAAS,GAAG,KAAK,EAAW;oBACzB,EAAS,GAAG,KAAK,EAAc;MAC7C,EAAS,QAAU,EAAe,EAAa;QAE/C,CACD,KAAK;EAAK,CAEb,MAAO;WACE,EAAQ,eAAe,WAAW,EAAQ,aAAa;;YAEtD,EAAQ,UAAU,YAAY,EAAW;;EAEnD,MAAM"}
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
import { GoldenTestCase } from "../types.js";
|
|
2
|
-
|
|
3
|
-
//#region src/adapters/vitest-adapter.d.ts
|
|
4
|
-
interface VitestAdapterOptions {
|
|
5
|
-
suiteName: string;
|
|
6
|
-
cases: GoldenTestCase[];
|
|
7
|
-
runnerImport: string;
|
|
8
|
-
runnerFunction: string;
|
|
9
|
-
}
|
|
10
|
-
declare function generateVitestSuite(options: VitestAdapterOptions): string;
|
|
11
|
-
//#endregion
|
|
12
|
-
export { VitestAdapterOptions, generateVitestSuite };
|
|
13
|
-
//# sourceMappingURL=vitest-adapter.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"vitest-adapter.d.ts","names":[],"sources":["../../src/adapters/vitest-adapter.ts"],"sourcesContent":[],"mappings":";;;UAGiB,oBAAA;;EAAA,KAAA,EAER,cAFQ,EAAoB;EAOrB,YAAA,EAAA,MAAA;;;iBAAA,mBAAA,UAA6B"}
|
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
import{serialize as e}from"../generator/assertion-builder.js";function t(t){let n=t.cases.map(n=>{let r=e(n.input),i=e(n.metadata??{}),a=n.success?[`const result = await ${t.runnerFunction}(input${n.id}, metadata${n.id});`,`expect(result).toEqual(${e(n.expectedOutput??null)});`]:[`await expect(${t.runnerFunction}(input${n.id}, metadata${n.id})).rejects.toMatchObject(${e(n.expectedError??{message:`expected failure`})});`];return`
|
|
2
|
-
it('${n.name}', async () => {
|
|
3
|
-
const input${n.id} = ${r};
|
|
4
|
-
const metadata${n.id} = ${i};
|
|
5
|
-
${a.join(`
|
|
6
|
-
`)}
|
|
7
|
-
});`}).join(`
|
|
8
|
-
`);return`
|
|
9
|
-
import { describe, it, expect } from 'vitest';
|
|
10
|
-
import { ${t.runnerFunction} } from '${t.runnerImport}';
|
|
11
|
-
|
|
12
|
-
describe('${t.suiteName}', () => {${n}
|
|
13
|
-
});
|
|
14
|
-
`.trim()}export{t as generateVitestSuite};
|
|
15
|
-
//# sourceMappingURL=vitest-adapter.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"vitest-adapter.js","names":[],"sources":["../../src/adapters/vitest-adapter.ts"],"sourcesContent":["import type { GoldenTestCase } from '../types';\nimport { serialize } from '../generator/assertion-builder';\n\nexport interface VitestAdapterOptions {\n suiteName: string;\n cases: GoldenTestCase[];\n runnerImport: string;\n runnerFunction: string;\n}\n\nexport function generateVitestSuite(options: VitestAdapterOptions) {\n const caseBlocks = options.cases\n .map((testCase) => {\n const inputConst = serialize(testCase.input);\n const metadataConst = serialize(testCase.metadata ?? {});\n const assertions = testCase.success\n ? [\n `const result = await ${options.runnerFunction}(input${testCase.id}, metadata${testCase.id});`,\n `expect(result).toEqual(${serialize(testCase.expectedOutput ?? null)});`,\n ]\n : [\n `await expect(${options.runnerFunction}(input${testCase.id}, metadata${testCase.id})).rejects.toMatchObject(${serialize(testCase.expectedError ?? { message: 'expected failure' })});`,\n ];\n\n return `\n it('${testCase.name}', async () => {\n const input${testCase.id} = ${inputConst};\n const metadata${testCase.id} = ${metadataConst};\n ${assertions.join('\\n ')}\n });`;\n })\n .join('\\n');\n\n return `\nimport { describe, it, expect } from 'vitest';\nimport { ${options.runnerFunction} } from '${options.runnerImport}';\n\ndescribe('${options.suiteName}', () => {${caseBlocks}\n});\n`.trim();\n}\n"],"mappings":"8DAUA,SAAgB,EAAoB,EAA+B,CACjE,IAAM,EAAa,EAAQ,MACxB,IAAK,GAAa,CACjB,IAAM,EAAa,EAAU,EAAS,MAAM,CACtC,EAAgB,EAAU,EAAS,UAAY,EAAE,CAAC,CAClD,EAAa,EAAS,QACxB,CACE,wBAAwB,EAAQ,eAAe,QAAQ,EAAS,GAAG,YAAY,EAAS,GAAG,IAC3F,0BAA0B,EAAU,EAAS,gBAAkB,KAAK,CAAC,IACtE,CACD,CACE,gBAAgB,EAAQ,eAAe,QAAQ,EAAS,GAAG,YAAY,EAAS,GAAG,2BAA2B,EAAU,EAAS,eAAiB,CAAE,QAAS,mBAAoB,CAAC,CAAC,IACpL,CAEL,MAAO;QACL,EAAS,KAAK;iBACL,EAAS,GAAG,KAAK,EAAW;oBACzB,EAAS,GAAG,KAAK,EAAc;MAC7C,EAAW,KAAK;MAAS,CAAC;QAE1B,CACD,KAAK;EAAK,CAEb,MAAO;;WAEE,EAAQ,eAAe,WAAW,EAAQ,aAAa;;YAEtD,EAAQ,UAAU,YAAY,EAAW;;EAEnD,MAAM"}
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
import { GoldenTestCase } from "../types.js";
|
|
2
|
-
|
|
3
|
-
//#region src/generator/assertion-builder.d.ts
|
|
4
|
-
interface AssertionContext {
|
|
5
|
-
runnerCall: string;
|
|
6
|
-
caseRef: string;
|
|
7
|
-
}
|
|
8
|
-
declare function buildAssertions(testCase: GoldenTestCase, ctx: AssertionContext): string;
|
|
9
|
-
declare function serialize(value: unknown): string;
|
|
10
|
-
//#endregion
|
|
11
|
-
export { AssertionContext, buildAssertions, serialize };
|
|
12
|
-
//# sourceMappingURL=assertion-builder.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"assertion-builder.d.ts","names":[],"sources":["../../src/generator/assertion-builder.ts"],"sourcesContent":[],"mappings":";;;UAEiB,gBAAA;;EAAA,OAAA,EAAA,MAAA;AAKjB;AAegB,iBAfA,eAAA,CAeS,QAAA,EAdb,cAca,EAAA,GAAA,EAblB,gBAakB,CAAA,EAAA,MAAA;iBAAT,SAAA"}
|
|
@@ -1,3 +0,0 @@
|
|
|
1
|
-
function e(e,n){return e.success?[`const result = await ${n.runnerCall};`,`expect(result).toEqual(${t(e.expectedOutput??null)});`].join(`
|
|
2
|
-
`):`await expect(${n.runnerCall}).rejects.toMatchObject(${t(e.expectedError??{message:`expected failure`})});`}function t(e){return JSON.stringify(e,(e,t)=>t instanceof Date?t.toISOString():t===void 0?null:t,2)}export{e as buildAssertions,t as serialize};
|
|
3
|
-
//# sourceMappingURL=assertion-builder.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"assertion-builder.js","names":[],"sources":["../../src/generator/assertion-builder.ts"],"sourcesContent":["import type { GoldenTestCase } from '../types';\n\nexport interface AssertionContext {\n runnerCall: string;\n caseRef: string;\n}\n\nexport function buildAssertions(\n testCase: GoldenTestCase,\n ctx: AssertionContext\n) {\n if (testCase.success) {\n return [\n `const result = await ${ctx.runnerCall};`,\n `expect(result).toEqual(${serialize(testCase.expectedOutput ?? null)});`,\n ].join('\\n ');\n }\n return `await expect(${ctx.runnerCall}).rejects.toMatchObject(${serialize(\n testCase.expectedError ?? { message: 'expected failure' }\n )});`;\n}\n\nexport function serialize(value: unknown) {\n return JSON.stringify(\n value,\n (_key, val) => {\n if (val instanceof Date) return val.toISOString();\n if (typeof val === 'undefined') return null;\n return val;\n },\n 2\n );\n}\n"],"mappings":"AAOA,SAAgB,EACd,EACA,EACA,CAOA,OANI,EAAS,QACJ,CACL,wBAAwB,EAAI,WAAW,GACvC,0BAA0B,EAAU,EAAS,gBAAkB,KAAK,CAAC,IACtE,CAAC,KAAK;QAAW,CAEb,gBAAgB,EAAI,WAAW,0BAA0B,EAC9D,EAAS,eAAiB,CAAE,QAAS,mBAAoB,CAC1D,CAAC,IAGJ,SAAgB,EAAU,EAAgB,CACxC,OAAO,KAAK,UACV,GACC,EAAM,IACD,aAAe,KAAa,EAAI,aAAa,CACtC,IAAQ,OAAoB,KAChC,EAET,EACD"}
|
|
@@ -1,27 +0,0 @@
|
|
|
1
|
-
import { GoldenTestCase, TrafficSnapshot } from "../types.js";
|
|
2
|
-
|
|
3
|
-
//#region src/generator/golden-test-generator.d.ts
|
|
4
|
-
interface GoldenTestGeneratorOptions {
|
|
5
|
-
suiteName: string;
|
|
6
|
-
runnerImport: string;
|
|
7
|
-
runnerFunction: string;
|
|
8
|
-
framework?: 'vitest' | 'jest';
|
|
9
|
-
serializeMetadata?: (snapshot: TrafficSnapshot) => Record<string, unknown>;
|
|
10
|
-
}
|
|
11
|
-
declare class GoldenTestGenerator {
|
|
12
|
-
private readonly serializeMetadata;
|
|
13
|
-
constructor(serializeMetadata?: GoldenTestGeneratorOptions['serializeMetadata']);
|
|
14
|
-
createCases(snapshots: TrafficSnapshot[]): GoldenTestCase[];
|
|
15
|
-
generate(snapshots: TrafficSnapshot[], options: GoldenTestGeneratorOptions): string;
|
|
16
|
-
}
|
|
17
|
-
type GoldenTestRunner = (input: unknown, metadata?: Record<string, unknown>) => Promise<unknown>;
|
|
18
|
-
interface GoldenTestRunResult {
|
|
19
|
-
caseId: string;
|
|
20
|
-
passed: boolean;
|
|
21
|
-
durationMs: number;
|
|
22
|
-
error?: unknown;
|
|
23
|
-
}
|
|
24
|
-
declare function runGoldenTests(cases: GoldenTestCase[], runner: GoldenTestRunner): Promise<GoldenTestRunResult[]>;
|
|
25
|
-
//#endregion
|
|
26
|
-
export { GoldenTestGenerator, GoldenTestGeneratorOptions, GoldenTestRunResult, GoldenTestRunner, runGoldenTests };
|
|
27
|
-
//# sourceMappingURL=golden-test-generator.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"golden-test-generator.d.ts","names":[],"sources":["../../src/generator/golden-test-generator.ts"],"sourcesContent":[],"mappings":";;;UAMiB,0BAAA;;EAAA,YAAA,EAAA,MAAA;EAQJ,cAAA,EAAA,MAAmB;EAEQ,SAAA,CAAA,EAAA,QAAA,GAAA,MAAA;EAUf,iBAAA,CAAA,EAAA,CAAA,QAAA,EAfQ,eAeR,EAAA,GAf4B,MAe5B,CAAA,MAAA,EAAA,OAAA,CAAA;;AAeV,cA3BF,mBAAA,CA2BE;EACF,iBAAA,iBAAA;EAA0B,WAAA,CAAA,iBAAA,CAAA,EA1BC,0BA0BD,CAAA,mBAAA,CAAA;EAoB3B,WAAA,CAAA,SAAgB,EApCH,eAsCZ,EACR,CAAA,EAvCwC,cAuCjC,EAAA;EAEK,QAAA,CAAA,SAAA,EA1BF,eA0BqB,EAAA,EAAA,OAAA,EAzBvB,0BAyBuB,CAAA,EAAA,MAAA;AAOpC;AACS,KAbG,gBAAA,GAaH,CAAA,KAAA,EAAA,OAAA,EAAA,QAAA,CAAA,EAXI,MAWJ,CAAA,MAAA,EAAA,OAAA,CAAA,EAAA,GAVJ,OAUI,CAAA,OAAA,CAAA;AACC,UATO,mBAAA,CASP;EACC,MAAA,EAAA,MAAA;EAAR,MAAA,EAAA,OAAA;EAAO,UAAA,EAAA,MAAA;;;iBAHY,cAAA,QACb,0BACC,mBACP,QAAQ"}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
import{generateVitestSuite as e}from"../adapters/vitest-adapter.js";import{generateJestSuite as t}from"../adapters/jest-adapter.js";import{randomUUID as n}from"node:crypto";import{performance as r}from"node:perf_hooks";var i=class{constructor(e=e=>({tenantId:e.tenantId,userId:e.userId,channel:e.channel})){this.serializeMetadata=e}createCases(e){return e.map((e,t)=>({id:e.id??n(),name:e.success?`case-${t+1}-success`:`case-${t+1}-failure`,input:e.input,expectedOutput:e.output,expectedError:e.error,success:e.success,metadata:this.serializeMetadata?.(e)}))}generate(n,r){let i=this.createCases(n);return r.framework===`jest`?t({suiteName:r.suiteName,cases:i,runnerImport:r.runnerImport,runnerFunction:r.runnerFunction}):e({suiteName:r.suiteName,cases:i,runnerImport:r.runnerImport,runnerFunction:r.runnerFunction})}};async function a(e,t){let n=[];for(let i of e){let e=r.now();try{let a=await t(i.input,i.metadata);if(!i.success){n.push({caseId:i.id,passed:!1,durationMs:r.now()-e,error:Error(`Expected failure but runner resolved`)});continue}let o=JSON.stringify(a)===JSON.stringify(i.expectedOutput??null);n.push({caseId:i.id,passed:o,durationMs:r.now()-e,error:o?void 0:{expected:i.expectedOutput,received:a}})}catch(t){let a=r.now()-e;i.success?n.push({caseId:i.id,passed:!1,durationMs:a,error:t}):n.push({caseId:i.id,passed:!0,durationMs:a})}}return n}export{i as GoldenTestGenerator,a as runGoldenTests};
|
|
2
|
-
//# sourceMappingURL=golden-test-generator.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"golden-test-generator.js","names":["serializeMetadata: GoldenTestGeneratorOptions['serializeMetadata']","results: GoldenTestRunResult[]"],"sources":["../../src/generator/golden-test-generator.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\nimport { performance } from 'node:perf_hooks';\nimport type { GoldenTestCase, TrafficSnapshot } from '../types';\nimport { generateVitestSuite } from '../adapters/vitest-adapter';\nimport { generateJestSuite } from '../adapters/jest-adapter';\n\nexport interface GoldenTestGeneratorOptions {\n suiteName: string;\n runnerImport: string;\n runnerFunction: string;\n framework?: 'vitest' | 'jest';\n serializeMetadata?: (snapshot: TrafficSnapshot) => Record<string, unknown>;\n}\n\nexport class GoldenTestGenerator {\n constructor(\n private readonly serializeMetadata: GoldenTestGeneratorOptions['serializeMetadata'] = (\n snapshot\n ) =>\n ({\n tenantId: snapshot.tenantId,\n userId: snapshot.userId,\n channel: snapshot.channel,\n }) as Record<string, unknown>\n ) {}\n\n createCases(snapshots: TrafficSnapshot[]): GoldenTestCase[] {\n return snapshots.map((snapshot, index) => ({\n id: snapshot.id ?? randomUUID(),\n name: snapshot.success\n ? `case-${index + 1}-success`\n : `case-${index + 1}-failure`,\n input: snapshot.input,\n expectedOutput: snapshot.output,\n expectedError: snapshot.error,\n success: snapshot.success,\n metadata: this.serializeMetadata?.(snapshot),\n }));\n }\n\n generate(\n snapshots: TrafficSnapshot[],\n options: GoldenTestGeneratorOptions\n ): string {\n const cases = this.createCases(snapshots);\n if (options.framework === 'jest') {\n return generateJestSuite({\n suiteName: options.suiteName,\n cases,\n runnerImport: options.runnerImport,\n runnerFunction: options.runnerFunction,\n });\n }\n return generateVitestSuite({\n suiteName: options.suiteName,\n cases,\n runnerImport: options.runnerImport,\n runnerFunction: options.runnerFunction,\n });\n }\n}\n\nexport type GoldenTestRunner = (\n input: unknown,\n metadata?: Record<string, unknown>\n) => Promise<unknown>;\n\nexport interface GoldenTestRunResult {\n caseId: string;\n passed: boolean;\n durationMs: number;\n error?: unknown;\n}\n\nexport async function runGoldenTests(\n cases: GoldenTestCase[],\n runner: GoldenTestRunner\n): Promise<GoldenTestRunResult[]> {\n const results: GoldenTestRunResult[] = [];\n for (const testCase of cases) {\n const startedAt = performance.now();\n try {\n const output = await runner(testCase.input, testCase.metadata);\n if (!testCase.success) {\n results.push({\n caseId: testCase.id,\n passed: false,\n durationMs: performance.now() - startedAt,\n error: new Error('Expected failure but runner resolved'),\n });\n continue;\n }\n const matches =\n JSON.stringify(output) ===\n JSON.stringify(testCase.expectedOutput ?? null);\n results.push({\n caseId: testCase.id,\n passed: matches,\n durationMs: performance.now() - startedAt,\n error: matches\n ? undefined\n : { expected: testCase.expectedOutput, received: output },\n });\n } catch (error) {\n const durationMs = performance.now() - startedAt;\n if (!testCase.success) {\n results.push({ caseId: testCase.id, passed: true, durationMs });\n } else {\n results.push({ caseId: testCase.id, passed: false, durationMs, error });\n }\n }\n }\n return results;\n}\n"],"mappings":"2NAcA,IAAa,EAAb,KAAiC,CAC/B,YACE,EACE,IAEC,CACC,SAAU,EAAS,SACnB,OAAQ,EAAS,OACjB,QAAS,EAAS,QACnB,EACH,CARiB,KAAA,kBAAA,EAUnB,YAAY,EAAgD,CAC1D,OAAO,EAAU,KAAK,EAAU,KAAW,CACzC,GAAI,EAAS,IAAM,GAAY,CAC/B,KAAM,EAAS,QACX,QAAQ,EAAQ,EAAE,UAClB,QAAQ,EAAQ,EAAE,UACtB,MAAO,EAAS,MAChB,eAAgB,EAAS,OACzB,cAAe,EAAS,MACxB,QAAS,EAAS,QAClB,SAAU,KAAK,oBAAoB,EAAS,CAC7C,EAAE,CAGL,SACE,EACA,EACQ,CACR,IAAM,EAAQ,KAAK,YAAY,EAAU,CASzC,OARI,EAAQ,YAAc,OACjB,EAAkB,CACvB,UAAW,EAAQ,UACnB,QACA,aAAc,EAAQ,aACtB,eAAgB,EAAQ,eACzB,CAAC,CAEG,EAAoB,CACzB,UAAW,EAAQ,UACnB,QACA,aAAc,EAAQ,aACtB,eAAgB,EAAQ,eACzB,CAAC,GAgBN,eAAsB,EACpB,EACA,EACgC,CAChC,IAAMC,EAAiC,EAAE,CACzC,IAAK,IAAM,KAAY,EAAO,CAC5B,IAAM,EAAY,EAAY,KAAK,CACnC,GAAI,CACF,IAAM,EAAS,MAAM,EAAO,EAAS,MAAO,EAAS,SAAS,CAC9D,GAAI,CAAC,EAAS,QAAS,CACrB,EAAQ,KAAK,CACX,OAAQ,EAAS,GACjB,OAAQ,GACR,WAAY,EAAY,KAAK,CAAG,EAChC,MAAW,MAAM,uCAAuC,CACzD,CAAC,CACF,SAEF,IAAM,EACJ,KAAK,UAAU,EAAO,GACtB,KAAK,UAAU,EAAS,gBAAkB,KAAK,CACjD,EAAQ,KAAK,CACX,OAAQ,EAAS,GACjB,OAAQ,EACR,WAAY,EAAY,KAAK,CAAG,EAChC,MAAO,EACH,IAAA,GACA,CAAE,SAAU,EAAS,eAAgB,SAAU,EAAQ,CAC5D,CAAC,OACK,EAAO,CACd,IAAM,EAAa,EAAY,KAAK,CAAG,EAClC,EAAS,QAGZ,EAAQ,KAAK,CAAE,OAAQ,EAAS,GAAI,OAAQ,GAAO,aAAY,QAAO,CAAC,CAFvE,EAAQ,KAAK,CAAE,OAAQ,EAAS,GAAI,OAAQ,GAAM,aAAY,CAAC,EAMrE,OAAO"}
|
package/dist/index.d.ts
DELETED
|
@@ -1,7 +0,0 @@
|
|
|
1
|
-
import { GoldenTestCase, RuntimeContract, TrafficSnapshot } from "./types.js";
|
|
2
|
-
import { InMemoryTrafficStore, RecordOperationInput, TrafficRecorder, TrafficRecorderOptions, TrafficStore } from "./recorder/traffic-recorder.js";
|
|
3
|
-
import { GoldenTestGenerator, GoldenTestGeneratorOptions, GoldenTestRunResult, GoldenTestRunner, runGoldenTests } from "./generator/golden-test-generator.js";
|
|
4
|
-
import { AssertionContext, buildAssertions, serialize } from "./generator/assertion-builder.js";
|
|
5
|
-
import { VitestAdapterOptions, generateVitestSuite } from "./adapters/vitest-adapter.js";
|
|
6
|
-
import { JestAdapterOptions, generateJestSuite } from "./adapters/jest-adapter.js";
|
|
7
|
-
export { AssertionContext, GoldenTestCase, GoldenTestGenerator, GoldenTestGeneratorOptions, GoldenTestRunResult, GoldenTestRunner, InMemoryTrafficStore, JestAdapterOptions, RecordOperationInput, RuntimeContract, TrafficRecorder, TrafficRecorderOptions, TrafficSnapshot, TrafficStore, VitestAdapterOptions, buildAssertions, generateJestSuite, generateVitestSuite, runGoldenTests, serialize };
|
package/dist/index.js
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
import{InMemoryTrafficStore as e,TrafficRecorder as t}from"./recorder/traffic-recorder.js";import{buildAssertions as n,serialize as r}from"./generator/assertion-builder.js";import{generateVitestSuite as i}from"./adapters/vitest-adapter.js";import{generateJestSuite as a}from"./adapters/jest-adapter.js";import{GoldenTestGenerator as o,runGoldenTests as s}from"./generator/golden-test-generator.js";export{o as GoldenTestGenerator,e as InMemoryTrafficStore,t as TrafficRecorder,n as buildAssertions,a as generateJestSuite,i as generateVitestSuite,s as runGoldenTests,r as serialize};
|
|
@@ -1,40 +0,0 @@
|
|
|
1
|
-
import { TrafficSnapshot } from "../types.js";
|
|
2
|
-
|
|
3
|
-
//#region src/recorder/traffic-recorder.d.ts
|
|
4
|
-
interface TrafficStore {
|
|
5
|
-
save(snapshot: TrafficSnapshot): Promise<void>;
|
|
6
|
-
list(operation?: TrafficSnapshot['operation']['name']): Promise<TrafficSnapshot[]>;
|
|
7
|
-
}
|
|
8
|
-
declare class InMemoryTrafficStore implements TrafficStore {
|
|
9
|
-
private readonly items;
|
|
10
|
-
save(snapshot: TrafficSnapshot): Promise<void>;
|
|
11
|
-
list(operation?: string): Promise<TrafficSnapshot[]>;
|
|
12
|
-
}
|
|
13
|
-
interface TrafficRecorderOptions {
|
|
14
|
-
store: TrafficStore;
|
|
15
|
-
sampleRate?: number;
|
|
16
|
-
sanitize?: (snapshot: TrafficSnapshot) => TrafficSnapshot;
|
|
17
|
-
}
|
|
18
|
-
interface RecordOperationInput {
|
|
19
|
-
operation: TrafficSnapshot['operation'];
|
|
20
|
-
input: unknown;
|
|
21
|
-
output?: unknown;
|
|
22
|
-
error?: TrafficSnapshot['error'];
|
|
23
|
-
success: boolean;
|
|
24
|
-
durationMs?: number;
|
|
25
|
-
tenantId?: string;
|
|
26
|
-
userId?: string;
|
|
27
|
-
channel?: string;
|
|
28
|
-
metadata?: Record<string, unknown>;
|
|
29
|
-
}
|
|
30
|
-
declare class TrafficRecorder {
|
|
31
|
-
private readonly store;
|
|
32
|
-
private readonly sampleRate;
|
|
33
|
-
private readonly sanitize?;
|
|
34
|
-
constructor(options: TrafficRecorderOptions);
|
|
35
|
-
record(input: RecordOperationInput): Promise<void>;
|
|
36
|
-
private shouldSample;
|
|
37
|
-
}
|
|
38
|
-
//#endregion
|
|
39
|
-
export { InMemoryTrafficStore, RecordOperationInput, TrafficRecorder, TrafficRecorderOptions, TrafficStore };
|
|
40
|
-
//# sourceMappingURL=traffic-recorder.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"traffic-recorder.d.ts","names":[],"sources":["../../src/recorder/traffic-recorder.ts"],"sourcesContent":[],"mappings":";;;UAGiB,YAAA;iBACA,kBAAkB;EADlB,IAAA,CAAA,SAAY,CAAA,EAGb,eAHa,CAAA,WAAA,CAAA,CAAA,MAAA,CAAA,CAAA,EAIxB,OAJwB,CAIhB,eAJgB,EAAA,CAAA;;AACM,cAMtB,oBAAA,YAAgC,YANV,CAAA;EAEnB,iBAAA,KAAA;EACH,IAAA,CAAA,QAAA,EAMU,eANV,CAAA,EAM4B,OAN5B,CAAA,IAAA,CAAA;EAAR,IAAA,CAAA,SAAA,CAAA,EAAA,MAAA,CAAA,EAU6B,OAV7B,CAUqC,eAVrC,EAAA,CAAA;;AAGQ,UAaI,sBAAA,CAbiB;EAGX,KAAA,EAWd,YAXc;EAAkB,UAAA,CAAA,EAAA,MAAA;EAIC,QAAA,CAAA,EAAA,CAAA,QAAA,EASlB,eATkB,EAAA,GASE,eATF;;AAPG,UAmB5B,oBAAA,CAnB4B;EAAY,SAAA,EAoB5C,eApB4C,CAAA,WAAA,CAAA;EAaxC,KAAA,EAAA,OAAA;EACR,MAAA,CAAA,EAAA,OAAA;EAEe,KAAA,CAAA,EAOd,eAPc,CAAA,OAAA,CAAA;EAAoB,OAAA,EAAA,OAAA;EAAe,UAAA,CAAA,EAAA,MAAA;EAG1C,QAAA,CAAA,EAAA,MAAA;EACJ,MAAA,CAAA,EAAA,MAAA;EAGH,OAAA,CAAA,EAAA,MAAA;EAMG,QAAA,CAAA,EAAA,MAAA,CAAA,MAAA,EAAA,OAAA,CAAA;;AAGA,cAAA,eAAA,CAAe;EAKL,iBAAA,KAAA;EAMD,iBAAA,UAAA;EAAoB,iBAAA,QAAA;EAAA,WAAA,CAAA,OAAA,EANnB,sBAMmB;gBAApB,uBAAoB"}
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
import{randomUUID as e}from"node:crypto";var t=class{items=[];async save(e){this.items.push(e)}async list(e){return e?this.items.filter(t=>t.operation.name===e):[...this.items]}},n=class{store;sampleRate;sanitize;constructor(e){this.store=e.store,this.sampleRate=e.sampleRate??1,this.sanitize=e.sanitize}async record(t){if(!this.shouldSample())return;let n={id:e(),operation:t.operation,input:r(t.input),output:r(t.output),error:t.error?r(t.error):void 0,success:t.success,timestamp:new Date,durationMs:t.durationMs,tenantId:t.tenantId,userId:t.userId,channel:t.channel,metadata:t.metadata},i=this.sanitize?this.sanitize(n):n;await this.store.save(i)}shouldSample(){return this.sampleRate>=1?!0:Math.random()<=this.sampleRate}};function r(e){if(e==null)return e??void 0;try{let t=globalThis.structuredClone;return typeof t==`function`?t(e):JSON.parse(JSON.stringify(e))}catch{return}}export{t as InMemoryTrafficStore,n as TrafficRecorder};
|
|
2
|
-
//# sourceMappingURL=traffic-recorder.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"traffic-recorder.js","names":["snapshot: TrafficSnapshot"],"sources":["../../src/recorder/traffic-recorder.ts"],"sourcesContent":["import { randomUUID } from 'node:crypto';\nimport type { TrafficSnapshot } from '../types';\n\nexport interface TrafficStore {\n save(snapshot: TrafficSnapshot): Promise<void>;\n list(\n operation?: TrafficSnapshot['operation']['name']\n ): Promise<TrafficSnapshot[]>;\n}\n\nexport class InMemoryTrafficStore implements TrafficStore {\n private readonly items: TrafficSnapshot[] = [];\n\n async save(snapshot: TrafficSnapshot): Promise<void> {\n this.items.push(snapshot);\n }\n\n async list(operation?: string): Promise<TrafficSnapshot[]> {\n if (!operation) return [...this.items];\n return this.items.filter((item) => item.operation.name === operation);\n }\n}\n\nexport interface TrafficRecorderOptions {\n store: TrafficStore;\n sampleRate?: number;\n sanitize?: (snapshot: TrafficSnapshot) => TrafficSnapshot;\n}\n\nexport interface RecordOperationInput {\n operation: TrafficSnapshot['operation'];\n input: unknown;\n output?: unknown;\n error?: TrafficSnapshot['error'];\n success: boolean;\n durationMs?: number;\n tenantId?: string;\n userId?: string;\n channel?: string;\n metadata?: Record<string, unknown>;\n}\n\nexport class TrafficRecorder {\n private readonly store: TrafficStore;\n private readonly sampleRate: number;\n private readonly sanitize?: (snapshot: TrafficSnapshot) => TrafficSnapshot;\n\n constructor(options: TrafficRecorderOptions) {\n this.store = options.store;\n this.sampleRate = options.sampleRate ?? 1;\n this.sanitize = options.sanitize;\n }\n\n async record(input: RecordOperationInput) {\n if (!this.shouldSample()) return;\n const snapshot: TrafficSnapshot = {\n id: randomUUID(),\n operation: input.operation,\n input: structuredCloneSafe(input.input),\n output: structuredCloneSafe(input.output),\n error: input.error ? structuredCloneSafe(input.error) : undefined,\n success: input.success,\n timestamp: new Date(),\n durationMs: input.durationMs,\n tenantId: input.tenantId,\n userId: input.userId,\n channel: input.channel,\n metadata: input.metadata,\n };\n const sanitized = this.sanitize ? this.sanitize(snapshot) : snapshot;\n await this.store.save(sanitized);\n }\n\n private shouldSample() {\n if (this.sampleRate >= 1) return true;\n return Math.random() <= this.sampleRate;\n }\n}\n\nfunction structuredCloneSafe<T>(value: T): T | undefined {\n if (value == null) return value ?? undefined;\n try {\n const clone = (globalThis as { structuredClone?: <R>(input: R) => R })\n .structuredClone;\n if (typeof clone === 'function') {\n return clone(value);\n }\n return JSON.parse(JSON.stringify(value));\n } catch {\n return undefined;\n }\n}\n"],"mappings":"yCAUA,IAAa,EAAb,KAA0D,CACxD,MAA4C,EAAE,CAE9C,MAAM,KAAK,EAA0C,CACnD,KAAK,MAAM,KAAK,EAAS,CAG3B,MAAM,KAAK,EAAgD,CAEzD,OADK,EACE,KAAK,MAAM,OAAQ,GAAS,EAAK,UAAU,OAAS,EAAU,CAD9C,CAAC,GAAG,KAAK,MAAM,GAwB7B,EAAb,KAA6B,CAC3B,MACA,WACA,SAEA,YAAY,EAAiC,CAC3C,KAAK,MAAQ,EAAQ,MACrB,KAAK,WAAa,EAAQ,YAAc,EACxC,KAAK,SAAW,EAAQ,SAG1B,MAAM,OAAO,EAA6B,CACxC,GAAI,CAAC,KAAK,cAAc,CAAE,OAC1B,IAAMA,EAA4B,CAChC,GAAI,GAAY,CAChB,UAAW,EAAM,UACjB,MAAO,EAAoB,EAAM,MAAM,CACvC,OAAQ,EAAoB,EAAM,OAAO,CACzC,MAAO,EAAM,MAAQ,EAAoB,EAAM,MAAM,CAAG,IAAA,GACxD,QAAS,EAAM,QACf,UAAW,IAAI,KACf,WAAY,EAAM,WAClB,SAAU,EAAM,SAChB,OAAQ,EAAM,OACd,QAAS,EAAM,QACf,SAAU,EAAM,SACjB,CACK,EAAY,KAAK,SAAW,KAAK,SAAS,EAAS,CAAG,EAC5D,MAAM,KAAK,MAAM,KAAK,EAAU,CAGlC,cAAuB,CAErB,OADI,KAAK,YAAc,EAAU,GAC1B,KAAK,QAAQ,EAAI,KAAK,aAIjC,SAAS,EAAuB,EAAyB,CACvD,GAAI,GAAS,KAAM,OAAO,GAAS,IAAA,GACnC,GAAI,CACF,IAAM,EAAS,WACZ,gBAIH,OAHI,OAAO,GAAU,WACZ,EAAM,EAAM,CAEd,KAAK,MAAM,KAAK,UAAU,EAAM,CAAC,MAClC,CACN"}
|
package/dist/types.d.ts
DELETED
|
@@ -1,43 +0,0 @@
|
|
|
1
|
-
import { ContractSpec, ResourceRefDescriptor } from "@lssm/lib.contracts";
|
|
2
|
-
import { AnySchemaModel } from "@lssm/lib.schema";
|
|
3
|
-
|
|
4
|
-
//#region src/types.d.ts
|
|
5
|
-
interface TrafficSnapshot {
|
|
6
|
-
id: string;
|
|
7
|
-
operation: {
|
|
8
|
-
name: string;
|
|
9
|
-
version: number;
|
|
10
|
-
};
|
|
11
|
-
input: unknown;
|
|
12
|
-
output?: unknown;
|
|
13
|
-
error?: {
|
|
14
|
-
name?: string;
|
|
15
|
-
message?: string;
|
|
16
|
-
stack?: string;
|
|
17
|
-
code?: string;
|
|
18
|
-
};
|
|
19
|
-
success: boolean;
|
|
20
|
-
timestamp: Date;
|
|
21
|
-
durationMs?: number;
|
|
22
|
-
tenantId?: string;
|
|
23
|
-
userId?: string;
|
|
24
|
-
channel?: string;
|
|
25
|
-
metadata?: Record<string, unknown>;
|
|
26
|
-
}
|
|
27
|
-
interface GoldenTestCase {
|
|
28
|
-
id: string;
|
|
29
|
-
name: string;
|
|
30
|
-
input: unknown;
|
|
31
|
-
expectedOutput?: unknown;
|
|
32
|
-
expectedError?: {
|
|
33
|
-
name?: string;
|
|
34
|
-
message?: string;
|
|
35
|
-
code?: string;
|
|
36
|
-
};
|
|
37
|
-
success: boolean;
|
|
38
|
-
metadata?: Record<string, unknown>;
|
|
39
|
-
}
|
|
40
|
-
type RuntimeContract = ContractSpec<AnySchemaModel, AnySchemaModel | ResourceRefDescriptor<boolean>>;
|
|
41
|
-
//#endregion
|
|
42
|
-
export { GoldenTestCase, RuntimeContract, TrafficSnapshot };
|
|
43
|
-
//# sourceMappingURL=types.d.ts.map
|
package/dist/types.d.ts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","names":[],"sources":["../src/types.ts"],"sourcesContent":[],"mappings":";;;;UAGiB,eAAA;;EAAA,SAAA,EAAA;IAoBA,IAAA,EAAA,MAAA;IAcL,OAAA,EAAA,MAAe;EACzB,CAAA;EACA,KAAA,EAAA,OAAA;EAAiB,MAAA,CAAA,EAAA,OAAA;EAFW,KAAA,CAAA,EAAA;IAAY,IAAA,CAAA,EAAA,MAAA;;;;;;aAtB7B;;;;;aAKA;;UAGI,cAAA;;;;;;;;;;;aAWJ;;KAGD,eAAA,GAAkB,aAC5B,gBACA,iBAAiB"}
|