crewly 1.8.9 → 1.8.11

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 (241) hide show
  1. package/config/constants.d.ts.map +1 -0
  2. package/config/index.d.ts.map +1 -0
  3. package/config/roles/_common/memory-instructions.md +6 -5
  4. package/config/roles/_common/wiki-instructions.md +49 -0
  5. package/config/roles/architect/prompt.md +2 -2
  6. package/config/roles/backend-developer/prompt.md +2 -2
  7. package/config/roles/designer/prompt.md +2 -2
  8. package/config/roles/developer/prompt.md +2 -2
  9. package/config/roles/frontend-developer/prompt.md +2 -2
  10. package/config/roles/fullstack-dev/prompt.md +2 -2
  11. package/config/roles/generalist/prompt.md +2 -2
  12. package/config/roles/ops/prompt.md +2 -2
  13. package/config/roles/orchestrator/prompt.md +135 -11
  14. package/config/roles/product-manager/prompt.md +2 -2
  15. package/config/roles/qa/prompt.md +2 -2
  16. package/config/roles/qa-engineer/prompt.md +2 -2
  17. package/config/roles/researcher/prompt.md +15 -6
  18. package/config/roles/sales/prompt.md +2 -2
  19. package/config/roles/support/prompt.md +2 -2
  20. package/config/roles/team-leader/prompt.md +17 -2
  21. package/config/roles/tpm/prompt.md +2 -2
  22. package/config/roles/ux-designer/prompt.md +2 -2
  23. package/config/skills/orchestrator/wiki-cleanup/SKILL.md +89 -0
  24. package/config/skills/orchestrator/wiki-cleanup/execute.sh +139 -0
  25. package/config/skills/orchestrator/wiki-lint/SKILL.md +75 -0
  26. package/config/skills/orchestrator/wiki-lint/execute.sh +66 -0
  27. package/config/skills/orchestrator/wiki-migrate/SKILL.md +103 -0
  28. package/config/skills/orchestrator/wiki-migrate/execute.sh +82 -0
  29. package/config/skills/orchestrator/wiki-process-queue/SKILL.md +9 -1
  30. package/dist/backend/backend/src/controllers/task-management/task-management.controller.d.ts +169 -0
  31. package/dist/backend/backend/src/controllers/task-management/task-management.controller.d.ts.map +1 -0
  32. package/dist/backend/backend/src/controllers/task-management/task-management.controller.js +1779 -0
  33. package/dist/backend/backend/src/controllers/task-management/task-management.controller.js.map +1 -0
  34. package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.d.ts +18 -0
  35. package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.d.ts.map +1 -1
  36. package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.js +63 -0
  37. package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.js.map +1 -1
  38. package/dist/backend/backend/src/controllers/task-pool/task-pool.routes.d.ts.map +1 -1
  39. package/dist/backend/backend/src/controllers/task-pool/task-pool.routes.js +5 -1
  40. package/dist/backend/backend/src/controllers/task-pool/task-pool.routes.js.map +1 -1
  41. package/dist/backend/backend/src/controllers/wiki/wiki.controller.d.ts +109 -0
  42. package/dist/backend/backend/src/controllers/wiki/wiki.controller.d.ts.map +1 -1
  43. package/dist/backend/backend/src/controllers/wiki/wiki.controller.js +418 -4
  44. package/dist/backend/backend/src/controllers/wiki/wiki.controller.js.map +1 -1
  45. package/dist/backend/backend/src/controllers/wiki/wiki.routes.d.ts.map +1 -1
  46. package/dist/backend/backend/src/controllers/wiki/wiki.routes.js +11 -1
  47. package/dist/backend/backend/src/controllers/wiki/wiki.routes.js.map +1 -1
  48. package/dist/backend/backend/src/index.d.ts.map +1 -1
  49. package/dist/backend/backend/src/index.js +64 -0
  50. package/dist/backend/backend/src/index.js.map +1 -1
  51. package/dist/backend/backend/src/index.js.orc-bak-20260529 +3130 -0
  52. package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.d.ts +513 -0
  53. package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.d.ts.map +1 -0
  54. package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.js +1568 -0
  55. package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.js.map +1 -0
  56. package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.d.ts +86 -0
  57. package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.d.ts.map +1 -0
  58. package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.js +147 -0
  59. package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.js.map +1 -0
  60. package/dist/backend/backend/src/services/agent/crewly-agent/api-client.d.ts +68 -0
  61. package/dist/backend/backend/src/services/agent/crewly-agent/api-client.d.ts.map +1 -0
  62. package/dist/backend/backend/src/services/agent/crewly-agent/api-client.js +131 -0
  63. package/dist/backend/backend/src/services/agent/crewly-agent/api-client.js.map +1 -0
  64. package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.d.ts +130 -0
  65. package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.d.ts.map +1 -0
  66. package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.js +263 -0
  67. package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.js.map +1 -0
  68. package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.d.ts +74 -0
  69. package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.d.ts.map +1 -0
  70. package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.js +140 -0
  71. package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.js.map +1 -0
  72. package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.d.ts +29 -0
  73. package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.d.ts.map +1 -0
  74. package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.js +279 -0
  75. package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.js.map +1 -0
  76. package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.d.ts +340 -0
  77. package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.d.ts.map +1 -0
  78. package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.js +1176 -0
  79. package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.js.map +1 -0
  80. package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.d.ts +79 -0
  81. package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.d.ts.map +1 -0
  82. package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.js +145 -0
  83. package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.js.map +1 -0
  84. package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.d.ts +79 -0
  85. package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.d.ts.map +1 -0
  86. package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.js +218 -0
  87. package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.js.map +1 -0
  88. package/dist/backend/backend/src/services/agent/crewly-agent/index.d.ts +16 -0
  89. package/dist/backend/backend/src/services/agent/crewly-agent/index.d.ts.map +1 -0
  90. package/dist/backend/backend/src/services/agent/crewly-agent/index.js +16 -0
  91. package/dist/backend/backend/src/services/agent/crewly-agent/index.js.map +1 -0
  92. package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.d.ts +135 -0
  93. package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.d.ts.map +1 -0
  94. package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.js +185 -0
  95. package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.js.map +1 -0
  96. package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.d.ts +141 -0
  97. package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.d.ts.map +1 -0
  98. package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.js +310 -0
  99. package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.js.map +1 -0
  100. package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.d.ts +91 -0
  101. package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.d.ts.map +1 -0
  102. package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.js +143 -0
  103. package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.js.map +1 -0
  104. package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.d.ts +103 -0
  105. package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.d.ts.map +1 -0
  106. package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.js +256 -0
  107. package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.js.map +1 -0
  108. package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.d.ts +143 -0
  109. package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.d.ts.map +1 -0
  110. package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.js +264 -0
  111. package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.js.map +1 -0
  112. package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.d.ts +13 -0
  113. package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.d.ts.map +1 -0
  114. package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.js +91 -0
  115. package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.js.map +1 -0
  116. package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.d.ts +135 -0
  117. package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.d.ts.map +1 -0
  118. package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.js +1937 -0
  119. package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.js.map +1 -0
  120. package/dist/backend/backend/src/services/ai/prompt-builder.service.js +1 -1
  121. package/dist/backend/backend/src/services/autonomous/auto-assign.service.d.ts +429 -0
  122. package/dist/backend/backend/src/services/autonomous/auto-assign.service.d.ts.map +1 -0
  123. package/dist/backend/backend/src/services/autonomous/auto-assign.service.js +852 -0
  124. package/dist/backend/backend/src/services/autonomous/auto-assign.service.js.map +1 -0
  125. package/dist/backend/backend/src/services/project/task-tracking.service.d.ts +171 -0
  126. package/dist/backend/backend/src/services/project/task-tracking.service.d.ts.map +1 -0
  127. package/dist/backend/backend/src/services/project/task-tracking.service.js +725 -0
  128. package/dist/backend/backend/src/services/project/task-tracking.service.js.map +1 -0
  129. package/dist/backend/backend/src/services/reconciler/reconciler-data-provider.d.ts.map +1 -1
  130. package/dist/backend/backend/src/services/reconciler/reconciler-data-provider.js +50 -0
  131. package/dist/backend/backend/src/services/reconciler/reconciler-data-provider.js.map +1 -1
  132. package/dist/backend/backend/src/services/task-pool/task-pool.service.d.ts +19 -0
  133. package/dist/backend/backend/src/services/task-pool/task-pool.service.d.ts.map +1 -1
  134. package/dist/backend/backend/src/services/task-pool/task-pool.service.js +45 -0
  135. package/dist/backend/backend/src/services/task-pool/task-pool.service.js.map +1 -1
  136. package/dist/backend/backend/src/services/v3/agent-auto-claim.service.d.ts.map +1 -1
  137. package/dist/backend/backend/src/services/v3/agent-auto-claim.service.js +34 -1
  138. package/dist/backend/backend/src/services/v3/agent-auto-claim.service.js.map +1 -1
  139. package/dist/backend/backend/src/services/v3/project-task-watcher.service.d.ts +118 -0
  140. package/dist/backend/backend/src/services/v3/project-task-watcher.service.d.ts.map +1 -0
  141. package/dist/backend/backend/src/services/v3/project-task-watcher.service.js +326 -0
  142. package/dist/backend/backend/src/services/v3/project-task-watcher.service.js.map +1 -0
  143. package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.d.ts +72 -0
  144. package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.d.ts.map +1 -0
  145. package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.js +186 -0
  146. package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.js.map +1 -0
  147. package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.d.ts +4 -1
  148. package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.d.ts.map +1 -1
  149. package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.js +24 -1
  150. package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.js.map +1 -1
  151. package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.d.ts +74 -0
  152. package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.d.ts.map +1 -0
  153. package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.js +154 -0
  154. package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.js.map +1 -0
  155. package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.d.ts +160 -0
  156. package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.d.ts.map +1 -0
  157. package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.js +399 -0
  158. package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.js.map +1 -0
  159. package/dist/backend/backend/src/services/wiki/wiki-lint.service.d.ts +182 -0
  160. package/dist/backend/backend/src/services/wiki/wiki-lint.service.d.ts.map +1 -0
  161. package/dist/backend/backend/src/services/wiki/wiki-lint.service.js +505 -0
  162. package/dist/backend/backend/src/services/wiki/wiki-lint.service.js.map +1 -0
  163. package/dist/backend/backend/src/services/wiki/wiki-migrate.service.d.ts +232 -0
  164. package/dist/backend/backend/src/services/wiki/wiki-migrate.service.d.ts.map +1 -0
  165. package/dist/backend/backend/src/services/wiki/wiki-migrate.service.js +1416 -0
  166. package/dist/backend/backend/src/services/wiki/wiki-migrate.service.js.map +1 -0
  167. package/dist/backend/backend/src/services/wiki/wiki-recent.service.d.ts +51 -0
  168. package/dist/backend/backend/src/services/wiki/wiki-recent.service.d.ts.map +1 -0
  169. package/dist/backend/backend/src/services/wiki/wiki-recent.service.js +102 -0
  170. package/dist/backend/backend/src/services/wiki/wiki-recent.service.js.map +1 -0
  171. package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.d.ts +84 -0
  172. package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.d.ts.map +1 -0
  173. package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.js +156 -0
  174. package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.js.map +1 -0
  175. package/dist/backend/backend/src/services/wiki/wiki-search.service.d.ts +90 -0
  176. package/dist/backend/backend/src/services/wiki/wiki-search.service.d.ts.map +1 -0
  177. package/dist/backend/backend/src/services/wiki/wiki-search.service.js +190 -0
  178. package/dist/backend/backend/src/services/wiki/wiki-search.service.js.map +1 -0
  179. package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.d.ts +164 -0
  180. package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.d.ts.map +1 -0
  181. package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.js +675 -0
  182. package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.js.map +1 -0
  183. package/dist/backend/backend/src/services/workflow/cron-task.service.d.ts.map +1 -1
  184. package/dist/backend/backend/src/services/workflow/cron-task.service.js +65 -0
  185. package/dist/backend/backend/src/services/workflow/cron-task.service.js.map +1 -1
  186. package/dist/backend/backend/src/types/auto-assign.types.d.ts +271 -0
  187. package/dist/backend/backend/src/types/auto-assign.types.d.ts.map +1 -0
  188. package/dist/backend/backend/src/types/auto-assign.types.js +136 -0
  189. package/dist/backend/backend/src/types/auto-assign.types.js.map +1 -0
  190. package/dist/backend/backend/src/types/cron-task.types.d.ts +16 -1
  191. package/dist/backend/backend/src/types/cron-task.types.d.ts.map +1 -1
  192. package/dist/backend/backend/src/utils/esm-require.utils.d.ts +111 -0
  193. package/dist/backend/backend/src/utils/esm-require.utils.d.ts.map +1 -0
  194. package/dist/backend/backend/src/utils/esm-require.utils.js +124 -0
  195. package/dist/backend/backend/src/utils/esm-require.utils.js.map +1 -0
  196. package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.d.ts +220 -0
  197. package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.d.ts.map +1 -0
  198. package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.js +37 -0
  199. package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.js.map +1 -0
  200. package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.d.ts +56 -0
  201. package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.d.ts.map +1 -0
  202. package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.js +91 -0
  203. package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.js.map +1 -0
  204. package/dist/cli/backend/src/services/knowledge/learnings-index.service.d.ts +159 -0
  205. package/dist/cli/backend/src/services/knowledge/learnings-index.service.d.ts.map +1 -0
  206. package/dist/cli/backend/src/services/knowledge/learnings-index.service.js +304 -0
  207. package/dist/cli/backend/src/services/knowledge/learnings-index.service.js.map +1 -0
  208. package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.d.ts +115 -0
  209. package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.d.ts.map +1 -0
  210. package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.js +215 -0
  211. package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.js.map +1 -0
  212. package/dist/cli/backend/src/services/memory/embedding-provider.d.ts +78 -0
  213. package/dist/cli/backend/src/services/memory/embedding-provider.d.ts.map +1 -0
  214. package/dist/cli/backend/src/services/memory/embedding-provider.js +179 -0
  215. package/dist/cli/backend/src/services/memory/embedding-provider.js.map +1 -0
  216. package/dist/cli/backend/src/services/memory/vector-store.service.d.ts +331 -0
  217. package/dist/cli/backend/src/services/memory/vector-store.service.d.ts.map +1 -0
  218. package/dist/cli/backend/src/services/memory/vector-store.service.js +814 -0
  219. package/dist/cli/backend/src/services/memory/vector-store.service.js.map +1 -0
  220. package/dist/cli/backend/src/services/project/task-tracking.service.d.ts +171 -0
  221. package/dist/cli/backend/src/services/project/task-tracking.service.d.ts.map +1 -0
  222. package/dist/cli/backend/src/services/project/task-tracking.service.js +725 -0
  223. package/dist/cli/backend/src/services/project/task-tracking.service.js.map +1 -0
  224. package/dist/cli/backend/src/services/task-pool/task-pool.service.d.ts +19 -0
  225. package/dist/cli/backend/src/services/task-pool/task-pool.service.d.ts.map +1 -1
  226. package/dist/cli/backend/src/services/task-pool/task-pool.service.js +45 -0
  227. package/dist/cli/backend/src/services/task-pool/task-pool.service.js.map +1 -1
  228. package/dist/cli/backend/src/types/auto-assign.types.d.ts +271 -0
  229. package/dist/cli/backend/src/types/auto-assign.types.d.ts.map +1 -0
  230. package/dist/cli/backend/src/types/auto-assign.types.js +136 -0
  231. package/dist/cli/backend/src/types/auto-assign.types.js.map +1 -0
  232. package/dist/cli/cli/src/index.js +0 -0
  233. package/frontend/dist/assets/{index-db3f5041.css → index-068bb4f6.css} +10 -1
  234. package/frontend/dist/assets/index-c24ceb15.js +4960 -0
  235. package/frontend/dist/index.html +2 -2
  236. package/package.json +1 -1
  237. package/config/skills/agent/core/query-knowledge/SKILL.md +0 -87
  238. package/config/skills/agent/core/query-knowledge/execute.sh +0 -30
  239. package/config/skills/orchestrator/query-knowledge/SKILL.md +0 -75
  240. package/config/skills/orchestrator/query-knowledge/execute.sh +0 -30
  241. package/frontend/dist/assets/index-cc115bb4.js +0 -4926
