@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.
- package/README.md +146 -232
- package/dist/archive.d.ts +4 -0
- package/dist/archive.d.ts.map +1 -0
- package/dist/archive.js +28 -0
- package/dist/archive.js.map +1 -0
- package/dist/attest.d.ts +3 -0
- package/dist/attest.d.ts.map +1 -0
- package/dist/attest.js +79 -0
- package/dist/attest.js.map +1 -0
- package/dist/certify.d.ts +3 -0
- package/dist/certify.d.ts.map +1 -0
- package/dist/certify.js +27 -0
- package/dist/certify.js.map +1 -0
- package/dist/errors.d.ts +11 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +21 -0
- package/dist/errors.js.map +1 -0
- package/dist/index.d.ts +8 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +7 -1
- package/dist/index.js.map +1 -1
- package/dist/providers/anthropic.d.ts +22 -0
- package/dist/providers/anthropic.d.ts.map +1 -0
- package/dist/providers/anthropic.js +61 -0
- package/dist/providers/anthropic.js.map +1 -0
- package/dist/providers/wrap.d.ts +7 -0
- package/dist/providers/wrap.d.ts.map +1 -0
- package/dist/providers/wrap.js +28 -0
- package/dist/providers/wrap.js.map +1 -0
- package/dist/run.d.ts +14 -0
- package/dist/run.d.ts.map +1 -0
- package/dist/run.js +62 -0
- package/dist/run.js.map +1 -0
- package/dist/snapshot.d.ts.map +1 -1
- package/dist/snapshot.js +15 -2
- package/dist/snapshot.js.map +1 -1
- package/dist/types.d.ts +100 -0
- package/dist/types.d.ts.map +1 -1
- package/fixtures/golden/golden-001.json +29 -0
- package/fixtures/golden/golden-002.json +38 -0
- package/fixtures/golden/golden-003.json +36 -0
- package/fixtures/golden/golden-004.json +29 -0
- package/fixtures/golden/golden-005.json +29 -0
- package/fixtures/vectors/vector-002.chain.json +101 -0
- package/package.json +15 -3
- package/dist/__tests__/fixtures.test.d.ts +0 -2
- package/dist/__tests__/fixtures.test.d.ts.map +0 -1
- package/dist/__tests__/fixtures.test.js +0 -37
- package/dist/__tests__/fixtures.test.js.map +0 -1
- package/dist/__tests__/vectors.test.d.ts +0 -2
- package/dist/__tests__/vectors.test.d.ts.map +0 -1
- package/dist/__tests__/vectors.test.js +0 -261
- 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.
|
|
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 +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 +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":"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"}
|