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.
- package/config/constants.d.ts.map +1 -0
- package/config/index.d.ts.map +1 -0
- package/config/roles/_common/memory-instructions.md +6 -5
- package/config/roles/_common/wiki-instructions.md +49 -0
- package/config/roles/architect/prompt.md +2 -2
- package/config/roles/backend-developer/prompt.md +2 -2
- package/config/roles/designer/prompt.md +2 -2
- package/config/roles/developer/prompt.md +2 -2
- package/config/roles/frontend-developer/prompt.md +2 -2
- package/config/roles/fullstack-dev/prompt.md +2 -2
- package/config/roles/generalist/prompt.md +2 -2
- package/config/roles/ops/prompt.md +2 -2
- package/config/roles/orchestrator/prompt.md +135 -11
- package/config/roles/product-manager/prompt.md +2 -2
- package/config/roles/qa/prompt.md +2 -2
- package/config/roles/qa-engineer/prompt.md +2 -2
- package/config/roles/researcher/prompt.md +15 -6
- package/config/roles/sales/prompt.md +2 -2
- package/config/roles/support/prompt.md +2 -2
- package/config/roles/team-leader/prompt.md +17 -2
- package/config/roles/tpm/prompt.md +2 -2
- package/config/roles/ux-designer/prompt.md +2 -2
- package/config/skills/orchestrator/wiki-cleanup/SKILL.md +89 -0
- package/config/skills/orchestrator/wiki-cleanup/execute.sh +139 -0
- package/config/skills/orchestrator/wiki-lint/SKILL.md +75 -0
- package/config/skills/orchestrator/wiki-lint/execute.sh +66 -0
- package/config/skills/orchestrator/wiki-migrate/SKILL.md +103 -0
- package/config/skills/orchestrator/wiki-migrate/execute.sh +82 -0
- package/config/skills/orchestrator/wiki-process-queue/SKILL.md +9 -1
- package/dist/backend/backend/src/controllers/task-management/task-management.controller.d.ts +169 -0
- package/dist/backend/backend/src/controllers/task-management/task-management.controller.d.ts.map +1 -0
- package/dist/backend/backend/src/controllers/task-management/task-management.controller.js +1779 -0
- package/dist/backend/backend/src/controllers/task-management/task-management.controller.js.map +1 -0
- package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.d.ts +18 -0
- package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.d.ts.map +1 -1
- package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.js +63 -0
- package/dist/backend/backend/src/controllers/task-pool/task-pool.controller.js.map +1 -1
- package/dist/backend/backend/src/controllers/task-pool/task-pool.routes.d.ts.map +1 -1
- package/dist/backend/backend/src/controllers/task-pool/task-pool.routes.js +5 -1
- package/dist/backend/backend/src/controllers/task-pool/task-pool.routes.js.map +1 -1
- package/dist/backend/backend/src/controllers/wiki/wiki.controller.d.ts +109 -0
- package/dist/backend/backend/src/controllers/wiki/wiki.controller.d.ts.map +1 -1
- package/dist/backend/backend/src/controllers/wiki/wiki.controller.js +418 -4
- package/dist/backend/backend/src/controllers/wiki/wiki.controller.js.map +1 -1
- package/dist/backend/backend/src/controllers/wiki/wiki.routes.d.ts.map +1 -1
- package/dist/backend/backend/src/controllers/wiki/wiki.routes.js +11 -1
- package/dist/backend/backend/src/controllers/wiki/wiki.routes.js.map +1 -1
- package/dist/backend/backend/src/index.d.ts.map +1 -1
- package/dist/backend/backend/src/index.js +64 -0
- package/dist/backend/backend/src/index.js.map +1 -1
- package/dist/backend/backend/src/index.js.orc-bak-20260529 +3130 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.d.ts +513 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.js +1568 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-runner.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.d.ts +86 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.js +147 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/agent-worker.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/api-client.d.ts +68 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/api-client.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/api-client.js +131 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/api-client.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.d.ts +130 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.js +263 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-log.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.d.ts +74 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.js +140 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/audit-trail.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.d.ts +29 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.js +279 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/auditor-tools.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.d.ts +340 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.js +1176 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/crewly-agent-runtime.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.d.ts +79 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.js +145 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/deepseek-sse-transform.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.d.ts +79 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.js +218 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/env-isolation.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/index.d.ts +16 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/index.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/index.js +16 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/index.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.d.ts +135 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.js +185 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/mcp-tool-bridge.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.d.ts +141 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.js +310 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/model-manager.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.d.ts +91 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.js +143 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/output-filter.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.d.ts +103 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.js +256 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/prompt-guard.service.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.d.ts +143 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.js +264 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/rate-limiter.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.d.ts +13 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.js +91 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/smoke-test.js.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.d.ts +135 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.d.ts.map +1 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.js +1937 -0
- package/dist/backend/backend/src/services/agent/crewly-agent/tool-registry.js.map +1 -0
- package/dist/backend/backend/src/services/ai/prompt-builder.service.js +1 -1
- package/dist/backend/backend/src/services/autonomous/auto-assign.service.d.ts +429 -0
- package/dist/backend/backend/src/services/autonomous/auto-assign.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/autonomous/auto-assign.service.js +852 -0
- package/dist/backend/backend/src/services/autonomous/auto-assign.service.js.map +1 -0
- package/dist/backend/backend/src/services/project/task-tracking.service.d.ts +171 -0
- package/dist/backend/backend/src/services/project/task-tracking.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/project/task-tracking.service.js +725 -0
- package/dist/backend/backend/src/services/project/task-tracking.service.js.map +1 -0
- package/dist/backend/backend/src/services/reconciler/reconciler-data-provider.d.ts.map +1 -1
- package/dist/backend/backend/src/services/reconciler/reconciler-data-provider.js +50 -0
- package/dist/backend/backend/src/services/reconciler/reconciler-data-provider.js.map +1 -1
- package/dist/backend/backend/src/services/task-pool/task-pool.service.d.ts +19 -0
- package/dist/backend/backend/src/services/task-pool/task-pool.service.d.ts.map +1 -1
- package/dist/backend/backend/src/services/task-pool/task-pool.service.js +45 -0
- package/dist/backend/backend/src/services/task-pool/task-pool.service.js.map +1 -1
- package/dist/backend/backend/src/services/v3/agent-auto-claim.service.d.ts.map +1 -1
- package/dist/backend/backend/src/services/v3/agent-auto-claim.service.js +34 -1
- package/dist/backend/backend/src/services/v3/agent-auto-claim.service.js.map +1 -1
- package/dist/backend/backend/src/services/v3/project-task-watcher.service.d.ts +118 -0
- package/dist/backend/backend/src/services/v3/project-task-watcher.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/v3/project-task-watcher.service.js +326 -0
- package/dist/backend/backend/src/services/v3/project-task-watcher.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.d.ts +72 -0
- package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.js +186 -0
- package/dist/backend/backend/src/services/wiki/wiki-backlinks.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.d.ts +4 -1
- package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.d.ts.map +1 -1
- package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.js +24 -1
- package/dist/backend/backend/src/services/wiki/wiki-bookkeep-trigger.service.js.map +1 -1
- package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.d.ts +74 -0
- package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.js +154 -0
- package/dist/backend/backend/src/services/wiki/wiki-chat-subscriber.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.d.ts +160 -0
- package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.js +399 -0
- package/dist/backend/backend/src/services/wiki/wiki-cleanup.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-lint.service.d.ts +182 -0
- package/dist/backend/backend/src/services/wiki/wiki-lint.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-lint.service.js +505 -0
- package/dist/backend/backend/src/services/wiki/wiki-lint.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-migrate.service.d.ts +232 -0
- package/dist/backend/backend/src/services/wiki/wiki-migrate.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-migrate.service.js +1416 -0
- package/dist/backend/backend/src/services/wiki/wiki-migrate.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-recent.service.d.ts +51 -0
- package/dist/backend/backend/src/services/wiki/wiki-recent.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-recent.service.js +102 -0
- package/dist/backend/backend/src/services/wiki/wiki-recent.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.d.ts +84 -0
- package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.js +156 -0
- package/dist/backend/backend/src/services/wiki/wiki-reflect-trigger.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-search.service.d.ts +90 -0
- package/dist/backend/backend/src/services/wiki/wiki-search.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-search.service.js +190 -0
- package/dist/backend/backend/src/services/wiki/wiki-search.service.js.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.d.ts +164 -0
- package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.d.ts.map +1 -0
- package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.js +675 -0
- package/dist/backend/backend/src/services/wiki/wiki-workitem-bridge.service.js.map +1 -0
- package/dist/backend/backend/src/services/workflow/cron-task.service.d.ts.map +1 -1
- package/dist/backend/backend/src/services/workflow/cron-task.service.js +65 -0
- package/dist/backend/backend/src/services/workflow/cron-task.service.js.map +1 -1
- package/dist/backend/backend/src/types/auto-assign.types.d.ts +271 -0
- package/dist/backend/backend/src/types/auto-assign.types.d.ts.map +1 -0
- package/dist/backend/backend/src/types/auto-assign.types.js +136 -0
- package/dist/backend/backend/src/types/auto-assign.types.js.map +1 -0
- package/dist/backend/backend/src/types/cron-task.types.d.ts +16 -1
- package/dist/backend/backend/src/types/cron-task.types.d.ts.map +1 -1
- package/dist/backend/backend/src/utils/esm-require.utils.d.ts +111 -0
- package/dist/backend/backend/src/utils/esm-require.utils.d.ts.map +1 -0
- package/dist/backend/backend/src/utils/esm-require.utils.js +124 -0
- package/dist/backend/backend/src/utils/esm-require.utils.js.map +1 -0
- package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.d.ts +220 -0
- package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.d.ts.map +1 -0
- package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.js +37 -0
- package/dist/cli/backend/src/services/ai/prompt-modules/prompt-module.interface.js.map +1 -0
- package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.d.ts +56 -0
- package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.d.ts.map +1 -0
- package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.js +91 -0
- package/dist/cli/backend/src/services/knowledge/fts5-search-strategy.js.map +1 -0
- package/dist/cli/backend/src/services/knowledge/learnings-index.service.d.ts +159 -0
- package/dist/cli/backend/src/services/knowledge/learnings-index.service.d.ts.map +1 -0
- package/dist/cli/backend/src/services/knowledge/learnings-index.service.js +304 -0
- package/dist/cli/backend/src/services/knowledge/learnings-index.service.js.map +1 -0
- package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.d.ts +115 -0
- package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.d.ts.map +1 -0
- package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.js +215 -0
- package/dist/cli/backend/src/services/knowledge/wiki-compiler.service.js.map +1 -0
- package/dist/cli/backend/src/services/memory/embedding-provider.d.ts +78 -0
- package/dist/cli/backend/src/services/memory/embedding-provider.d.ts.map +1 -0
- package/dist/cli/backend/src/services/memory/embedding-provider.js +179 -0
- package/dist/cli/backend/src/services/memory/embedding-provider.js.map +1 -0
- package/dist/cli/backend/src/services/memory/vector-store.service.d.ts +331 -0
- package/dist/cli/backend/src/services/memory/vector-store.service.d.ts.map +1 -0
- package/dist/cli/backend/src/services/memory/vector-store.service.js +814 -0
- package/dist/cli/backend/src/services/memory/vector-store.service.js.map +1 -0
- package/dist/cli/backend/src/services/project/task-tracking.service.d.ts +171 -0
- package/dist/cli/backend/src/services/project/task-tracking.service.d.ts.map +1 -0
- package/dist/cli/backend/src/services/project/task-tracking.service.js +725 -0
- package/dist/cli/backend/src/services/project/task-tracking.service.js.map +1 -0
- package/dist/cli/backend/src/services/task-pool/task-pool.service.d.ts +19 -0
- package/dist/cli/backend/src/services/task-pool/task-pool.service.d.ts.map +1 -1
- package/dist/cli/backend/src/services/task-pool/task-pool.service.js +45 -0
- package/dist/cli/backend/src/services/task-pool/task-pool.service.js.map +1 -1
- package/dist/cli/backend/src/types/auto-assign.types.d.ts +271 -0
- package/dist/cli/backend/src/types/auto-assign.types.d.ts.map +1 -0
- package/dist/cli/backend/src/types/auto-assign.types.js +136 -0
- package/dist/cli/backend/src/types/auto-assign.types.js.map +1 -0
- package/dist/cli/cli/src/index.js +0 -0
- package/frontend/dist/assets/{index-db3f5041.css → index-068bb4f6.css} +10 -1
- package/frontend/dist/assets/index-c24ceb15.js +4960 -0
- package/frontend/dist/index.html +2 -2
- package/package.json +1 -1
- package/config/skills/agent/core/query-knowledge/SKILL.md +0 -87
- package/config/skills/agent/core/query-knowledge/execute.sh +0 -30
- package/config/skills/orchestrator/query-knowledge/SKILL.md +0 -75
- package/config/skills/orchestrator/query-knowledge/execute.sh +0 -30
- 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
|