@@ -0,0 +1,37 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ /**
4
+ * Load a role-specific fragment file from config/roles/{role}/fragments/{fragmentName}.md.
5
+ * Returns null if the file doesn't exist.
6
+ *
7
+ * @param projectRoot - Project root path (where config/ lives)
8
+ * @param role - Agent role (e.g. 'orchestrator')
9
+ * @param fragmentName - Fragment file name without .md extension
10
+ * @returns Fragment content string or null
11
+ */
12
+ export function loadRoleFragment(projectRoot, role, fragmentName) {
13
+ try {
14
+ const fragmentPath = path.join(projectRoot, 'config', 'roles', role, 'fragments', `${fragmentName}.md`);
15
+ if (fs.existsSync(fragmentPath)) {
16
+ return fs.readFileSync(fragmentPath, 'utf-8');
17
+ }
18
+ }
19
+ catch {
20
+ // Fragment not found — fall back to inline content
21
+ }
22
+ return null;
23
+ }
24
+ /**
25
+ * Estimate token count from a string.
26
+ * Uses the rough heuristic of ~4 characters per token (suitable for English/code mix).
27
+ *
28
+ * @param text - Text to estimate tokens for
29
+ * @returns Estimated token count
30
+ */
31
+ export function estimateTokens(text) {
32
+ if (!text)
33
+ return 0;
34
+ // ~4 chars per token is a reasonable estimate for English + code
35
+ return Math.ceil(text.length / 4);
36
+ }
37
+ //# sourceMappingURL=prompt-module.interface.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompt-module.interface.js","sourceRoot":"","sources":["../../../../../../../backend/src/services/ai/prompt-modules/prompt-module.interface.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,IAAI,CAAC;AACzB,OAAO,KAAK,IAAI,MAAM,MAAM,CAAC;AAyM7B;;;;;;;;GAQG;AACH,MAAM,UAAU,gBAAgB,CAC/B,WAAmB,EACnB,IAAY,EACZ,YAAoB;IAEpB,IAAI,CAAC;QACJ,MAAM,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,GAAG,YAAY,KAAK,CAAC,CAAC;QACxG,IAAI,EAAE,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE,CAAC;YACjC,OAAO,EAAE,CAAC,YAAY,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;QAC/C,CAAC;IACF,CAAC;IAAC,MAAM,CAAC;QACR,mDAAmD;IACpD,CAAC;IACD,OAAO,IAAI,CAAC;AACb,CAAC;AAED;;;;;;GAMG;AACH,MAAM,UAAU,cAAc,CAAC,IAAY;IAC1C,IAAI,CAAC,IAAI;QAAE,OAAO,CAAC,CAAC;IACpB,iEAAiE;IACjE,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;AACnC,CAAC"}
@@ -0,0 +1,56 @@
1
+ /**
2
+ * FTS5 Search Strategy
3
+ *
4
+ * Implements the KnowledgeSearchStrategy interface using the SQLite FTS5
5
+ * full-text search index. Maps FTS5 BM25-ranked results to the existing
6
+ * ScoredDocument format used by KnowledgeSearchService.
7
+ *
8
+ * @module services/knowledge/fts5-search-strategy
9
+ */
10
+ import type { KnowledgeSearchStrategy, ScoredDocument } from './knowledge-search.service.js';
11
+ import type { KnowledgeDocumentSummary } from '../../types/knowledge.types.js';
12
+ import { WikiCompilerService } from './wiki-compiler.service.js';
13
+ /**
14
+ * FTS5-based search strategy for the Wiki-First Knowledge System.
15
+ *
16
+ * This strategy provides pure full-text search via SQLite FTS5 with BM25 ranking.
17
+ *
18
+ * Note on semantic fallback (design doc Section 10): The 10% semantic fallback
19
+ * weight is achieved at the MemoryService level, which runs both FTS5 knowledge
20
+ * search AND vector-based semantic search in parallel. This strategy intentionally
21
+ * does not mix semantic results — that composition happens upstream in the recall
22
+ * pipeline (MemoryService.recall -> searchKnowledgeDocuments + semanticSearch).
23
+ *
24
+ * Uses SQLite FTS5 MATCH with BM25 ranking for fast, zero-cost retrieval.
25
+ * Results are mapped back to the KnowledgeDocumentSummary objects provided
26
+ * by the caller, falling back to minimal summaries when a document is found
27
+ * in the FTS5 index but not in the provided documents array.
28
+ *
29
+ * @example
30
+ * ```typescript
31
+ * const strategy = new Fts5SearchStrategy(wikiCompiler);
32
+ * const results = await strategy.search('deployment guide', documents);
33
+ * ```
34
+ */
35
+ export declare class Fts5SearchStrategy implements KnowledgeSearchStrategy {
36
+ private readonly wikiCompiler;
37
+ private readonly logger;
38
+ /**
39
+ * Create an Fts5SearchStrategy.
40
+ *
41
+ * @param wikiCompiler - The WikiCompilerService instance providing FTS5 search
42
+ */
43
+ constructor(wikiCompiler: WikiCompilerService);
44
+ /**
45
+ * Search for documents matching the query using FTS5 full-text search.
46
+ *
47
+ * FTS5 BM25 rank values are negative (more negative = more relevant),
48
+ * so scores are converted to positive values for the ScoredDocument interface.
49
+ *
50
+ * @param query - The search query text
51
+ * @param documents - Candidate document summaries to match against
52
+ * @returns Documents sorted by descending relevance score
53
+ */
54
+ search(query: string, documents: KnowledgeDocumentSummary[]): Promise<ScoredDocument[]>;
55
+ }
56
+ //# sourceMappingURL=fts5-search-strategy.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fts5-search-strategy.d.ts","sourceRoot":"","sources":["../../../../../../backend/src/services/knowledge/fts5-search-strategy.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,KAAK,EAAE,uBAAuB,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAC7F,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,gCAAgC,CAAC;AAC/E,OAAO,EAAE,mBAAmB,EAAE,MAAM,4BAA4B,CAAC;AAGjE;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,qBAAa,kBAAmB,YAAW,uBAAuB;IAChE,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAsB;IACnD,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAkB;IAEzC;;;;OAIG;gBACS,YAAY,EAAE,mBAAmB;IAK7C;;;;;;;;;OASG;IACG,MAAM,CAAC,KAAK,EAAE,MAAM,EAAE,SAAS,EAAE,wBAAwB,EAAE,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC;CAyC9F"}
@@ -0,0 +1,91 @@
1
+ /**
2
+ * FTS5 Search Strategy
3
+ *
4
+ * Implements the KnowledgeSearchStrategy interface using the SQLite FTS5
5
+ * full-text search index. Maps FTS5 BM25-ranked results to the existing
6
+ * ScoredDocument format used by KnowledgeSearchService.
7
+ *
8
+ * @module services/knowledge/fts5-search-strategy
9
+ */
10
+ import { LoggerService } from '../core/logger.service.js';
11
+ /**
12
+ * FTS5-based search strategy for the Wiki-First Knowledge System.
13
+ *
14
+ * This strategy provides pure full-text search via SQLite FTS5 with BM25 ranking.
15
+ *
16
+ * Note on semantic fallback (design doc Section 10): The 10% semantic fallback
17
+ * weight is achieved at the MemoryService level, which runs both FTS5 knowledge
18
+ * search AND vector-based semantic search in parallel. This strategy intentionally
19
+ * does not mix semantic results — that composition happens upstream in the recall
20
+ * pipeline (MemoryService.recall -> searchKnowledgeDocuments + semanticSearch).
21
+ *
22
+ * Uses SQLite FTS5 MATCH with BM25 ranking for fast, zero-cost retrieval.
23
+ * Results are mapped back to the KnowledgeDocumentSummary objects provided
24
+ * by the caller, falling back to minimal summaries when a document is found
25
+ * in the FTS5 index but not in the provided documents array.
26
+ *
27
+ * @example
28
+ * ```typescript
29
+ * const strategy = new Fts5SearchStrategy(wikiCompiler);
30
+ * const results = await strategy.search('deployment guide', documents);
31
+ * ```
32
+ */
33
+ export class Fts5SearchStrategy {
34
+ wikiCompiler;
35
+ logger;
36
+ /**
37
+ * Create an Fts5SearchStrategy.
38
+ *
39
+ * @param wikiCompiler - The WikiCompilerService instance providing FTS5 search
40
+ */
41
+ constructor(wikiCompiler) {
42
+ this.wikiCompiler = wikiCompiler;
43
+ this.logger = LoggerService.getInstance().createComponentLogger('Fts5SearchStrategy');
44
+ }
45
+ /**
46
+ * Search for documents matching the query using FTS5 full-text search.
47
+ *
48
+ * FTS5 BM25 rank values are negative (more negative = more relevant),
49
+ * so scores are converted to positive values for the ScoredDocument interface.
50
+ *
51
+ * @param query - The search query text
52
+ * @param documents - Candidate document summaries to match against
53
+ * @returns Documents sorted by descending relevance score
54
+ */
55
+ async search(query, documents) {
56
+ const ftsResults = this.wikiCompiler.search(query, { limit: 10 });
57
+ if (ftsResults.length === 0) {
58
+ this.logger.debug('FTS5 search returned no results', { query });
59
+ return [];
60
+ }
61
+ // Build a lookup map from the provided documents array
62
+ const docById = new Map(documents.map((d) => [d.id, d]));
63
+ const scored = [];
64
+ for (const ftsResult of ftsResults) {
65
+ // FTS5 rank is negative; convert to positive score (higher = more relevant)
66
+ const score = Math.abs(ftsResult.rank);
67
+ const existingDoc = docById.get(ftsResult.id);
68
+ if (existingDoc) {
69
+ scored.push({ document: existingDoc, score });
70
+ }
71
+ else {
72
+ // Create a minimal summary for documents in FTS5 but not in the provided list
73
+ const minimalDoc = {
74
+ id: ftsResult.id,
75
+ title: ftsResult.title,
76
+ category: ftsResult.category,
77
+ tags: ftsResult.tags ? ftsResult.tags.split(/,\s*/) : [],
78
+ preview: ftsResult.content.slice(0, 200),
79
+ scope: 'global',
80
+ createdBy: 'system',
81
+ updatedBy: 'system',
82
+ createdAt: new Date().toISOString(),
83
+ updatedAt: new Date().toISOString(),
84
+ };
85
+ scored.push({ document: minimalDoc, score });
86
+ }
87
+ }
88
+ return scored.sort((a, b) => b.score - a.score);
89
+ }
90
+ }
91
+ //# sourceMappingURL=fts5-search-strategy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fts5-search-strategy.js","sourceRoot":"","sources":["../../../../../../backend/src/services/knowledge/fts5-search-strategy.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAKH,OAAO,EAAE,aAAa,EAAwB,MAAM,2BAA2B,CAAC;AAEhF;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,MAAM,OAAO,kBAAkB;IACZ,YAAY,CAAsB;IAClC,MAAM,CAAkB;IAEzC;;;;OAIG;IACH,YAAY,YAAiC;QAC3C,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;QACjC,IAAI,CAAC,MAAM,GAAG,aAAa,CAAC,WAAW,EAAE,CAAC,qBAAqB,CAAC,oBAAoB,CAAC,CAAC;IACxF,CAAC;IAED;;;;;;;;;OASG;IACH,KAAK,CAAC,MAAM,CAAC,KAAa,EAAE,SAAqC;QAC/D,MAAM,UAAU,GAAG,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,CAAC;QAElE,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5B,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,iCAAiC,EAAE,EAAE,KAAK,EAAE,CAAC,CAAC;YAChE,OAAO,EAAE,CAAC;QACZ,CAAC;QAED,uDAAuD;QACvD,MAAM,OAAO,GAAG,IAAI,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QAEzD,MAAM,MAAM,GAAqB,EAAE,CAAC;QAEpC,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE,CAAC;YACnC,4EAA4E;YAC5E,MAAM,KAAK,GAAG,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;YAEvC,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;YAE9C,IAAI,WAAW,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,WAAW,EAAE,KAAK,EAAE,CAAC,CAAC;YAChD,CAAC;iBAAM,CAAC;gBACN,8EAA8E;gBAC9E,MAAM,UAAU,GAA6B;oBAC3C,EAAE,EAAE,SAAS,CAAC,EAAE;oBAChB,KAAK,EAAE,SAAS,CAAC,KAAK;oBACtB,QAAQ,EAAE,SAAS,CAAC,QAAQ;oBAC5B,IAAI,EAAE,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE;oBACxD,OAAO,EAAE,SAAS,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC;oBACxC,KAAK,EAAE,QAAQ;oBACf,SAAS,EAAE,QAAQ;oBACnB,SAAS,EAAE,QAAQ;oBACnB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;oBACnC,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;iBACpC,CAAC;gBACF,MAAM,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,KAAK,EAAE,CAAC,CAAC;YAC/C,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC;IAClD,CAAC;CACF"}
@@ -0,0 +1,159 @@
1
+ /**
2
+ * Learnings Index Service
3
+ *
4
+ * Surfaces project-scoped record-learning entries (`.crewly/knowledge/learnings.md`)
5
+ * as virtual `KnowledgeDocumentSummary` entries so they appear in the `/knowledge`
6
+ * UI and are returned by recall searches.
7
+ *
8
+ * Background (memory.f1, 2026-04-28): the `record-learning` skill writes to
9
+ * `learnings.md` (an append-only log owned by `ProjectMemoryService`), but the
10
+ * `/knowledge` UI reads `KnowledgeService.listDocuments()` which only sees
11
+ * `.crewly/docs/`. As a result, learnings persisted to disk were not visible
12
+ * in the UI and were not returned by recall. This service bridges that gap by
13
+ * parsing the learnings log on demand and exposing each entry under a stable
14
+ * synthetic ID (`learning:<sha1-prefix>`) that the rest of the knowledge stack
15
+ * can carry through `listDocuments` / `getDocument` / search.
16
+ *
17
+ * @module services/knowledge/learnings-index.service
18
+ */
19
+ import type { KnowledgeDocument, KnowledgeDocumentSummary, KnowledgeScope } from '../../types/knowledge.types.js';
20
+ /** Prefix that marks a synthetic learning document ID. */
21
+ export declare const LEARNING_ID_PREFIX = "learning:";
22
+ /** Category label used for learnings surfaced in the knowledge UI. */
23
+ export declare const LEARNINGS_CATEGORY = "Learnings";
24
+ /**
25
+ * A learning entry parsed out of `learnings.md`.
26
+ *
27
+ * @internal
28
+ */
29
+ export interface ParsedLearning {
30
+ /** Stable synthetic ID — `learning:` prefix + sha1 hash of (timestamp + author + content). */
31
+ id: string;
32
+ /** Title — Karpathy-lite `[[Entity]]` opener if present, else first sentence (capped). */
33
+ title: string;
34
+ /** Tags — `learning` always, plus any `[tag-name]` tokens found in the body. */
35
+ tags: string[];
36
+ /** Full body of the entry (date / agent header lines stripped). */
37
+ content: string;
38
+ /** ISO timestamp parsed from `## YYYY-MM-DD` and `### [role/agent] HH:MM:SS` headers. */
39
+ createdAt: string;
40
+ /** Same as `createdAt` — learnings are append-only, no separate update timestamp. */
41
+ updatedAt: string;
42
+ /** Agent author string from the `[role/agentId]` header (or `unknown`). */
43
+ author: string;
44
+ }
45
+ /**
46
+ * Parse `learnings.md` content into discrete learning entries.
47
+ *
48
+ * The expected format produced by `ProjectMemoryService.recordLearning` is:
49
+ *
50
+ * ```
51
+ * ## YYYY-MM-DD
52
+ *
53
+ * ### [role/agentId] HH:MM:SS
54
+ * <content>
55
+ *
56
+ * ---
57
+ * ```
58
+ *
59
+ * Older entries may lack the `### [role/agentId]` header — those are still
60
+ * parsed, with `author = 'unknown'` and time defaulting to `00:00:00`. Entries
61
+ * containing only the file-level header (`# Project Learnings`) are skipped.
62
+ *
63
+ * Exported for testing only — callers should use {@link LearningsIndexService}.
64
+ *
65
+ * @param raw - Full file content (YAML frontmatter is stripped if present)
66
+ * @returns Parsed entries in file order
67
+ */
68
+ export declare function parseLearnings(raw: string): ParsedLearning[];
69
+ /**
70
+ * Service that surfaces `learnings.md` entries as virtual knowledge documents.
71
+ *
72
+ * Singleton, file-system-backed, no caching — `learnings.md` is small enough
73
+ * (a single project's append-only log) that re-parsing on every list call is
74
+ * acceptable and avoids stale-cache bugs across writes from the agent skill.
75
+ */
76
+ export declare class LearningsIndexService {
77
+ private static instance;
78
+ private readonly logger;
79
+ private constructor();
80
+ /**
81
+ * Get the singleton instance.
82
+ *
83
+ * @returns LearningsIndexService instance
84
+ */
85
+ static getInstance(): LearningsIndexService;
86
+ /**
87
+ * Reset the singleton (for testing).
88
+ */
89
+ static resetInstance(): void;
90
+ /**
91
+ * Detect whether an ID belongs to this service's synthetic namespace.
92
+ *
93
+ * @param id - Document ID to test
94
+ * @returns True if the ID has the learning prefix
95
+ */
96
+ static isLearningId(id: string): boolean;
97
+ /**
98
+ * Resolve the absolute path to `learnings.md` for the given scope.
99
+ *
100
+ * Learnings only exist at project scope today (the `record-learning` skill
101
+ * always requires a `projectPath`). Global scope returns null so callers
102
+ * can short-circuit cleanly.
103
+ *
104
+ * @param scope - 'global' or 'project'
105
+ * @param projectPath - Required when scope is 'project'
106
+ * @returns Absolute file path, or null if learnings cannot exist for this scope
107
+ */
108
+ private learningsPath;
109
+ /**
110
+ * Read and parse `learnings.md` for the given scope.
111
+ *
112
+ * Missing-file and read-error cases both return an empty array — the
113
+ * indexer must never fail callers just because no learnings exist yet.
114
+ *
115
+ * @param scope - Document scope
116
+ * @param projectPath - Project path (required for 'project' scope)
117
+ * @returns Parsed learning entries (empty if file missing or unreadable)
118
+ */
119
+ private readLearnings;
120
+ /**
121
+ * List all learnings for the given scope as `KnowledgeDocumentSummary` entries.
122
+ *
123
+ * Sorted newest-first so the UI shows recent learnings at the top of the
124
+ * unified knowledge list (matching the user mental model that learnings
125
+ * are a chronological journal).
126
+ *
127
+ * @param scope - 'global' or 'project'
128
+ * @param projectPath - Required when scope is 'project'
129
+ * @returns Array of summaries (empty if scope is 'global' or no learnings exist)
130
+ */
131
+ listLearnings(scope: KnowledgeScope, projectPath?: string): Promise<KnowledgeDocumentSummary[]>;
132
+ /**
133
+ * Fetch a single learning by its synthetic ID.
134
+ *
135
+ * @param id - Synthetic document ID (must start with `learning:`)
136
+ * @param scope - Document scope
137
+ * @param projectPath - Required when scope is 'project'
138
+ * @returns Full document, or null if id is not a learning id or not found
139
+ */
140
+ getLearning(id: string, scope: KnowledgeScope, projectPath?: string): Promise<KnowledgeDocument | null>;
141
+ /**
142
+ * Project a parsed entry into a `KnowledgeDocumentSummary`.
143
+ *
144
+ * @param e - Parsed learning entry
145
+ * @param scope - Target scope label
146
+ * @returns Summary suitable for the knowledge list API
147
+ */
148
+ private toSummary;
149
+ /**
150
+ * Project a parsed entry into a full `KnowledgeDocument`.
151
+ *
152
+ * @param e - Parsed learning entry
153
+ * @param scope - Target scope label
154
+ * @param projectPath - Carried into the document for project-scoped entries
155
+ * @returns Full document suitable for the knowledge document API
156
+ */
157
+ private toDocument;
158
+ }
159
+ //# sourceMappingURL=learnings-index.service.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"learnings-index.service.d.ts","sourceRoot":"","sources":["../../../../../../backend/src/services/knowledge/learnings-index.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;GAiBG;AAQH,OAAO,KAAK,EACV,iBAAiB,EACjB,wBAAwB,EACxB,cAAc,EACf,MAAM,gCAAgC,CAAC;AAGxC,0DAA0D;AAC1D,eAAO,MAAM,kBAAkB,cAAc,CAAC;AAE9C,sEAAsE;AACtE,eAAO,MAAM,kBAAkB,cAAc,CAAC;AAQ9C;;;;GAIG;AACH,MAAM,WAAW,cAAc;IAC7B,8FAA8F;IAC9F,EAAE,EAAE,MAAM,CAAC;IACX,0FAA0F;IAC1F,KAAK,EAAE,MAAM,CAAC;IACd,gFAAgF;IAChF,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,mEAAmE;IACnE,OAAO,EAAE,MAAM,CAAC;IAChB,yFAAyF;IACzF,SAAS,EAAE,MAAM,CAAC;IAClB,qFAAqF;IACrF,SAAS,EAAE,MAAM,CAAC;IAClB,2EAA2E;IAC3E,MAAM,EAAE,MAAM,CAAC;CAChB;AAED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG,cAAc,EAAE,CAiG5D;AAED;;;;;;GAMG;AACH,qBAAa,qBAAqB;IAChC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAsC;IAC7D,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAkB;IAEzC,OAAO;IAIP;;;;OAIG;IACH,MAAM,CAAC,WAAW,IAAI,qBAAqB;IAO3C;;OAEG;IACH,MAAM,CAAC,aAAa,IAAI,IAAI;IAI5B;;;;;OAKG;IACH,MAAM,CAAC,YAAY,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO;IAIxC;;;;;;;;;;OAUG;IACH,OAAO,CAAC,aAAa;IAYrB;;;;;;;;;OASG;YACW,aAAa;IAiB3B;;;;;;;;;;OAUG;IACG,aAAa,CAAC,KAAK,EAAE,cAAc,EAAE,WAAW,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,wBAAwB,EAAE,CAAC;IAOrG;;;;;;;OAOG;IACG,WAAW,CACf,EAAE,EAAE,MAAM,EACV,KAAK,EAAE,cAAc,EACrB,WAAW,CAAC,EAAE,MAAM,GACnB,OAAO,CAAC,iBAAiB,GAAG,IAAI,CAAC;IASpC;;;;;;OAMG;IACH,OAAO,CAAC,SAAS;IAejB;;;;;;;OAOG;IACH,OAAO,CAAC,UAAU;CAmBnB"}
@@ -0,0 +1,304 @@
1
+ /**
2
+ * Learnings Index Service
3
+ *
4
+ * Surfaces project-scoped record-learning entries (`.crewly/knowledge/learnings.md`)
5
+ * as virtual `KnowledgeDocumentSummary` entries so they appear in the `/knowledge`
6
+ * UI and are returned by recall searches.
7
+ *
8
+ * Background (memory.f1, 2026-04-28): the `record-learning` skill writes to
9
+ * `learnings.md` (an append-only log owned by `ProjectMemoryService`), but the
10
+ * `/knowledge` UI reads `KnowledgeService.listDocuments()` which only sees
11
+ * `.crewly/docs/`. As a result, learnings persisted to disk were not visible
12
+ * in the UI and were not returned by recall. This service bridges that gap by
13
+ * parsing the learnings log on demand and exposing each entry under a stable
14
+ * synthetic ID (`learning:<sha1-prefix>`) that the rest of the knowledge stack
15
+ * can carry through `listDocuments` / `getDocument` / search.
16
+ *
17
+ * @module services/knowledge/learnings-index.service
18
+ */
19
+ import * as path from 'path';
20
+ import * as fs from 'fs/promises';
21
+ import { existsSync } from 'fs';
22
+ import { createHash } from 'crypto';
23
+ import { LoggerService } from '../core/logger.service.js';
24
+ import { CREWLY_CONSTANTS, MEMORY_CONSTANTS } from '../../constants.js';
25
+ import { KNOWLEDGE_CONSTANTS } from '../../types/knowledge.types.js';
26
+ /** Prefix that marks a synthetic learning document ID. */
27
+ export const LEARNING_ID_PREFIX = 'learning:';
28
+ /** Category label used for learnings surfaced in the knowledge UI. */
29
+ export const LEARNINGS_CATEGORY = 'Learnings';
30
+ /** Maximum title length for a synthesized learning summary (kept short for list rendering). */
31
+ const MAX_TITLE_LENGTH = 100;
32
+ /** Length of the sha1 hash prefix used for synthetic IDs (16 hex chars = 64 bits collision space). */
33
+ const ID_HASH_LENGTH = 16;
34
+ /**
35
+ * Parse `learnings.md` content into discrete learning entries.
36
+ *
37
+ * The expected format produced by `ProjectMemoryService.recordLearning` is:
38
+ *
39
+ * ```
40
+ * ## YYYY-MM-DD
41
+ *
42
+ * ### [role/agentId] HH:MM:SS
43
+ * <content>
44
+ *
45
+ * ---
46
+ * ```
47
+ *
48
+ * Older entries may lack the `### [role/agentId]` header — those are still
49
+ * parsed, with `author = 'unknown'` and time defaulting to `00:00:00`. Entries
50
+ * containing only the file-level header (`# Project Learnings`) are skipped.
51
+ *
52
+ * Exported for testing only — callers should use {@link LearningsIndexService}.
53
+ *
54
+ * @param raw - Full file content (YAML frontmatter is stripped if present)
55
+ * @returns Parsed entries in file order
56
+ */
57
+ export function parseLearnings(raw) {
58
+ // Strip YAML frontmatter (if present) so the first `---` after it doesn't
59
+ // get treated as a section separator.
60
+ let body = raw;
61
+ if (body.startsWith('---')) {
62
+ const end = body.indexOf('\n---', 3);
63
+ if (end !== -1) {
64
+ body = body.slice(end + 4);
65
+ }
66
+ }
67
+ // Sections are separated by a markdown horizontal rule (a line containing
68
+ // only `---`). The recordLearning writer always emits this between entries.
69
+ const sections = body
70
+ .split(/^---\s*$/m)
71
+ .map((s) => s.trim())
72
+ .filter((s) => s.length > 0);
73
+ const entries = [];
74
+ for (const section of sections) {
75
+ // Skip file-level headers (`# Project Learnings: ...`).
76
+ if (/^#\s+(Project Learnings|Learnings from)/i.test(section)) {
77
+ continue;
78
+ }
79
+ const dateMatch = section.match(/^##\s+(\d{4}-\d{2}-\d{2})/m);
80
+ const date = dateMatch?.[1] ?? '';
81
+ const agentMatch = section.match(/^###\s+\[([^\]]+)\]\s+(\d{2}:\d{2}:\d{2})/m);
82
+ const author = agentMatch?.[1] ?? 'unknown';
83
+ const time = agentMatch?.[2] ?? '00:00:00';
84
+ // ISO timestamp: combine date + time as UTC. If date is missing (truly
85
+ // malformed), fall back to epoch so the entry still surfaces but sorts
86
+ // last. This is intentionally permissive: better to show a learning with
87
+ // a wrong-looking timestamp than to drop it silently.
88
+ const createdAt = date ? `${date}T${time}.000Z` : new Date(0).toISOString();
89
+ // Strip the headers we already parsed from the content body so they don't
90
+ // appear duplicated in the synthesized document.
91
+ const content = section
92
+ .replace(/^##\s+\d{4}-\d{2}-\d{2}\s*$/m, '')
93
+ .replace(/^###\s+\[[^\]]+\]\s+\d{2}:\d{2}:\d{2}\s*$/m, '')
94
+ .trim();
95
+ if (!content) {
96
+ continue;
97
+ }
98
+ // Title selection priority:
99
+ // 1. Karpathy-lite `[[Entity]]` opener — entity name as title.
100
+ // 2. First sentence/line of the body, capped at MAX_TITLE_LENGTH.
101
+ const entityMatch = content.match(/^\[\[([^\]]+)\]\]/);
102
+ let title;
103
+ if (entityMatch) {
104
+ title = entityMatch[1].trim().slice(0, MAX_TITLE_LENGTH);
105
+ }
106
+ else {
107
+ const firstSentence = content.split(/[.\n]/)[0].trim();
108
+ title = firstSentence.slice(0, MAX_TITLE_LENGTH) || 'Learning';
109
+ }
110
+ // Tags: 'learning' plus any `[tag-name]` markers in the body. Matches the
111
+ // existing convention used by record-learning callers (e.g. `[best-practice]`,
112
+ // `[gotcha]`, `[architecture-decision]`).
113
+ const tags = ['learning'];
114
+ const tagPattern = /\[([a-z][a-z0-9-]{0,49})\]/g;
115
+ let tagMatch;
116
+ while ((tagMatch = tagPattern.exec(content)) !== null) {
117
+ const tag = tagMatch[1];
118
+ if (!tags.includes(tag) && tags.length < KNOWLEDGE_CONSTANTS.MAX_TAGS) {
119
+ tags.push(tag);
120
+ }
121
+ }
122
+ // Stable synthetic ID. We hash (timestamp + author + content prefix) so
123
+ // that re-parsing the same file produces the same IDs, but two entries
124
+ // written at the same second by the same author with different content
125
+ // get distinct IDs. SHA-1 + 16 hex chars is sufficient (64 bits) for the
126
+ // append-only log scale.
127
+ const idHash = createHash('sha1')
128
+ .update(`${createdAt}|${author}|${content.slice(0, 200)}`)
129
+ .digest('hex')
130
+ .slice(0, ID_HASH_LENGTH);
131
+ entries.push({
132
+ id: `${LEARNING_ID_PREFIX}${idHash}`,
133
+ title,
134
+ tags,
135
+ content,
136
+ createdAt,
137
+ updatedAt: createdAt,
138
+ author,
139
+ });
140
+ }
141
+ return entries;
142
+ }
143
+ /**
144
+ * Service that surfaces `learnings.md` entries as virtual knowledge documents.
145
+ *
146
+ * Singleton, file-system-backed, no caching — `learnings.md` is small enough
147
+ * (a single project's append-only log) that re-parsing on every list call is
148
+ * acceptable and avoids stale-cache bugs across writes from the agent skill.
149
+ */
150
+ export class LearningsIndexService {
151
+ static instance = null;
152
+ logger;
153
+ constructor() {
154
+ this.logger = LoggerService.getInstance().createComponentLogger('LearningsIndexService');
155
+ }
156
+ /**
157
+ * Get the singleton instance.
158
+ *
159
+ * @returns LearningsIndexService instance
160
+ */
161
+ static getInstance() {
162
+ if (!LearningsIndexService.instance) {
163
+ LearningsIndexService.instance = new LearningsIndexService();
164
+ }
165
+ return LearningsIndexService.instance;
166
+ }
167
+ /**
168
+ * Reset the singleton (for testing).
169
+ */
170
+ static resetInstance() {
171
+ LearningsIndexService.instance = null;
172
+ }
173
+ /**
174
+ * Detect whether an ID belongs to this service's synthetic namespace.
175
+ *
176
+ * @param id - Document ID to test
177
+ * @returns True if the ID has the learning prefix
178
+ */
179
+ static isLearningId(id) {
180
+ return typeof id === 'string' && id.startsWith(LEARNING_ID_PREFIX);
181
+ }
182
+ /**
183
+ * Resolve the absolute path to `learnings.md` for the given scope.
184
+ *
185
+ * Learnings only exist at project scope today (the `record-learning` skill
186
+ * always requires a `projectPath`). Global scope returns null so callers
187
+ * can short-circuit cleanly.
188
+ *
189
+ * @param scope - 'global' or 'project'
190
+ * @param projectPath - Required when scope is 'project'
191
+ * @returns Absolute file path, or null if learnings cannot exist for this scope
192
+ */
193
+ learningsPath(scope, projectPath) {
194
+ if (scope !== 'project' || !projectPath) {
195
+ return null;
196
+ }
197
+ return path.join(projectPath, CREWLY_CONSTANTS.PATHS.CREWLY_HOME, MEMORY_CONSTANTS.PATHS.KNOWLEDGE_DIR, MEMORY_CONSTANTS.PROJECT_FILES.LEARNINGS);
198
+ }
199
+ /**
200
+ * Read and parse `learnings.md` for the given scope.
201
+ *
202
+ * Missing-file and read-error cases both return an empty array — the
203
+ * indexer must never fail callers just because no learnings exist yet.
204
+ *
205
+ * @param scope - Document scope
206
+ * @param projectPath - Project path (required for 'project' scope)
207
+ * @returns Parsed learning entries (empty if file missing or unreadable)
208
+ */
209
+ async readLearnings(scope, projectPath) {
210
+ const filePath = this.learningsPath(scope, projectPath);
211
+ if (!filePath || !existsSync(filePath)) {
212
+ return [];
213
+ }
214
+ try {
215
+ const raw = await fs.readFile(filePath, 'utf-8');
216
+ return parseLearnings(raw);
217
+ }
218
+ catch (error) {
219
+ this.logger.warn('Failed to read learnings.md', {
220
+ path: filePath,
221
+ error: error instanceof Error ? error.message : String(error),
222
+ });
223
+ return [];
224
+ }
225
+ }
226
+ /**
227
+ * List all learnings for the given scope as `KnowledgeDocumentSummary` entries.
228
+ *
229
+ * Sorted newest-first so the UI shows recent learnings at the top of the
230
+ * unified knowledge list (matching the user mental model that learnings
231
+ * are a chronological journal).
232
+ *
233
+ * @param scope - 'global' or 'project'
234
+ * @param projectPath - Required when scope is 'project'
235
+ * @returns Array of summaries (empty if scope is 'global' or no learnings exist)
236
+ */
237
+ async listLearnings(scope, projectPath) {
238
+ const parsed = await this.readLearnings(scope, projectPath);
239
+ return parsed
240
+ .map((e) => this.toSummary(e, scope))
241
+ .sort((a, b) => b.createdAt.localeCompare(a.createdAt));
242
+ }
243
+ /**
244
+ * Fetch a single learning by its synthetic ID.
245
+ *
246
+ * @param id - Synthetic document ID (must start with `learning:`)
247
+ * @param scope - Document scope
248
+ * @param projectPath - Required when scope is 'project'
249
+ * @returns Full document, or null if id is not a learning id or not found
250
+ */
251
+ async getLearning(id, scope, projectPath) {
252
+ if (!LearningsIndexService.isLearningId(id)) {
253
+ return null;
254
+ }
255
+ const parsed = await this.readLearnings(scope, projectPath);
256
+ const found = parsed.find((e) => e.id === id);
257
+ return found ? this.toDocument(found, scope, projectPath) : null;
258
+ }
259
+ /**
260
+ * Project a parsed entry into a `KnowledgeDocumentSummary`.
261
+ *
262
+ * @param e - Parsed learning entry
263
+ * @param scope - Target scope label
264
+ * @returns Summary suitable for the knowledge list API
265
+ */
266
+ toSummary(e, scope) {
267
+ return {
268
+ id: e.id,
269
+ title: e.title,
270
+ category: LEARNINGS_CATEGORY,
271
+ tags: e.tags,
272
+ preview: e.content.slice(0, KNOWLEDGE_CONSTANTS.PREVIEW_LENGTH),
273
+ scope,
274
+ createdBy: e.author,
275
+ updatedBy: e.author,
276
+ createdAt: e.createdAt,
277
+ updatedAt: e.updatedAt,
278
+ };
279
+ }
280
+ /**
281
+ * Project a parsed entry into a full `KnowledgeDocument`.
282
+ *
283
+ * @param e - Parsed learning entry
284
+ * @param scope - Target scope label
285
+ * @param projectPath - Carried into the document for project-scoped entries
286
+ * @returns Full document suitable for the knowledge document API
287
+ */
288
+ toDocument(e, scope, projectPath) {
289
+ return {
290
+ id: e.id,
291
+ title: e.title,
292
+ category: LEARNINGS_CATEGORY,
293
+ tags: e.tags,
294
+ content: e.content,
295
+ scope,
296
+ projectPath: scope === 'project' ? projectPath : undefined,
297
+ createdBy: e.author,
298
+ updatedBy: e.author,
299
+ createdAt: e.createdAt,
300
+ updatedAt: e.updatedAt,
301
+ };
302
+ }
303
+ }
304
+ //# sourceMappingURL=learnings-index.service.js.map