elasticdash-test 0.1.20-alpha-2 → 0.1.20-alpha-5

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 (110) hide show
  1. package/dist/ci/benchmark.d.ts +15 -0
  2. package/dist/ci/benchmark.d.ts.map +1 -0
  3. package/dist/ci/benchmark.js +36 -0
  4. package/dist/ci/benchmark.js.map +1 -0
  5. package/dist/ci/ed-runner.d.ts +32 -0
  6. package/dist/ci/ed-runner.d.ts.map +1 -0
  7. package/dist/ci/ed-runner.js +149 -0
  8. package/dist/ci/ed-runner.js.map +1 -0
  9. package/dist/ci/git-info.d.ts +5 -1
  10. package/dist/ci/git-info.d.ts.map +1 -1
  11. package/dist/ci/git-info.js +63 -13
  12. package/dist/ci/git-info.js.map +1 -1
  13. package/dist/ci/measurement.d.ts +9 -0
  14. package/dist/ci/measurement.d.ts.map +1 -0
  15. package/dist/ci/measurement.js +15 -0
  16. package/dist/ci/measurement.js.map +1 -0
  17. package/dist/ci/replay.d.ts +31 -0
  18. package/dist/ci/replay.d.ts.map +1 -0
  19. package/dist/ci/replay.js +96 -0
  20. package/dist/ci/replay.js.map +1 -0
  21. package/dist/ci/reporters/default.d.ts +8 -0
  22. package/dist/ci/reporters/default.d.ts.map +1 -0
  23. package/dist/ci/reporters/default.js +46 -0
  24. package/dist/ci/reporters/default.js.map +1 -0
  25. package/dist/ci/reporters/index.d.ts +8 -0
  26. package/dist/ci/reporters/index.d.ts.map +1 -0
  27. package/dist/ci/reporters/index.js +14 -0
  28. package/dist/ci/reporters/index.js.map +1 -0
  29. package/dist/ci/reporters/json.d.ts +8 -0
  30. package/dist/ci/reporters/json.d.ts.map +1 -0
  31. package/dist/ci/reporters/json.js +14 -0
  32. package/dist/ci/reporters/json.js.map +1 -0
  33. package/dist/ci/reporters/junit.d.ts +8 -0
  34. package/dist/ci/reporters/junit.d.ts.map +1 -0
  35. package/dist/ci/reporters/junit.js +48 -0
  36. package/dist/ci/reporters/junit.js.map +1 -0
  37. package/dist/ci/runner.d.ts.map +1 -1
  38. package/dist/ci/runner.js +6 -0
  39. package/dist/ci/runner.js.map +1 -1
  40. package/dist/ci/test-discovery.d.ts +5 -0
  41. package/dist/ci/test-discovery.d.ts.map +1 -0
  42. package/dist/ci/test-discovery.js +11 -0
  43. package/dist/ci/test-discovery.js.map +1 -0
  44. package/dist/ci/test-loader.d.ts +19 -0
  45. package/dist/ci/test-loader.d.ts.map +1 -0
  46. package/dist/ci/test-loader.js +146 -0
  47. package/dist/ci/test-loader.js.map +1 -0
  48. package/dist/ci/test-registry.d.ts +24 -0
  49. package/dist/ci/test-registry.d.ts.map +1 -0
  50. package/dist/ci/test-registry.js +11 -0
  51. package/dist/ci/test-registry.js.map +1 -0
  52. package/dist/ci/trace-schema.d.ts +29 -0
  53. package/dist/ci/trace-schema.d.ts.map +1 -0
  54. package/dist/ci/trace-schema.js +56 -0
  55. package/dist/ci/trace-schema.js.map +1 -0
  56. package/dist/ci/trace-writer.d.ts +8 -0
  57. package/dist/ci/trace-writer.d.ts.map +1 -0
  58. package/dist/ci/trace-writer.js +60 -0
  59. package/dist/ci/trace-writer.js.map +1 -0
  60. package/dist/ci/upload-client.d.ts +56 -0
  61. package/dist/ci/upload-client.d.ts.map +1 -0
  62. package/dist/ci/upload-client.js +127 -0
  63. package/dist/ci/upload-client.js.map +1 -0
  64. package/dist/cli.js +72 -0
  65. package/dist/cli.js.map +1 -1
  66. package/dist/index.cjs +852 -111
  67. package/dist/index.d.ts +10 -0
  68. package/dist/index.d.ts.map +1 -1
  69. package/dist/index.js +7 -0
  70. package/dist/index.js.map +1 -1
  71. package/dist/interceptors/telemetry-push.d.ts +2 -0
  72. package/dist/interceptors/telemetry-push.d.ts.map +1 -1
  73. package/dist/interceptors/telemetry-push.js +2 -0
  74. package/dist/interceptors/telemetry-push.js.map +1 -1
  75. package/dist/interceptors/tool.d.ts.map +1 -1
  76. package/dist/interceptors/tool.js +8 -0
  77. package/dist/interceptors/tool.js.map +1 -1
  78. package/dist/interceptors/workflow-ai.d.ts.map +1 -1
  79. package/dist/interceptors/workflow-ai.js +8 -0
  80. package/dist/interceptors/workflow-ai.js.map +1 -1
  81. package/dist/observability.d.ts.map +1 -1
  82. package/dist/observability.js +6 -0
  83. package/dist/observability.js.map +1 -1
  84. package/dist/workflow-runner.d.ts.map +1 -1
  85. package/dist/workflow-runner.js +4 -1
  86. package/dist/workflow-runner.js.map +1 -1
  87. package/package.json +3 -2
  88. package/src/ci/benchmark.ts +57 -0
  89. package/src/ci/ed-runner.ts +203 -0
  90. package/src/ci/git-info.ts +72 -11
  91. package/src/ci/measurement.ts +25 -0
  92. package/src/ci/replay.ts +127 -0
  93. package/src/ci/reporters/default.ts +50 -0
  94. package/src/ci/reporters/index.ts +21 -0
  95. package/src/ci/reporters/json.ts +18 -0
  96. package/src/ci/reporters/junit.ts +61 -0
  97. package/src/ci/runner.ts +6 -0
  98. package/src/ci/test-discovery.ts +16 -0
  99. package/src/ci/test-loader.ts +184 -0
  100. package/src/ci/test-registry.ts +36 -0
  101. package/src/ci/trace-schema.ts +87 -0
  102. package/src/ci/trace-writer.ts +64 -0
  103. package/src/ci/upload-client.ts +203 -0
  104. package/src/cli.ts +85 -0
  105. package/src/index.ts +14 -0
  106. package/src/interceptors/telemetry-push.ts +3 -0
  107. package/src/interceptors/tool.ts +9 -0
  108. package/src/interceptors/workflow-ai.ts +9 -0
  109. package/src/observability.ts +6 -0
  110. package/src/workflow-runner.ts +4 -1
