@dewtech/dare-cli 3.8.2 → 3.10.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 (256) hide show
  1. package/README.md +19 -0
  2. package/dist/__tests__/drift-regression.test.d.ts +2 -0
  3. package/dist/__tests__/drift-regression.test.d.ts.map +1 -0
  4. package/dist/__tests__/drift-regression.test.js +118 -0
  5. package/dist/__tests__/drift-regression.test.js.map +1 -0
  6. package/dist/__tests__/ide-command-parity.test.js +1 -0
  7. package/dist/__tests__/ide-command-parity.test.js.map +1 -1
  8. package/dist/__tests__/project-generator.test.js +10 -1
  9. package/dist/__tests__/project-generator.test.js.map +1 -1
  10. package/dist/__tests__/secure-executor-regression.test.d.ts +2 -0
  11. package/dist/__tests__/secure-executor-regression.test.d.ts.map +1 -0
  12. package/dist/__tests__/secure-executor-regression.test.js +294 -0
  13. package/dist/__tests__/secure-executor-regression.test.js.map +1 -0
  14. package/dist/__tests__/semantic-regression.test.d.ts +2 -0
  15. package/dist/__tests__/semantic-regression.test.d.ts.map +1 -0
  16. package/dist/__tests__/semantic-regression.test.js +248 -0
  17. package/dist/__tests__/semantic-regression.test.js.map +1 -0
  18. package/dist/agent/__tests__/budget.test.d.ts +2 -0
  19. package/dist/agent/__tests__/budget.test.d.ts.map +1 -0
  20. package/dist/agent/__tests__/budget.test.js +36 -0
  21. package/dist/agent/__tests__/budget.test.js.map +1 -0
  22. package/dist/agent/__tests__/claude-driver.test.d.ts +2 -0
  23. package/dist/agent/__tests__/claude-driver.test.d.ts.map +1 -0
  24. package/dist/agent/__tests__/claude-driver.test.js +88 -0
  25. package/dist/agent/__tests__/claude-driver.test.js.map +1 -0
  26. package/dist/agent/__tests__/cost-telemetry.test.d.ts +2 -0
  27. package/dist/agent/__tests__/cost-telemetry.test.d.ts.map +1 -0
  28. package/dist/agent/__tests__/cost-telemetry.test.js +61 -0
  29. package/dist/agent/__tests__/cost-telemetry.test.js.map +1 -0
  30. package/dist/agent/__tests__/driver.types.test.d.ts +2 -0
  31. package/dist/agent/__tests__/driver.types.test.d.ts.map +1 -0
  32. package/dist/agent/__tests__/driver.types.test.js +33 -0
  33. package/dist/agent/__tests__/driver.types.test.js.map +1 -0
  34. package/dist/agent/__tests__/mock-driver.test.d.ts +2 -0
  35. package/dist/agent/__tests__/mock-driver.test.d.ts.map +1 -0
  36. package/dist/agent/__tests__/mock-driver.test.js +100 -0
  37. package/dist/agent/__tests__/mock-driver.test.js.map +1 -0
  38. package/dist/agent/__tests__/no-llm-in-core.test.d.ts +2 -0
  39. package/dist/agent/__tests__/no-llm-in-core.test.d.ts.map +1 -0
  40. package/dist/agent/__tests__/no-llm-in-core.test.js +46 -0
  41. package/dist/agent/__tests__/no-llm-in-core.test.js.map +1 -0
  42. package/dist/agent/budget.d.ts +15 -0
  43. package/dist/agent/budget.d.ts.map +1 -0
  44. package/dist/agent/budget.js +23 -0
  45. package/dist/agent/budget.js.map +1 -0
  46. package/dist/agent/driver.d.ts +46 -0
  47. package/dist/agent/driver.d.ts.map +1 -0
  48. package/dist/agent/driver.js +2 -0
  49. package/dist/agent/driver.js.map +1 -0
  50. package/dist/agent/drivers/claude.d.ts +43 -0
  51. package/dist/agent/drivers/claude.d.ts.map +1 -0
  52. package/dist/agent/drivers/claude.js +134 -0
  53. package/dist/agent/drivers/claude.js.map +1 -0
  54. package/dist/agent/drivers/mock.d.ts +3 -0
  55. package/dist/agent/drivers/mock.d.ts.map +1 -0
  56. package/dist/agent/drivers/mock.js +56 -0
  57. package/dist/agent/drivers/mock.js.map +1 -0
  58. package/dist/agent/drivers/noop.d.ts +3 -0
  59. package/dist/agent/drivers/noop.d.ts.map +1 -0
  60. package/dist/agent/drivers/noop.js +13 -0
  61. package/dist/agent/drivers/noop.js.map +1 -0
  62. package/dist/agent/telemetry.d.ts +4 -0
  63. package/dist/agent/telemetry.d.ts.map +1 -0
  64. package/dist/agent/telemetry.js +25 -0
  65. package/dist/agent/telemetry.js.map +1 -0
  66. package/dist/bin/dare.js +2 -0
  67. package/dist/bin/dare.js.map +1 -1
  68. package/dist/commands/__tests__/execute-agent.test.d.ts +2 -0
  69. package/dist/commands/__tests__/execute-agent.test.d.ts.map +1 -0
  70. package/dist/commands/__tests__/execute-agent.test.js +191 -0
  71. package/dist/commands/__tests__/execute-agent.test.js.map +1 -0
  72. package/dist/commands/__tests__/execute.telemetry.spec.js +3 -0
  73. package/dist/commands/__tests__/execute.telemetry.spec.js.map +1 -1
  74. package/dist/commands/__tests__/graph-drift.test.d.ts +2 -0
  75. package/dist/commands/__tests__/graph-drift.test.d.ts.map +1 -0
  76. package/dist/commands/__tests__/graph-drift.test.js +169 -0
  77. package/dist/commands/__tests__/graph-drift.test.js.map +1 -0
  78. package/dist/commands/__tests__/require-approval.test.d.ts +2 -0
  79. package/dist/commands/__tests__/require-approval.test.d.ts.map +1 -0
  80. package/dist/commands/__tests__/require-approval.test.js +158 -0
  81. package/dist/commands/__tests__/require-approval.test.js.map +1 -0
  82. package/dist/commands/execute.d.ts +68 -0
  83. package/dist/commands/execute.d.ts.map +1 -1
  84. package/dist/commands/execute.js +567 -2
  85. package/dist/commands/execute.js.map +1 -1
  86. package/dist/commands/graph.d.ts.map +1 -1
  87. package/dist/commands/graph.js +204 -1
  88. package/dist/commands/graph.js.map +1 -1
  89. package/dist/commands/guard.d.ts +13 -0
  90. package/dist/commands/guard.d.ts.map +1 -0
  91. package/dist/commands/guard.js +338 -0
  92. package/dist/commands/guard.js.map +1 -0
  93. package/dist/dag-runner/graph-ingest.d.ts.map +1 -1
  94. package/dist/dag-runner/graph-ingest.js +13 -7
  95. package/dist/dag-runner/graph-ingest.js.map +1 -1
  96. package/dist/graphrag/__tests__/drift-config.test.d.ts +2 -0
  97. package/dist/graphrag/__tests__/drift-config.test.d.ts.map +1 -0
  98. package/dist/graphrag/__tests__/drift-config.test.js +33 -0
  99. package/dist/graphrag/__tests__/drift-config.test.js.map +1 -0
  100. package/dist/graphrag/__tests__/drift.test.d.ts +2 -0
  101. package/dist/graphrag/__tests__/drift.test.d.ts.map +1 -0
  102. package/dist/graphrag/__tests__/drift.test.js +152 -0
  103. package/dist/graphrag/__tests__/drift.test.js.map +1 -0
  104. package/dist/graphrag/__tests__/hybrid-integration.test.d.ts +2 -0
  105. package/dist/graphrag/__tests__/hybrid-integration.test.d.ts.map +1 -0
  106. package/dist/graphrag/__tests__/hybrid-integration.test.js +158 -0
  107. package/dist/graphrag/__tests__/hybrid-integration.test.js.map +1 -0
  108. package/dist/graphrag/__tests__/hybrid.test.d.ts +2 -0
  109. package/dist/graphrag/__tests__/hybrid.test.d.ts.map +1 -0
  110. package/dist/graphrag/__tests__/hybrid.test.js +71 -0
  111. package/dist/graphrag/__tests__/hybrid.test.js.map +1 -0
  112. package/dist/graphrag/__tests__/incremental-index.test.d.ts +2 -0
  113. package/dist/graphrag/__tests__/incremental-index.test.d.ts.map +1 -0
  114. package/dist/graphrag/__tests__/incremental-index.test.js +135 -0
  115. package/dist/graphrag/__tests__/incremental-index.test.js.map +1 -0
  116. package/dist/graphrag/__tests__/no-heavy-dep-in-core.test.d.ts +2 -0
  117. package/dist/graphrag/__tests__/no-heavy-dep-in-core.test.d.ts.map +1 -0
  118. package/dist/graphrag/__tests__/no-heavy-dep-in-core.test.js +48 -0
  119. package/dist/graphrag/__tests__/no-heavy-dep-in-core.test.js.map +1 -0
  120. package/dist/graphrag/__tests__/requirement-hash.test.d.ts +2 -0
  121. package/dist/graphrag/__tests__/requirement-hash.test.d.ts.map +1 -0
  122. package/dist/graphrag/__tests__/requirement-hash.test.js +74 -0
  123. package/dist/graphrag/__tests__/requirement-hash.test.js.map +1 -0
  124. package/dist/graphrag/__tests__/semantic-config.test.d.ts +2 -0
  125. package/dist/graphrag/__tests__/semantic-config.test.d.ts.map +1 -0
  126. package/dist/graphrag/__tests__/semantic-config.test.js +39 -0
  127. package/dist/graphrag/__tests__/semantic-config.test.js.map +1 -0
  128. package/dist/graphrag/__tests__/vector-persistence.test.d.ts +2 -0
  129. package/dist/graphrag/__tests__/vector-persistence.test.d.ts.map +1 -0
  130. package/dist/graphrag/__tests__/vector-persistence.test.js +112 -0
  131. package/dist/graphrag/__tests__/vector-persistence.test.js.map +1 -0
  132. package/dist/graphrag/__tests__/vector-search.test.d.ts +2 -0
  133. package/dist/graphrag/__tests__/vector-search.test.d.ts.map +1 -0
  134. package/dist/graphrag/__tests__/vector-search.test.js +36 -0
  135. package/dist/graphrag/__tests__/vector-search.test.js.map +1 -0
  136. package/dist/graphrag/drift.d.ts +22 -0
  137. package/dist/graphrag/drift.d.ts.map +1 -0
  138. package/dist/graphrag/drift.js +204 -0
  139. package/dist/graphrag/drift.js.map +1 -0
  140. package/dist/graphrag/embeddings.d.ts +15 -0
  141. package/dist/graphrag/embeddings.d.ts.map +1 -0
  142. package/dist/graphrag/embeddings.js +107 -0
  143. package/dist/graphrag/embeddings.js.map +1 -0
  144. package/dist/graphrag/graph-rag.d.ts +17 -1
  145. package/dist/graphrag/graph-rag.d.ts.map +1 -1
  146. package/dist/graphrag/graph-rag.js +170 -2
  147. package/dist/graphrag/graph-rag.js.map +1 -1
  148. package/dist/graphrag/hybrid.d.ts +8 -0
  149. package/dist/graphrag/hybrid.d.ts.map +1 -0
  150. package/dist/graphrag/hybrid.js +203 -0
  151. package/dist/graphrag/hybrid.js.map +1 -0
  152. package/dist/graphrag/incremental-index.d.ts +18 -0
  153. package/dist/graphrag/incremental-index.d.ts.map +1 -0
  154. package/dist/graphrag/incremental-index.js +159 -0
  155. package/dist/graphrag/incremental-index.js.map +1 -0
  156. package/dist/graphrag/index.d.ts +1 -1
  157. package/dist/graphrag/index.d.ts.map +1 -1
  158. package/dist/graphrag/json-graph.d.ts +2 -1
  159. package/dist/graphrag/json-graph.d.ts.map +1 -1
  160. package/dist/graphrag/json-graph.js +14 -0
  161. package/dist/graphrag/json-graph.js.map +1 -1
  162. package/dist/graphrag/knowledge-graph.d.ts +5 -0
  163. package/dist/graphrag/knowledge-graph.d.ts.map +1 -1
  164. package/dist/graphrag/neo4j-graph.d.ts +2 -1
  165. package/dist/graphrag/neo4j-graph.d.ts.map +1 -1
  166. package/dist/graphrag/neo4j-graph.js +40 -5
  167. package/dist/graphrag/neo4j-graph.js.map +1 -1
  168. package/dist/graphrag/requirement-ingest.d.ts +4 -1
  169. package/dist/graphrag/requirement-ingest.d.ts.map +1 -1
  170. package/dist/graphrag/requirement-ingest.js +18 -8
  171. package/dist/graphrag/requirement-ingest.js.map +1 -1
  172. package/dist/graphrag/types.d.ts +1 -0
  173. package/dist/graphrag/types.d.ts.map +1 -1
  174. package/dist/graphrag/vector-search.d.ts +10 -0
  175. package/dist/graphrag/vector-search.d.ts.map +1 -0
  176. package/dist/graphrag/vector-search.js +38 -0
  177. package/dist/graphrag/vector-search.js.map +1 -0
  178. package/dist/guard/__tests__/boundary.test.d.ts +2 -0
  179. package/dist/guard/__tests__/boundary.test.d.ts.map +1 -0
  180. package/dist/guard/__tests__/boundary.test.js +37 -0
  181. package/dist/guard/__tests__/boundary.test.js.map +1 -0
  182. package/dist/guard/__tests__/guard-command.test.d.ts +2 -0
  183. package/dist/guard/__tests__/guard-command.test.d.ts.map +1 -0
  184. package/dist/guard/__tests__/guard-command.test.js +88 -0
  185. package/dist/guard/__tests__/guard-command.test.js.map +1 -0
  186. package/dist/guard/__tests__/guard-config.test.d.ts +2 -0
  187. package/dist/guard/__tests__/guard-config.test.d.ts.map +1 -0
  188. package/dist/guard/__tests__/guard-config.test.js +34 -0
  189. package/dist/guard/__tests__/guard-config.test.js.map +1 -0
  190. package/dist/guard/__tests__/guard-integration.test.d.ts +2 -0
  191. package/dist/guard/__tests__/guard-integration.test.d.ts.map +1 -0
  192. package/dist/guard/__tests__/guard-integration.test.js +218 -0
  193. package/dist/guard/__tests__/guard-integration.test.js.map +1 -0
  194. package/dist/guard/__tests__/guard-types.test.d.ts +2 -0
  195. package/dist/guard/__tests__/guard-types.test.d.ts.map +1 -0
  196. package/dist/guard/__tests__/guard-types.test.js +28 -0
  197. package/dist/guard/__tests__/guard-types.test.js.map +1 -0
  198. package/dist/guard/__tests__/provenance.test.d.ts +2 -0
  199. package/dist/guard/__tests__/provenance.test.d.ts.map +1 -0
  200. package/dist/guard/__tests__/provenance.test.js +80 -0
  201. package/dist/guard/__tests__/provenance.test.js.map +1 -0
  202. package/dist/guard/__tests__/scan.test.d.ts +2 -0
  203. package/dist/guard/__tests__/scan.test.d.ts.map +1 -0
  204. package/dist/guard/__tests__/scan.test.js +62 -0
  205. package/dist/guard/__tests__/scan.test.js.map +1 -0
  206. package/dist/guard/__tests__/unicode.test.d.ts +2 -0
  207. package/dist/guard/__tests__/unicode.test.d.ts.map +1 -0
  208. package/dist/guard/__tests__/unicode.test.js +75 -0
  209. package/dist/guard/__tests__/unicode.test.js.map +1 -0
  210. package/dist/guard/boundary.d.ts +9 -0
  211. package/dist/guard/boundary.d.ts.map +1 -0
  212. package/dist/guard/boundary.js +17 -0
  213. package/dist/guard/boundary.js.map +1 -0
  214. package/dist/guard/config.d.ts +60 -0
  215. package/dist/guard/config.d.ts.map +1 -0
  216. package/dist/guard/config.js +64 -0
  217. package/dist/guard/config.js.map +1 -0
  218. package/dist/guard/pipeline.d.ts +13 -0
  219. package/dist/guard/pipeline.d.ts.map +1 -0
  220. package/dist/guard/pipeline.js +120 -0
  221. package/dist/guard/pipeline.js.map +1 -0
  222. package/dist/guard/provenance.d.ts +18 -0
  223. package/dist/guard/provenance.d.ts.map +1 -0
  224. package/dist/guard/provenance.js +108 -0
  225. package/dist/guard/provenance.js.map +1 -0
  226. package/dist/guard/scan.d.ts +6 -0
  227. package/dist/guard/scan.d.ts.map +1 -0
  228. package/dist/guard/scan.js +84 -0
  229. package/dist/guard/scan.js.map +1 -0
  230. package/dist/guard/types.d.ts +28 -0
  231. package/dist/guard/types.d.ts.map +1 -0
  232. package/dist/guard/types.js +2 -0
  233. package/dist/guard/types.js.map +1 -0
  234. package/dist/guard/unicode.d.ts +8 -0
  235. package/dist/guard/unicode.d.ts.map +1 -0
  236. package/dist/guard/unicode.js +126 -0
  237. package/dist/guard/unicode.js.map +1 -0
  238. package/dist/mcp-server/server.d.ts.map +1 -1
  239. package/dist/mcp-server/server.js +8 -1
  240. package/dist/mcp-server/server.js.map +1 -1
  241. package/dist/utils/UpdateApplier.d.ts.map +1 -1
  242. package/dist/utils/UpdateApplier.js +3 -1
  243. package/dist/utils/UpdateApplier.js.map +1 -1
  244. package/dist/utils/project-generator.d.ts.map +1 -1
  245. package/dist/utils/project-generator.js +36 -5
  246. package/dist/utils/project-generator.js.map +1 -1
  247. package/dist/verification/__tests__/telemetry.test.js +3 -0
  248. package/dist/verification/__tests__/telemetry.test.js.map +1 -1
  249. package/dist/verification/config.d.ts +82 -0
  250. package/dist/verification/config.d.ts.map +1 -1
  251. package/dist/verification/config.js +146 -0
  252. package/dist/verification/config.js.map +1 -1
  253. package/package.json +5 -1
  254. package/templates/ide/antigravity/.agents/skills/dare-guard/SKILL.md +16 -0
  255. package/templates/ide/claude/.claude/commands/dare-guard.md +16 -0
  256. package/templates/ide/cursor/.cursor/commands/dare-guard.md +16 -0
