@nexart/ai-execution 0.1.0 → 0.3.0

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 (53) hide show
  1. package/README.md +146 -232
  2. package/dist/archive.d.ts +4 -0
  3. package/dist/archive.d.ts.map +1 -0
  4. package/dist/archive.js +28 -0
  5. package/dist/archive.js.map +1 -0
  6. package/dist/attest.d.ts +3 -0
  7. package/dist/attest.d.ts.map +1 -0
  8. package/dist/attest.js +79 -0
  9. package/dist/attest.js.map +1 -0
  10. package/dist/certify.d.ts +3 -0
  11. package/dist/certify.d.ts.map +1 -0
  12. package/dist/certify.js +27 -0
  13. package/dist/certify.js.map +1 -0
  14. package/dist/errors.d.ts +11 -0
  15. package/dist/errors.d.ts.map +1 -0
  16. package/dist/errors.js +21 -0
  17. package/dist/errors.js.map +1 -0
  18. package/dist/index.d.ts +8 -2
  19. package/dist/index.d.ts.map +1 -1
  20. package/dist/index.js +7 -1
  21. package/dist/index.js.map +1 -1
  22. package/dist/providers/anthropic.d.ts +22 -0
  23. package/dist/providers/anthropic.d.ts.map +1 -0
  24. package/dist/providers/anthropic.js +61 -0
  25. package/dist/providers/anthropic.js.map +1 -0
  26. package/dist/providers/wrap.d.ts +7 -0
  27. package/dist/providers/wrap.d.ts.map +1 -0
  28. package/dist/providers/wrap.js +28 -0
  29. package/dist/providers/wrap.js.map +1 -0
  30. package/dist/run.d.ts +14 -0
  31. package/dist/run.d.ts.map +1 -0
  32. package/dist/run.js +62 -0
  33. package/dist/run.js.map +1 -0
  34. package/dist/snapshot.d.ts.map +1 -1
  35. package/dist/snapshot.js +15 -2
  36. package/dist/snapshot.js.map +1 -1
  37. package/dist/types.d.ts +100 -0
  38. package/dist/types.d.ts.map +1 -1
  39. package/fixtures/golden/golden-001.json +29 -0
  40. package/fixtures/golden/golden-002.json +38 -0
  41. package/fixtures/golden/golden-003.json +36 -0
  42. package/fixtures/golden/golden-004.json +29 -0
  43. package/fixtures/golden/golden-005.json +29 -0
  44. package/fixtures/vectors/vector-002.chain.json +101 -0
  45. package/package.json +15 -3
  46. package/dist/__tests__/fixtures.test.d.ts +0 -2
  47. package/dist/__tests__/fixtures.test.d.ts.map +0 -1
  48. package/dist/__tests__/fixtures.test.js +0 -37
  49. package/dist/__tests__/fixtures.test.js.map +0 -1
  50. package/dist/__tests__/vectors.test.d.ts +0 -2
  51. package/dist/__tests__/vectors.test.d.ts.map +0 -1
  52. package/dist/__tests__/vectors.test.js +0 -261
  53. package/dist/__tests__/vectors.test.js.map +0 -1