@@ -0,0 +1,15 @@
1
+ import type { TestMeasurement } from './measurement.js';
2
+ import type { TestBenchmarks } from './test-registry.js';
3
+ export interface MetricResult {
4
+ name: 'duration_ms' | 'tokens_total';
5
+ value: number;
6
+ threshold: number;
7
+ passed: boolean;
8
+ }
9
+ export interface BenchmarkResult {
10
+ passed: boolean;
11
+ failure_reason?: string;
12
+ metrics: MetricResult[];
13
+ }
14
+ export declare function compareBenchmarks(measurement: TestMeasurement, benchmarks: TestBenchmarks): BenchmarkResult;
15
+ //# sourceMappingURL=benchmark.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"benchmark.d.ts","sourceRoot":"","sources":["../../src/ci/benchmark.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,kBAAkB,CAAA;AACvD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,oBAAoB,CAAA;AAExD,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,aAAa,GAAG,cAAc,CAAA;IACpC,KAAK,EAAE,MAAM,CAAA;IACb,SAAS,EAAE,MAAM,CAAA;IACjB,MAAM,EAAE,OAAO,CAAA;CAChB;AAED,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,OAAO,CAAA;IACf,cAAc,CAAC,EAAE,MAAM,CAAA;IACvB,OAAO,EAAE,YAAY,EAAE,CAAA;CACxB;AAED,wBAAgB,iBAAiB,CAC/B,WAAW,EAAE,eAAe,EAC5B,UAAU,EAAE,cAAc,GACzB,eAAe,CAqCjB"}
@@ -0,0 +1,36 @@
1
+ export function compareBenchmarks(measurement, benchmarks) {
2
+ const metrics = [];
3
+ let firstFailure;
4
+ if (benchmarks.max_duration_ms !== undefined) {
5
+ const passed = measurement.duration_ms <= benchmarks.max_duration_ms;
6
+ metrics.push({
7
+ name: 'duration_ms',
8
+ value: measurement.duration_ms,
9
+ threshold: benchmarks.max_duration_ms,
10
+ passed,
11
+ });
12
+ if (!passed && !firstFailure) {
13
+ firstFailure = `duration_ms (${measurement.duration_ms}) exceeded max threshold (${benchmarks.max_duration_ms})`;
14
+ }
15
+ }
16
+ if (benchmarks.max_tokens_total !== undefined) {
17
+ const value = measurement.tokens_total ?? 0;
18
+ const passed = value <= benchmarks.max_tokens_total;
19
+ metrics.push({
20
+ name: 'tokens_total',
21
+ value,
22
+ threshold: benchmarks.max_tokens_total,
23
+ passed,
24
+ });
25
+ if (!passed && !firstFailure) {
26
+ firstFailure = `tokens_total (${value}) exceeded max threshold (${benchmarks.max_tokens_total})`;
27
+ }
28
+ }
29
+ const allPassed = metrics.every(m => m.passed);
30
+ return {
31
+ passed: allPassed,
32
+ failure_reason: firstFailure,
33
+ metrics,
34
+ };
35
+ }
36
+ //# sourceMappingURL=benchmark.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"benchmark.js","sourceRoot":"","sources":["../../src/ci/benchmark.ts"],"names":[],"mappings":"AAgBA,MAAM,UAAU,iBAAiB,CAC/B,WAA4B,EAC5B,UAA0B;IAE1B,MAAM,OAAO,GAAmB,EAAE,CAAA;IAClC,IAAI,YAAgC,CAAA;IAEpC,IAAI,UAAU,CAAC,eAAe,KAAK,SAAS,EAAE,CAAC;QAC7C,MAAM,MAAM,GAAG,WAAW,CAAC,WAAW,IAAI,UAAU,CAAC,eAAe,CAAA;QACpE,OAAO,CAAC,IAAI,CAAC;YACX,IAAI,EAAE,aAAa;YACnB,KAAK,EAAE,WAAW,CAAC,WAAW;YAC9B,SAAS,EAAE,UAAU,CAAC,eAAe;YACrC,MAAM;SACP,CAAC,CAAA;QACF,IAAI,CAAC,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;YAC7B,YAAY,GAAG,gBAAgB,WAAW,CAAC,WAAW,6BAA6B,UAAU,CAAC,eAAe,GAAG,CAAA;QAClH,CAAC;IACH,CAAC;IAED,IAAI,UAAU,CAAC,gBAAgB,KAAK,SAAS,EAAE,CAAC;QAC9C,MAAM,KAAK,GAAG,WAAW,CAAC,YAAY,IAAI,CAAC,CAAA;QAC3C,MAAM,MAAM,GAAG,KAAK,IAAI,UAAU,CAAC,gBAAgB,CAAA;QACnD,OAAO,CAAC,IAAI,CAAC;YACX,IAAI,EAAE,cAAc;YACpB,KAAK;YACL,SAAS,EAAE,UAAU,CAAC,gBAAgB;YACtC,MAAM;SACP,CAAC,CAAA;QACF,IAAI,CAAC,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;YAC7B,YAAY,GAAG,iBAAiB,KAAK,6BAA6B,UAAU,CAAC,gBAAgB,GAAG,CAAA;QAClG,CAAC;IACH,CAAC;IAED,MAAM,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAA;IAC9C,OAAO;QACL,MAAM,EAAE,SAAS;QACjB,cAAc,EAAE,YAAY;QAC5B,OAAO;KACR,CAAA;AACH,CAAC"}
@@ -0,0 +1,32 @@
1
+ import type { TestMeasurement } from './measurement.js';
2
+ import type { BenchmarkResult } from './benchmark.js';
3
+ export interface EdTestRunOptions {
4
+ cwd?: string;
5
+ filter?: string;
6
+ failFast?: boolean;
7
+ noUpload?: boolean;
8
+ reporter?: 'default' | 'json' | 'junit';
9
+ }
10
+ export interface EdTestResult {
11
+ testId: string;
12
+ testName: string;
13
+ status: 'pass' | 'fail';
14
+ failureReason?: string;
15
+ measurement?: TestMeasurement;
16
+ benchmarkResult?: BenchmarkResult;
17
+ traceRef?: string;
18
+ target?: {
19
+ type: string;
20
+ step_id: string;
21
+ };
22
+ durationMs: number;
23
+ }
24
+ export interface EdTestRunResult {
25
+ runId: string;
26
+ startedAt: string;
27
+ finishedAt: string;
28
+ results: EdTestResult[];
29
+ sdkVersion: string;
30
+ }
31
+ export declare function runEdTests(options?: EdTestRunOptions): Promise<EdTestRunResult>;
32
+ //# sourceMappingURL=ed-runner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ed-runner.d.ts","sourceRoot":"","sources":["../../src/ci/ed-runner.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,kBAAkB,CAAA;AACvD,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAA;AAQrD,MAAM,WAAW,gBAAgB;IAC/B,GAAG,CAAC,EAAE,MAAM,CAAA;IACZ,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,QAAQ,CAAC,EAAE,OAAO,CAAA;IAClB,QAAQ,CAAC,EAAE,SAAS,GAAG,MAAM,GAAG,OAAO,CAAA;CACxC;AAED,MAAM,WAAW,YAAY;IAC3B,MAAM,EAAE,MAAM,CAAA;IACd,QAAQ,EAAE,MAAM,CAAA;IAChB,MAAM,EAAE,MAAM,GAAG,MAAM,CAAA;IACvB,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,WAAW,CAAC,EAAE,eAAe,CAAA;IAC7B,eAAe,CAAC,EAAE,eAAe,CAAA;IACjC,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,MAAM,CAAC,EAAE;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,MAAM,CAAA;KAAE,CAAA;IAC1C,UAAU,EAAE,MAAM,CAAA;CACnB;AAED,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,MAAM,CAAA;IACb,SAAS,EAAE,MAAM,CAAA;IACjB,UAAU,EAAE,MAAM,CAAA;IAClB,OAAO,EAAE,YAAY,EAAE,CAAA;IACvB,UAAU,EAAE,MAAM,CAAA;CACnB;AAID,wBAAsB,UAAU,CAAC,OAAO,CAAC,EAAE,gBAAgB,GAAG,OAAO,CAAC,eAAe,CAAC,CA4CrF"}
@@ -0,0 +1,149 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { createRequire } from 'node:module';
3
+ import { loadTests } from './test-loader.js';
4
+ import { createReplayContext, installReplay, uninstallReplay, ReplayMissError } from './replay.js';
5
+ import { collectMeasurement } from './measurement.js';
6
+ import { compareBenchmarks } from './benchmark.js';
7
+ const require = createRequire(import.meta.url);
8
+ const { version: SDK_VERSION } = require('../../package.json');
9
+ // ─── Runner ─────────────────────────────────────────────────
10
+ export async function runEdTests(options) {
11
+ const cwd = options?.cwd ?? process.cwd();
12
+ const runId = randomUUID();
13
+ const startedAt = new Date().toISOString();
14
+ const results = [];
15
+ const { tests, errors } = await loadTests({ cwd });
16
+ // Report validation errors as failed tests
17
+ for (const err of errors) {
18
+ results.push({
19
+ testId: err.testName ?? 'unknown',
20
+ testName: err.testName ?? 'unknown',
21
+ status: 'fail',
22
+ failureReason: `validation error: ${err.message}`,
23
+ durationMs: 0,
24
+ });
25
+ }
26
+ // Filter tests if pattern provided
27
+ let testsToRun = tests;
28
+ if (options?.filter) {
29
+ const pattern = options.filter;
30
+ testsToRun = tests.filter(t => matchGlob(t.name, pattern));
31
+ }
32
+ for (const test of testsToRun) {
33
+ const result = await runSingleTest(test);
34
+ results.push(result);
35
+ if (options?.failFast && result.status === 'fail') {
36
+ break;
37
+ }
38
+ }
39
+ const finishedAt = new Date().toISOString();
40
+ return {
41
+ runId,
42
+ startedAt,
43
+ finishedAt,
44
+ results,
45
+ sdkVersion: SDK_VERSION,
46
+ };
47
+ }
48
+ // ─── Single test execution ──────────────────────────────────
49
+ async function runSingleTest(test) {
50
+ const startMs = Date.now();
51
+ const base = {
52
+ testId: test.name,
53
+ testName: test.name,
54
+ traceRef: test.trace,
55
+ target: { type: test.target.type, step_id: test.target.step_id },
56
+ };
57
+ // Check run function exists
58
+ if (!test.run || typeof test.run !== 'function') {
59
+ return {
60
+ ...base,
61
+ testId: test.name,
62
+ testName: test.name,
63
+ status: 'fail',
64
+ failureReason: 'test has no run function',
65
+ durationMs: Date.now() - startMs,
66
+ };
67
+ }
68
+ const replayCtx = createReplayContext(test.traceData, test.target.step_id);
69
+ installReplay(replayCtx);
70
+ try {
71
+ const timeoutMs = test.timeout_ms ?? 60000;
72
+ await Promise.race([
73
+ test.run(),
74
+ new Promise((_, reject) => setTimeout(() => reject(new TimeoutError(timeoutMs)), timeoutMs)),
75
+ ]);
76
+ // Collect measurement from the target step
77
+ const measurement = collectMeasurement(replayCtx);
78
+ if (!measurement) {
79
+ return {
80
+ ...base,
81
+ testId: test.name,
82
+ testName: test.name,
83
+ status: 'fail',
84
+ failureReason: `target step "${test.target.step_id}" was not replayed during execution`,
85
+ durationMs: Date.now() - startMs,
86
+ };
87
+ }
88
+ // Compare against benchmarks
89
+ const benchmarkResult = compareBenchmarks(measurement, test.benchmarks);
90
+ return {
91
+ ...base,
92
+ testId: test.name,
93
+ testName: test.name,
94
+ status: benchmarkResult.passed ? 'pass' : 'fail',
95
+ failureReason: benchmarkResult.failure_reason,
96
+ measurement,
97
+ benchmarkResult,
98
+ durationMs: Date.now() - startMs,
99
+ };
100
+ }
101
+ catch (err) {
102
+ if (err instanceof ReplayMissError) {
103
+ return {
104
+ ...base,
105
+ testId: test.name,
106
+ testName: test.name,
107
+ status: 'fail',
108
+ failureReason: `replay miss: ${err.callType}::${err.callName}`,
109
+ durationMs: Date.now() - startMs,
110
+ };
111
+ }
112
+ if (err instanceof TimeoutError) {
113
+ return {
114
+ ...base,
115
+ testId: test.name,
116
+ testName: test.name,
117
+ status: 'fail',
118
+ failureReason: `test timed out after ${err.timeoutMs}ms`,
119
+ durationMs: Date.now() - startMs,
120
+ };
121
+ }
122
+ return {
123
+ ...base,
124
+ testId: test.name,
125
+ testName: test.name,
126
+ status: 'fail',
127
+ failureReason: `execution error: ${err instanceof Error ? err.message : String(err)}`,
128
+ durationMs: Date.now() - startMs,
129
+ };
130
+ }
131
+ finally {
132
+ uninstallReplay();
133
+ }
134
+ }
135
+ // ─── Helpers ────────────────────────────────────────────────
136
+ class TimeoutError extends Error {
137
+ timeoutMs;
138
+ constructor(timeoutMs) {
139
+ super(`Test timed out after ${timeoutMs}ms`);
140
+ this.timeoutMs = timeoutMs;
141
+ this.name = 'TimeoutError';
142
+ }
143
+ }
144
+ function matchGlob(name, pattern) {
145
+ // Simple glob: convert * to .* and ? to .
146
+ const regex = new RegExp('^' + pattern.replace(/[.+^${}()|[\]\\]/g, '\\$&').replace(/\*/g, '.*').replace(/\?/g, '.') + '$');
147
+ return regex.test(name);
148
+ }
149
+ //# sourceMappingURL=ed-runner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ed-runner.js","sourceRoot":"","sources":["../../src/ci/ed-runner.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAA;AACxC,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAA;AAC3C,OAAO,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAA;AAC5C,OAAO,EAAE,mBAAmB,EAAE,aAAa,EAAE,eAAe,EAAE,eAAe,EAAE,MAAM,aAAa,CAAA;AAClG,OAAO,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAA;AACrD,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAA;AAKlD,MAAM,OAAO,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA;AAC9C,MAAM,EAAE,OAAO,EAAE,WAAW,EAAE,GAAG,OAAO,CAAC,oBAAoB,CAAwB,CAAA;AAgCrF,+DAA+D;AAE/D,MAAM,CAAC,KAAK,UAAU,UAAU,CAAC,OAA0B;IACzD,MAAM,GAAG,GAAG,OAAO,EAAE,GAAG,IAAI,OAAO,CAAC,GAAG,EAAE,CAAA;IACzC,MAAM,KAAK,GAAG,UAAU,EAAE,CAAA;IAC1B,MAAM,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAA;IAC1C,MAAM,OAAO,GAAmB,EAAE,CAAA;IAElC,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,EAAE,GAAG,EAAE,CAAC,CAAA;IAElD,2CAA2C;IAC3C,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;QACzB,OAAO,CAAC,IAAI,CAAC;YACX,MAAM,EAAE,GAAG,CAAC,QAAQ,IAAI,SAAS;YACjC,QAAQ,EAAE,GAAG,CAAC,QAAQ,IAAI,SAAS;YACnC,MAAM,EAAE,MAAM;YACd,aAAa,EAAE,qBAAqB,GAAG,CAAC,OAAO,EAAE;YACjD,UAAU,EAAE,CAAC;SACd,CAAC,CAAA;IACJ,CAAC;IAED,mCAAmC;IACnC,IAAI,UAAU,GAAoB,KAAK,CAAA;IACvC,IAAI,OAAO,EAAE,MAAM,EAAE,CAAC;QACpB,MAAM,OAAO,GAAG,OAAO,CAAC,MAAM,CAAA;QAC9B,UAAU,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAA;IAC5D,CAAC;IAED,KAAK,MAAM,IAAI,IAAI,UAAU,EAAE,CAAC;QAC9B,MAAM,MAAM,GAAG,MAAM,aAAa,CAAC,IAAI,CAAC,CAAA;QACxC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;QAEpB,IAAI,OAAO,EAAE,QAAQ,IAAI,MAAM,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;YAClD,MAAK;QACP,CAAC;IACH,CAAC;IAED,MAAM,UAAU,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAA;IAE3C,OAAO;QACL,KAAK;QACL,SAAS;QACT,UAAU;QACV,OAAO;QACP,UAAU,EAAE,WAAW;KACxB,CAAA;AACH,CAAC;AAED,+DAA+D;AAE/D,KAAK,UAAU,aAAa,CAAC,IAAmB;IAC9C,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAA;IAC1B,MAAM,IAAI,GAA0B;QAClC,MAAM,EAAE,IAAI,CAAC,IAAI;QACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;QACnB,QAAQ,EAAE,IAAI,CAAC,KAAK;QACpB,MAAM,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE;KACjE,CAAA;IAED,4BAA4B;IAC5B,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,OAAO,IAAI,CAAC,GAAG,KAAK,UAAU,EAAE,CAAC;QAChD,OAAO;YACL,GAAG,IAAI;YACP,MAAM,EAAE,IAAI,CAAC,IAAI;YACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;YACnB,MAAM,EAAE,MAAM;YACd,aAAa,EAAE,0BAA0B;YACzC,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;SACjC,CAAA;IACH,CAAC;IAED,MAAM,SAAS,GAAG,mBAAmB,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAA;IAC1E,aAAa,CAAC,SAAS,CAAC,CAAA;IAExB,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,IAAI,KAAK,CAAA;QAE1C,MAAM,OAAO,CAAC,IAAI,CAAC;YACjB,IAAI,CAAC,GAAG,EAAE;YACV,IAAI,OAAO,CAAQ,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,CAC/B,UAAU,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,YAAY,CAAC,SAAS,CAAC,CAAC,EAAE,SAAS,CAAC,CACjE;SACF,CAAC,CAAA;QAEF,2CAA2C;QAC3C,MAAM,WAAW,GAAG,kBAAkB,CAAC,SAAS,CAAC,CAAA;QACjD,IAAI,CAAC,WAAW,EAAE,CAAC;YACjB,OAAO;gBACL,GAAG,IAAI;gBACP,MAAM,EAAE,IAAI,CAAC,IAAI;gBACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;gBACnB,MAAM,EAAE,MAAM;gBACd,aAAa,EAAE,gBAAgB,IAAI,CAAC,MAAM,CAAC,OAAO,qCAAqC;gBACvF,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;aACjC,CAAA;QACH,CAAC;QAED,6BAA6B;QAC7B,MAAM,eAAe,GAAG,iBAAiB,CAAC,WAAW,EAAE,IAAI,CAAC,UAAU,CAAC,CAAA;QAEvE,OAAO;YACL,GAAG,IAAI;YACP,MAAM,EAAE,IAAI,CAAC,IAAI;YACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;YACnB,MAAM,EAAE,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM;YAChD,aAAa,EAAE,eAAe,CAAC,cAAc;YAC7C,WAAW;YACX,eAAe;YACf,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;SACjC,CAAA;IACH,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,IAAI,GAAG,YAAY,eAAe,EAAE,CAAC;YACnC,OAAO;gBACL,GAAG,IAAI;gBACP,MAAM,EAAE,IAAI,CAAC,IAAI;gBACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;gBACnB,MAAM,EAAE,MAAM;gBACd,aAAa,EAAE,gBAAgB,GAAG,CAAC,QAAQ,KAAK,GAAG,CAAC,QAAQ,EAAE;gBAC9D,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;aACjC,CAAA;QACH,CAAC;QACD,IAAI,GAAG,YAAY,YAAY,EAAE,CAAC;YAChC,OAAO;gBACL,GAAG,IAAI;gBACP,MAAM,EAAE,IAAI,CAAC,IAAI;gBACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;gBACnB,MAAM,EAAE,MAAM;gBACd,aAAa,EAAE,wBAAwB,GAAG,CAAC,SAAS,IAAI;gBACxD,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;aACjC,CAAA;QACH,CAAC;QACD,OAAO;YACL,GAAG,IAAI;YACP,MAAM,EAAE,IAAI,CAAC,IAAI;YACjB,QAAQ,EAAE,IAAI,CAAC,IAAI;YACnB,MAAM,EAAE,MAAM;YACd,aAAa,EAAE,oBAAoB,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE;YACrF,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO;SACjC,CAAA;IACH,CAAC;YAAS,CAAC;QACT,eAAe,EAAE,CAAA;IACnB,CAAC;AACH,CAAC;AAED,+DAA+D;AAE/D,MAAM,YAAa,SAAQ,KAAK;IACX;IAAnB,YAAmB,SAAiB;QAClC,KAAK,CAAC,wBAAwB,SAAS,IAAI,CAAC,CAAA;QAD3B,cAAS,GAAT,SAAS,CAAQ;QAElC,IAAI,CAAC,IAAI,GAAG,cAAc,CAAA;IAC5B,CAAC;CACF;AAED,SAAS,SAAS,CAAC,IAAY,EAAE,OAAe;IAC9C,0CAA0C;IAC1C,MAAM,KAAK,GAAG,IAAI,MAAM,CACtB,GAAG,GAAG,OAAO,CAAC,OAAO,CAAC,mBAAmB,EAAE,MAAM,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,GAAG,GAAG,CAClG,CAAA;IACD,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;AACzB,CAAC"}
@@ -2,12 +2,16 @@ export interface GitInfo {
2
2
  branch?: string;
3
3
  commit?: string;
4
4
  commitMessage?: string;
5
+ baseBranch?: string;
5
6
  prNumber?: number;
6
7
  prUrl?: string;
8
+ repo?: string;
9
+ ciProvider?: string;
10
+ ciRunUrl?: string;
7
11
  }