@@ -0,0 +1,248 @@
1
+ import { createHash } from 'node:crypto';
2
+ import { readFileSync, readdirSync } from 'node:fs';
3
+ import os from 'node:os';
4
+ import path from 'node:path';
5
+ import { fileURLToPath } from 'node:url';
6
+ import fs from 'fs-extra';
7
+ import { describe, expect, it, vi } from 'vitest';
8
+ import { EmbeddingModelMissingError } from '../graphrag/embeddings.js';
9
+ import * as embeddingsModule from '../graphrag/embeddings.js';
10
+ import { GraphRAG } from '../graphrag/graph-rag.js';
11
+ import { hybridSearch } from '../graphrag/hybrid.js';
12
+ import { JsonGraph } from '../graphrag/json-graph.js';
13
+ const TEST_DIR = path.dirname(fileURLToPath(import.meta.url));
14
+ const FIXTURE_PATH = path.join(TEST_DIR, '__fixtures__', 'semantic-queries.json');
15
+ const PKG_ROOT = path.resolve(TEST_DIR, '..', '..');
16
+ const ALLOWED_REL = path.join('src', 'graphrag', 'embeddings.ts');
17
+ const FORBIDDEN_IMPORT_RE = /(?:import|require)\s*\(?['"](?:onnxruntime|@xenova\/transformers)|from\s+['"](?:onnxruntime|@xenova\/transformers)/;
18
+ const EMBEDDING_DIM = 8;
19
+ function canonicalSemanticKey(text) {
20
+ const normalized = text.toLowerCase();
21
+ if (normalized.includes('refund'))
22
+ return 'concept:refund_policy';
23
+ if (normalized.includes('billing') || normalized.includes('invoice'))
24
+ return 'concept:billing_invoice';
25
+ if (normalized.includes('cancel') ||
26
+ normalized.includes('termination') ||
27
+ normalized.includes('stop recurring')) {
28
+ return 'concept:cancel_subscription';
29
+ }
30
+ if (normalized.includes('downgrade') ||
31
+ normalized.includes('cheaper tier') ||
32
+ normalized.includes('lower tier')) {
33
+ return 'concept:downgrade_plan';
34
+ }
35
+ return `concept:${normalized.replace(/\s+/g, '_')}`;
36
+ }
37
+ function deterministicVector(text) {
38
+ const digest = createHash('sha256').update(canonicalSemanticKey(text)).digest();
39
+ const vector = new Float32Array(EMBEDDING_DIM);
40
+ for (let i = 0; i < EMBEDDING_DIM; i++) {
41
+ const left = digest[i] ?? 0;
42
+ const right = digest[i + EMBEDDING_DIM] ?? 0;
43
+ vector[i] = (left - right) / 255;
44
+ }
45
+ return vector;
46
+ }
47
+ function createDeterministicEmbedder() {
48
+ return {
49
+ dim: EMBEDDING_DIM,
50
+ async embed(text) {
51
+ return deterministicVector(text);
52
+ },
53
+ };
54
+ }
55
+ function buildFixtureNodes() {
56
+ return [
57
+ {
58
+ id: 'node:refund',
59
+ type: 'task',
60
+ label: 'refund policy handbook',
61
+ description: 'official refund policy for digital orders',
62
+ vector: Array.from(deterministicVector('refund policy')),
63
+ },
64
+ {
65
+ id: 'node:billing',
66
+ type: 'task',
67
+ label: 'billing invoice reconciliation',
68
+ description: 'billing invoice closing steps',
69
+ vector: Array.from(deterministicVector('billing invoice')),
70
+ },
71
+ {
72
+ id: 'node:cancel',
73
+ type: 'task',
74
+ label: 'subscription termination playbook',
75
+ description: 'stop recurring charge workflow',
76
+ vector: Array.from(deterministicVector('cancel subscription')),
77
+ },
78
+ {
79
+ id: 'node:downgrade',
80
+ type: 'task',
81
+ label: 'plan tier adjustment guide',
82
+ description: 'switch to a lower tier safely',
83
+ vector: Array.from(deterministicVector('downgrade plan')),
84
+ },
85
+ {
86
+ id: 'node:noise',
87
+ type: 'task',
88
+ label: 'engineering onboarding notes',
89
+ description: 'team process and meeting cadence',
90
+ vector: Array.from(deterministicVector('engineering onboarding')),
91
+ },
92
+ ];
93
+ }
94
+ function readLabeledQueries() {
95
+ return JSON.parse(readFileSync(FIXTURE_PATH, 'utf8'));
96
+ }
97
+ function reciprocalRank(results, relevantNodeIds) {
98
+ for (let i = 0; i < results.length; i++) {
99
+ const item = results[i];
100
+ if (item && relevantNodeIds.has(item.node.id)) {
101
+ return 1 / (i + 1);
102
+ }
103
+ }
104
+ return 0;
105
+ }
106
+ function recallAtK(results, relevantNodeIds, k) {
107
+ if (relevantNodeIds.size === 0)
108
+ return 0;
109
+ const matched = new Set(results.slice(0, k).filter((result) => relevantNodeIds.has(result.node.id)).map((result) => result.node.id));
110
+ return matched.size / relevantNodeIds.size;
111
+ }
112
+ function walkTsFiles(dir, base = PKG_ROOT) {
113
+ const entries = readdirSync(dir, { withFileTypes: true });
114
+ const files = [];
115
+ for (const entry of entries) {
116
+ const full = path.join(dir, entry.name);
117
+ if (entry.isDirectory()) {
118
+ if (entry.name === '__tests__' || entry.name === 'node_modules')
119
+ continue;
120
+ files.push(...walkTsFiles(full, base));
121
+ continue;
122
+ }
123
+ if (!entry.name.endsWith('.ts'))
124
+ continue;
125
+ if (entry.name.endsWith('.test.ts') || entry.name.endsWith('.spec.ts'))
126
+ continue;
127
+ files.push(path.relative(base, full));
128
+ }
129
+ return files;
130
+ }
131
+ describe('semantic regression audit (task-809)', () => {
132
+ it('hybrid_not_worse_than_keyword', async () => {
133
+ const graphPath = path.join(os.tmpdir(), `semantic-regression-${Date.now()}-${Math.random()}.json`);
134
+ const graph = new JsonGraph(graphPath);
135
+ await graph.init();
136
+ for (const node of buildFixtureNodes()) {
137
+ graph.addNode(node);
138
+ }
139
+ try {
140
+ const queries = readLabeledQueries();
141
+ const embedder = createDeterministicEmbedder();
142
+ let semanticImproved = false;
143
+ for (const query of queries) {
144
+ const relevant = new Set(query.relevantNodeIds);
145
+ const keywordRanked = graph.searchNodes(query.query, 5);
146
+ const hybridRanked = await hybridSearch(graph, embedder, query.query, { k: 5, rrfK: 60 });
147
+ const keywordMrr = reciprocalRank(keywordRanked, relevant);
148
+ const hybridMrr = reciprocalRank(hybridRanked, relevant);
149
+ const keywordRecall = recallAtK(keywordRanked, relevant, 5);
150
+ const hybridRecall = recallAtK(hybridRanked, relevant, 5);
151
+ expect(hybridMrr, `MRR regressed for query "${query.id}" (${query.query})`).toBeGreaterThanOrEqual(keywordMrr);
152
+ expect(hybridRecall, `Recall@5 regressed for query "${query.id}" (${query.query})`).toBeGreaterThanOrEqual(keywordRecall);
153
+ if (query.kind === 'semantic' &&
154
+ (hybridMrr > keywordMrr || hybridRecall > keywordRecall)) {
155
+ semanticImproved = true;
156
+ }
157
+ }
158
+ expect(semanticImproved).toBe(true);
159
+ }
160
+ finally {
161
+ graph.close();
162
+ await fs.remove(graphPath).catch(() => undefined);
163
+ }
164
+ });
165
+ it('fallback_without_model', async () => {
166
+ const projectRoot = await fs.mkdtemp(path.join(os.tmpdir(), 'semantic-fallback-'));
167
+ const dbPath = path.join(projectRoot, '.dare', 'graph.db');
168
+ await fs.ensureDir(path.dirname(dbPath));
169
+ await fs.writeJson(path.join(projectRoot, 'dare.config.json'), {
170
+ graphrag: {
171
+ backend: 'sqlite',
172
+ semantic: {
173
+ enabled: true,
174
+ model: 'all-MiniLM-L6-v2',
175
+ modelHash: 'sha256:missing',
176
+ rrfK: 60,
177
+ },
178
+ },
179
+ });
180
+ const loadEmbedderSpy = vi
181
+ .spyOn(embeddingsModule, 'loadEmbedder')
182
+ .mockRejectedValue(new EmbeddingModelMissingError('missing-runtime'));
183
+ const infoSpy = vi.spyOn(console, 'info').mockImplementation(() => undefined);
184
+ const graph = new GraphRAG(dbPath);
185
+ await graph.init();
186
+ graph.addNode({
187
+ id: 'node:keyword-fallback',
188
+ type: 'task',
189
+ label: 'fallback keyword target',
190
+ description: 'keyword fallback should still resolve this node',
191
+ });
192
+ try {
193
+ const keywordRanked = graph.searchNodes('fallback keyword', 5);
194
+ const hybridRanked = await graph.searchNodesHybrid('fallback keyword', 5);
195
+ expect(hybridRanked.map((entry) => entry.node.id)).toEqual(keywordRanked.map((entry) => entry.node.id));
196
+ expect(loadEmbedderSpy).toHaveBeenCalled();
197
+ expect(infoSpy).toHaveBeenCalledWith(expect.stringContaining('semantic disabled, using keyword fallback'));
198
+ }
199
+ finally {
200
+ vi.restoreAllMocks();
201
+ await Promise.resolve(graph.close());
202
+ await fs.remove(projectRoot).catch(() => undefined);
203
+ }
204
+ });
205
+ it('no_heavy_dep_in_core', () => {
206
+ const sourceFiles = walkTsFiles(path.join(PKG_ROOT, 'src')).filter((rel) => path.normalize(rel) !== path.normalize(ALLOWED_REL));
207
+ const offenders = sourceFiles.filter((rel) => {
208
+ const content = readFileSync(path.join(PKG_ROOT, rel), 'utf8');
209
+ return FORBIDDEN_IMPORT_RE.test(content);
210
+ });
211
+ expect(offenders).toEqual([]);
212
+ });
213
+ it('legacy_graph_without_vectors', async () => {
214
+ const graphPath = path.join(os.tmpdir(), `legacy-semantic-${Date.now()}-${Math.random()}.json`);
215
+ const graph = new JsonGraph(graphPath);
216
+ await graph.init();
217
+ graph.addNode({
218
+ id: 'legacy:1',
219
+ type: 'task',
220
+ label: 'legacy keyword anchor',
221
+ description: 'legacy graph has no vectors',
222
+ });
223
+ graph.addNode({
224
+ id: 'legacy:2',
225
+ type: 'task',
226
+ label: 'legacy keyword secondary',
227
+ description: 'secondary keyword fallback candidate',
228
+ });
229
+ try {
230
+ const keywordRanked = graph.searchNodes('legacy keyword', 5);
231
+ const hybridRanked = await hybridSearch(graph, createDeterministicEmbedder(), 'legacy keyword', { k: 5, rrfK: 60 });
232
+ expect(hybridRanked.map((entry) => entry.node.id)).toEqual(keywordRanked.map((entry) => entry.node.id));
233
+ }
234
+ finally {
235
+ graph.close();
236
+ await fs.remove(graphPath).catch(() => undefined);
237
+ }
238
+ });
239
+ it('embeddings_deterministic', async () => {
240
+ const embedder = createDeterministicEmbedder();
241
+ const first = await embedder.embed('cancel my plan');
242
+ const second = await embedder.embed('cancel my plan');
243
+ const different = await embedder.embed('move to cheaper tier');
244
+ expect(Array.from(first)).toEqual(Array.from(second));
245
+ expect(Array.from(first)).not.toEqual(Array.from(different));
246
+ });
247
+ });
248
+ //# sourceMappingURL=semantic-regression.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"semantic-regression.test.js","sourceRoot":"","sources":["../../src/__tests__/semantic-regression.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=budget.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"budget.test.d.ts","sourceRoot":"","sources":["../../../src/agent/__tests__/budget.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,36 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ import { BudgetTracker } from '../budget.js';
3
+ const usage = (input, output, cost = 0) => ({
4
+ inputTokens: input,
5
+ outputTokens: output,
6
+ costUsd: cost,
7
+ model: 'mock',
8
+ });
9
+ describe('BudgetTracker', () => {
10
+ it('should_sum_all_best_of_n_candidates', () => {
11
+ const budget = new BudgetTracker(1000);
12
+ budget.add(usage(100, 50, 0.01));
13
+ budget.add(usage(200, 100, 0.02));
14
+ budget.add(usage(50, 25, 0.005));
15
+ expect(budget.spent.tokens).toBe(525);
16
+ expect(budget.spent.usd).toBeCloseTo(0.035);
17
+ });
18
+ it('should_be_unlimited_when_null', () => {
19
+ const budget = new BudgetTracker(null);
20
+ budget.add(usage(999999, 999999));
21
+ expect(budget.remaining()).toBe(Number.POSITIVE_INFINITY);
22
+ expect(budget.exhausted()).toBe(false);
23
+ });
24
+ it('should_exhaust_at_limit', () => {
25
+ const budget = new BudgetTracker(100);
26
+ budget.add(usage(60, 40));
27
+ expect(budget.exhausted()).toBe(true);
28
+ expect(budget.remaining()).toBe(0);
29
+ });
30
+ it('should_not_go_negative', () => {
31
+ const budget = new BudgetTracker(50);
32
+ budget.add(usage(100, 100));
33
+ expect(budget.remaining()).toBe(0);
34
+ });
35
+ });
36
+ //# sourceMappingURL=budget.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"budget.test.js","sourceRoot":"","sources":["../../../src/agent/__tests__/budget.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,aAAa,EAAE,MAAM,cAAc,CAAC;AAG7C,MAAM,KAAK,GAAG,CAAC,KAAa,EAAE,MAAc,EAAE,IAAI,GAAG,CAAC,EAAc,EAAE,CAAC,CAAC;IACtE,WAAW,EAAE,KAAK;IAClB,YAAY,EAAE,MAAM;IACpB,OAAO,EAAE,IAAI;IACb,KAAK,EAAE,MAAM;CACd,CAAC,CAAC;AAEH,QAAQ,CAAC,eAAe,EAAE,GAAG,EAAE;IAC7B,EAAE,CAAC,qCAAqC,EAAE,GAAG,EAAE;QAC7C,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,IAAI,CAAC,CAAC;QACvC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC,CAAC;QAClC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC,CAAC;QACjC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACtC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;IAC9C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+BAA+B,EAAE,GAAG,EAAE;QACvC,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,IAAI,CAAC,CAAC;QACvC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,MAAO,EAAE,MAAO,CAAC,CAAC,CAAC;QACpC,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAC1D,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yBAAyB,EAAE,GAAG,EAAE;QACjC,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,GAAG,CAAC,CAAC;QACtC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,CAAC;QAC1B,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wBAAwB,EAAE,GAAG,EAAE;QAChC,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,EAAE,CAAC,CAAC;QACrC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC,CAAC;QAC5B,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=claude-driver.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claude-driver.test.d.ts","sourceRoot":"","sources":["../../../src/agent/__tests__/claude-driver.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,88 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
2
+ import { AgentSdkMissingError, createClaudeDriver, setClaudeSdkImporterForTests, } from '../drivers/claude.js';
3
+ function makeInput(signal) {
4
+ return {
5
+ taskId: 'task-605',
6
+ spec: '# spec',
7
+ steering: [],
8
+ worktree: '/tmp/worktree',
9
+ budgetRemaining: 1000,
10
+ signal,
11
+ };
12
+ }
13
+ describe('claude driver', () => {
14
+ const originalKey = process.env.ANTHROPIC_API_KEY;
15
+ beforeEach(() => {
16
+ vi.restoreAllMocks();
17
+ setClaudeSdkImporterForTests(null);
18
+ process.env.ANTHROPIC_API_KEY = 'sk-ant-test-key';
19
+ });
20
+ afterEach(() => {
21
+ setClaudeSdkImporterForTests(null);
22
+ if (originalKey === undefined) {
23
+ delete process.env.ANTHROPIC_API_KEY;
24
+ }
25
+ else {
26
+ process.env.ANTHROPIC_API_KEY = originalKey;
27
+ }
28
+ });
29
+ it('throws_AgentSdkMissingError_when_absent', async () => {
30
+ setClaudeSdkImporterForTests(async () => {
31
+ throw new Error('module missing');
32
+ });
33
+ await expect(createClaudeDriver({ model: 'claude-sonnet-4-5' })).rejects.toBeInstanceOf(AgentSdkMissingError);
34
+ });
35
+ it('does_not_log_api_key', async () => {
36
+ const key = 'sk-ant-secret-should-not-leak';
37
+ process.env.ANTHROPIC_API_KEY = key;
38
+ const createSpy = vi.fn(async () => ({
39
+ usage: { input_tokens: 5, output_tokens: 3 },
40
+ content: [{ type: 'text', text: 'ok' }],
41
+ }));
42
+ class MockAnthropic {
43
+ constructor() {
44
+ this.messages = { create: createSpy };
45
+ }
46
+ }
47
+ setClaudeSdkImporterForTests(async () => ({ default: MockAnthropic }));
48
+ const logSpy = vi.spyOn(console, 'log').mockImplementation(() => undefined);
49
+ const infoSpy = vi.spyOn(console, 'info').mockImplementation(() => undefined);
50
+ const warnSpy = vi.spyOn(console, 'warn').mockImplementation(() => undefined);
51
+ const errorSpy = vi.spyOn(console, 'error').mockImplementation(() => undefined);
52
+ const driver = await createClaudeDriver({ model: 'claude-sonnet-4-5' });
53
+ await driver.run(makeInput(new AbortController().signal));
54
+ const output = [...logSpy.mock.calls, ...infoSpy.mock.calls, ...warnSpy.mock.calls, ...errorSpy.mock.calls]
55
+ .flat()
56
+ .map((value) => String(value))
57
+ .join('\n');
58
+ expect(output).not.toContain(key);
59
+ });
60
+ it('run_returns_usage', async () => {
61
+ const createSpy = vi.fn(async () => ({
62
+ usage: { input_tokens: 120, output_tokens: 60 },
63
+ content: [{ type: 'text', text: 'implemented patch' }],
64
+ }));
65
+ let receivedKey = '';
66
+ class MockAnthropic {
67
+ constructor(options) {
68
+ this.messages = { create: createSpy };
69
+ receivedKey = options.apiKey;
70
+ }
71
+ }
72
+ setClaudeSdkImporterForTests(async () => ({ default: MockAnthropic }));
73
+ const driver = await createClaudeDriver({ model: 'claude-sonnet-4-5', maxTokens: 512 });
74
+ const signal = new AbortController().signal;
75
+ const result = await driver.run(makeInput(signal));
76
+ expect(receivedKey).toBe('sk-ant-test-key');
77
+ expect(createSpy).toHaveBeenCalledWith(expect.objectContaining({
78
+ model: 'claude-sonnet-4-5',
79
+ max_tokens: 512,
80
+ }), expect.objectContaining({ signal }));
81
+ expect(result.status).toBe('implemented');
82
+ expect(result.usage.inputTokens).toBe(120);
83
+ expect(result.usage.outputTokens).toBe(60);
84
+ expect(result.usage.costUsd).toBeCloseTo(0.00126, 6);
85
+ expect(result.usage.model).toBe('claude-sonnet-4-5');
86
+ });
87
+ });
88
+ //# sourceMappingURL=claude-driver.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"claude-driver.test.js","sourceRoot":"","sources":["../../../src/agent/__tests__/claude-driver.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=cost-telemetry.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cost-telemetry.test.d.ts","sourceRoot":"","sources":["../../../src/agent/__tests__/cost-telemetry.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,61 @@
1
+ import os from 'node:os';
2
+ import path from 'node:path';
3
+ import fs from 'fs-extra';
4
+ import { beforeEach, afterEach, describe, expect, it } from 'vitest';
5
+ import { JsonGraph } from '../../graphrag/json-graph.js';
6
+ import { recordCostTelemetry } from '../telemetry.js';
7
+ const sampleUsage = {
8
+ inputTokens: 120,
9
+ outputTokens: 45,
10
+ costUsd: 0.03125,
11
+ model: 'claude-sonnet',
12
+ };
13
+ describe('recordCostTelemetry', () => {
14
+ let filePath;
15
+ let graph;
16
+ beforeEach(async () => {
17
+ filePath = path.join(os.tmpdir(), `dare-cost-telemetry-${Date.now()}-${Math.random().toString(36).slice(2)}.json`);
18
+ graph = new JsonGraph(filePath);
19
+ await graph.init();
20
+ });
21
+ afterEach(async () => {
22
+ graph.close();
23
+ await fs.remove(filePath).catch(() => undefined);
24
+ });
25
+ it('writes_cost_metadata_on_task_node', () => {
26
+ graph.addNode({
27
+ id: 'task:task-607',
28
+ type: 'task',
29
+ label: 'task-607',
30
+ metadata: { status: 'DONE' },
31
+ });
32
+ recordCostTelemetry(graph, 'task-607', sampleUsage, 3);
33
+ const taskNode = graph.getNode('task:task-607');
34
+ expect(taskNode?.metadata?.status).toBe('DONE');
35
+ expect(taskNode?.metadata?.inputTokens).toBe(120);
36
+ expect(taskNode?.metadata?.outputTokens).toBe(45);
37
+ expect(taskNode?.metadata?.costUsd).toBe(0.03125);
38
+ expect(taskNode?.metadata?.model).toBe('claude-sonnet');
39
+ expect(taskNode?.metadata?.attempts).toBe(3);
40
+ });
41
+ it('is_queryable_via_graph', () => {
42
+ graph.addNode({
43
+ id: 'task:task-608',
44
+ type: 'task',
45
+ label: 'task-608',
46
+ });
47
+ recordCostTelemetry(graph, 'task-608', sampleUsage, 1);
48
+ const fetched = graph.getNode('task:task-608');
49
+ expect(fetched?.metadata?.costUsd).toBe(0.03125);
50
+ const matches = graph.searchNodes('task-608', 5);
51
+ expect(matches.some((entry) => entry.node.id === 'task:task-608')).toBe(true);
52
+ const queried = matches.find((entry) => entry.node.id === 'task:task-608');
53
+ expect(queried?.node.metadata?.inputTokens).toBe(120);
54
+ });
55
+ it('no_throw_on_missing_node', () => {
56
+ expect(() => {
57
+ recordCostTelemetry(graph, 'task-missing', sampleUsage, 2);
58
+ }).not.toThrow();
59
+ });
60
+ });
61
+ //# sourceMappingURL=cost-telemetry.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cost-telemetry.test.js","sourceRoot":"","sources":["../../../src/agent/__tests__/cost-telemetry.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=driver.types.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"driver.types.test.d.ts","sourceRoot":"","sources":["../../../src/agent/__tests__/driver.types.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,33 @@
1
+ import { describe, it, expect } from 'vitest';
2
+ describe('agent/driver types', () => {
3
+ it('exports_all_symbols', () => {
4
+ const usage = {
5
+ inputTokens: 0,
6
+ outputTokens: 0,
7
+ costUsd: 0,
8
+ model: 'mock',
9
+ };
10
+ const steering = [];
11
+ const input = {
12
+ taskId: 'task-601',
13
+ spec: '# spec',
14
+ steering,
15
+ worktree: '/tmp/wt',
16
+ budgetRemaining: 1000,
17
+ signal: new AbortController().signal,
18
+ };
19
+ const result = {
20
+ status: 'implemented',
21
+ worktree: input.worktree,
22
+ summary: 'done',
23
+ usage,
24
+ };
25
+ const driver = {
26
+ id: 'mock',
27
+ requiresNetwork: false,
28
+ run: async () => result,
29
+ };
30
+ expect(driver.id).toBe('mock');
31
+ });
32
+ });
33
+ //# sourceMappingURL=driver.types.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"driver.types.test.js","sourceRoot":"","sources":["../../../src/agent/__tests__/driver.types.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAS9C,QAAQ,CAAC,oBAAoB,EAAE,GAAG,EAAE;IAClC,EAAE,CAAC,qBAAqB,EAAE,GAAG,EAAE;QAC7B,MAAM,KAAK,GAAe;YACxB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;YACf,OAAO,EAAE,CAAC;YACV,KAAK,EAAE,MAAM;SACd,CAAC;QACF,MAAM,QAAQ,GAAmC,EAAE,CAAC;QACpD,MAAM,KAAK,GAAkB;YAC3B,MAAM,EAAE,UAAU;YAClB,IAAI,EAAE,QAAQ;YACd,QAAQ;YACR,QAAQ,EAAE,SAAS;YACnB,eAAe,EAAE,IAAI;YACrB,MAAM,EAAE,IAAI,eAAe,EAAE,CAAC,MAAM;SACrC,CAAC;QACF,MAAM,MAAM,GAAmB;YAC7B,MAAM,EAAE,aAAa;YACrB,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,OAAO,EAAE,MAAM;YACf,KAAK;SACN,CAAC;QACF,MAAM,MAAM,GAAgB;YAC1B,EAAE,EAAE,MAAM;YACV,eAAe,EAAE,KAAK;YACtB,GAAG,EAAE,KAAK,IAAI,EAAE,CAAC,MAAM;SACxB,CAAC;QACF,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=mock-driver.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mock-driver.test.d.ts","sourceRoot":"","sources":["../../../src/agent/__tests__/mock-driver.test.ts"],"names":[],"mappings":""}