@@ -0,0 +1,36 @@
1
+ {
2
+ "bundleType": "cer.ai.execution.v1",
3
+ "certificateHash": "sha256:59e14f23d7c6e9a8ca04f54e7b2d118292f23ebcbf97b4200c78c21d4da0632c",
4
+ "createdAt": "2026-02-12T00:00:00.000Z",
5
+ "version": "0.1",
6
+ "snapshot": {
7
+ "type": "ai.execution.v1",
8
+ "protocolVersion": "1.2.0",
9
+ "executionSurface": "ai",
10
+ "executionId": "golden-003",
11
+ "timestamp": "2026-02-12T00:00:00.000Z",
12
+ "provider": "anthropic",
13
+ "model": "claude-3-opus",
14
+ "modelVersion": "2026-01-15",
15
+ "prompt": "Summarize.",
16
+ "input": "Long text about AI safety and alignment research.",
17
+ "inputHash": "sha256:0293e7e89bc1d34ab2639d976daba18de9a9055c9dfc154a522759662f47109c",
18
+ "parameters": {
19
+ "temperature": 0.3,
20
+ "maxTokens": 2048,
21
+ "topP": null,
22
+ "seed": null
23
+ },
24
+ "output": "AI safety research focuses on alignment.",
25
+ "outputHash": "sha256:bd13453d2b1868ad264dbcad17384231c7b9fd2be518b478c1effa79dc50cc60",
26
+ "sdkVersion": "0.1.0",
27
+ "appId": null
28
+ },
29
+ "meta": {
30
+ "source": "test-suite",
31
+ "tags": [
32
+ "golden",
33
+ "meta"
34
+ ]
35
+ }
36
+ }
@@ -0,0 +1,29 @@
1
+ {
2
+ "bundleType": "cer.ai.execution.v1",
3
+ "certificateHash": "sha256:b7600a7f066320305d2b322d0bfed2a435af587ee4841eb017fa78a45f76b928",
4
+ "createdAt": "2026-02-12T00:00:00.000Z",
5
+ "version": "0.1",
6
+ "snapshot": {
7
+ "type": "ai.execution.v1",
8
+ "protocolVersion": "1.2.0",
9
+ "executionSurface": "ai",
10
+ "executionId": "golden-004",
11
+ "timestamp": "2026-02-12T00:00:00.000Z",
12
+ "provider": "openai",
13
+ "model": "gpt-4o-mini",
14
+ "modelVersion": null,
15
+ "prompt": "Reply briefly.",
16
+ "input": "Hello",
17
+ "inputHash": "sha256:185f8db32271fe25f561a6fc938b2e264306ec304eda518007d1764826381969",
18
+ "parameters": {
19
+ "temperature": 1,
20
+ "maxTokens": 64,
21
+ "topP": null,
22
+ "seed": null
23
+ },
24
+ "output": "Hi!",
25
+ "outputHash": "sha256:ca51ce1fb15acc6d69b8a5700256172fcc507e02073e6f19592e341bd6508ab8",
26
+ "sdkVersion": "0.1.0",
27
+ "appId": null
28
+ }
29
+ }
@@ -0,0 +1,29 @@
1
+ {
2
+ "bundleType": "cer.ai.execution.v1",
3
+ "certificateHash": "sha256:184d658b451dbc6618007203efe35195f18841c8e11fa55f7e73b98f65c69efe",
4
+ "createdAt": "2026-02-12T00:00:00.000Z",
5
+ "version": "0.1",
6
+ "snapshot": {
7
+ "type": "ai.execution.v1",
8
+ "protocolVersion": "1.2.0",
9
+ "executionSurface": "ai",
10
+ "executionId": "golden-005",
11
+ "timestamp": "2026-02-12T00:00:00.000Z",
12
+ "provider": "openai",
13
+ "model": "gpt-4o",
14
+ "modelVersion": "2026-02-01",
15
+ "prompt": "Be precise.",
16
+ "input": "Calculate pi to 5 decimals.",
17
+ "inputHash": "sha256:b4794eea4e9c2108260389b08c97d4939c35cd93e432783d5a2e98f7088060c3",
18
+ "parameters": {
19
+ "temperature": 0,
20
+ "maxTokens": 128,
21
+ "topP": 0.95,
22
+ "seed": 12345
23
+ },
24
+ "output": "3.14159",
25
+ "outputHash": "sha256:c0740dd25c9de39b9c8d5ab452e8b69bcc0bf86f2a60ed7e527e79d0a3035852",
26
+ "sdkVersion": "0.1.0",
27
+ "appId": "pi-calc"
28
+ }
29
+ }
@@ -0,0 +1,101 @@
1
+ [
2
+ {
3
+ "snapshot": {
4
+ "type": "ai.execution.v1",
5
+ "protocolVersion": "1.2.0",
6
+ "executionSurface": "ai",
7
+ "executionId": "vec-002-step-0",
8
+ "timestamp": "2026-03-01T00:00:00.000Z",
9
+ "provider": "openai",
10
+ "model": "gpt-4o",
11
+ "modelVersion": null,
12
+ "prompt": "Plan the analysis.",
13
+ "input": "Analyze sales data for Q1",
14
+ "inputHash": "sha256:11503825ac14fd8f8b118eafca0f657bbc4b3154243dcadb822624872629aa34",
15
+ "parameters": {
16
+ "temperature": 0.3,
17
+ "maxTokens": 512,
18
+ "topP": null,
19
+ "seed": null
20
+ },
21
+ "output": "I will: 1) load data, 2) compute totals, 3) summarize.",
22
+ "outputHash": "sha256:87eede0dfd7afd3d7bdb828633f8d99d67bd6809e3d1449ba241101d10753fc5",
23
+ "sdkVersion": "0.2.0",
24
+ "appId": null,
25
+ "runId": "run-vec-002",
26
+ "stepId": "step-0",
27
+ "stepIndex": 0,
28
+ "workflowId": "wf-analysis",
29
+ "conversationId": null,
30
+ "prevStepHash": null
31
+ },
32
+ "expectedCertificateHash": "sha256:a159ff1a6591384bd102d289040f36317ae35265095b76a71d485dc922db4cc6",
33
+ "cerCreatedAt": "2026-03-01T00:00:00.000Z"
34
+ },
35
+ {
36
+ "snapshot": {
37
+ "type": "ai.execution.v1",
38
+ "protocolVersion": "1.2.0",
39
+ "executionSurface": "ai",
40
+ "executionId": "vec-002-step-1",
41
+ "timestamp": "2026-03-01T00:00:00.000Z",
42
+ "provider": "openai",
43
+ "model": "gpt-4o",
44
+ "modelVersion": null,
45
+ "prompt": "Execute step 1 of the plan.",
46
+ "input": "Load and total the Q1 data.",
47
+ "inputHash": "sha256:544b18ed4cf2caa45e3aad3dbabdca50a7f7bb8e354ce35c406c96f3929b0b34",
48
+ "parameters": {
49
+ "temperature": 0.3,
50
+ "maxTokens": 512,
51
+ "topP": null,
52
+ "seed": null
53
+ },
54
+ "output": "Total revenue: $1.2M across 3 regions.",
55
+ "outputHash": "sha256:50f990992c3b9a261ac4594e535e176d528a4a320c9d8eb5ff5631ac1fbba80a",
56
+ "sdkVersion": "0.2.0",
57
+ "appId": null,
58
+ "runId": "run-vec-002",
59
+ "stepId": "step-1",
60
+ "stepIndex": 1,
61
+ "workflowId": "wf-analysis",
62
+ "conversationId": null,
63
+ "prevStepHash": "sha256:a159ff1a6591384bd102d289040f36317ae35265095b76a71d485dc922db4cc6"
64
+ },
65
+ "expectedCertificateHash": "sha256:9f1116538932bb1be27b01a33d5037dacca22548c7704a3bc3ddcd5e75c9bf2e",
66
+ "cerCreatedAt": "2026-03-01T00:00:00.000Z"
67
+ },
68
+ {
69
+ "snapshot": {
70
+ "type": "ai.execution.v1",
71
+ "protocolVersion": "1.2.0",
72
+ "executionSurface": "ai",
73
+ "executionId": "vec-002-step-2",
74
+ "timestamp": "2026-03-01T00:00:00.000Z",
75
+ "provider": "openai",
76
+ "model": "gpt-4o",
77
+ "modelVersion": null,
78
+ "prompt": "Summarize findings.",
79
+ "input": "Summarize: Total revenue $1.2M, 3 regions.",
80
+ "inputHash": "sha256:9707eccac666818b01558a57356bcf590ed83ae342c091c80a380372c5a05958",
81
+ "parameters": {
82
+ "temperature": 0.3,
83
+ "maxTokens": 512,
84
+ "topP": null,
85
+ "seed": null
86
+ },
87
+ "output": "Q1 revenue was $1.2M distributed across North, South, and West regions.",
88
+ "outputHash": "sha256:399b3ce0ef2b056df2f2bc78fb252385be40af12cae260e6750f478391418a7f",
89
+ "sdkVersion": "0.2.0",
90
+ "appId": null,
91
+ "runId": "run-vec-002",
92
+ "stepId": "step-2",
93
+ "stepIndex": 2,
94
+ "workflowId": "wf-analysis",
95
+ "conversationId": null,
96
+ "prevStepHash": "sha256:9f1116538932bb1be27b01a33d5037dacca22548c7704a3bc3ddcd5e75c9bf2e"
97
+ },
98
+ "expectedCertificateHash": "sha256:7cb3f19a32e6905170fd80b203d8c75b3a3ea67e090d0231e9ee7ea95ec57b8d",
99
+ "cerCreatedAt": "2026-03-01T00:00:00.000Z"
100
+ }
101
+ ]
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@nexart/ai-execution",
3
- "version": "0.1.0",
3
+ "version": "0.3.0",
4
4
  "description": "AI Execution Integrity — tamper-evident records and Certified Execution Records (CER) for AI operations",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",