8
12
  /**
9
13
  * Auto-detect git info from CI environment variables.
10
- * Supports GitHub Actions, GitLab CI, and generic CI.
14
+ * Supports GitHub Actions, GitLab CI, CircleCI, Buildkite, and local git.
11
15
  */
12
16
  export declare function detectGitInfo(): GitInfo;
13
17
  //# sourceMappingURL=git-info.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"git-info.d.ts","sourceRoot":"","sources":["../../src/ci/git-info.ts"],"names":[],"mappings":"AAEA,MAAM,WAAW,OAAO;IACtB,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,KAAK,CAAC,EAAE,MAAM,CAAA;CACf;AAED;;;GAGG;AACH,wBAAgB,aAAa,IAAI,OAAO,CAmDvC"}
1
+ {"version":3,"file":"git-info.d.ts","sourceRoot":"","sources":["../../src/ci/git-info.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,OAAO;IACtB,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB,KAAK,CAAC,EAAE,MAAM,CAAA;IACd,IAAI,CAAC,EAAE,MAAM,CAAA;IACb,UAAU,CAAC,EAAE,MAAM,CAAA;IACnB,QAAQ,CAAC,EAAE,MAAM,CAAA;CAClB;AAED;;;GAGG;AACH,wBAAgB,aAAa,IAAI,OAAO,CAwFvC"}
@@ -1,7 +1,7 @@
1
- // ─── CI Environment Auto-Detection ──────────────────────────
1
+ import { execSync } from 'node:child_process';
2
2
  /**
3
3
  * Auto-detect git info from CI environment variables.
4
- * Supports GitHub Actions, GitLab CI, and generic CI.
4
+ * Supports GitHub Actions, GitLab CI, CircleCI, Buildkite, and local git.
5
5
  */
6
6
  export function detectGitInfo() {
7
7
  const env = process.env;
@@ -10,16 +10,24 @@ export function detectGitInfo() {
10
10
  const prNumber = env.GITHUB_EVENT_NAME === 'pull_request'
11
11
  ? parseInt(env.GITHUB_REF?.match(/refs\/pull\/(\d+)/)?.[1] ?? '', 10) || undefined
12
12
  : undefined;
13
- const repo = env.GITHUB_REPOSITORY; // e.g. "owner/repo"
13
+ const repo = env.GITHUB_REPOSITORY;
14
14
  const prUrl = prNumber && repo
15
15
  ? `https://github.com/${repo}/pull/${prNumber}`
16
16
  : undefined;
17
+ const serverUrl = env.GITHUB_SERVER_URL || 'https://github.com';
18
+ const ciRunUrl = repo && env.GITHUB_RUN_ID
19
+ ? `${serverUrl}/${repo}/actions/runs/${env.GITHUB_RUN_ID}`
20
+ : undefined;
17
21
  return {
18
22
  branch: env.GITHUB_HEAD_REF || env.GITHUB_REF_NAME,
19
23
  commit: env.GITHUB_SHA,
20
24
  commitMessage: env.GITHUB_COMMIT_MESSAGE,
25
+ baseBranch: env.GITHUB_BASE_REF || undefined,
21
26
  prNumber,
22
27
  prUrl,
28
+ repo,
29
+ ciProvider: 'github-actions',
30
+ ciRunUrl,
23
31
  };
24
32
  }
25
33
  // GitLab CI
@@ -31,22 +39,64 @@ export function detectGitInfo() {
31
39
  branch: env.CI_MERGE_REQUEST_SOURCE_BRANCH_NAME || env.CI_COMMIT_BRANCH,
32
40
  commit: env.CI_COMMIT_SHA,
33
41
  commitMessage: env.CI_COMMIT_MESSAGE,
42
+ baseBranch: env.CI_MERGE_REQUEST_TARGET_BRANCH_NAME || undefined,
34
43
  prNumber,
35
44
  prUrl: env.CI_MERGE_REQUEST_PROJECT_URL && prNumber
36
45
  ? `${env.CI_MERGE_REQUEST_PROJECT_URL}/-/merge_requests/${prNumber}`
37
46
  : undefined,
47
+ repo: env.CI_PROJECT_PATH,
48
+ ciProvider: 'gitlab-ci',
49
+ ciRunUrl: env.CI_JOB_URL,
38
50
  };
39
51
  }
40
- // Generic CI (CircleCI, Travis, Bitbucket Pipelines, etc.)
41
- return {
42
- branch: env.CIRCLE_BRANCH || env.TRAVIS_BRANCH || env.BITBUCKET_BRANCH || env.CI_BRANCH,
43
- commit: env.CIRCLE_SHA1 || env.TRAVIS_COMMIT || env.BITBUCKET_COMMIT || env.CI_COMMIT_SHA,
44
- commitMessage: env.CI_COMMIT_MESSAGE,
45
- prNumber: env.CIRCLE_PULL_REQUEST
52
+ // CircleCI
53
+ if (env.CIRCLECI === 'true') {
54
+ const prNumber = env.CIRCLE_PULL_REQUEST
46
55
  ? parseInt(env.CIRCLE_PULL_REQUEST.split('/').pop() ?? '', 10) || undefined
47
- : env.TRAVIS_PULL_REQUEST && env.TRAVIS_PULL_REQUEST !== 'false'
48
- ? parseInt(env.TRAVIS_PULL_REQUEST, 10) || undefined
49
- : undefined,
50
- };
56
+ : undefined;
57
+ return {
58
+ branch: env.CIRCLE_BRANCH,
59
+ commit: env.CIRCLE_SHA1,
60
+ prNumber,
61
+ repo: env.CIRCLE_PROJECT_REPONAME,
62
+ ciProvider: 'circleci',
63
+ ciRunUrl: env.CIRCLE_BUILD_URL,
64
+ };
65
+ }
66
+ // Buildkite
67
+ if (env.BUILDKITE === 'true') {
68
+ const prNumber = env.BUILDKITE_PULL_REQUEST && env.BUILDKITE_PULL_REQUEST !== 'false'
69
+ ? parseInt(env.BUILDKITE_PULL_REQUEST, 10) || undefined
70
+ : undefined;
71
+ return {
72
+ branch: env.BUILDKITE_BRANCH,
73
+ commit: env.BUILDKITE_COMMIT,
74
+ baseBranch: env.BUILDKITE_PULL_REQUEST_BASE_BRANCH || undefined,
75
+ prNumber,
76
+ repo: env.BUILDKITE_REPO,
77
+ ciProvider: 'buildkite',
78
+ ciRunUrl: env.BUILDKITE_BUILD_URL,
79
+ };
80
+ }
81
+ // Local fallback: shell out to git
82
+ return detectLocalGitInfo();
83
+ }
84
+ function detectLocalGitInfo() {
85
+ try {
86
+ const commit = execSync('git rev-parse HEAD', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
87
+ const branch = execSync('git rev-parse --abbrev-ref HEAD', { encoding: 'utf-8', stdio: ['pipe', 'pipe', 'pipe'] }).trim();
88
+ return {
89
+ branch: branch || undefined,
90
+ commit: commit || undefined,
91
+ ciProvider: 'local',
92
+ };
93
+ }
94
+ catch {
95
+ return {
96
+ branch: 'unknown',
97
+ commit: 'unknown',
98
+ ciProvider: 'local',
99
+ };
100
+ }
51
101
  }
52
102
  //# sourceMappingURL=git-info.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"git-info.js","sourceRoot":"","sources":["../../src/ci/git-info.ts"],"names":[],"mappings":"AAAA,+DAA+D;AAU/D;;;GAGG;AACH,MAAM,UAAU,aAAa;IAC3B,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAA;IAEvB,iBAAiB;IACjB,IAAI,GAAG,CAAC,cAAc,KAAK,MAAM,EAAE,CAAC;QAClC,MAAM,QAAQ,GAAG,GAAG,CAAC,iBAAiB,KAAK,cAAc;YACvD,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,CAAC,mBAAmB,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,EAAE,EAAE,CAAC,IAAI,SAAS;YAClF,CAAC,CAAC,SAAS,CAAA;QAEb,MAAM,IAAI,GAAG,GAAG,CAAC,iBAAiB,CAAA,CAAC,oBAAoB;QACvD,MAAM,KAAK,GAAG,QAAQ,IAAI,IAAI;YAC5B,CAAC,CAAC,sBAAsB,IAAI,SAAS,QAAQ,EAAE;YAC/C,CAAC,CAAC,SAAS,CAAA;QAEb,OAAO;YACL,MAAM,EAAE,GAAG,CAAC,eAAe,IAAI,GAAG,CAAC,eAAe;YAClD,MAAM,EAAE,GAAG,CAAC,UAAU;YACtB,aAAa,EAAE,GAAG,CAAC,qBAAqB;YACxC,QAAQ;YACR,KAAK;SACN,CAAA;IACH,CAAC;IAED,YAAY;IACZ,IAAI,GAAG,CAAC,SAAS,KAAK,MAAM,EAAE,CAAC;QAC7B,MAAM,QAAQ,GAAG,GAAG,CAAC,oBAAoB;YACvC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,oBAAoB,EAAE,EAAE,CAAC,IAAI,SAAS;YACrD,CAAC,CAAC,SAAS,CAAA;QAEb,OAAO;YACL,MAAM,EAAE,GAAG,CAAC,mCAAmC,IAAI,GAAG,CAAC,gBAAgB;YACvE,MAAM,EAAE,GAAG,CAAC,aAAa;YACzB,aAAa,EAAE,GAAG,CAAC,iBAAiB;YACpC,QAAQ;YACR,KAAK,EAAE,GAAG,CAAC,4BAA4B,IAAI,QAAQ;gBACjD,CAAC,CAAC,GAAG,GAAG,CAAC,4BAA4B,qBAAqB,QAAQ,EAAE;gBACpE,CAAC,CAAC,SAAS;SACd,CAAA;IACH,CAAC;IAED,2DAA2D;IAC3D,OAAO;QACL,MAAM,EAAE,GAAG,CAAC,aAAa,IAAI,GAAG,CAAC,aAAa,IAAI,GAAG,CAAC,gBAAgB,IAAI,GAAG,CAAC,SAAS;QACvF,MAAM,EAAE,GAAG,CAAC,WAAW,IAAI,GAAG,CAAC,aAAa,IAAI,GAAG,CAAC,gBAAgB,IAAI,GAAG,CAAC,aAAa;QACzF,aAAa,EAAE,GAAG,CAAC,iBAAiB;QACpC,QAAQ,EAAE,GAAG,CAAC,mBAAmB;YAC/B,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,mBAAmB,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC,IAAI,SAAS;YAC3E,CAAC,CAAC,GAAG,CAAC,mBAAmB,IAAI,GAAG,CAAC,mBAAmB,KAAK,OAAO;gBAC9D,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,mBAAmB,EAAE,EAAE,CAAC,IAAI,SAAS;gBACpD,CAAC,CAAC,SAAS;KAChB,CAAA;AACH,CAAC"}
1
+ {"version":3,"file":"git-info.js","sourceRoot":"","sources":["../../src/ci/git-info.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { EdReplayContext } from './replay.js';
2
+ export interface TestMeasurement {
3
+ duration_ms: number;
4
+ tokens_input?: number;
5
+ tokens_output?: number;
6
+ tokens_total?: number;
7
+ }
8
+ export declare function collectMeasurement(ctx: EdReplayContext): TestMeasurement | null;
9
+ //# sourceMappingURL=measurement.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"measurement.d.ts","sourceRoot":"","sources":["../../src/ci/measurement.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,aAAa,CAAA;AAElD,MAAM,WAAW,eAAe;IAC9B,WAAW,EAAE,MAAM,CAAA;IACnB,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,YAAY,CAAC,EAAE,MAAM,CAAA;CACtB;AAED,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,eAAe,GAAG,eAAe,GAAG,IAAI,CAe/E"}
@@ -0,0 +1,15 @@
1
+ export function collectMeasurement(ctx) {
2
+ const m = ctx.targetMeasurement;
3
+ if (!m)
4
+ return null;
5
+ const result = {
6
+ duration_ms: m.duration_ms,
7
+ };
8
+ if (m.tokens) {
9
+ result.tokens_input = m.tokens.input;
10
+ result.tokens_output = m.tokens.output;
11
+ result.tokens_total = m.tokens.total;
12
+ }
13
+ return result;
14
+ }
15
+ //# sourceMappingURL=measurement.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"measurement.js","sourceRoot":"","sources":["../../src/ci/measurement.ts"],"names":[],"mappings":"AASA,MAAM,UAAU,kBAAkB,CAAC,GAAoB;IACrD,MAAM,CAAC,GAAG,GAAG,CAAC,iBAAiB,CAAA;IAC/B,IAAI,CAAC,CAAC;QAAE,OAAO,IAAI,CAAA;IAEnB,MAAM,MAAM,GAAoB;QAC9B,WAAW,EAAE,CAAC,CAAC,WAAW;KAC3B,CAAA;IAED,IAAI,CAAC,CAAC,MAAM,EAAE,CAAC;QACb,MAAM,CAAC,YAAY,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAA;QACpC,MAAM,CAAC,aAAa,GAAG,CAAC,CAAC,MAAM,CAAC,MAAM,CAAA;QACtC,MAAM,CAAC,YAAY,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAA;IACtC,CAAC;IAED,OAAO,MAAM,CAAA;AACf,CAAC"}
@@ -0,0 +1,31 @@
1
+ import type { DiskTrace, DiskTraceStep } from './trace-schema.js';
2
+ export interface ReplayMeasurement {
3
+ duration_ms: number;
4
+ tokens?: {
5
+ input: number;
6
+ output: number;
7
+ total: number;
8
+ } | null;
9
+ }
10
+ export interface EdReplayContext {
11
+ steps: DiskTraceStep[];
12
+ consumed: boolean[];
13
+ targetStepId: string;
14
+ targetMeasurement: ReplayMeasurement | null;
15
+ }
16
+ export declare class ReplayMissError extends Error {
17
+ callType: string;
18
+ callName: string;
19
+ constructor(callType: string, callName: string);
20
+ }
21
+ export declare function getEdReplayContext(): EdReplayContext | undefined;
22
+ export declare function installReplay(ctx: EdReplayContext): void;
23
+ export declare function uninstallReplay(): void;
24
+ export declare function createReplayContext(trace: DiskTrace, targetStepId: string): EdReplayContext;
25
+ export declare function canonicalizeInput(input: unknown): string;
26
+ export declare function computeMatchKey(type: 'tool_call' | 'ai_call', name: string, input: unknown): string;
27
+ export declare function replayCall(ctx: EdReplayContext, type: 'tool_call' | 'ai_call', name: string, input: unknown): {
28
+ output: unknown;
29
+ measurement: ReplayMeasurement | null;
30
+ };
31
+ //# sourceMappingURL=replay.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"replay.d.ts","sourceRoot":"","sources":["../../src/ci/replay.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,SAAS,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAA;AAIjE,MAAM,WAAW,iBAAiB;IAChC,WAAW,EAAE,MAAM,CAAA;IACnB,MAAM,CAAC,EAAE;QAAE,KAAK,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI,CAAA;CACjE;AAED,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,aAAa,EAAE,CAAA;IACtB,QAAQ,EAAE,OAAO,EAAE,CAAA;IACnB,YAAY,EAAE,MAAM,CAAA;IACpB,iBAAiB,EAAE,iBAAiB,GAAG,IAAI,CAAA;CAC5C;AAED,qBAAa,eAAgB,SAAQ,KAAK;IACrB,QAAQ,EAAE,MAAM;IAAS,QAAQ,EAAE,MAAM;gBAAzC,QAAQ,EAAE,MAAM,EAAS,QAAQ,EAAE,MAAM;CAI7D;AAWD,wBAAgB,kBAAkB,IAAI,eAAe,GAAG,SAAS,CAEhE;AAED,wBAAgB,aAAa,CAAC,GAAG,EAAE,eAAe,GAAG,IAAI,CAExD;AAED,wBAAgB,eAAe,IAAI,IAAI,CAEtC;AAID,wBAAgB,mBAAmB,CAAC,KAAK,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,GAAG,eAAe,CAO3F;AAID,wBAAgB,iBAAiB,CAAC,KAAK,EAAE,OAAO,GAAG,MAAM,CAExD;AA0BD,wBAAgB,eAAe,CAAC,IAAI,EAAE,WAAW,GAAG,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,GAAG,MAAM,CAInG;AAID,wBAAgB,UAAU,CACxB,GAAG,EAAE,eAAe,EACpB,IAAI,EAAE,WAAW,GAAG,SAAS,EAC7B,IAAI,EAAE,MAAM,EACZ,KAAK,EAAE,OAAO,GACb;IAAE,MAAM,EAAE,OAAO,CAAC;IAAC,WAAW,EAAE,iBAAiB,GAAG,IAAI,CAAA;CAAE,CA0B5D"}
@@ -0,0 +1,96 @@
1
+ import { createHash } from 'node:crypto';
2
+ import { AsyncLocalStorage } from 'node:async_hooks';
3
+ export class ReplayMissError extends Error {
4
+ callType;
5
+ callName;
6
+ constructor(callType, callName) {
7
+ super(`replay miss: ${callType}::${callName}`);
8
+ this.callType = callType;
9
+ this.callName = callName;
10
+ this.name = 'ReplayMissError';
11
+ }
12
+ }
13
+ // ─── ALS-backed context ─────────────────────────────────────
14
+ const g = globalThis;
15
+ const ED_REPLAY_ALS_KEY = '__elasticdash_ed_replay_als__';
16
+ const edReplayAls = g[ED_REPLAY_ALS_KEY] ??
17
+ new AsyncLocalStorage();
18
+ if (!g[ED_REPLAY_ALS_KEY])
19
+ g[ED_REPLAY_ALS_KEY] = edReplayAls;
20
+ export function getEdReplayContext() {
21
+ return edReplayAls.getStore();
22
+ }
23
+ export function installReplay(ctx) {
24
+ edReplayAls.enterWith(ctx);
25
+ }
26
+ export function uninstallReplay() {
27
+ edReplayAls.enterWith(undefined);
28
+ }
29
+ // ─── Context creation ───────────────────────────────────────
30
+ export function createReplayContext(trace, targetStepId) {
31
+ return {
32
+ steps: trace.steps,
33
+ consumed: new Array(trace.steps.length).fill(false),
34
+ targetStepId,
35
+ targetMeasurement: null,
36
+ };
37
+ }
38
+ // ─── Input canonicalization & match key ─────────────────────
39
+ export function canonicalizeInput(input) {
40
+ return JSON.stringify(sortKeys(input));
41
+ }
42
+ function sortKeys(value) {
43
+ if (value === null || value === undefined)
44
+ return value;
45
+ if (Array.isArray(value))
46
+ return value.map(sortKeys);
47
+ if (typeof value === 'number') {
48
+ // Round floats to 6 decimal places
49
+ if (!Number.isInteger(value)) {
50
+ return Math.round(value * 1e6) / 1e6;
51
+ }
52
+ return value;
53
+ }
54
+ if (typeof value === 'string') {
55
+ // Normalize whitespace
56
+ return value.replace(/\s+/g, ' ').trim();
57
+ }
58
+ if (typeof value === 'object') {
59
+ const sorted = {};
60
+ for (const key of Object.keys(value).sort()) {
61
+ sorted[key] = sortKeys(value[key]);
62
+ }
63
+ return sorted;
64
+ }
65
+ return value;
66
+ }
67
+ export function computeMatchKey(type, name, input) {
68
+ const canonical = canonicalizeInput(input);
69
+ const hash = createHash('sha256').update(canonical).digest('hex').slice(0, 16);
70
+ return `${type}::${name}::${hash}`;
71
+ }
72
+ // ─── Replay call ────────────────────────────────────────────
73
+ export function replayCall(ctx, type, name, input) {
74
+ const runtimeKey = computeMatchKey(type, name, input);
75
+ // Find next unconsumed step with matching key
76
+ for (let i = 0; i < ctx.steps.length; i++) {
77
+ if (ctx.consumed[i])
78
+ continue;
79
+ const step = ctx.steps[i];
80
+ const stepKey = computeMatchKey(step.type, step.name, step.input);
81
+ if (stepKey === runtimeKey) {
82
+ ctx.consumed[i] = true;
83
+ const measurement = {
84
+ duration_ms: step.duration_ms,
85
+ tokens: step.tokens ?? null,
86
+ };
87
+ // Capture measurement if this is the target step
88
+ if (step.step_id === ctx.targetStepId) {
89
+ ctx.targetMeasurement = measurement;
90
+ }
91
+ return { output: step.output, measurement };
92
+ }
93
+ }
94
+ throw new ReplayMissError(type, name);
95
+ }
96
+ //# sourceMappingURL=replay.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"replay.js","sourceRoot":"","sources":["../../src/ci/replay.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAA;AACxC,OAAO,EAAE,iBAAiB,EAAE,MAAM,kBAAkB,CAAA;AAiBpD,MAAM,OAAO,eAAgB,SAAQ,KAAK;IACrB;IAAyB;IAA5C,YAAmB,QAAgB,EAAS,QAAgB;QAC1D,KAAK,CAAC,gBAAgB,QAAQ,KAAK,QAAQ,EAAE,CAAC,CAAA;QAD7B,aAAQ,GAAR,QAAQ,CAAQ;QAAS,aAAQ,GAAR,QAAQ,CAAQ;QAE1D,IAAI,CAAC,IAAI,GAAG,iBAAiB,CAAA;IAC/B,CAAC;CACF;AAED,+DAA+D;AAE/D,MAAM,CAAC,GAAG,UAAqC,CAAA;AAC/C,MAAM,iBAAiB,GAAG,+BAA+B,CAAA;AACzD,MAAM,WAAW,GACd,CAAC,CAAC,iBAAiB,CAAoD;IACxE,IAAI,iBAAiB,EAA+B,CAAA;AACtD,IAAI,CAAC,CAAC,CAAC,iBAAiB,CAAC;IAAE,CAAC,CAAC,iBAAiB,CAAC,GAAG,WAAW,CAAA;AAE7D,MAAM,UAAU,kBAAkB;IAChC,OAAO,WAAW,CAAC,QAAQ,EAAE,CAAA;AAC/B,CAAC;AAED,MAAM,UAAU,aAAa,CAAC,GAAoB;IAChD,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,CAAA;AAC5B,CAAC;AAED,MAAM,UAAU,eAAe;IAC7B,WAAW,CAAC,SAAS,CAAC,SAAS,CAAC,CAAA;AAClC,CAAC;AAED,+DAA+D;AAE/D,MAAM,UAAU,mBAAmB,CAAC,KAAgB,EAAE,YAAoB;IACxE,OAAO;QACL,KAAK,EAAE,KAAK,CAAC,KAAK;QAClB,QAAQ,EAAE,IAAI,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC;QACnD,YAAY;QACZ,iBAAiB,EAAE,IAAI;KACxB,CAAA;AACH,CAAC;AAED,+DAA+D;AAE/D,MAAM,UAAU,iBAAiB,CAAC,KAAc;IAC9C,OAAO,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAA;AACxC,CAAC;AAED,SAAS,QAAQ,CAAC,KAAc;IAC9B,IAAI,KAAK,KAAK,IAAI,IAAI,KAAK,KAAK,SAAS;QAAE,OAAO,KAAK,CAAA;IACvD,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAAE,OAAO,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAA;IACpD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,mCAAmC;QACnC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC;YAC7B,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,CAAC,GAAG,GAAG,CAAA;QACtC,CAAC;QACD,OAAO,KAAK,CAAA;IACd,CAAC;IACD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,uBAAuB;QACvB,OAAO,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,IAAI,EAAE,CAAA;IAC1C,CAAC;IACD,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,MAAM,MAAM,GAA4B,EAAE,CAAA;QAC1C,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,KAAgC,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC;YACvE,MAAM,CAAC,GAAG,CAAC,GAAG,QAAQ,CAAE,KAAiC,CAAC,GAAG,CAAC,CAAC,CAAA;QACjE,CAAC;QACD,OAAO,MAAM,CAAA;IACf,CAAC;IACD,OAAO,KAAK,CAAA;AACd,CAAC;AAED,MAAM,UAAU,eAAe,CAAC,IAA6B,EAAE,IAAY,EAAE,KAAc;IACzF,MAAM,SAAS,GAAG,iBAAiB,CAAC,KAAK,CAAC,CAAA;IAC1C,MAAM,IAAI,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAA;IAC9E,OAAO,GAAG,IAAI,KAAK,IAAI,KAAK,IAAI,EAAE,CAAA;AACpC,CAAC;AAED,+DAA+D;AAE/D,MAAM,UAAU,UAAU,CACxB,GAAoB,EACpB,IAA6B,EAC7B,IAAY,EACZ,KAAc;IAEd,MAAM,UAAU,GAAG,eAAe,CAAC,IAAI,EAAE,IAAI,EAAE,KAAK,CAAC,CAAA;IAErD,8CAA8C;IAC9C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QAC1C,IAAI,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC;YAAE,SAAQ;QAC7B,MAAM,IAAI,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAA;QACzB,MAAM,OAAO,GAAG,eAAe,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAA;QACjE,IAAI,OAAO,KAAK,UAAU,EAAE,CAAC;YAC3B,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,IAAI,CAAA;YAEtB,MAAM,WAAW,GAAsB;gBACrC,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,MAAM,EAAE,IAAI,CAAC,MAAM,IAAI,IAAI;aAC5B,CAAA;YAED,iDAAiD;YACjD,IAAI,IAAI,CAAC,OAAO,KAAK,GAAG,CAAC,YAAY,EAAE,CAAC;gBACtC,GAAG,CAAC,iBAAiB,GAAG,WAAW,CAAA;YACrC,CAAC;YAED,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,WAAW,EAAE,CAAA;QAC7C,CAAC;IACH,CAAC;IAED,MAAM,IAAI,eAAe,CAAC,IAAI,EAAE,IAAI,CAAC,CAAA;AACvC,CAAC"}
@@ -0,0 +1,8 @@
1
+ import type { Reporter } from './index.js';
2
+ import type { EdTestResult, EdTestRunResult } from '../ed-runner.js';
3
+ export declare class DefaultReporter implements Reporter {
4
+ onTestStart(name: string): void;
5
+ onTestResult(result: EdTestResult): void;
6
+ onRunComplete(runResult: EdTestRunResult, uploadUrl?: string): void;
7
+ }
8
+ //# sourceMappingURL=default.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"default.d.ts","sourceRoot":"","sources":["../../../src/ci/reporters/default.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAA;AAC1C,OAAO,KAAK,EAAE,YAAY,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAA;AAEpE,qBAAa,eAAgB,YAAW,QAAQ;IAC9C,WAAW,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;IAI/B,YAAY,CAAC,MAAM,EAAE,YAAY,GAAG,IAAI;IAuBxC,aAAa,CAAC,SAAS,EAAE,eAAe,EAAE,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI;CAiBpE"}
@@ -0,0 +1,46 @@
1
+ import chalk from 'chalk';
2
+ export class DefaultReporter {
3
+ onTestStart(name) {
4
+ process.stdout.write(chalk.gray(` ${name} ... `));
5
+ }
6
+ onTestResult(result) {
7
+ if (result.status === 'pass') {
8
+ console.log(chalk.green('PASS'));
9
+ if (result.benchmarkResult) {
10
+ for (const m of result.benchmarkResult.metrics) {
11
+ console.log(chalk.gray(` ${m.name}: ${m.value} (threshold: ${m.threshold}) `) + chalk.green('✓'));
12
+ }
13
+ }
14
+ }
15
+ else {
16
+ console.log(chalk.red('FAIL'));
17
+ if (result.benchmarkResult) {
18
+ for (const m of result.benchmarkResult.metrics) {
19
+ const icon = m.passed ? chalk.green('✓') : chalk.red('✗');
20
+ console.log(chalk.gray(` ${m.name}: ${m.value} (threshold: ${m.threshold}) `) + icon);
21
+ }
22
+ }
23
+ if (result.failureReason) {
24
+ console.log(chalk.red(` → ${result.failureReason}`));
25
+ }
26
+ }
27
+ console.log();
28
+ }
29
+ onRunComplete(runResult, uploadUrl) {
30
+ const passed = runResult.results.filter(r => r.status === 'pass').length;
31
+ const failed = runResult.results.filter(r => r.status === 'fail').length;
32
+ const total = runResult.results.length;
33
+ console.log(chalk.white.bold('─'.repeat(45)));
34
+ const parts = [];
35
+ if (passed > 0)
36
+ parts.push(chalk.green(`${passed} passed`));
37
+ if (failed > 0)
38
+ parts.push(chalk.red(`${failed} failed`));
39
+ console.log(`${parts.join(', ')}, ${total} total`);
40
+ if (uploadUrl) {
41
+ console.log(chalk.gray(`Uploaded to ${uploadUrl}`));
42
+ }
43
+ console.log();
44
+ }
45
+ }
46
+ //# sourceMappingURL=default.js.map