prjct-cli 1.22.0 → 1.23.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 (448) hide show
  1. package/CHANGELOG.md +147 -0
  2. package/bin/prjct +30 -13
  3. package/dist/bin/prjct.mjs +917 -35845
  4. package/dist/bin/prjct.mjs.map +7 -0
  5. package/dist/cli/linear.mjs +16 -0
  6. package/dist/cli/linear.mjs.map +7 -0
  7. package/dist/templates.json +1 -0
  8. package/package.json +4 -5
  9. package/bin/prjct.ts +0 -342
  10. package/core/__tests__/agentic/analysis-injection.test.ts +0 -377
  11. package/core/__tests__/agentic/cache-eviction.test.ts +0 -294
  12. package/core/__tests__/agentic/command-context.test.ts +0 -281
  13. package/core/__tests__/agentic/command-executor.test.ts +0 -659
  14. package/core/__tests__/agentic/domain-classifier.test.ts +0 -330
  15. package/core/__tests__/agentic/injection-validator.test.ts +0 -255
  16. package/core/__tests__/agentic/memory-system.test.ts +0 -281
  17. package/core/__tests__/agentic/plan-mode.test.ts +0 -386
  18. package/core/__tests__/agentic/prompt-assembly.test.ts +0 -298
  19. package/core/__tests__/agentic/prompt-builder.test.ts +0 -243
  20. package/core/__tests__/agentic/response-validator.test.ts +0 -263
  21. package/core/__tests__/agentic/semantic-matching.test.ts +0 -131
  22. package/core/__tests__/agentic/smart-context.test.ts +0 -372
  23. package/core/__tests__/agentic/tech-normalizer.test.ts +0 -136
  24. package/core/__tests__/agentic/token-budget.test.ts +0 -294
  25. package/core/__tests__/ai-tools/formatters.test.ts +0 -476
  26. package/core/__tests__/domain/bm25.test.ts +0 -225
  27. package/core/__tests__/domain/change-propagator.test.ts +0 -100
  28. package/core/__tests__/domain/fibonacci.test.ts +0 -113
  29. package/core/__tests__/domain/file-hasher.test.ts +0 -146
  30. package/core/__tests__/domain/file-ranker.test.ts +0 -169
  31. package/core/__tests__/domain/git-cochange.test.ts +0 -121
  32. package/core/__tests__/domain/import-graph.test.ts +0 -156
  33. package/core/__tests__/domain/velocity.test.ts +0 -623
  34. package/core/__tests__/infrastructure/performance-tracker.test.ts +0 -328
  35. package/core/__tests__/schemas/model.test.ts +0 -272
  36. package/core/__tests__/services/dependency-validator.test.ts +0 -175
  37. package/core/__tests__/services/hierarchical-agent-resolver.test.ts +0 -359
  38. package/core/__tests__/services/nested-context-resolver.test.ts +0 -443
  39. package/core/__tests__/services/project-index.test.ts +0 -355
  40. package/core/__tests__/services/staleness-checker.test.ts +0 -204
  41. package/core/__tests__/storage/analysis-storage.test.ts +0 -641
  42. package/core/__tests__/storage/archive-storage.test.ts +0 -455
  43. package/core/__tests__/storage/safe-reader.test.ts +0 -262
  44. package/core/__tests__/storage/sqlite-migration.test.ts +0 -1016
  45. package/core/__tests__/storage/state-storage-feedback.test.ts +0 -463
  46. package/core/__tests__/storage/state-storage-history.test.ts +0 -469
  47. package/core/__tests__/storage/storage-manager.test.ts +0 -383
  48. package/core/__tests__/storage/subtask-handoff.test.ts +0 -237
  49. package/core/__tests__/types/fs.test.ts +0 -125
  50. package/core/__tests__/utils/date-helper.test.ts +0 -449
  51. package/core/__tests__/utils/output.test.ts +0 -278
  52. package/core/__tests__/utils/preserve-sections.test.ts +0 -216
  53. package/core/__tests__/utils/project-commands.test.ts +0 -71
  54. package/core/__tests__/utils/retry.test.ts +0 -381
  55. package/core/__tests__/workflow/state-machine.test.ts +0 -216
  56. package/core/agentic/agent-router.ts +0 -150
  57. package/core/agentic/anti-hallucination.ts +0 -141
  58. package/core/agentic/chain-of-thought.ts +0 -234
  59. package/core/agentic/command-classifier.ts +0 -141
  60. package/core/agentic/command-context.ts +0 -168
  61. package/core/agentic/command-executor.ts +0 -471
  62. package/core/agentic/context-builder.ts +0 -285
  63. package/core/agentic/domain-classifier.ts +0 -525
  64. package/core/agentic/environment-block.ts +0 -102
  65. package/core/agentic/ground-truth.ts +0 -706
  66. package/core/agentic/index.ts +0 -193
  67. package/core/agentic/injection-validator.ts +0 -208
  68. package/core/agentic/loop-detector.ts +0 -451
  69. package/core/agentic/memory-system.ts +0 -1547
  70. package/core/agentic/orchestrator-executor.ts +0 -579
  71. package/core/agentic/plan-mode.ts +0 -525
  72. package/core/agentic/prompt-builder.ts +0 -1069
  73. package/core/agentic/response-validator.ts +0 -98
  74. package/core/agentic/services.ts +0 -167
  75. package/core/agentic/skill-loader.ts +0 -106
  76. package/core/agentic/smart-context.ts +0 -393
  77. package/core/agentic/tech-normalizer.ts +0 -167
  78. package/core/agentic/template-executor.ts +0 -272
  79. package/core/agentic/template-loader.ts +0 -109
  80. package/core/agentic/token-budget.ts +0 -226
  81. package/core/agentic/tool-registry.ts +0 -146
  82. package/core/agents/index.ts +0 -28
  83. package/core/agents/performance.ts +0 -429
  84. package/core/ai-tools/formatters.ts +0 -341
  85. package/core/ai-tools/generator.ts +0 -144
  86. package/core/ai-tools/index.ts +0 -15
  87. package/core/ai-tools/registry.ts +0 -201
  88. package/core/bus/bus.ts +0 -314
  89. package/core/bus/index.ts +0 -8
  90. package/core/cli/linear.ts +0 -500
  91. package/core/cli/lint-meta-commentary.ts +0 -177
  92. package/core/cli/start.ts +0 -386
  93. package/core/commands/analysis.ts +0 -1274
  94. package/core/commands/analytics.ts +0 -342
  95. package/core/commands/base.ts +0 -118
  96. package/core/commands/cleanup.ts +0 -157
  97. package/core/commands/command-data.ts +0 -463
  98. package/core/commands/commands.ts +0 -306
  99. package/core/commands/context.ts +0 -238
  100. package/core/commands/design.ts +0 -77
  101. package/core/commands/index.ts +0 -19
  102. package/core/commands/maintenance.ts +0 -77
  103. package/core/commands/performance.ts +0 -114
  104. package/core/commands/planning.ts +0 -662
  105. package/core/commands/register.ts +0 -127
  106. package/core/commands/registry.ts +0 -444
  107. package/core/commands/setup.ts +0 -280
  108. package/core/commands/shipping.ts +0 -267
  109. package/core/commands/snapshots.ts +0 -297
  110. package/core/commands/uninstall.ts +0 -542
  111. package/core/commands/velocity.ts +0 -149
  112. package/core/commands/workflow.ts +0 -505
  113. package/core/config/command-context.config.json +0 -66
  114. package/core/constants/index.ts +0 -379
  115. package/core/context/generator.ts +0 -368
  116. package/core/context-tools/files-tool.ts +0 -577
  117. package/core/context-tools/imports-tool.ts +0 -400
  118. package/core/context-tools/index.ts +0 -434
  119. package/core/context-tools/recent-tool.ts +0 -301
  120. package/core/context-tools/signatures-tool.ts +0 -495
  121. package/core/context-tools/summary-tool.ts +0 -301
  122. package/core/context-tools/token-counter.ts +0 -273
  123. package/core/context-tools/types.ts +0 -253
  124. package/core/domain/agent-generator.ts +0 -186
  125. package/core/domain/agent-loader.ts +0 -419
  126. package/core/domain/analyzer.ts +0 -387
  127. package/core/domain/architecture-generator.ts +0 -108
  128. package/core/domain/bm25.ts +0 -525
  129. package/core/domain/change-propagator.ts +0 -162
  130. package/core/domain/context-estimator.ts +0 -175
  131. package/core/domain/fibonacci.ts +0 -128
  132. package/core/domain/file-hasher.ts +0 -296
  133. package/core/domain/file-ranker.ts +0 -151
  134. package/core/domain/git-cochange.ts +0 -250
  135. package/core/domain/import-graph.ts +0 -315
  136. package/core/domain/snapshot-manager.ts +0 -415
  137. package/core/domain/task-stack.ts +0 -578
  138. package/core/domain/velocity.ts +0 -470
  139. package/core/errors.ts +0 -335
  140. package/core/events/events.ts +0 -85
  141. package/core/events/index.ts +0 -8
  142. package/core/index.ts +0 -481
  143. package/core/infrastructure/agent-detector.ts +0 -135
  144. package/core/infrastructure/ai-provider.ts +0 -578
  145. package/core/infrastructure/author-detector.ts +0 -133
  146. package/core/infrastructure/capability-installer.ts +0 -76
  147. package/core/infrastructure/claude-agent.ts +0 -297
  148. package/core/infrastructure/command-installer.ts +0 -752
  149. package/core/infrastructure/config-manager.ts +0 -364
  150. package/core/infrastructure/editors-config.ts +0 -172
  151. package/core/infrastructure/path-manager.ts +0 -571
  152. package/core/infrastructure/performance-tracker.ts +0 -326
  153. package/core/infrastructure/permission-manager.ts +0 -289
  154. package/core/infrastructure/setup.ts +0 -1061
  155. package/core/infrastructure/update-checker.ts +0 -246
  156. package/core/integrations/issue-tracker/enricher.ts +0 -271
  157. package/core/integrations/issue-tracker/index.ts +0 -8
  158. package/core/integrations/issue-tracker/manager.ts +0 -286
  159. package/core/integrations/issue-tracker/types.ts +0 -310
  160. package/core/integrations/jira/cache.ts +0 -57
  161. package/core/integrations/jira/client.ts +0 -688
  162. package/core/integrations/jira/index.ts +0 -23
  163. package/core/integrations/jira/service.ts +0 -244
  164. package/core/integrations/linear/cache.ts +0 -68
  165. package/core/integrations/linear/client.ts +0 -436
  166. package/core/integrations/linear/index.ts +0 -20
  167. package/core/integrations/linear/service.ts +0 -260
  168. package/core/integrations/linear/sync.ts +0 -314
  169. package/core/outcomes/analyzer.ts +0 -286
  170. package/core/outcomes/index.ts +0 -34
  171. package/core/outcomes/recorder.ts +0 -195
  172. package/core/plugin/builtin/webhook.ts +0 -148
  173. package/core/plugin/hooks.ts +0 -315
  174. package/core/plugin/index.ts +0 -50
  175. package/core/plugin/loader.ts +0 -354
  176. package/core/plugin/registry.ts +0 -326
  177. package/core/schemas/agents.ts +0 -27
  178. package/core/schemas/analysis.ts +0 -530
  179. package/core/schemas/classification.ts +0 -91
  180. package/core/schemas/command-context.ts +0 -29
  181. package/core/schemas/enriched-task.ts +0 -291
  182. package/core/schemas/ideas.ts +0 -114
  183. package/core/schemas/index.ts +0 -53
  184. package/core/schemas/issues.ts +0 -159
  185. package/core/schemas/llm-output.ts +0 -170
  186. package/core/schemas/metrics.ts +0 -143
  187. package/core/schemas/model.ts +0 -153
  188. package/core/schemas/outcomes.ts +0 -487
  189. package/core/schemas/performance.ts +0 -128
  190. package/core/schemas/permissions.ts +0 -180
  191. package/core/schemas/prd.ts +0 -450
  192. package/core/schemas/project.ts +0 -57
  193. package/core/schemas/roadmap.ts +0 -322
  194. package/core/schemas/schemas.ts +0 -38
  195. package/core/schemas/shipped.ts +0 -109
  196. package/core/schemas/state.ts +0 -284
  197. package/core/schemas/velocity.ts +0 -103
  198. package/core/server/index.ts +0 -21
  199. package/core/server/routes-extended.ts +0 -566
  200. package/core/server/routes.ts +0 -176
  201. package/core/server/server.ts +0 -149
  202. package/core/server/sse.ts +0 -192
  203. package/core/services/agent-generator.ts +0 -385
  204. package/core/services/agent-service.ts +0 -168
  205. package/core/services/breakdown-service.ts +0 -124
  206. package/core/services/context-generator.ts +0 -445
  207. package/core/services/context-selector.ts +0 -429
  208. package/core/services/dependency-validator.ts +0 -318
  209. package/core/services/diff-generator.ts +0 -313
  210. package/core/services/doctor-service.ts +0 -423
  211. package/core/services/file-categorizer.ts +0 -448
  212. package/core/services/file-scorer.ts +0 -270
  213. package/core/services/git-analyzer.ts +0 -293
  214. package/core/services/hierarchical-agent-resolver.ts +0 -236
  215. package/core/services/hooks-service.ts +0 -685
  216. package/core/services/index.ts +0 -46
  217. package/core/services/local-state-generator.ts +0 -158
  218. package/core/services/memory-service.ts +0 -181
  219. package/core/services/nested-context-resolver.ts +0 -842
  220. package/core/services/project-index.ts +0 -911
  221. package/core/services/project-service.ts +0 -155
  222. package/core/services/session-tracker.ts +0 -287
  223. package/core/services/skill-installer.ts +0 -447
  224. package/core/services/skill-lock.ts +0 -132
  225. package/core/services/skill-service.ts +0 -306
  226. package/core/services/stack-detector.ts +0 -229
  227. package/core/services/staleness-checker.ts +0 -327
  228. package/core/services/sync-service.ts +0 -1515
  229. package/core/services/sync-verifier.ts +0 -253
  230. package/core/services/watch-service.ts +0 -312
  231. package/core/session/compaction.ts +0 -248
  232. package/core/session/index.ts +0 -35
  233. package/core/session/log-migration.ts +0 -88
  234. package/core/session/metrics.ts +0 -323
  235. package/core/session/session-log-manager.ts +0 -307
  236. package/core/session/task-session-manager.ts +0 -404
  237. package/core/session/utils.ts +0 -51
  238. package/core/storage/analysis-storage.ts +0 -373
  239. package/core/storage/archive-storage.ts +0 -205
  240. package/core/storage/database.ts +0 -575
  241. package/core/storage/ideas-storage.ts +0 -298
  242. package/core/storage/index-storage.ts +0 -523
  243. package/core/storage/index.ts +0 -79
  244. package/core/storage/metrics-storage.ts +0 -321
  245. package/core/storage/migrate-json.ts +0 -720
  246. package/core/storage/queue-storage.ts +0 -336
  247. package/core/storage/safe-reader.ts +0 -105
  248. package/core/storage/shipped-storage.ts +0 -253
  249. package/core/storage/state-storage.ts +0 -1035
  250. package/core/storage/storage-manager.ts +0 -205
  251. package/core/storage/storage.ts +0 -177
  252. package/core/storage/velocity-storage.ts +0 -149
  253. package/core/sync/auth-config.ts +0 -138
  254. package/core/sync/index.ts +0 -31
  255. package/core/sync/oauth-handler.ts +0 -143
  256. package/core/sync/sync-client.ts +0 -251
  257. package/core/sync/sync-manager.ts +0 -327
  258. package/core/tsconfig.json +0 -22
  259. package/core/types/agentic.ts +0 -760
  260. package/core/types/agents.ts +0 -150
  261. package/core/types/bus.ts +0 -193
  262. package/core/types/citations.ts +0 -22
  263. package/core/types/commands.ts +0 -399
  264. package/core/types/config.ts +0 -92
  265. package/core/types/core.ts +0 -96
  266. package/core/types/diff.ts +0 -41
  267. package/core/types/domain.ts +0 -71
  268. package/core/types/errors.ts +0 -111
  269. package/core/types/events.ts +0 -42
  270. package/core/types/fs.ts +0 -72
  271. package/core/types/index.ts +0 -510
  272. package/core/types/infrastructure.ts +0 -210
  273. package/core/types/integrations.ts +0 -31
  274. package/core/types/jira.ts +0 -51
  275. package/core/types/logger.ts +0 -17
  276. package/core/types/memory.ts +0 -313
  277. package/core/types/outcomes.ts +0 -190
  278. package/core/types/output.ts +0 -47
  279. package/core/types/plugin.ts +0 -25
  280. package/core/types/project-sync.ts +0 -129
  281. package/core/types/provider.ts +0 -163
  282. package/core/types/server.ts +0 -71
  283. package/core/types/services.ts +0 -84
  284. package/core/types/session.ts +0 -135
  285. package/core/types/stack.ts +0 -19
  286. package/core/types/storage.ts +0 -318
  287. package/core/types/sync-verifier.ts +0 -33
  288. package/core/types/sync.ts +0 -121
  289. package/core/types/task.ts +0 -72
  290. package/core/types/template.ts +0 -24
  291. package/core/types/utils.ts +0 -92
  292. package/core/types/workflow.ts +0 -23
  293. package/core/utils/agent-stream.ts +0 -140
  294. package/core/utils/animations.ts +0 -251
  295. package/core/utils/branding.ts +0 -88
  296. package/core/utils/cache.ts +0 -187
  297. package/core/utils/citations.ts +0 -39
  298. package/core/utils/collection-filters.ts +0 -209
  299. package/core/utils/date-helper.ts +0 -176
  300. package/core/utils/error-messages.ts +0 -38
  301. package/core/utils/file-helper.ts +0 -277
  302. package/core/utils/fs-helpers.ts +0 -14
  303. package/core/utils/help.ts +0 -314
  304. package/core/utils/jsonl-helper.ts +0 -290
  305. package/core/utils/keychain.ts +0 -127
  306. package/core/utils/logger.ts +0 -77
  307. package/core/utils/markdown-builder.ts +0 -280
  308. package/core/utils/next-steps.ts +0 -95
  309. package/core/utils/output.ts +0 -403
  310. package/core/utils/preserve-sections.ts +0 -218
  311. package/core/utils/project-commands.ts +0 -126
  312. package/core/utils/project-credentials.ts +0 -143
  313. package/core/utils/provider-cache.ts +0 -49
  314. package/core/utils/retry.ts +0 -318
  315. package/core/utils/runtime.ts +0 -108
  316. package/core/utils/session-helper.ts +0 -278
  317. package/core/utils/subtask-table.ts +0 -227
  318. package/core/utils/version.ts +0 -128
  319. package/core/wizard/index.ts +0 -13
  320. package/core/wizard/onboarding.ts +0 -633
  321. package/core/workflow/index.ts +0 -7
  322. package/core/workflow/state-machine.ts +0 -198
  323. package/core/workflow/workflow-preferences.ts +0 -294
  324. package/dist/core/infrastructure/command-installer.js +0 -1141
  325. package/dist/core/infrastructure/editors-config.js +0 -177
  326. package/dist/core/infrastructure/setup.js +0 -2244
  327. package/dist/core/utils/version.js +0 -141
  328. package/templates/agentic/agent-routing.md +0 -45
  329. package/templates/agentic/agents/uxui.md +0 -63
  330. package/templates/agentic/checklist-routing.md +0 -98
  331. package/templates/agentic/orchestrator.md +0 -68
  332. package/templates/agentic/task-fragmentation.md +0 -89
  333. package/templates/agents/AGENTS.md +0 -68
  334. package/templates/analysis/analyze.md +0 -84
  335. package/templates/analysis/patterns.md +0 -60
  336. package/templates/antigravity/SKILL.md +0 -39
  337. package/templates/architect/discovery.md +0 -67
  338. package/templates/architect/phases.md +0 -59
  339. package/templates/checklists/architecture.md +0 -28
  340. package/templates/checklists/code-quality.md +0 -28
  341. package/templates/checklists/data.md +0 -33
  342. package/templates/checklists/documentation.md +0 -33
  343. package/templates/checklists/infrastructure.md +0 -33
  344. package/templates/checklists/performance.md +0 -33
  345. package/templates/checklists/security.md +0 -33
  346. package/templates/checklists/testing.md +0 -33
  347. package/templates/checklists/ux-ui.md +0 -37
  348. package/templates/commands/analyze.md +0 -56
  349. package/templates/commands/auth.md +0 -234
  350. package/templates/commands/bug.md +0 -163
  351. package/templates/commands/cleanup.md +0 -19
  352. package/templates/commands/dash.md +0 -99
  353. package/templates/commands/design.md +0 -15
  354. package/templates/commands/done.md +0 -291
  355. package/templates/commands/enrich.md +0 -174
  356. package/templates/commands/git.md +0 -295
  357. package/templates/commands/history.md +0 -389
  358. package/templates/commands/idea.md +0 -88
  359. package/templates/commands/impact.md +0 -864
  360. package/templates/commands/init.md +0 -54
  361. package/templates/commands/jira.md +0 -278
  362. package/templates/commands/linear.md +0 -288
  363. package/templates/commands/merge.md +0 -206
  364. package/templates/commands/next.md +0 -80
  365. package/templates/commands/p.md +0 -67
  366. package/templates/commands/p.toml +0 -37
  367. package/templates/commands/pause.md +0 -136
  368. package/templates/commands/plan.md +0 -696
  369. package/templates/commands/prd.md +0 -356
  370. package/templates/commands/resume.md +0 -171
  371. package/templates/commands/review.md +0 -276
  372. package/templates/commands/serve.md +0 -118
  373. package/templates/commands/setup.md +0 -91
  374. package/templates/commands/ship.md +0 -475
  375. package/templates/commands/skill.md +0 -259
  376. package/templates/commands/spec.md +0 -218
  377. package/templates/commands/status.md +0 -207
  378. package/templates/commands/sync.md +0 -104
  379. package/templates/commands/task.md +0 -312
  380. package/templates/commands/test.md +0 -93
  381. package/templates/commands/update.md +0 -63
  382. package/templates/commands/verify.md +0 -204
  383. package/templates/commands/workflow.md +0 -150
  384. package/templates/config/skill-mappings.json +0 -82
  385. package/templates/context/dashboard.md +0 -256
  386. package/templates/context/roadmap.md +0 -221
  387. package/templates/cursor/commands/bug.md +0 -8
  388. package/templates/cursor/commands/done.md +0 -4
  389. package/templates/cursor/commands/pause.md +0 -6
  390. package/templates/cursor/commands/resume.md +0 -4
  391. package/templates/cursor/commands/ship.md +0 -8
  392. package/templates/cursor/commands/sync.md +0 -4
  393. package/templates/cursor/commands/task.md +0 -8
  394. package/templates/cursor/p.md +0 -29
  395. package/templates/cursor/router.mdc +0 -28
  396. package/templates/design/api.md +0 -95
  397. package/templates/design/architecture.md +0 -77
  398. package/templates/design/component.md +0 -89
  399. package/templates/design/database.md +0 -78
  400. package/templates/design/flow.md +0 -94
  401. package/templates/global/ANTIGRAVITY.md +0 -254
  402. package/templates/global/CLAUDE.md +0 -497
  403. package/templates/global/CURSOR.mdc +0 -266
  404. package/templates/global/GEMINI.md +0 -293
  405. package/templates/global/STORAGE-SPEC.md +0 -391
  406. package/templates/global/WINDSURF.md +0 -266
  407. package/templates/global/modules/CLAUDE-commands.md +0 -70
  408. package/templates/global/modules/CLAUDE-core.md +0 -105
  409. package/templates/global/modules/CLAUDE-git.md +0 -50
  410. package/templates/global/modules/CLAUDE-intelligence.md +0 -92
  411. package/templates/global/modules/CLAUDE-storage.md +0 -50
  412. package/templates/global/modules/module-config.json +0 -36
  413. package/templates/mcp-config.json +0 -19
  414. package/templates/permissions/default.jsonc +0 -60
  415. package/templates/permissions/permissive.jsonc +0 -49
  416. package/templates/permissions/strict.jsonc +0 -58
  417. package/templates/planning-methodology.md +0 -195
  418. package/templates/skills/code-review.md +0 -47
  419. package/templates/skills/debug.md +0 -61
  420. package/templates/skills/refactor.md +0 -47
  421. package/templates/subagents/agent-base.md +0 -20
  422. package/templates/subagents/domain/backend.md +0 -109
  423. package/templates/subagents/domain/database.md +0 -121
  424. package/templates/subagents/domain/devops.md +0 -152
  425. package/templates/subagents/domain/frontend.md +0 -103
  426. package/templates/subagents/domain/testing.md +0 -169
  427. package/templates/subagents/pm-expert.md +0 -366
  428. package/templates/subagents/workflow/chief-architect.md +0 -657
  429. package/templates/subagents/workflow/prjct-planner.md +0 -159
  430. package/templates/subagents/workflow/prjct-shipper.md +0 -188
  431. package/templates/subagents/workflow/prjct-workflow.md +0 -98
  432. package/templates/tools/bash.txt +0 -22
  433. package/templates/tools/edit.txt +0 -18
  434. package/templates/tools/glob.txt +0 -19
  435. package/templates/tools/grep.txt +0 -21
  436. package/templates/tools/read.txt +0 -14
  437. package/templates/tools/task.txt +0 -20
  438. package/templates/tools/webfetch.txt +0 -16
  439. package/templates/tools/websearch.txt +0 -18
  440. package/templates/tools/write.txt +0 -17
  441. package/templates/windsurf/router.md +0 -28
  442. package/templates/windsurf/workflows/bug.md +0 -8
  443. package/templates/windsurf/workflows/done.md +0 -4
  444. package/templates/windsurf/workflows/pause.md +0 -4
  445. package/templates/windsurf/workflows/resume.md +0 -4
  446. package/templates/windsurf/workflows/ship.md +0 -8
  447. package/templates/windsurf/workflows/sync.md +0 -4
  448. package/templates/windsurf/workflows/task.md +0 -8