@@ -13,11 +13,20 @@
13
13
  "./providers/openai": {
14
14
  "types": "./dist/providers/openai.d.ts",
15
15
  "import": "./dist/providers/openai.js"
16
+ },
17
+ "./providers/anthropic": {
18
+ "types": "./dist/providers/anthropic.d.ts",
19
+ "import": "./dist/providers/anthropic.js"
20
+ },
21
+ "./providers/wrap": {
22
+ "types": "./dist/providers/wrap.d.ts",
23
+ "import": "./dist/providers/wrap.js"
16
24
  }
17
25
  },
18
26
  "scripts": {
19
27
  "build": "tsc",
20
- "test": "node --test dist/__tests__/vectors.test.js dist/__tests__/fixtures.test.js",
28
+ "test": "node --test dist/__tests__/vectors.test.js dist/__tests__/fixtures.test.js dist/__tests__/v020.test.js dist/__tests__/v030.test.js",
29
+ "release": "npm run build && npm test && npm version patch && npm publish --access public",
21
30
  "prepublishOnly": "npm run build && npm run test"
22
31
  },
23
32
  "keywords": [
@@ -29,7 +38,9 @@
29
38
  "cer",
30
39
  "snapshot",
31
40
  "sha256",
32
- "tamper-evident"
41
+ "tamper-evident",
42
+ "agentic",
43
+ "workflow"
33
44
  ],
34
45
  "author": "NexArt",
35
46
  "license": "MIT",
@@ -38,6 +49,7 @@
38
49
  },
