@clinebot/core 0.0.21 → 0.0.23

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 (260) hide show
  1. package/dist/ClineCore.d.ts +110 -0
  2. package/dist/ClineCore.d.ts.map +1 -0
  3. package/dist/account/cline-account-service.d.ts +2 -1
  4. package/dist/account/cline-account-service.d.ts.map +1 -1
  5. package/dist/account/index.d.ts +1 -1
  6. package/dist/account/index.d.ts.map +1 -1
  7. package/dist/account/rpc.d.ts +3 -1
  8. package/dist/account/rpc.d.ts.map +1 -1
  9. package/dist/account/types.d.ts +3 -0
  10. package/dist/account/types.d.ts.map +1 -1
  11. package/dist/agents/plugin-loader.d.ts.map +1 -1
  12. package/dist/agents/plugin-sandbox-bootstrap.js +17 -17
  13. package/dist/auth/client.d.ts +1 -1
  14. package/dist/auth/client.d.ts.map +1 -1
  15. package/dist/auth/cline.d.ts +1 -1
  16. package/dist/auth/cline.d.ts.map +1 -1
  17. package/dist/auth/codex.d.ts +1 -1
  18. package/dist/auth/codex.d.ts.map +1 -1
  19. package/dist/auth/oca.d.ts +1 -1
  20. package/dist/auth/oca.d.ts.map +1 -1
  21. package/dist/auth/utils.d.ts +2 -2
  22. package/dist/auth/utils.d.ts.map +1 -1
  23. package/dist/index.d.ts +50 -5
  24. package/dist/index.d.ts.map +1 -1
  25. package/dist/index.js +949 -0
  26. package/dist/providers/local-provider-service.d.ts +4 -4
  27. package/dist/providers/local-provider-service.d.ts.map +1 -1
  28. package/dist/runtime/runtime-builder.d.ts +1 -0
  29. package/dist/runtime/runtime-builder.d.ts.map +1 -1
  30. package/dist/runtime/session-runtime.d.ts +2 -1
  31. package/dist/runtime/session-runtime.d.ts.map +1 -1
  32. package/dist/runtime/team-runtime-registry.d.ts +13 -0
  33. package/dist/runtime/team-runtime-registry.d.ts.map +1 -0
  34. package/dist/session/default-session-manager.d.ts +2 -2
  35. package/dist/session/default-session-manager.d.ts.map +1 -1
  36. package/dist/session/rpc-runtime-ensure.d.ts +53 -0
  37. package/dist/session/rpc-runtime-ensure.d.ts.map +1 -0
  38. package/dist/session/session-config-builder.d.ts +2 -3
  39. package/dist/session/session-config-builder.d.ts.map +1 -1
  40. package/dist/session/session-host.d.ts +8 -18
  41. package/dist/session/session-host.d.ts.map +1 -1
  42. package/dist/session/session-manager.d.ts +1 -1
  43. package/dist/session/session-manager.d.ts.map +1 -1
  44. package/dist/session/session-manifest.d.ts +1 -2
  45. package/dist/session/session-manifest.d.ts.map +1 -1
  46. package/dist/session/unified-session-persistence-service.d.ts +2 -2
  47. package/dist/session/unified-session-persistence-service.d.ts.map +1 -1
  48. package/dist/session/utils/helpers.d.ts +1 -1
  49. package/dist/session/utils/helpers.d.ts.map +1 -1
  50. package/dist/session/utils/types.d.ts +1 -1
  51. package/dist/session/utils/types.d.ts.map +1 -1
  52. package/dist/storage/provider-settings-legacy-migration.d.ts.map +1 -1
  53. package/dist/telemetry/OpenTelemetryProvider.d.ts.map +1 -1
  54. package/dist/telemetry/distinct-id.d.ts +2 -0
  55. package/dist/telemetry/distinct-id.d.ts.map +1 -0
  56. package/dist/telemetry/{opentelemetry.d.ts → index.d.ts} +1 -1
  57. package/dist/telemetry/index.d.ts.map +1 -0
  58. package/dist/telemetry/index.js +28 -0
  59. package/dist/tools/constants.d.ts +1 -1
  60. package/dist/tools/constants.d.ts.map +1 -1
  61. package/dist/tools/definitions.d.ts +3 -3
  62. package/dist/tools/definitions.d.ts.map +1 -1
  63. package/dist/tools/executors/apply-patch.d.ts +1 -1
  64. package/dist/tools/executors/apply-patch.d.ts.map +1 -1
  65. package/dist/tools/executors/bash.d.ts +1 -1
  66. package/dist/tools/executors/bash.d.ts.map +1 -1
  67. package/dist/tools/executors/editor.d.ts +1 -1
  68. package/dist/tools/executors/editor.d.ts.map +1 -1
  69. package/dist/tools/executors/file-read.d.ts +1 -1
  70. package/dist/tools/executors/file-read.d.ts.map +1 -1
  71. package/dist/tools/executors/index.d.ts +14 -14
  72. package/dist/tools/executors/index.d.ts.map +1 -1
  73. package/dist/tools/executors/search.d.ts +1 -1
  74. package/dist/tools/executors/search.d.ts.map +1 -1
  75. package/dist/tools/executors/web-fetch.d.ts +1 -1
  76. package/dist/tools/executors/web-fetch.d.ts.map +1 -1
  77. package/dist/tools/helpers.d.ts +1 -1
  78. package/dist/tools/helpers.d.ts.map +1 -1
  79. package/dist/tools/index.d.ts +10 -10
  80. package/dist/tools/index.d.ts.map +1 -1
  81. package/dist/tools/model-tool-routing.d.ts +1 -1
  82. package/dist/tools/model-tool-routing.d.ts.map +1 -1
  83. package/dist/tools/presets.d.ts +1 -1
  84. package/dist/tools/presets.d.ts.map +1 -1
  85. package/dist/types/common.d.ts +17 -8
  86. package/dist/types/common.d.ts.map +1 -1
  87. package/dist/types/config.d.ts +4 -3
  88. package/dist/types/config.d.ts.map +1 -1
  89. package/dist/types/provider-settings.d.ts +1 -1
  90. package/dist/types/provider-settings.d.ts.map +1 -1
  91. package/dist/types.d.ts +5 -2
  92. package/dist/types.d.ts.map +1 -1
  93. package/dist/version.d.ts +2 -0
  94. package/dist/version.d.ts.map +1 -0
  95. package/package.json +44 -38
  96. package/src/ClineCore.ts +137 -0
  97. package/src/account/cline-account-service.test.ts +101 -0
  98. package/src/account/cline-account-service.ts +300 -0
  99. package/src/account/featurebase-token.test.ts +175 -0
  100. package/src/account/index.ts +23 -0
  101. package/src/account/rpc.test.ts +63 -0
  102. package/src/account/rpc.ts +185 -0
  103. package/src/account/types.ts +102 -0
  104. package/src/agents/agent-config-loader.test.ts +236 -0
  105. package/src/agents/agent-config-loader.ts +108 -0
  106. package/src/agents/agent-config-parser.ts +198 -0
  107. package/src/agents/hooks-config-loader.test.ts +20 -0
  108. package/src/agents/hooks-config-loader.ts +118 -0
  109. package/src/agents/index.ts +85 -0
  110. package/src/agents/plugin-config-loader.test.ts +140 -0
  111. package/src/agents/plugin-config-loader.ts +97 -0
  112. package/src/agents/plugin-loader.test.ts +210 -0
  113. package/src/agents/plugin-loader.ts +175 -0
  114. package/src/agents/plugin-sandbox-bootstrap.ts +448 -0
  115. package/src/agents/plugin-sandbox.test.ts +296 -0
  116. package/src/agents/plugin-sandbox.ts +341 -0
  117. package/src/agents/unified-config-file-watcher.test.ts +196 -0
  118. package/src/agents/unified-config-file-watcher.ts +483 -0
  119. package/src/agents/user-instruction-config-loader.test.ts +158 -0
  120. package/src/agents/user-instruction-config-loader.ts +438 -0
  121. package/src/auth/client.test.ts +40 -0
  122. package/src/auth/client.ts +25 -0
  123. package/src/auth/cline.test.ts +130 -0
  124. package/src/auth/cline.ts +420 -0
  125. package/src/auth/codex.test.ts +170 -0
  126. package/src/auth/codex.ts +491 -0
  127. package/src/auth/oca.test.ts +215 -0
  128. package/src/auth/oca.ts +573 -0
  129. package/src/auth/server.ts +216 -0
  130. package/src/auth/types.ts +81 -0
  131. package/src/auth/utils.test.ts +128 -0
  132. package/src/auth/utils.ts +247 -0
  133. package/src/chat/chat-schema.ts +82 -0
  134. package/src/index.ts +479 -0
  135. package/src/input/file-indexer.d.ts +11 -0
  136. package/src/input/file-indexer.test.ts +127 -0
  137. package/src/input/file-indexer.ts +327 -0
  138. package/src/input/index.ts +7 -0
  139. package/src/input/mention-enricher.test.ts +85 -0
  140. package/src/input/mention-enricher.ts +122 -0
  141. package/src/mcp/config-loader.test.ts +238 -0
  142. package/src/mcp/config-loader.ts +219 -0
  143. package/src/mcp/index.ts +26 -0
  144. package/src/mcp/manager.test.ts +106 -0
  145. package/src/mcp/manager.ts +262 -0
  146. package/src/mcp/types.ts +88 -0
  147. package/src/providers/local-provider-registry.ts +232 -0
  148. package/src/providers/local-provider-service.test.ts +783 -0
  149. package/src/providers/local-provider-service.ts +471 -0
  150. package/src/runtime/commands.test.ts +98 -0
  151. package/src/runtime/commands.ts +83 -0
  152. package/src/runtime/hook-file-hooks.test.ts +237 -0
  153. package/src/runtime/hook-file-hooks.ts +859 -0
  154. package/src/runtime/index.ts +37 -0
  155. package/src/runtime/rules.ts +34 -0
  156. package/src/runtime/runtime-builder.team-persistence.test.ts +245 -0
  157. package/src/runtime/runtime-builder.test.ts +371 -0
  158. package/src/runtime/runtime-builder.ts +631 -0
  159. package/src/runtime/runtime-parity.test.ts +143 -0
  160. package/src/runtime/sandbox/subprocess-sandbox.ts +231 -0
  161. package/src/runtime/session-runtime.ts +49 -0
  162. package/src/runtime/skills.ts +44 -0
  163. package/src/runtime/team-runtime-registry.ts +46 -0
  164. package/src/runtime/tool-approval.ts +104 -0
  165. package/src/runtime/workflows.test.ts +119 -0
  166. package/src/runtime/workflows.ts +45 -0
  167. package/src/session/default-session-manager.e2e.test.ts +384 -0
  168. package/src/session/default-session-manager.test.ts +1931 -0
  169. package/src/session/default-session-manager.ts +1422 -0
  170. package/src/session/file-session-service.ts +280 -0
  171. package/src/session/index.ts +45 -0
  172. package/src/session/rpc-runtime-ensure.ts +521 -0
  173. package/src/session/rpc-session-service.ts +107 -0
  174. package/src/session/rpc-spawn-lease.test.ts +49 -0
  175. package/src/session/rpc-spawn-lease.ts +122 -0
  176. package/src/session/runtime-oauth-token-manager.test.ts +137 -0
  177. package/src/session/runtime-oauth-token-manager.ts +272 -0
  178. package/src/session/session-agent-events.ts +248 -0
  179. package/src/session/session-artifacts.ts +106 -0
  180. package/src/session/session-config-builder.ts +113 -0
  181. package/src/session/session-graph.ts +92 -0
  182. package/src/session/session-host.test.ts +89 -0
  183. package/src/session/session-host.ts +205 -0
  184. package/src/session/session-manager.ts +69 -0
  185. package/src/session/session-manifest.ts +29 -0
  186. package/src/session/session-service.team-persistence.test.ts +48 -0
  187. package/src/session/session-service.ts +673 -0
  188. package/src/session/session-team-coordination.ts +229 -0
  189. package/src/session/session-telemetry.ts +100 -0
  190. package/src/session/sqlite-rpc-session-backend.ts +303 -0
  191. package/src/session/unified-session-persistence-service.test.ts +85 -0
  192. package/src/session/unified-session-persistence-service.ts +994 -0
  193. package/src/session/utils/helpers.ts +139 -0
  194. package/src/session/utils/types.ts +57 -0
  195. package/src/session/utils/usage.ts +32 -0
  196. package/src/session/workspace-manager.ts +98 -0
  197. package/src/session/workspace-manifest.ts +100 -0
  198. package/src/storage/artifact-store.ts +1 -0
  199. package/src/storage/file-team-store.ts +257 -0
  200. package/src/storage/index.ts +11 -0
  201. package/src/storage/provider-settings-legacy-migration.test.ts +424 -0
  202. package/src/storage/provider-settings-legacy-migration.ts +826 -0
  203. package/src/storage/provider-settings-manager.test.ts +191 -0
  204. package/src/storage/provider-settings-manager.ts +152 -0
  205. package/src/storage/session-store.ts +1 -0
  206. package/src/storage/sqlite-session-store.ts +275 -0
  207. package/src/storage/sqlite-team-store.ts +454 -0
  208. package/src/storage/team-store.ts +40 -0
  209. package/src/team/index.ts +4 -0
  210. package/src/team/projections.ts +285 -0
  211. package/src/telemetry/ITelemetryAdapter.ts +94 -0
  212. package/src/telemetry/LoggerTelemetryAdapter.test.ts +42 -0
  213. package/src/telemetry/LoggerTelemetryAdapter.ts +114 -0
  214. package/src/telemetry/OpenTelemetryAdapter.test.ts +157 -0
  215. package/src/telemetry/OpenTelemetryAdapter.ts +348 -0
  216. package/src/telemetry/OpenTelemetryProvider.test.ts +113 -0
  217. package/src/telemetry/OpenTelemetryProvider.ts +325 -0
  218. package/src/telemetry/TelemetryService.test.ts +134 -0
  219. package/src/telemetry/TelemetryService.ts +141 -0
  220. package/src/telemetry/core-events.ts +400 -0
  221. package/src/telemetry/distinct-id.test.ts +57 -0
  222. package/src/telemetry/distinct-id.ts +58 -0
  223. package/src/telemetry/index.ts +20 -0
  224. package/src/tools/constants.ts +35 -0
  225. package/src/tools/definitions.test.ts +704 -0
  226. package/src/tools/definitions.ts +709 -0
  227. package/src/tools/executors/apply-patch-parser.ts +520 -0
  228. package/src/tools/executors/apply-patch.ts +359 -0
  229. package/src/tools/executors/bash.test.ts +87 -0
  230. package/src/tools/executors/bash.ts +207 -0
  231. package/src/tools/executors/editor.test.ts +35 -0
  232. package/src/tools/executors/editor.ts +219 -0
  233. package/src/tools/executors/file-read.test.ts +49 -0
  234. package/src/tools/executors/file-read.ts +110 -0
  235. package/src/tools/executors/index.ts +87 -0
  236. package/src/tools/executors/search.ts +278 -0
  237. package/src/tools/executors/web-fetch.ts +259 -0
  238. package/src/tools/helpers.ts +130 -0
  239. package/src/tools/index.ts +169 -0
  240. package/src/tools/model-tool-routing.test.ts +86 -0
  241. package/src/tools/model-tool-routing.ts +132 -0
  242. package/src/tools/presets.test.ts +62 -0
  243. package/src/tools/presets.ts +168 -0
  244. package/src/tools/schemas.ts +327 -0
  245. package/src/tools/types.ts +329 -0
  246. package/src/types/common.ts +26 -0
  247. package/src/types/config.ts +86 -0
  248. package/src/types/events.ts +74 -0
  249. package/src/types/index.ts +24 -0
  250. package/src/types/provider-settings.ts +43 -0
  251. package/src/types/sessions.ts +16 -0
  252. package/src/types/storage.ts +64 -0
  253. package/src/types/workspace.ts +7 -0
  254. package/src/types.ts +132 -0
  255. package/src/version.ts +3 -0
  256. package/dist/index.node.d.ts +0 -47
  257. package/dist/index.node.d.ts.map +0 -1
  258. package/dist/index.node.js +0 -948
  259. package/dist/telemetry/opentelemetry.d.ts.map +0 -1
  260. package/dist/telemetry/opentelemetry.js +0 -27