@@ -1,842 +0,0 @@
1
- /**
2
- * NestedContextResolver - Discovers and resolves nested PRJCT.md and AGENTS.md files in monorepos
3
- *
4
- * Responsible for:
5
- * - Finding all PRJCT.md and AGENTS.md files in a monorepo
6
- * - Building a hierarchy of context (root → packages)
7
- * - Resolving inheritance between parent and child contexts/agents
8
- *
9
- * Pattern from OpenAI Codex AGENTS.md: "Scope = directory tree. Deeper files take precedence."
10
- */
11
-
12
- import fs from 'node:fs/promises'
13
- import path from 'node:path'
14
- import pathManager, {
15
- type MonorepoInfo,
16
- type MonorepoPackage,
17
- } from '../infrastructure/path-manager'
18
- import * as fileHelper from '../utils/file-helper'
19
-
20
- // ============================================================================
21
- // TYPES
22
- // ============================================================================
23
-
24
- export interface NestedContext {
25
- /** Absolute path to the PRJCT.md file */
26
- path: string
27
- /** Relative path from monorepo root */
28
- relativePath: string
29
- /** Depth in the directory tree (0 = root) */
30
- depth: number
31
- /** Parent context (null for root) */
32
- parent: NestedContext | null
33
- /** Child contexts */
34
- children: NestedContext[]
35
- /** Raw content of the PRJCT.md file */
36
- content: string
37
- /** Parsed sections from the PRJCT.md */
38
- sections: ContextSection[]
39
- /** Associated package info (if in a monorepo package) */
40
- package: MonorepoPackage | null
41
- }
42
-
43
- export interface ContextSection {
44
- /** Section name (e.g., "Rules", "Patterns", "Stack") */
45
- name: string
46
- /** Section content */
47
- content: string
48
- /** Whether this section should override parent */
49
- override: boolean
50
- }
51
-
52
- export interface ResolvedContext {
53
- /** The final merged content */
54
- content: string
55
- /** Sources that contributed to this context (from root to leaf) */
56
- sources: string[]
57
- /** Sections that were overridden */
58
- overrides: string[]
59
- }
60
-
61
- // ============================================================================
62
- // AGENTS.md TYPES
63
- // ============================================================================
64
-
65
- export interface AgentDefinition {
66
- /** Agent name (e.g., "frontend", "backend", "database") */
67
- name: string
68
- /** Description of what this agent handles */
69
- description: string
70
- /** Domain this agent specializes in */
71
- domain?: string
72
- /** Trigger phrases that activate this agent */
73
- triggers?: string[]
74
- /** Rules/guidelines for this agent */
75
- rules?: string[]
76
- /** Code patterns this agent follows */
77
- patterns?: string[]
78
- /** Example interactions */
79
- examples?: string[]
80
- /** Whether this agent overrides parent definition */
81
- override?: boolean
82
- }
83
-
84
- export interface NestedAgents {
85
- /** Absolute path to the AGENTS.md file */
86
- path: string
87
- /** Relative path from monorepo root */
88
- relativePath: string
89
- /** Depth in the directory tree (0 = root) */
90
- depth: number
91
- /** Parent agents file (null for root) */
92
- parent: NestedAgents | null
93
- /** Child agents files */
94
- children: NestedAgents[]
95
- /** Raw content of the AGENTS.md file */
96
- content: string
97
- /** Parsed agent definitions */
98
- agents: AgentDefinition[]
99
- /** Associated package info (if in a monorepo package) */
100
- package: MonorepoPackage | null
101
- }
102
-
103
- export interface ResolvedAgents {
104
- /** The merged agent definitions (deeper overrides shallower) */
105
- agents: AgentDefinition[]
106
- /** Sources that contributed to these agents (from root to leaf) */
107
- sources: string[]
108
- /** Agents that were overridden */
109
- overrides: string[]
110
- }
111
-
112
- // ============================================================================
113
- // NESTED CONTEXT RESOLVER
114
- // ============================================================================
115
-
116
- export class NestedContextResolver {
117
- private rootPath: string
118
- private monoInfo: MonorepoInfo | null = null
119
-
120
- constructor(rootPath: string) {
121
- this.rootPath = path.resolve(rootPath)
122
- }
123
-
124
- /**
125
- * Initialize the resolver with monorepo detection
126
- */
127
- async initialize(): Promise<void> {
128
- this.monoInfo = await pathManager.detectMonorepo(this.rootPath)
129
- }
130
-
131
- /**
132
- * Discover all PRJCT.md files in the project/monorepo
133
- */
134
- async discoverContextFiles(): Promise<NestedContext[]> {
135
- const contexts: NestedContext[] = []
136
-
137
- // Always check root
138
- const rootPrjctPath = path.join(this.rootPath, 'PRJCT.md')
139
- if (await fileHelper.fileExists(rootPrjctPath)) {
140
- const rootContext = await this.loadContext(rootPrjctPath, null)
141
- contexts.push(rootContext)
142
- }
143
-
144
- // If monorepo, check each package
145
- if (this.monoInfo?.isMonorepo) {
146
- for (const pkg of this.monoInfo.packages) {
147
- const pkgPrjctPath = path.join(pkg.path, 'PRJCT.md')
148
- if (await fileHelper.fileExists(pkgPrjctPath)) {
149
- const parentContext = contexts.find((c) => c.depth === 0) || null
150
- const pkgContext = await this.loadContext(pkgPrjctPath, parentContext, pkg)
151
- contexts.push(pkgContext)
152
-
153
- if (parentContext) {
154
- parentContext.children.push(pkgContext)
155
- }
156
- }
157
- }
158
- }
159
-
160
- // Also scan for any PRJCT.md in subdirectories (non-package)
161
- const additionalContexts = await this.scanForNestedContexts(this.rootPath, contexts)
162
- contexts.push(...additionalContexts)
163
-
164
- return contexts
165
- }
166
-
167
- /**
168
- * Load a single PRJCT.md file into a NestedContext
169
- */
170
- private async loadContext(
171
- filePath: string,
172
- parent: NestedContext | null,
173
- pkg: MonorepoPackage | null = null
174
- ): Promise<NestedContext> {
175
- const content = await fs.readFile(filePath, 'utf-8')
176
- const relativePath = path.relative(this.rootPath, filePath)
177
- const depth = relativePath.split(path.sep).length - 1
178
-
179
- return {
180
- path: filePath,
181
- relativePath,
182
- depth,
183
- parent,
184
- children: [],
185
- content,
186
- sections: this.parseSections(content),
187
- package: pkg,
188
- }
189
- }
190
-
191
- /**
192
- * Parse PRJCT.md content into sections
193
- */
194
- private parseSections(content: string): ContextSection[] {
195
- const sections: ContextSection[] = []
196
- const lines = content.split('\n')
197
-
198
- let currentSection: ContextSection | null = null
199
- let currentContent: string[] = []
200
-
201
- for (const line of lines) {
202
- // Check for section header (## or ###)
203
- const headerMatch = line.match(/^##\s+(.+)$/)
204
- if (headerMatch) {
205
- // Save previous section
206
- if (currentSection) {
207
- currentSection.content = currentContent.join('\n').trim()
208
- sections.push(currentSection)
209
- }
210
-
211
- // Check for override marker
212
- const sectionName = headerMatch[1]
213
- const override = sectionName.includes('@override') || sectionName.includes('(override)')
214
-
215
- currentSection = {
216
- name: sectionName.replace(/@override|\(override\)/gi, '').trim(),
217
- content: '',
218
- override,
219
- }
220
- currentContent = []
221
- } else if (currentSection) {
222
- currentContent.push(line)
223
- }
224
- }
225
-
226
- // Save last section
227
- if (currentSection) {
228
- currentSection.content = currentContent.join('\n').trim()
229
- sections.push(currentSection)
230
- }
231
-
232
- return sections
233
- }
234
-
235
- /**
236
- * Scan for additional nested PRJCT.md files not in packages
237
- */
238
- private async scanForNestedContexts(
239
- dir: string,
240
- existing: NestedContext[]
241
- ): Promise<NestedContext[]> {
242
- const found: NestedContext[] = []
243
- const existingPaths = new Set(existing.map((c) => c.path))
244
-
245
- const scan = async (currentDir: string, depth: number): Promise<void> => {
246
- // Limit depth to avoid infinite recursion
247
- if (depth > 5) return
248
-
249
- try {
250
- const entries = await fs.readdir(currentDir, { withFileTypes: true })
251
-
252
- for (const entry of entries) {
253
- // Skip common non-project directories
254
- if (
255
- entry.name.startsWith('.') ||
256
- entry.name === 'node_modules' ||
257
- entry.name === 'dist' ||
258
- entry.name === 'build' ||
259
- entry.name === 'coverage'
260
- ) {
261
- continue
262
- }
263
-
264
- if (entry.isDirectory()) {
265
- const subDir = path.join(currentDir, entry.name)
266
- const prjctPath = path.join(subDir, 'PRJCT.md')
267
-
268
- if ((await fileHelper.fileExists(prjctPath)) && !existingPaths.has(prjctPath)) {
269
- // Find parent context (closest ancestor with PRJCT.md)
270
- const parent = this.findParentContext(prjctPath, existing.concat(found))
271
- const context = await this.loadContext(prjctPath, parent)
272
- found.push(context)
273
- existingPaths.add(prjctPath)
274
-
275
- if (parent) {
276
- parent.children.push(context)
277
- }
278
- }
279
-
280
- // Continue scanning subdirectories
281
- await scan(subDir, depth + 1)
282
- }
283
- }
284
- } catch {
285
- // Permission denied or other error, skip
286
- }
287
- }
288
-
289
- await scan(dir, 0)
290
- return found
291
- }
292
-
293
- /**
294
- * Find the parent context for a given path
295
- */
296
- private findParentContext(filePath: string, contexts: NestedContext[]): NestedContext | null {
297
- const fileDir = path.dirname(filePath)
298
-
299
- // Sort by depth descending to find closest parent
300
- const sorted = [...contexts].sort((a, b) => b.depth - a.depth)
301
-
302
- for (const ctx of sorted) {
303
- const ctxDir = path.dirname(ctx.path)
304
- if (fileDir.startsWith(ctxDir) && fileDir !== ctxDir) {
305
- return ctx
306
- }
307
- }
308
-
309
- return null
310
- }
311
-
312
- /**
313
- * Resolve context for a specific path by merging parent contexts
314
- * Deeper files take precedence (can override parent sections)
315
- */
316
- async resolveContextForPath(targetPath: string): Promise<ResolvedContext> {
317
- const contexts = await this.discoverContextFiles()
318
-
319
- // Find the most specific context for this path
320
- const targetDir = path.resolve(targetPath)
321
- let bestMatch: NestedContext | null = null
322
-
323
- for (const ctx of contexts) {
324
- const ctxDir = path.dirname(ctx.path)
325
- if (targetDir.startsWith(ctxDir)) {
326
- if (!bestMatch || ctx.depth > bestMatch.depth) {
327
- bestMatch = ctx
328
- }
329
- }
330
- }
331
-
332
- if (!bestMatch) {
333
- return {
334
- content: '',
335
- sources: [],
336
- overrides: [],
337
- }
338
- }
339
-
340
- // Build inheritance chain (from root to leaf)
341
- const chain: NestedContext[] = []
342
- let current: NestedContext | null = bestMatch
343
- while (current) {
344
- chain.unshift(current)
345
- current = current.parent
346
- }
347
-
348
- // Merge sections following inheritance
349
- return this.mergeContextChain(chain)
350
- }
351
-
352
- /**
353
- * Merge a chain of contexts following inheritance rules
354
- */
355
- private mergeContextChain(chain: NestedContext[]): ResolvedContext {
356
- const mergedSections = new Map<string, string>()
357
- const sources: string[] = []
358
- const overrides: string[] = []
359
-
360
- for (const ctx of chain) {
361
- sources.push(ctx.relativePath)
362
-
363
- for (const section of ctx.sections) {
364
- if (section.override || !mergedSections.has(section.name)) {
365
- mergedSections.set(section.name, section.content)
366
- if (section.override) {
367
- overrides.push(`${ctx.relativePath}:${section.name}`)
368
- }
369
- } else {
370
- // Append to existing section
371
- const existing = mergedSections.get(section.name) || ''
372
- mergedSections.set(section.name, `${existing}\n\n${section.content}`)
373
- }
374
- }
375
- }
376
-
377
- // Rebuild content from merged sections
378
- const parts: string[] = []
379
- for (const [name, content] of mergedSections) {
380
- parts.push(`## ${name}\n\n${content}`)
381
- }
382
-
383
- return {
384
- content: parts.join('\n\n---\n\n'),
385
- sources,
386
- overrides,
387
- }
388
- }
389
-
390
- /**
391
- * Get context for a specific monorepo package
392
- */
393
- async getPackageContext(packageName: string): Promise<ResolvedContext | null> {
394
- if (!this.monoInfo?.isMonorepo) {
395
- return null
396
- }
397
-
398
- const pkg = this.monoInfo.packages.find((p) => p.name === packageName)
399
- if (!pkg) {
400
- return null
401
- }
402
-
403
- return this.resolveContextForPath(pkg.path)
404
- }
405
-
406
- /**
407
- * Get all package contexts in the monorepo
408
- */
409
- async getAllPackageContexts(): Promise<Map<string, ResolvedContext>> {
410
- const results = new Map<string, ResolvedContext>()
411
-
412
- if (!this.monoInfo?.isMonorepo) {
413
- return results
414
- }
415
-
416
- for (const pkg of this.monoInfo.packages) {
417
- const ctx = await this.resolveContextForPath(pkg.path)
418
- results.set(pkg.name, ctx)
419
- }
420
-
421
- return results
422
- }
423
-
424
- // ==========================================================================
425
- // AGENTS.md DISCOVERY AND RESOLUTION
426
- // ==========================================================================
427
-
428
- /**
429
- * Discover all AGENTS.md files in the project/monorepo
430
- */
431
- async discoverAgentFiles(): Promise<NestedAgents[]> {
432
- const agentFiles: NestedAgents[] = []
433
-
434
- // Always check root
435
- const rootAgentsPath = path.join(this.rootPath, 'AGENTS.md')
436
- if (await fileHelper.fileExists(rootAgentsPath)) {
437
- const rootAgents = await this.loadAgents(rootAgentsPath, null)
438
- agentFiles.push(rootAgents)
439
- }
440
-
441
- // If monorepo, check each package
442
- if (this.monoInfo?.isMonorepo) {
443
- for (const pkg of this.monoInfo.packages) {
444
- const pkgAgentsPath = path.join(pkg.path, 'AGENTS.md')
445
- if (await fileHelper.fileExists(pkgAgentsPath)) {
446
- const parentAgents = agentFiles.find((a) => a.depth === 0) || null
447
- const pkgAgents = await this.loadAgents(pkgAgentsPath, parentAgents, pkg)
448
- agentFiles.push(pkgAgents)
449
-
450
- if (parentAgents) {
451
- parentAgents.children.push(pkgAgents)
452
- }
453
- }
454
- }
455
- }
456
-
457
- // Also scan for any AGENTS.md in subdirectories (non-package)
458
- const additionalAgents = await this.scanForNestedAgents(this.rootPath, agentFiles)
459
- agentFiles.push(...additionalAgents)
460
-
461
- return agentFiles
462
- }
463
-
464
- /**
465
- * Load a single AGENTS.md file into a NestedAgents structure
466
- */
467
- private async loadAgents(
468
- filePath: string,
469
- parent: NestedAgents | null,
470
- pkg: MonorepoPackage | null = null
471
- ): Promise<NestedAgents> {
472
- const content = await fs.readFile(filePath, 'utf-8')
473
- const relativePath = path.relative(this.rootPath, filePath)
474
- const depth = relativePath.split(path.sep).length - 1
475
-
476
- return {
477
- path: filePath,
478
- relativePath,
479
- depth,
480
- parent,
481
- children: [],
482
- content,
483
- agents: this.parseAgents(content),
484
- package: pkg,
485
- }
486
- }
487
-
488
- /**
489
- * Parse AGENTS.md content into agent definitions
490
- *
491
- * Expected format:
492
- * ## AgentName @override?
493
- *
494
- * Description text here.
495
- *
496
- * ### Triggers
497
- * - trigger phrase 1
498
- * - trigger phrase 2
499
- *
500
- * ### Rules
501
- * - rule 1
502
- * - rule 2
503
- *
504
- * ### Patterns
505
- * ```typescript
506
- * // code pattern
507
- * ```
508
- */
509
- private parseAgents(content: string): AgentDefinition[] {
510
- const agents: AgentDefinition[] = []
511
- const lines = content.split('\n')
512
-
513
- let currentAgent: AgentDefinition | null = null
514
- let currentSubsection: string | null = null
515
- let currentContent: string[] = []
516
-
517
- const saveCurrentContent = () => {
518
- if (!currentAgent) return
519
-
520
- if (currentSubsection) {
521
- const contentStr = currentContent.join('\n').trim()
522
- switch (currentSubsection.toLowerCase()) {
523
- case 'triggers':
524
- currentAgent.triggers = this.parseListItems(contentStr)
525
- break
526
- case 'rules':
527
- currentAgent.rules = this.parseListItems(contentStr)
528
- break
529
- case 'patterns':
530
- currentAgent.patterns = this.parseCodeBlocks(contentStr)
531
- break
532
- case 'examples':
533
- currentAgent.examples = this.parseListItems(contentStr)
534
- break
535
- case 'domain':
536
- currentAgent.domain = contentStr
537
- break
538
- }
539
- } else {
540
- // This is the description
541
- const desc = currentContent.join('\n').trim()
542
- if (desc && !currentAgent.description) {
543
- currentAgent.description = desc
544
- }
545
- }
546
- currentContent = []
547
- }
548
-
549
- for (const line of lines) {
550
- // Check for agent header (## AgentName)
551
- const agentMatch = line.match(/^##\s+([^#].+)$/)
552
- if (agentMatch) {
553
- // Save previous agent
554
- if (currentAgent) {
555
- saveCurrentContent()
556
- agents.push(currentAgent)
557
- }
558
-
559
- // Check for override marker
560
- const agentName = agentMatch[1]
561
- const override = agentName.includes('@override') || agentName.includes('(override)')
562
-
563
- currentAgent = {
564
- name: agentName.replace(/@override|\(override\)/gi, '').trim(),
565
- description: '',
566
- override,
567
- }
568
- currentSubsection = null
569
- currentContent = []
570
- continue
571
- }
572
-
573
- // Check for subsection header (### Triggers, ### Rules, etc.)
574
- const subsectionMatch = line.match(/^###\s+(.+)$/)
575
- if (subsectionMatch && currentAgent) {
576
- saveCurrentContent()
577
- currentSubsection = subsectionMatch[1].trim()
578
- currentContent = []
579
- continue
580
- }
581
-
582
- // Accumulate content
583
- if (currentAgent) {
584
- currentContent.push(line)
585
- }
586
- }
587
-
588
- // Save last agent
589
- if (currentAgent) {
590
- saveCurrentContent()
591
- agents.push(currentAgent)
592
- }
593
-
594
- return agents
595
- }
596
-
597
- /**
598
- * Parse list items from content (lines starting with - or *)
599
- */
600
- private parseListItems(content: string): string[] {
601
- return content
602
- .split('\n')
603
- .filter((line) => line.match(/^\s*[-*]\s+/))
604
- .map((line) => line.replace(/^\s*[-*]\s+/, '').trim())
605
- .filter((item) => item.length > 0)
606
- }
607
-
608
- /**
609
- * Parse code blocks from content
610
- */
611
- private parseCodeBlocks(content: string): string[] {
612
- const blocks: string[] = []
613
- const codeBlockRegex = /```[\w]*\n([\s\S]*?)```/g
614
- let match: RegExpExecArray | null
615
-
616
- while ((match = codeBlockRegex.exec(content)) !== null) {
617
- blocks.push(match[1].trim())
618
- }
619
-
620
- // If no code blocks, treat entire content as a pattern
621
- if (blocks.length === 0 && content.trim()) {
622
- blocks.push(content.trim())
623
- }
624
-
625
- return blocks
626
- }
627
-
628
- /**
629
- * Scan for additional nested AGENTS.md files not in packages
630
- */
631
- private async scanForNestedAgents(
632
- dir: string,
633
- existing: NestedAgents[]
634
- ): Promise<NestedAgents[]> {
635
- const found: NestedAgents[] = []
636
- const existingPaths = new Set(existing.map((a) => a.path))
637
-
638
- const scan = async (currentDir: string, depth: number): Promise<void> => {
639
- // Limit depth to avoid infinite recursion
640
- if (depth > 5) return
641
-
642
- try {
643
- const entries = await fs.readdir(currentDir, { withFileTypes: true })
644
-
645
- for (const entry of entries) {
646
- // Skip common non-project directories
647
- if (
648
- entry.name.startsWith('.') ||
649
- entry.name === 'node_modules' ||
650
- entry.name === 'dist' ||
651
- entry.name === 'build' ||
652
- entry.name === 'coverage'
653
- ) {
654
- continue
655
- }
656
-
657
- if (entry.isDirectory()) {
658
- const subDir = path.join(currentDir, entry.name)
659
- const agentsPath = path.join(subDir, 'AGENTS.md')
660
-
661
- if ((await fileHelper.fileExists(agentsPath)) && !existingPaths.has(agentsPath)) {
662
- // Find parent agents file (closest ancestor with AGENTS.md)
663
- const parent = this.findParentAgents(agentsPath, existing.concat(found))
664
- const agents = await this.loadAgents(agentsPath, parent)
665
- found.push(agents)
666
- existingPaths.add(agentsPath)
667
-
668
- if (parent) {
669
- parent.children.push(agents)
670
- }
671
- }
672
-
673
- // Continue scanning subdirectories
674
- await scan(subDir, depth + 1)
675
- }
676
- }
677
- } catch {
678
- // Permission denied or other error, skip
679
- }
680
- }
681
-
682
- await scan(dir, 0)
683
- return found
684
- }
685
-
686
- /**
687
- * Find the parent agents file for a given path
688
- */
689
- private findParentAgents(filePath: string, agentFiles: NestedAgents[]): NestedAgents | null {
690
- const fileDir = path.dirname(filePath)
691
-
692
- // Sort by depth descending to find closest parent
693
- const sorted = [...agentFiles].sort((a, b) => b.depth - a.depth)
694
-
695
- for (const agents of sorted) {
696
- const agentsDir = path.dirname(agents.path)
697
- if (fileDir.startsWith(agentsDir) && fileDir !== agentsDir) {
698
- return agents
699
- }
700
- }
701
-
702
- return null
703
- }
704
-
705
- /**
706
- * Resolve agents for a specific path by merging parent agent definitions
707
- * Deeper files take precedence (can override parent agents)
708
- */
709
- async resolveAgentsForPath(targetPath: string): Promise<ResolvedAgents> {
710
- const agentFiles = await this.discoverAgentFiles()
711
-
712
- // Find the most specific agents file for this path
713
- const targetDir = path.resolve(targetPath)
714
- let bestMatch: NestedAgents | null = null
715
-
716
- for (const agents of agentFiles) {
717
- const agentsDir = path.dirname(agents.path)
718
- if (targetDir.startsWith(agentsDir)) {
719
- if (!bestMatch || agents.depth > bestMatch.depth) {
720
- bestMatch = agents
721
- }
722
- }
723
- }
724
-
725
- if (!bestMatch) {
726
- return {
727
- agents: [],
728
- sources: [],
729
- overrides: [],
730
- }
731
- }
732
-
733
- // Build inheritance chain (from root to leaf)
734
- const chain: NestedAgents[] = []
735
- let current: NestedAgents | null = bestMatch
736
- while (current) {
737
- chain.unshift(current)
738
- current = current.parent
739
- }
740
-
741
- // Merge agents following inheritance
742
- return this.mergeAgentsChain(chain)
743
- }
744
-
745
- /**
746
- * Merge a chain of agent files following inheritance rules
747
- */
748
- private mergeAgentsChain(chain: NestedAgents[]): ResolvedAgents {
749
- const mergedAgents = new Map<string, AgentDefinition>()
750
- const sources: string[] = []
751
- const overrides: string[] = []
752
-
753
- for (const agentFile of chain) {
754
- sources.push(agentFile.relativePath)
755
-
756
- for (const agent of agentFile.agents) {
757
- const existing = mergedAgents.get(agent.name)
758
-
759
- if (agent.override || !existing) {
760
- // Override or new agent
761
- mergedAgents.set(agent.name, { ...agent })
762
- if (agent.override && existing) {
763
- overrides.push(`${agentFile.relativePath}:${agent.name}`)
764
- }
765
- } else {
766
- // Merge with existing agent
767
- const merged: AgentDefinition = { ...existing }
768
-
769
- // Append arrays
770
- if (agent.triggers) {
771
- merged.triggers = [...(existing.triggers || []), ...agent.triggers]
772
- }
773
- if (agent.rules) {
774
- merged.rules = [...(existing.rules || []), ...agent.rules]
775
- }
776
- if (agent.patterns) {
777
- merged.patterns = [...(existing.patterns || []), ...agent.patterns]
778
- }
779
- if (agent.examples) {
780
- merged.examples = [...(existing.examples || []), ...agent.examples]
781
- }
782
-
783
- // Override scalar values if provided
784
- if (agent.description) {
785
- merged.description = `${existing.description}\n\n${agent.description}`
786
- }
787
- if (agent.domain) {
788
- merged.domain = agent.domain
789
- }
790
-
791
- mergedAgents.set(agent.name, merged)
792
- }
793
- }
794
- }
795
-
796
- return {
797
- agents: Array.from(mergedAgents.values()),
798
- sources,
799
- overrides,
800
- }
801
- }
802
-
803
- /**
804
- * Get agents for a specific monorepo package
805
- */
806
- async getPackageAgents(packageName: string): Promise<ResolvedAgents | null> {
807
- if (!this.monoInfo?.isMonorepo) {
808
- return null
809
- }
810
-
811
- const pkg = this.monoInfo.packages.find((p) => p.name === packageName)
812
- if (!pkg) {
813
- return null
814
- }
815
-
816
- return this.resolveAgentsForPath(pkg.path)
817
- }
818
-
819
- /**
820
- * Get all package agents in the monorepo
821
- */
822
- async getAllPackageAgents(): Promise<Map<string, ResolvedAgents>> {
823
- const results = new Map<string, ResolvedAgents>()
824
-
825
- if (!this.monoInfo?.isMonorepo) {
826
- return results
827
- }
828
-
829
- for (const pkg of this.monoInfo.packages) {
830
- const agents = await this.resolveAgentsForPath(pkg.path)
831
- results.set(pkg.name, agents)
832
- }
833
-
834
- return results
835
- }
836
- }
837
-
838
- // ============================================================================
839
- // EXPORTS
840
- // ============================================================================
841
-
842
- export default NestedContextResolver