39
50
  "files": [
40
51
  "dist",
52
+ "!dist/__tests__",
41
53
  "fixtures",
42
54
  "README.md"
43
55
  ],
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=fixtures.test.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"fixtures.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/fixtures.test.ts"],"names":[],"mappings":""}
@@ -1,37 +0,0 @@
1
- import { describe, it } from 'node:test';
2
- import assert from 'node:assert/strict';
3
- import { readFileSync } from 'node:fs';
4
- import { fileURLToPath } from 'node:url';
5
- import { dirname, join } from 'node:path';
6
- import { verifySnapshot } from '../snapshot.js';
7
- import { sealCer, verifyCer } from '../cer.js';
8
- const __filename = fileURLToPath(import.meta.url);
9
- const __dirname = dirname(__filename);
10
- const fixturesDir = join(__dirname, '..', '..', 'fixtures', 'vectors');
11
- function loadJson(filename) {
12
- return JSON.parse(readFileSync(join(fixturesDir, filename), 'utf-8'));
13
- }
14
- describe('fixture: vector-001', () => {
15
- const snapshot = loadJson('vector-001.snapshot.json');
16
- const expected = loadJson('vector-001.expected.json');
17
- it('snapshot inputHash matches expected', () => {
18
- assert.equal(snapshot.inputHash, expected.inputHash);
19
- });
20
- it('snapshot outputHash matches expected', () => {
21
- assert.equal(snapshot.outputHash, expected.outputHash);
22
- });
23
- it('snapshot passes verification', () => {
24
- const result = verifySnapshot(snapshot);
25
- assert.equal(result.ok, true, `verification errors: ${result.errors.join('; ')}`);
26
- });
27
- it('CER certificateHash matches expected', () => {
28
- const bundle = sealCer(snapshot, { createdAt: expected.cerCreatedAt });
29
- assert.equal(bundle.certificateHash, expected.certificateHash);
30
- });
31
- it('CER bundle passes verification', () => {
32
- const bundle = sealCer(snapshot, { createdAt: expected.cerCreatedAt });
33
- const result = verifyCer(bundle);
34
- assert.equal(result.ok, true, `verification errors: ${result.errors.join('; ')}`);
35
- });
36
- });
37
- //# sourceMappingURL=fixtures.test.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"fixtures.test.js","sourceRoot":"","sources":["../../src/__tests__/fixtures.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,EAAE,aAAa,EAAE,MAAM,UAAU,CAAC;AACzC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAG/C,MAAM,UAAU,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AAClD,MAAM,SAAS,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;AACtC,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,EAAE,IAAI,EAAE,IAAI,EAAE,UAAU,EAAE,SAAS,CAAC,CAAC;AAEvE,SAAS,QAAQ,CAAC,QAAgB;IAChC,OAAO,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;AACxE,CAAC;AAED,QAAQ,CAAC,qBAAqB,EAAE,GAAG,EAAE;IACnC,MAAM,QAAQ,GAAG,QAAQ,CAAC,0BAA0B,CAA0B,CAAC;IAC/E,MAAM,QAAQ,GAAG,QAAQ,CAAC,0BAA0B,CAKnD,CAAC;IAEF,EAAE,CAAC,qCAAqC,EAAE,GAAG,EAAE;QAC7C,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,SAAS,EAAE,QAAQ,CAAC,SAAS,CAAC,CAAC;IACvD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,UAAU,EAAE,QAAQ,CAAC,UAAU,CAAC,CAAC;IACzD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,MAAM,GAAG,cAAc,CAAC,QAAQ,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,IAAI,EAAE,wBAAwB,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACpF,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,MAAM,GAAG,OAAO,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,QAAQ,CAAC,YAAY,EAAE,CAAC,CAAC;QACvE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,eAAe,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACjE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;QACxC,MAAM,MAAM,GAAG,OAAO,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,QAAQ,CAAC,YAAY,EAAE,CAAC,CAAC;QACvE,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QACjC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,IAAI,EAAE,wBAAwB,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACpF,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -1,2 +0,0 @@
1
- export {};
2
- //# sourceMappingURL=vectors.test.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"vectors.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/vectors.test.ts"],"names":[],"mappings":""}
@@ -1,261 +0,0 @@
1
- import { describe, it } from 'node:test';
2
- import assert from 'node:assert/strict';
3
- import { toCanonicalJson } from '../canonicalJson.js';
4
- import { sha256Hex, hashUtf8, hashCanonicalJson, computeInputHash, computeOutputHash } from '../hash.js';
5
- import { createSnapshot, verifySnapshot } from '../snapshot.js';
6
- import { sealCer, verifyCer } from '../cer.js';
7
- const FIXED_TIMESTAMP = '2026-02-12T00:00:00.000Z';
8
- const FIXED_EXECUTION_ID = 'test-exec-001';
9
- describe('canonicalJson', () => {
10
- it('sorts object keys lexicographically', () => {
11
- const result = toCanonicalJson({ z: 1, a: 2, m: 3 });
12
- assert.equal(result, '{"a":2,"m":3,"z":1}');
13
- });
14
- it('does not sort arrays', () => {
15
- const result = toCanonicalJson([3, 1, 2]);
16
- assert.equal(result, '[3,1,2]');
17
- });
18
- it('handles nested objects', () => {
19
- const result = toCanonicalJson({ b: { d: 1, c: 2 }, a: 3 });
20
- assert.equal(result, '{"a":3,"b":{"c":2,"d":1}}');
21
- });
22
- it('handles null', () => {
23
- assert.equal(toCanonicalJson(null), 'null');
24
- });
25
- it('handles strings with escapes', () => {
26
- assert.equal(toCanonicalJson('hello "world"'), '"hello \\"world\\""');
27
- });
28
- it('rejects NaN', () => {
29
- assert.throws(() => toCanonicalJson(NaN), /Non-finite/);
30
- });
31
- it('rejects Infinity', () => {
32
- assert.throws(() => toCanonicalJson(Infinity), /Non-finite/);
33
- });
34
- it('omits undefined values in objects', () => {
35
- const result = toCanonicalJson({ a: 1, b: undefined, c: 3 });
36
- assert.equal(result, '{"a":1,"c":3}');
37
- });
38
- it('produces no whitespace', () => {
39
- const result = toCanonicalJson({ key: 'value', arr: [1, 2] });
40
- assert.ok(!result.includes(' '));
41
- assert.ok(!result.includes('\n'));
42
- });
43
- });
44
- describe('hashing', () => {
45
- it('sha256Hex produces correct hex for empty string', () => {
46
- const result = sha256Hex('');
47
- assert.equal(result, 'e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855');
48
- });
49
- it('hashUtf8 prefixes with sha256:', () => {
50
- const result = hashUtf8('hello');
51
- assert.ok(result.startsWith('sha256:'));
52
- assert.equal(result, 'sha256:2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824');
53
- });
54
- it('hashCanonicalJson is stable for same object', () => {
55
- const obj = { b: 2, a: 1 };
56
- const h1 = hashCanonicalJson(obj);
57
- const h2 = hashCanonicalJson({ a: 1, b: 2 });
58
- assert.equal(h1, h2);
59
- });
60
- it('computeInputHash for string uses hashUtf8', () => {
61
- const result = computeInputHash('test input');
62
- assert.equal(result, hashUtf8('test input'));
63
- });
64
- it('computeInputHash for object uses hashCanonicalJson', () => {
65
- const obj = { role: 'user', content: 'hello' };
66
- const result = computeInputHash(obj);
67
- assert.equal(result, hashCanonicalJson(obj));
68
- });
69
- });
70
- describe('snapshot', () => {
71
- const baseParams = {
72
- executionId: FIXED_EXECUTION_ID,
73
- timestamp: FIXED_TIMESTAMP,
74
- provider: 'openai',
75
- model: 'gpt-4o',
76
- modelVersion: '2026-01-01',
77
- prompt: 'You are a helpful assistant.',
78
- input: 'What is 2+2?',
79
- parameters: {
80
- temperature: 0.7,
81
- maxTokens: 1024,
82
- topP: null,
83
- seed: null,
84
- },
85
- output: 'The answer is 4.',
86
- sdkVersion: '0.1.0',
87
- appId: null,
88
- };
89
- it('creates snapshot with correct type fields', () => {
90
- const snap = createSnapshot(baseParams);
91
- assert.equal(snap.type, 'ai.execution.v1');
92
- assert.equal(snap.protocolVersion, '1.2.0');
93
- assert.equal(snap.executionSurface, 'ai');
94
- assert.equal(snap.executionId, FIXED_EXECUTION_ID);
95
- assert.equal(snap.timestamp, FIXED_TIMESTAMP);
96
- });
97
- it('computes correct inputHash for text input', () => {
98
- const snap = createSnapshot(baseParams);
99
- const expected = computeInputHash('What is 2+2?');
100
- assert.equal(snap.inputHash, expected);
101
- });
102
- it('computes correct outputHash for text output', () => {
103
- const snap = createSnapshot(baseParams);
104
- const expected = computeOutputHash('The answer is 4.');
105
- assert.equal(snap.outputHash, expected);
106
- });
107
- it('computes correct hashes for JSON input/output', () => {
108
- const jsonParams = {
109
- ...baseParams,
110
- input: { messages: [{ role: 'user', content: 'hello' }] },
111
- output: { result: 'world', confidence: 0.95 },
112
- };
113
- const snap = createSnapshot(jsonParams);
114
- const expectedInput = computeInputHash(jsonParams.input);
115
- const expectedOutput = computeOutputHash(jsonParams.output);
116
- assert.equal(snap.inputHash, expectedInput);
117
- assert.equal(snap.outputHash, expectedOutput);
118
- });
119
- it('verifySnapshot passes for valid snapshot', () => {
120
- const snap = createSnapshot(baseParams);
121
- const result = verifySnapshot(snap);
122
- assert.equal(result.ok, true);
123
- assert.equal(result.errors.length, 0);
124
- });
125
- it('verifySnapshot fails on tampered output', () => {
126
- const snap = createSnapshot(baseParams);
127
- snap.output = 'Tampered output!';
128
- const result = verifySnapshot(snap);
129
- assert.equal(result.ok, false);
130
- assert.ok(result.errors.some(e => e.includes('outputHash mismatch')));
131
- });
132
- it('rejects non-finite temperature', () => {
133
- assert.throws(() => createSnapshot({ ...baseParams, parameters: { ...baseParams.parameters, temperature: NaN } }), /temperature/);
134
- });
135
- });
136
- describe('cer', () => {
137
- const baseParams = {
138
- executionId: FIXED_EXECUTION_ID,
139
- timestamp: FIXED_TIMESTAMP,
140
- provider: 'openai',
141
- model: 'gpt-4o',
142
- modelVersion: '2026-01-01',
143
- prompt: 'You are a helpful assistant.',
144
- input: 'What is 2+2?',
145
- parameters: {
146
- temperature: 0.7,
147
- maxTokens: 1024,
148
- topP: null,
149
- seed: null,
150
- },
151
- output: 'The answer is 4.',
152
- sdkVersion: '0.1.0',
153
- appId: null,
154
- };
155
- it('seals a CER bundle with correct fields', () => {
156
- const snap = createSnapshot(baseParams);
157
- const bundle = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
158
- assert.equal(bundle.bundleType, 'cer.ai.execution.v1');
159
- assert.equal(bundle.version, '0.1');
160
- assert.equal(bundle.createdAt, FIXED_TIMESTAMP);
161
- assert.ok(bundle.certificateHash.startsWith('sha256:'));
162
- assert.deepStrictEqual(bundle.snapshot, snap);
163
- });
164
- it('certificateHash is stable for same snapshot + createdAt', () => {
165
- const snap = createSnapshot(baseParams);
166
- const b1 = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
167
- const b2 = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
168
- assert.equal(b1.certificateHash, b2.certificateHash);
169
- });
170
- it('certificateHash changes when createdAt changes', () => {
171
- const snap = createSnapshot(baseParams);
172
- const b1 = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
173
- const b2 = sealCer(snap, { createdAt: '2026-02-13T00:00:00.000Z' });
174
- assert.notEqual(b1.certificateHash, b2.certificateHash);
175
- });
176
- it('verifyCer passes for valid bundle', () => {
177
- const snap = createSnapshot(baseParams);
178
- const bundle = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
179
- const result = verifyCer(bundle);
180
- assert.equal(result.ok, true);
181
- assert.equal(result.errors.length, 0);
182
- });
183
- it('verifyCer fails on tampered certificateHash', () => {
184
- const snap = createSnapshot(baseParams);
185
- const bundle = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
186
- bundle.certificateHash = 'sha256:0000000000000000000000000000000000000000000000000000000000000000';
187
- const result = verifyCer(bundle);
188
- assert.equal(result.ok, false);
189
- assert.ok(result.errors.some(e => e.includes('certificateHash mismatch')));
190
- });
191
- it('verifyCer fails on tampered snapshot output', () => {
192
- const snap = createSnapshot(baseParams);
193
- const bundle = sealCer(snap, { createdAt: FIXED_TIMESTAMP });
194
- bundle.snapshot.output = 'Tampered!';
195
- const result = verifyCer(bundle);
196
- assert.equal(result.ok, false);
197
- assert.ok(result.errors.some(e => e.includes('outputHash mismatch')));
198
- });
199
- it('includes meta when provided', () => {
200
- const snap = createSnapshot(baseParams);
201
- const bundle = sealCer(snap, {
202
- createdAt: FIXED_TIMESTAMP,
203
- meta: { source: 'test', tags: ['unit-test'] },
204
- });
205
- assert.deepStrictEqual(bundle.meta, { source: 'test', tags: ['unit-test'] });
206
- });
207
- });
208
- describe('deterministic test vectors', () => {
209
- it('text input/text output vector', () => {
210
- const snap = createSnapshot({
211
- executionId: 'vec-text-001',
212
- timestamp: '2026-01-01T00:00:00.000Z',
213
- provider: 'openai',
214
- model: 'gpt-4o',
215
- modelVersion: null,
216
- prompt: 'system prompt',
217
- input: 'hello world',
218
- parameters: { temperature: 0, maxTokens: 100, topP: null, seed: null },
219
- output: 'goodbye world',
220
- sdkVersion: '0.1.0',
221
- appId: null,
222
- });
223
- assert.equal(snap.inputHash, 'sha256:b94d27b9934d3e08a52e52d7da7dabfac484efe37a5380ee9088f7ace2efcde9');
224
- assert.equal(snap.outputHash, 'sha256:9150e02727e29ca8522c29ad4aa5a8343c21ccf909b40f73c41bf478df7e6fc3');
225
- const bundle = sealCer(snap, { createdAt: '2026-01-01T00:00:00.000Z' });
226
- assert.ok(bundle.certificateHash.startsWith('sha256:'));
227
- const v1 = verifySnapshot(snap);
228
- assert.equal(v1.ok, true);
229
- const v2 = verifyCer(bundle);
230
- assert.equal(v2.ok, true);
231
- });
232
- it('JSON input/JSON output vector', () => {
233
- const snap = createSnapshot({
234
- executionId: 'vec-json-001',
235
- timestamp: '2026-01-01T00:00:00.000Z',
236
- provider: 'openai',
237
- model: 'gpt-4o',
238
- modelVersion: '2026-01-01',
239
- prompt: 'structured',
240
- input: { query: 'test', context: [1, 2, 3] },
241
- parameters: { temperature: 0.5, maxTokens: 256, topP: 0.9, seed: 42 },
242
- output: { answer: 'result', score: 0.99 },
243
- sdkVersion: '0.1.0',
244
- appId: 'myapp',
245
- });
246
- const expectedInputCanonical = '{"context":[1,2,3],"query":"test"}';
247
- const expectedOutputCanonical = '{"answer":"result","score":0.99}';
248
- assert.equal(toCanonicalJson(snap.input), expectedInputCanonical);
249
- assert.equal(toCanonicalJson(snap.output), expectedOutputCanonical);
250
- assert.equal(snap.inputHash, computeInputHash({ query: 'test', context: [1, 2, 3] }));
251
- assert.equal(snap.outputHash, computeOutputHash({ answer: 'result', score: 0.99 }));
252
- const v1 = verifySnapshot(snap);
253
- assert.equal(v1.ok, true);
254
- const bundle = sealCer(snap, { createdAt: '2026-01-01T00:00:00.000Z' });
255
- const v2 = verifyCer(bundle);
256
- assert.equal(v2.ok, true);
257
- const bundle2 = sealCer(snap, { createdAt: '2026-01-01T00:00:00.000Z' });
258
- assert.equal(bundle.certificateHash, bundle2.certificateHash);
259
- });
260
- });
261
- //# sourceMappingURL=vectors.test.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"vectors.test.js","sourceRoot":"","sources":["../../src/__tests__/vectors.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,MAAM,MAAM,oBAAoB,CAAC;AACxC,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,SAAS,EAAE,QAAQ,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,MAAM,YAAY,CAAC;AACzG,OAAO,EAAE,cAAc,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAChE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAG/C,MAAM,eAAe,GAAG,0BAA0B,CAAC;AACnD,MAAM,kBAAkB,GAAG,eAAe,CAAC;AAE3C,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;IAC7B,EAAE,CAAC,qCAAqC,EAAE,GAAG,EAAE;QAC7C,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QACrD,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,qBAAqB,CAAC,CAAC;IAC9C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sBAAsB,EAAE,GAAG,EAAE;QAC9B,MAAM,MAAM,GAAG,eAAe,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wBAAwB,EAAE,GAAG,EAAE;QAChC,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QAC5D,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,2BAA2B,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,cAAc,EAAE,GAAG,EAAE;QACtB,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,MAAM,CAAC,CAAC;IAC9C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,eAAe,CAAC,EAAE,qBAAqB,CAAC,CAAC;IACxE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,aAAa,EAAE,GAAG,EAAE;QACrB,MAAM,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,GAAG,CAAC,EAAE,YAAY,CAAC,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kBAAkB,EAAE,GAAG,EAAE;QAC1B,MAAM,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,QAAQ,CAAC,EAAE,YAAY,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mCAAmC,EAAE,GAAG,EAAE;QAC3C,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QAC7D,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,eAAe,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wBAAwB,EAAE,GAAG,EAAE;QAChC,MAAM,MAAM,GAAG,eAAe,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC;QAC9D,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,SAAS,EAAE,GAAG,EAAE;IACvB,EAAE,CAAC,iDAAiD,EAAE,GAAG,EAAE;QACzD,MAAM,MAAM,GAAG,SAAS,CAAC,EAAE,CAAC,CAAC;QAC7B,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,kEAAkE,CAAC,CAAC;IAC3F,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;QACxC,MAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,CAAC,CAAC;QACjC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,yEAAyE,CAAC,CAAC;IAClG,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;QACrD,MAAM,GAAG,GAAG,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC;QAC3B,MAAM,EAAE,GAAG,iBAAiB,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,EAAE,GAAG,iBAAiB,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;QAC7C,MAAM,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;IACvB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,MAAM,GAAG,gBAAgB,CAAC,YAAY,CAAC,CAAC;QAC9C,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oDAAoD,EAAE,GAAG,EAAE;QAC5D,MAAM,GAAG,GAAG,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC;QAC/C,MAAM,MAAM,GAAG,gBAAgB,CAAC,GAAG,CAAC,CAAC;QACrC,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,iBAAiB,CAAC,GAAG,CAAC,CAAC,CAAC;IAC/C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,UAAU,EAAE,GAAG,EAAE;IACxB,MAAM,UAAU,GAAG;QACjB,WAAW,EAAE,kBAAkB;QAC/B,SAAS,EAAE,eAAe;QAC1B,QAAQ,EAAE,QAAQ;QAClB,KAAK,EAAE,QAAQ;QACf,YAAY,EAAE,YAA6B;QAC3C,MAAM,EAAE,8BAA8B;QACtC,KAAK,EAAE,cAAc;QACrB,UAAU,EAAE;YACV,WAAW,EAAE,GAAG;YAChB,SAAS,EAAE,IAAI;YACf,IAAI,EAAE,IAAqB;YAC3B,IAAI,EAAE,IAAqB;SAC5B;QACD,MAAM,EAAE,kBAAkB;QAC1B,UAAU,EAAE,OAAwB;QACpC,KAAK,EAAE,IAAqB;KAC7B,CAAC;IAEF,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE,iBAAiB,CAAC,CAAC;QAC3C,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;QAC5C,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,gBAAgB,EAAE,IAAI,CAAC,CAAC;QAC1C,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,WAAW,EAAE,kBAAkB,CAAC,CAAC;QACnD,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC;IAChD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;QACnD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,QAAQ,GAAG,gBAAgB,CAAC,cAAc,CAAC,CAAC;QAClD,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;QACrD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,QAAQ,GAAG,iBAAiB,CAAC,kBAAkB,CAAC,CAAC;QACvD,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;IAC1C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;QACvD,MAAM,UAAU,GAAG;YACjB,GAAG,UAAU;YACb,KAAK,EAAE,EAAE,QAAQ,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC,EAA6B;YACpF,MAAM,EAAE,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,EAAE,IAAI,EAA6B;SACzE,CAAC;QACF,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,aAAa,GAAG,gBAAgB,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QACzD,MAAM,cAAc,GAAG,iBAAiB,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAC5D,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,aAAa,CAAC,CAAC;QAC5C,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;IAChD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC;QACpC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAC9B,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;QACjD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACvC,IAA2C,CAAC,MAAM,GAAG,kBAAkB,CAAC;QACzE,MAAM,MAAM,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC;QACpC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QAC/B,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IACxE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;QACxC,MAAM,CAAC,MAAM,CACX,GAAG,EAAE,CAAC,cAAc,CAAC,EAAE,GAAG,UAAU,EAAE,UAAU,EAAE,EAAE,GAAG,UAAU,CAAC,UAAU,EAAE,WAAW,EAAE,GAAG,EAAE,EAAE,CAAC,EACnG,aAAa,CACd,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,KAAK,EAAE,GAAG,EAAE;IACnB,MAAM,UAAU,GAAG;QACjB,WAAW,EAAE,kBAAkB;QAC/B,SAAS,EAAE,eAAe;QAC1B,QAAQ,EAAE,QAAQ;QAClB,KAAK,EAAE,QAAQ;QACf,YAAY,EAAE,YAA6B;QAC3C,MAAM,EAAE,8BAA8B;QACtC,KAAK,EAAE,cAAc;QACrB,UAAU,EAAE;YACV,WAAW,EAAE,GAAG;YAChB,SAAS,EAAE,IAAI;YACf,IAAI,EAAE,IAAqB;YAC3B,IAAI,EAAE,IAAqB;SAC5B;QACD,MAAM,EAAE,kBAAkB;QAC1B,UAAU,EAAE,OAAwB;QACpC,KAAK,EAAE,IAAqB;KAC7B,CAAC;IAEF,EAAE,CAAC,wCAAwC,EAAE,GAAG,EAAE;QAChD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QAC7D,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,UAAU,EAAE,qBAAqB,CAAC,CAAC;QACvD,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QACpC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,eAAe,CAAC,CAAC;QAChD,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,eAAe,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC;QACxD,MAAM,CAAC,eAAe,CAAC,MAAM,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;IAChD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yDAAyD,EAAE,GAAG,EAAE;QACjE,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QACzD,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QACzD,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,eAAe,EAAE,EAAE,CAAC,eAAe,CAAC,CAAC;IACvD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QACzD,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,0BAA0B,EAAE,CAAC,CAAC;QACpE,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC,eAAe,EAAE,EAAE,CAAC,eAAe,CAAC,CAAC;IAC1D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mCAAmC,EAAE,GAAG,EAAE;QAC3C,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QAC7D,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QACjC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAC9B,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IACxC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;QACrD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QAC7D,MAAM,CAAC,eAAe,GAAG,yEAAyE,CAAC;QACnG,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QACjC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QAC/B,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,0BAA0B,CAAC,CAAC,CAAC,CAAC;IAC7E,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,GAAG,EAAE;QACrD,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,eAAe,EAAE,CAAC,CAAC;QAC5D,MAAM,CAAC,QAA+C,CAAC,MAAM,GAAG,WAAW,CAAC;QAC7E,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QACjC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;QAC/B,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;IACxE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE,GAAG,EAAE;QACrC,MAAM,IAAI,GAAG,cAAc,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE;YAC3B,SAAS,EAAE,eAAe;YAC1B,IAAI,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,WAAW,CAAC,EAAE;SAC9C,CAAC,CAAC;QACH,MAAM,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IAC/E,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,4BAA4B,EAAE,GAAG,EAAE;IAC1C,EAAE,CAAC,+BAA+B,EAAE,GAAG,EAAE;QACvC,MAAM,IAAI,GAAG,cAAc,CAAC;YAC1B,WAAW,EAAE,cAAc;YAC3B,SAAS,EAAE,0BAA0B;YACrC,QAAQ,EAAE,QAAQ;YAClB,KAAK,EAAE,QAAQ;YACf,YAAY,EAAE,IAAI;YAClB,MAAM,EAAE,eAAe;YACvB,KAAK,EAAE,aAAa;YACpB,UAAU,EAAE,EAAE,WAAW,EAAE,CAAC,EAAE,SAAS,EAAE,GAAG,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE;YACtE,MAAM,EAAE,eAAe;YACvB,UAAU,EAAE,OAAO;YACnB,KAAK,EAAE,IAAI;SACZ,CAAC,CAAC;QAEH,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,yEAAyE,CAAC,CAAC;QACxG,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,yEAAyE,CAAC,CAAC;QAEzG,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,0BAA0B,EAAE,CAAC,CAAC;QACxE,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,eAAe,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC,CAAC;QAExD,MAAM,EAAE,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC;QAChC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAE1B,MAAM,EAAE,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QAC7B,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+BAA+B,EAAE,GAAG,EAAE;QACvC,MAAM,IAAI,GAAG,cAAc,CAAC;YAC1B,WAAW,EAAE,cAAc;YAC3B,SAAS,EAAE,0BAA0B;YACrC,QAAQ,EAAE,QAAQ;YAClB,KAAK,EAAE,QAAQ;YACf,YAAY,EAAE,YAAY;YAC1B,MAAM,EAAE,YAAY;YACpB,KAAK,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,EAAE;YAC5C,UAAU,EAAE,EAAE,WAAW,EAAE,GAAG,EAAE,SAAS,EAAE,GAAG,EAAE,IAAI,EAAE,GAAG,EAAE,IAAI,EAAE,EAAE,EAAE;YACrE,MAAM,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,IAAI,EAAE;YACzC,UAAU,EAAE,OAAO;YACnB,KAAK,EAAE,OAAO;SACf,CAAC,CAAC;QAEH,MAAM,sBAAsB,GAAG,oCAAoC,CAAC;QACpE,MAAM,uBAAuB,GAAG,kCAAkC,CAAC;QAEnE,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,sBAAsB,CAAC,CAAC;QAClE,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,uBAAuB,CAAC,CAAC;QAEpE,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,EAAE,gBAAgB,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACtF,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,EAAE,iBAAiB,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;QAEpF,MAAM,EAAE,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC;QAChC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAE1B,MAAM,MAAM,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,0BAA0B,EAAE,CAAC,CAAC;QACxE,MAAM,EAAE,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;QAC7B,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,EAAE,EAAE,IAAI,CAAC,CAAC;QAE1B,MAAM,OAAO,GAAG,OAAO,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,0BAA0B,EAAE,CAAC,CAAC;QACzE,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,eAAe,EAAE,OAAO,CAAC,eAAe,CAAC,CAAC;IAChE,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}