@@ -0,0 +1,327 @@
1
+ import { spawn } from "node:child_process";
2
+ import { readdir } from "node:fs/promises";
3
+ import path from "node:path";
4
+ import { isMainThread, parentPort, Worker } from "node:worker_threads";
5
+
6
+ const DEFAULT_INDEX_TTL_MS = 15_000;
7
+ const STALE_CACHE_EVICTION_MS = 10 * 60_000;
8
+ const WORKER_INDEX_REQUEST_TIMEOUT_MS = 1_000;
9
+ const DEFAULT_EXCLUDE_DIRS = new Set([
10
+ ".git",
11
+ "node_modules",
12
+ "dist",
13
+ "build",
14
+ ".next",
15
+ "coverage",
16
+ ".turbo",
17
+ ".cache",
18
+ "target",
19
+ "out",
20
+ ]);
21
+
22
+ interface CacheEntry {
23
+ files: Set<string>;
24
+ lastBuiltAt: number;
25
+ lastAccessedAt: number;
26
+ pending: Promise<Set<string>> | null;
27
+ }
28
+
29
+ export interface FastFileIndexOptions {
30
+ ttlMs?: number;
31
+ }
32
+
33
+ interface IndexRequestMessage {
34
+ type: "index";
35
+ requestId: number;
36
+ cwd: string;
37
+ }
38
+
39
+ interface IndexResponseMessage {
40
+ type: "indexResult";
41
+ requestId: number;
42
+ files?: string[];
43
+ error?: string;
44
+ }
45
+
46
+ const CACHE = new Map<string, CacheEntry>();
47
+
48
+ function pruneStaleCacheEntries(now: number): void {
49
+ if (CACHE.size <= 1) {
50
+ return;
51
+ }
52
+ for (const [cwd, entry] of CACHE.entries()) {
53
+ if (entry.pending) {
54
+ continue;
55
+ }
56
+ if (now - entry.lastAccessedAt > STALE_CACHE_EVICTION_MS) {
57
+ CACHE.delete(cwd);
58
+ }
59
+ }
60
+ }
61
+
62
+ function toPosixRelative(cwd: string, absolutePath: string): string {
63
+ return path.relative(cwd, absolutePath).split(path.sep).join("/");
64
+ }
65
+
66
+ async function listFilesWithRg(cwd: string): Promise<Set<string>> {
67
+ const output = await new Promise<string>((resolve, reject) => {
68
+ const child = spawn("rg", ["--files", "--hidden", "-g", "!.git"], {
69
+ cwd,
70
+ stdio: ["ignore", "pipe", "pipe"],
71
+ });
72
+
73
+ let stdout = "";
74
+ let stderr = "";
75
+
76
+ child.stdout.on("data", (chunk: Buffer | string) => {
77
+ stdout += chunk.toString();
78
+ });
79
+ child.stderr.on("data", (chunk: Buffer | string) => {
80
+ stderr += chunk.toString();
81
+ });
82
+ child.on("error", reject);
83
+ child.on("close", (code: number | null) => {
84
+ if (code === 0) {
85
+ resolve(stdout);
86
+ return;
87
+ }
88
+ reject(new Error(stderr || `rg exited with code ${code}`));
89
+ });
90
+ });
91
+
92
+ const files = output
93
+ .split(/\r?\n/)
94
+ .map((line) => line.trim())
95
+ .filter((line) => line.length > 0)
96
+ .map((line) => line.replace(/\\/g, "/"));
97
+
98
+ return new Set(files);
99
+ }
100
+
101
+ async function walkDir(
102
+ cwd: string,
103
+ dir: string,
104
+ files: Set<string>,
105
+ ): Promise<void> {
106
+ const entries = await readdir(dir, { withFileTypes: true });
107
+ for (const entry of entries) {
108
+ const absolutePath = path.join(dir, entry.name);
109
+ if (entry.isDirectory()) {
110
+ if (DEFAULT_EXCLUDE_DIRS.has(entry.name)) {
111
+ continue;
112
+ }
113
+ await walkDir(cwd, absolutePath, files);
114
+ continue;
115
+ }
116
+ if (entry.isFile()) {
117
+ files.add(toPosixRelative(cwd, absolutePath));
118
+ }
119
+ }
120
+ }
121
+
122
+ async function listFilesFallback(cwd: string): Promise<Set<string>> {
123
+ const files = new Set<string>();
124
+ await walkDir(cwd, cwd, files);
125
+ return files;
126
+ }
127
+
128
+ async function buildIndex(cwd: string): Promise<Set<string>> {
129
+ try {
130
+ return await listFilesWithRg(cwd);
131
+ } catch {
132
+ return listFilesFallback(cwd);
133
+ }
134
+ }
135
+
136
+ function startWorkerServer(): void {
137
+ if (isMainThread || !parentPort) {
138
+ return;
139
+ }
140
+ const port = parentPort;
141
+
142
+ port.on("message", (message: IndexRequestMessage) => {
143
+ if (message.type !== "index") {
144
+ return;
145
+ }
146
+
147
+ void buildIndex(message.cwd)
148
+ .then((files) => {
149
+ const response: IndexResponseMessage = {
150
+ type: "indexResult",
151
+ requestId: message.requestId,
152
+ files: Array.from(files),
153
+ };
154
+ port.postMessage(response);
155
+ })
156
+ .catch((error: unknown) => {
157
+ const response: IndexResponseMessage = {
158
+ type: "indexResult",
159
+ requestId: message.requestId,
160
+ error:
161
+ error instanceof Error
162
+ ? error.message
163
+ : "Failed to build file index",
164
+ };
165
+ port.postMessage(response);
166
+ });
167
+ });
168
+ }
169
+
170
+ class FileIndexWorkerClient {
171
+ private readonly worker = new Worker(new URL(import.meta.url));
172
+ private nextRequestId = 0;
173
+ private pending = new Map<
174
+ number,
175
+ {
176
+ resolve: (files: string[]) => void;
177
+ reject: (reason: Error) => void;
178
+ }
179
+ >();
180
+
181
+ constructor() {
182
+ // Keep indexing opportunistic: this worker should never block process exit.
183
+ this.worker.unref();
184
+ this.worker.on("message", (message: IndexResponseMessage) => {
185
+ if (message.type !== "indexResult") {
186
+ return;
187
+ }
188
+ const request = this.pending.get(message.requestId);
189
+ if (!request) {
190
+ return;
191
+ }
192
+ this.pending.delete(message.requestId);
193
+ if (message.error) {
194
+ request.reject(new Error(message.error));
195
+ return;
196
+ }
197
+ request.resolve(message.files ?? []);
198
+ });
199
+
200
+ this.worker.on("error", (error: Error) => {
201
+ this.flushPending(error);
202
+ });
203
+
204
+ this.worker.on("exit", (code) => {
205
+ if (code !== 0) {
206
+ this.flushPending(
207
+ new Error(`File index worker exited with code ${code}`),
208
+ );
209
+ }
210
+ });
211
+ }
212
+
213
+ requestIndex(cwd: string): Promise<string[]> {
214
+ const requestId = ++this.nextRequestId;
215
+ const result = new Promise<string[]>((resolve, reject) => {
216
+ const timeout = setTimeout(() => {
217
+ this.pending.delete(requestId);
218
+ reject(new Error("Timed out waiting for file index worker response"));
219
+ }, WORKER_INDEX_REQUEST_TIMEOUT_MS);
220
+ timeout.unref();
221
+ this.pending.set(requestId, {
222
+ resolve: (files) => {
223
+ clearTimeout(timeout);
224
+ resolve(files);
225
+ },
226
+ reject: (reason) => {
227
+ clearTimeout(timeout);
228
+ reject(reason);
229
+ },
230
+ });
231
+ });
232
+
233
+ const message: IndexRequestMessage = {
234
+ type: "index",
235
+ requestId,
236
+ cwd,
237
+ };
238
+ this.worker.postMessage(message);
239
+ return result;
240
+ }
241
+
242
+ private flushPending(error: Error): void {
243
+ for (const [requestId, request] of this.pending.entries()) {
244
+ request.reject(error);
245
+ this.pending.delete(requestId);
246
+ }
247
+ }
248
+ }
249
+
250
+ startWorkerServer();
251
+
252
+ let workerClient: FileIndexWorkerClient | null | undefined;
253
+
254
+ function getWorkerClient(): FileIndexWorkerClient | null {
255
+ if (!isMainThread) {
256
+ return null;
257
+ }
258
+ if (workerClient === undefined) {
259
+ workerClient = new FileIndexWorkerClient();
260
+ }
261
+ return workerClient;
262
+ }
263
+
264
+ async function buildIndexInBackground(cwd: string): Promise<Set<string>> {
265
+ const workerClient = getWorkerClient();
266
+ if (!workerClient) {
267
+ return buildIndex(cwd);
268
+ }
269
+
270
+ try {
271
+ const files = await workerClient.requestIndex(cwd);
272
+ return new Set(files);
273
+ } catch {
274
+ return buildIndex(cwd);
275
+ }
276
+ }
277
+
278
+ export async function getFileIndex(
279
+ cwd: string,
280
+ options: FastFileIndexOptions = {},
281
+ ): Promise<Set<string>> {
282
+ const ttlMs = options.ttlMs ?? DEFAULT_INDEX_TTL_MS;
283
+ const now = Date.now();
284
+ pruneStaleCacheEntries(now);
285
+ const existing = CACHE.get(cwd);
286
+
287
+ if (
288
+ existing &&
289
+ ttlMs > 0 &&
290
+ now - existing.lastBuiltAt <= ttlMs &&
291
+ existing.files.size > 0
292
+ ) {
293
+ existing.lastAccessedAt = now;
294
+ return existing.files;
295
+ }
296
+
297
+ if (existing?.pending) {
298
+ existing.lastAccessedAt = now;
299
+ return existing.pending;
300
+ }
301
+
302
+ const pending = buildIndexInBackground(cwd).then((files) => {
303
+ CACHE.set(cwd, {
304
+ files,
305
+ lastBuiltAt: Date.now(),
306
+ lastAccessedAt: Date.now(),
307
+ pending: null,
308
+ });
309
+ return files;
310
+ });
311
+
312
+ CACHE.set(cwd, {
313
+ files: existing?.files ?? new Set<string>(),
314
+ lastBuiltAt: existing?.lastBuiltAt ?? 0,
315
+ lastAccessedAt: now,
316
+ pending,
317
+ });
318
+
319
+ return pending;
320
+ }
321
+
322
+ export async function prewarmFileIndex(
323
+ cwd: string,
324
+ options: FastFileIndexOptions = {},
325
+ ): Promise<void> {
326
+ await getFileIndex(cwd, { ...options, ttlMs: 0 });
327
+ }
@@ -0,0 +1,7 @@
1
+ export type { FastFileIndexOptions } from "./file-indexer";
2
+ export { getFileIndex, prewarmFileIndex } from "./file-indexer";
3
+ export type {
4
+ MentionEnricherOptions,
5
+ MentionEnrichmentResult,
6
+ } from "./mention-enricher";
7
+ export { enrichPromptWithMentions } from "./mention-enricher";
@@ -0,0 +1,85 @@
1
+ import { mkdir, mkdtemp, rm, writeFile } from "node:fs/promises";
2
+ import os from "node:os";
3
+ import path from "node:path";
4
+ import { describe, expect, it, vi } from "vitest";
5
+ import { enrichPromptWithMentions } from "./mention-enricher";
6
+
7
+ vi.mock("node:worker_threads", async () => {
8
+ const actual = await vi.importActual<typeof import("node:worker_threads")>(
9
+ "node:worker_threads",
10
+ );
11
+ return {
12
+ ...actual,
13
+ isMainThread: false,
14
+ parentPort: null,
15
+ };
16
+ });
17
+
18
+ async function createTempWorkspace(): Promise<string> {
19
+ return mkdtemp(path.join(os.tmpdir(), "core-mentions-"));
20
+ }
21
+
22
+ describe("enrichPromptWithMentions", () => {
23
+ it("returns matched files for matching @path mentions", async () => {
24
+ const cwd = await createTempWorkspace();
25
+ try {
26
+ const sourcePath = path.join(cwd, "src", "index.ts");
27
+ await mkdir(path.dirname(sourcePath), { recursive: true });
28
+ await writeFile(sourcePath, "export const answer = 42\n", "utf8");
29
+
30
+ const result = await enrichPromptWithMentions(
31
+ "Review @src/index.ts",
32
+ cwd,
33
+ );
34
+
35
+ expect(result.mentions).toEqual(["src/index.ts"]);
36
+ expect(result.matchedFiles).toEqual(["src/index.ts"]);
37
+ expect(result.ignoredMentions).toEqual([]);
38
+ expect(result.prompt).toBe("Review @src/index.ts");
39
+ } finally {
40
+ await rm(cwd, { recursive: true, force: true });
41
+ }
42
+ });
43
+
44
+ it("ignores emails and unmatched mentions", async () => {
45
+ const cwd = await createTempWorkspace();
46
+ try {
47
+ await writeFile(path.join(cwd, "README.md"), "# Demo\n", "utf8");
48
+
49
+ const result = await enrichPromptWithMentions(
50
+ "Ping me at test@example.com and check @missing/file.ts.",
51
+ cwd,
52
+ );
53
+
54
+ expect(result.mentions).toEqual(["missing/file.ts"]);
55
+ expect(result.matchedFiles).toEqual([]);
56
+ expect(result.ignoredMentions).toEqual(["missing/file.ts"]);
57
+ expect(result.prompt).toBe(
58
+ "Ping me at test@example.com and check @missing/file.ts.",
59
+ );
60
+ } finally {
61
+ await rm(cwd, { recursive: true, force: true });
62
+ }
63
+ });
64
+
65
+ it("respects maxTotalBytes while keeping prompt unchanged", async () => {
66
+ const cwd = await createTempWorkspace();
67
+ try {
68
+ await writeFile(path.join(cwd, "a.ts"), "123", "utf8");
69
+ await writeFile(path.join(cwd, "b.ts"), "const b = 2\n", "utf8");
70
+
71
+ const result = await enrichPromptWithMentions(
72
+ "Use @a.ts and @b.ts",
73
+ cwd,
74
+ { maxTotalBytes: 5, maxFiles: 2, maxFileBytes: 5 },
75
+ );
76
+
77
+ expect(result.mentions).toEqual(["a.ts", "b.ts"]);
78
+ expect(result.matchedFiles).toEqual(["a.ts"]);
79
+ expect(result.ignoredMentions).toEqual(["b.ts"]);
80
+ expect(result.prompt).toBe("Use @a.ts and @b.ts");
81
+ } finally {
82
+ await rm(cwd, { recursive: true, force: true });
83
+ }
84
+ });
85
+ });
@@ -0,0 +1,122 @@
1
+ import { stat } from "node:fs/promises";
2
+ import path from "node:path";
3
+ import { type FastFileIndexOptions, getFileIndex } from "./file-indexer";
4
+
5
+ const TRAILING_PUNCTUATION = /[),.:;!?`'"]+$/;
6
+ const LEADING_WRAPPERS = /^[(`'"]+/;
7
+
8
+ export interface MentionEnricherOptions extends FastFileIndexOptions {
9
+ maxFiles?: number;
10
+ maxFileBytes?: number;
11
+ maxTotalBytes?: number;
12
+ }
13
+
14
+ export interface MentionEnrichmentResult {
15
+ prompt: string;
16
+ mentions: string[];
17
+ matchedFiles: string[];
18
+ ignoredMentions: string[];
19
+ }
20
+
21
+ function extractMentionTokens(input: string): string[] {
22
+ const matches = input.matchAll(/(^|[\s])@([^\s]+)/g);
23
+ const mentions: string[] = [];
24
+ for (const match of matches) {
25
+ const token = (match[2] ?? "").trim();
26
+ if (token.length === 0) {
27
+ continue;
28
+ }
29
+ const normalized = token
30
+ .replace(LEADING_WRAPPERS, "")
31
+ .replace(TRAILING_PUNCTUATION, "");
32
+ if (normalized.length === 0 || normalized.includes("@")) {
33
+ continue;
34
+ }
35
+ mentions.push(normalized);
36
+ }
37
+ return Array.from(new Set(mentions));
38
+ }
39
+
40
+ function normalizeMentionPath(
41
+ mention: string,
42
+ cwd: string,
43
+ ): string | undefined {
44
+ const candidate = mention.replace(/\\/g, "/");
45
+ const maybeAbsolute = path.isAbsolute(candidate)
46
+ ? path.resolve(candidate)
47
+ : path.resolve(cwd, candidate);
48
+ const relative = path.relative(cwd, maybeAbsolute);
49
+ if (relative.startsWith("..") || path.isAbsolute(relative)) {
50
+ return undefined;
51
+ }
52
+ return relative.split(path.sep).join("/");
53
+ }
54
+
55
+ export async function enrichPromptWithMentions(
56
+ input: string,
57
+ cwd: string,
58
+ options: MentionEnricherOptions = {},
59
+ ): Promise<MentionEnrichmentResult> {
60
+ const mentions = extractMentionTokens(input);
61
+ if (mentions.length === 0) {
62
+ return {
63
+ prompt: input,
64
+ mentions: [],
65
+ matchedFiles: [],
66
+ ignoredMentions: [],
67
+ };
68
+ }
69
+
70
+ const maxFiles = options.maxFiles;
71
+ const maxFileBytes = options.maxFileBytes;
72
+ const maxTotalBytes = options.maxTotalBytes;
73
+ const fileList = await getFileIndex(cwd, { ttlMs: options.ttlMs });
74
+ const matched: string[] = [];
75
+ const ignored: string[] = [];
76
+ const attachments: Array<{ path: string; content: string }> = [];
77
+ let totalBytes = 0;
78
+
79
+ for (const mention of mentions) {
80
+ if (maxFiles && attachments.length >= maxFiles) {
81
+ ignored.push(mention);
82
+ continue;
83
+ }
84
+
85
+ const relativePath = normalizeMentionPath(mention, cwd);
86
+ if (!relativePath || !fileList.has(relativePath)) {
87
+ ignored.push(mention);
88
+ continue;
89
+ }
90
+
91
+ if (!maxFileBytes || !maxTotalBytes) {
92
+ matched.push(relativePath);
93
+ continue;
94
+ }
95
+
96
+ const absolutePath = path.join(cwd, relativePath);
97
+ try {
98
+ const fileStat = await stat(absolutePath);
99
+ if (!fileStat.isFile()) {
100
+ ignored.push(mention);
101
+ continue;
102
+ }
103
+ const nextBytes = totalBytes + maxFileBytes;
104
+ if (nextBytes > maxTotalBytes) {
105
+ ignored.push(mention);
106
+ continue;
107
+ }
108
+
109
+ totalBytes += nextBytes;
110
+ matched.push(relativePath);
111
+ } catch {
112
+ ignored.push(mention);
113
+ }
114
+ }
115
+
116
+ return {
117
+ prompt: input,
118
+ mentions,
119
+ matchedFiles: matched,
120
+ ignoredMentions: ignored,
121
+ };
122
+ }