prjct-cli 1.22.0 → 1.23.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +147 -0
- package/bin/prjct +30 -13
- package/dist/bin/prjct.mjs +917 -35845
- package/dist/bin/prjct.mjs.map +7 -0
- package/dist/cli/linear.mjs +16 -0
- package/dist/cli/linear.mjs.map +7 -0
- package/dist/templates.json +1 -0
- package/package.json +4 -5
- package/bin/prjct.ts +0 -342
- package/core/__tests__/agentic/analysis-injection.test.ts +0 -377
- package/core/__tests__/agentic/cache-eviction.test.ts +0 -294
- package/core/__tests__/agentic/command-context.test.ts +0 -281
- package/core/__tests__/agentic/command-executor.test.ts +0 -659
- package/core/__tests__/agentic/domain-classifier.test.ts +0 -330
- package/core/__tests__/agentic/injection-validator.test.ts +0 -255
- package/core/__tests__/agentic/memory-system.test.ts +0 -281
- package/core/__tests__/agentic/plan-mode.test.ts +0 -386
- package/core/__tests__/agentic/prompt-assembly.test.ts +0 -298
- package/core/__tests__/agentic/prompt-builder.test.ts +0 -243
- package/core/__tests__/agentic/response-validator.test.ts +0 -263
- package/core/__tests__/agentic/semantic-matching.test.ts +0 -131
- package/core/__tests__/agentic/smart-context.test.ts +0 -372
- package/core/__tests__/agentic/tech-normalizer.test.ts +0 -136
- package/core/__tests__/agentic/token-budget.test.ts +0 -294
- package/core/__tests__/ai-tools/formatters.test.ts +0 -476
- package/core/__tests__/domain/bm25.test.ts +0 -225
- package/core/__tests__/domain/change-propagator.test.ts +0 -100
- package/core/__tests__/domain/fibonacci.test.ts +0 -113
- package/core/__tests__/domain/file-hasher.test.ts +0 -146
- package/core/__tests__/domain/file-ranker.test.ts +0 -169
- package/core/__tests__/domain/git-cochange.test.ts +0 -121
- package/core/__tests__/domain/import-graph.test.ts +0 -156
- package/core/__tests__/domain/velocity.test.ts +0 -623
- package/core/__tests__/infrastructure/performance-tracker.test.ts +0 -328
- package/core/__tests__/schemas/model.test.ts +0 -272
- package/core/__tests__/services/dependency-validator.test.ts +0 -175
- package/core/__tests__/services/hierarchical-agent-resolver.test.ts +0 -359
- package/core/__tests__/services/nested-context-resolver.test.ts +0 -443
- package/core/__tests__/services/project-index.test.ts +0 -355
- package/core/__tests__/services/staleness-checker.test.ts +0 -204
- package/core/__tests__/storage/analysis-storage.test.ts +0 -641
- package/core/__tests__/storage/archive-storage.test.ts +0 -455
- package/core/__tests__/storage/safe-reader.test.ts +0 -262
- package/core/__tests__/storage/sqlite-migration.test.ts +0 -1016
- package/core/__tests__/storage/state-storage-feedback.test.ts +0 -463
- package/core/__tests__/storage/state-storage-history.test.ts +0 -469
- package/core/__tests__/storage/storage-manager.test.ts +0 -383
- package/core/__tests__/storage/subtask-handoff.test.ts +0 -237
- package/core/__tests__/types/fs.test.ts +0 -125
- package/core/__tests__/utils/date-helper.test.ts +0 -449
- package/core/__tests__/utils/output.test.ts +0 -278
- package/core/__tests__/utils/preserve-sections.test.ts +0 -216
- package/core/__tests__/utils/project-commands.test.ts +0 -71
- package/core/__tests__/utils/retry.test.ts +0 -381
- package/core/__tests__/workflow/state-machine.test.ts +0 -216
- package/core/agentic/agent-router.ts +0 -150
- package/core/agentic/anti-hallucination.ts +0 -141
- package/core/agentic/chain-of-thought.ts +0 -234
- package/core/agentic/command-classifier.ts +0 -141
- package/core/agentic/command-context.ts +0 -168
- package/core/agentic/command-executor.ts +0 -471
- package/core/agentic/context-builder.ts +0 -285
- package/core/agentic/domain-classifier.ts +0 -525
- package/core/agentic/environment-block.ts +0 -102
- package/core/agentic/ground-truth.ts +0 -706
- package/core/agentic/index.ts +0 -193
- package/core/agentic/injection-validator.ts +0 -208
- package/core/agentic/loop-detector.ts +0 -451
- package/core/agentic/memory-system.ts +0 -1547
- package/core/agentic/orchestrator-executor.ts +0 -579
- package/core/agentic/plan-mode.ts +0 -525
- package/core/agentic/prompt-builder.ts +0 -1069
- package/core/agentic/response-validator.ts +0 -98
- package/core/agentic/services.ts +0 -167
- package/core/agentic/skill-loader.ts +0 -106
- package/core/agentic/smart-context.ts +0 -393
- package/core/agentic/tech-normalizer.ts +0 -167
- package/core/agentic/template-executor.ts +0 -272
- package/core/agentic/template-loader.ts +0 -109
- package/core/agentic/token-budget.ts +0 -226
- package/core/agentic/tool-registry.ts +0 -146
- package/core/agents/index.ts +0 -28
- package/core/agents/performance.ts +0 -429
- package/core/ai-tools/formatters.ts +0 -341
- package/core/ai-tools/generator.ts +0 -144
- package/core/ai-tools/index.ts +0 -15
- package/core/ai-tools/registry.ts +0 -201
- package/core/bus/bus.ts +0 -314
- package/core/bus/index.ts +0 -8
- package/core/cli/linear.ts +0 -500
- package/core/cli/lint-meta-commentary.ts +0 -177
- package/core/cli/start.ts +0 -386
- package/core/commands/analysis.ts +0 -1274
- package/core/commands/analytics.ts +0 -342
- package/core/commands/base.ts +0 -118
- package/core/commands/cleanup.ts +0 -157
- package/core/commands/command-data.ts +0 -463
- package/core/commands/commands.ts +0 -306
- package/core/commands/context.ts +0 -238
- package/core/commands/design.ts +0 -77
- package/core/commands/index.ts +0 -19
- package/core/commands/maintenance.ts +0 -77
- package/core/commands/performance.ts +0 -114
- package/core/commands/planning.ts +0 -662
- package/core/commands/register.ts +0 -127
- package/core/commands/registry.ts +0 -444
- package/core/commands/setup.ts +0 -280
- package/core/commands/shipping.ts +0 -267
- package/core/commands/snapshots.ts +0 -297
- package/core/commands/uninstall.ts +0 -542
- package/core/commands/velocity.ts +0 -149
- package/core/commands/workflow.ts +0 -505
- package/core/config/command-context.config.json +0 -66
- package/core/constants/index.ts +0 -379
- package/core/context/generator.ts +0 -368
- package/core/context-tools/files-tool.ts +0 -577
- package/core/context-tools/imports-tool.ts +0 -400
- package/core/context-tools/index.ts +0 -434
- package/core/context-tools/recent-tool.ts +0 -301
- package/core/context-tools/signatures-tool.ts +0 -495
- package/core/context-tools/summary-tool.ts +0 -301
- package/core/context-tools/token-counter.ts +0 -273
- package/core/context-tools/types.ts +0 -253
- package/core/domain/agent-generator.ts +0 -186
- package/core/domain/agent-loader.ts +0 -419
- package/core/domain/analyzer.ts +0 -387
- package/core/domain/architecture-generator.ts +0 -108
- package/core/domain/bm25.ts +0 -525
- package/core/domain/change-propagator.ts +0 -162
- package/core/domain/context-estimator.ts +0 -175
- package/core/domain/fibonacci.ts +0 -128
- package/core/domain/file-hasher.ts +0 -296
- package/core/domain/file-ranker.ts +0 -151
- package/core/domain/git-cochange.ts +0 -250
- package/core/domain/import-graph.ts +0 -315
- package/core/domain/snapshot-manager.ts +0 -415
- package/core/domain/task-stack.ts +0 -578
- package/core/domain/velocity.ts +0 -470
- package/core/errors.ts +0 -335
- package/core/events/events.ts +0 -85
- package/core/events/index.ts +0 -8
- package/core/index.ts +0 -481
- package/core/infrastructure/agent-detector.ts +0 -135
- package/core/infrastructure/ai-provider.ts +0 -578
- package/core/infrastructure/author-detector.ts +0 -133
- package/core/infrastructure/capability-installer.ts +0 -76
- package/core/infrastructure/claude-agent.ts +0 -297
- package/core/infrastructure/command-installer.ts +0 -752
- package/core/infrastructure/config-manager.ts +0 -364
- package/core/infrastructure/editors-config.ts +0 -172
- package/core/infrastructure/path-manager.ts +0 -571
- package/core/infrastructure/performance-tracker.ts +0 -326
- package/core/infrastructure/permission-manager.ts +0 -289
- package/core/infrastructure/setup.ts +0 -1061
- package/core/infrastructure/update-checker.ts +0 -246
- package/core/integrations/issue-tracker/enricher.ts +0 -271
- package/core/integrations/issue-tracker/index.ts +0 -8
- package/core/integrations/issue-tracker/manager.ts +0 -286
- package/core/integrations/issue-tracker/types.ts +0 -310
- package/core/integrations/jira/cache.ts +0 -57
- package/core/integrations/jira/client.ts +0 -688
- package/core/integrations/jira/index.ts +0 -23
- package/core/integrations/jira/service.ts +0 -244
- package/core/integrations/linear/cache.ts +0 -68
- package/core/integrations/linear/client.ts +0 -436
- package/core/integrations/linear/index.ts +0 -20
- package/core/integrations/linear/service.ts +0 -260
- package/core/integrations/linear/sync.ts +0 -314
- package/core/outcomes/analyzer.ts +0 -286
- package/core/outcomes/index.ts +0 -34
- package/core/outcomes/recorder.ts +0 -195
- package/core/plugin/builtin/webhook.ts +0 -148
- package/core/plugin/hooks.ts +0 -315
- package/core/plugin/index.ts +0 -50
- package/core/plugin/loader.ts +0 -354
- package/core/plugin/registry.ts +0 -326
- package/core/schemas/agents.ts +0 -27
- package/core/schemas/analysis.ts +0 -530
- package/core/schemas/classification.ts +0 -91
- package/core/schemas/command-context.ts +0 -29
- package/core/schemas/enriched-task.ts +0 -291
- package/core/schemas/ideas.ts +0 -114
- package/core/schemas/index.ts +0 -53
- package/core/schemas/issues.ts +0 -159
- package/core/schemas/llm-output.ts +0 -170
- package/core/schemas/metrics.ts +0 -143
- package/core/schemas/model.ts +0 -153
- package/core/schemas/outcomes.ts +0 -487
- package/core/schemas/performance.ts +0 -128
- package/core/schemas/permissions.ts +0 -180
- package/core/schemas/prd.ts +0 -450
- package/core/schemas/project.ts +0 -57
- package/core/schemas/roadmap.ts +0 -322
- package/core/schemas/schemas.ts +0 -38
- package/core/schemas/shipped.ts +0 -109
- package/core/schemas/state.ts +0 -284
- package/core/schemas/velocity.ts +0 -103
- package/core/server/index.ts +0 -21
- package/core/server/routes-extended.ts +0 -566
- package/core/server/routes.ts +0 -176
- package/core/server/server.ts +0 -149
- package/core/server/sse.ts +0 -192
- package/core/services/agent-generator.ts +0 -385
- package/core/services/agent-service.ts +0 -168
- package/core/services/breakdown-service.ts +0 -124
- package/core/services/context-generator.ts +0 -445
- package/core/services/context-selector.ts +0 -429
- package/core/services/dependency-validator.ts +0 -318
- package/core/services/diff-generator.ts +0 -313
- package/core/services/doctor-service.ts +0 -423
- package/core/services/file-categorizer.ts +0 -448
- package/core/services/file-scorer.ts +0 -270
- package/core/services/git-analyzer.ts +0 -293
- package/core/services/hierarchical-agent-resolver.ts +0 -236
- package/core/services/hooks-service.ts +0 -685
- package/core/services/index.ts +0 -46
- package/core/services/local-state-generator.ts +0 -158
- package/core/services/memory-service.ts +0 -181
- package/core/services/nested-context-resolver.ts +0 -842
- package/core/services/project-index.ts +0 -911
- package/core/services/project-service.ts +0 -155
- package/core/services/session-tracker.ts +0 -287
- package/core/services/skill-installer.ts +0 -447
- package/core/services/skill-lock.ts +0 -132
- package/core/services/skill-service.ts +0 -306
- package/core/services/stack-detector.ts +0 -229
- package/core/services/staleness-checker.ts +0 -327
- package/core/services/sync-service.ts +0 -1515
- package/core/services/sync-verifier.ts +0 -253
- package/core/services/watch-service.ts +0 -312
- package/core/session/compaction.ts +0 -248
- package/core/session/index.ts +0 -35
- package/core/session/log-migration.ts +0 -88
- package/core/session/metrics.ts +0 -323
- package/core/session/session-log-manager.ts +0 -307
- package/core/session/task-session-manager.ts +0 -404
- package/core/session/utils.ts +0 -51
- package/core/storage/analysis-storage.ts +0 -373
- package/core/storage/archive-storage.ts +0 -205
- package/core/storage/database.ts +0 -575
- package/core/storage/ideas-storage.ts +0 -298
- package/core/storage/index-storage.ts +0 -523
- package/core/storage/index.ts +0 -79
- package/core/storage/metrics-storage.ts +0 -321
- package/core/storage/migrate-json.ts +0 -720
- package/core/storage/queue-storage.ts +0 -336
- package/core/storage/safe-reader.ts +0 -105
- package/core/storage/shipped-storage.ts +0 -253
- package/core/storage/state-storage.ts +0 -1035
- package/core/storage/storage-manager.ts +0 -205
- package/core/storage/storage.ts +0 -177
- package/core/storage/velocity-storage.ts +0 -149
- package/core/sync/auth-config.ts +0 -138
- package/core/sync/index.ts +0 -31
- package/core/sync/oauth-handler.ts +0 -143
- package/core/sync/sync-client.ts +0 -251
- package/core/sync/sync-manager.ts +0 -327
- package/core/tsconfig.json +0 -22
- package/core/types/agentic.ts +0 -760
- package/core/types/agents.ts +0 -150
- package/core/types/bus.ts +0 -193
- package/core/types/citations.ts +0 -22
- package/core/types/commands.ts +0 -399
- package/core/types/config.ts +0 -92
- package/core/types/core.ts +0 -96
- package/core/types/diff.ts +0 -41
- package/core/types/domain.ts +0 -71
- package/core/types/errors.ts +0 -111
- package/core/types/events.ts +0 -42
- package/core/types/fs.ts +0 -72
- package/core/types/index.ts +0 -510
- package/core/types/infrastructure.ts +0 -210
- package/core/types/integrations.ts +0 -31
- package/core/types/jira.ts +0 -51
- package/core/types/logger.ts +0 -17
- package/core/types/memory.ts +0 -313
- package/core/types/outcomes.ts +0 -190
- package/core/types/output.ts +0 -47
- package/core/types/plugin.ts +0 -25
- package/core/types/project-sync.ts +0 -129
- package/core/types/provider.ts +0 -163
- package/core/types/server.ts +0 -71
- package/core/types/services.ts +0 -84
- package/core/types/session.ts +0 -135
- package/core/types/stack.ts +0 -19
- package/core/types/storage.ts +0 -318
- package/core/types/sync-verifier.ts +0 -33
- package/core/types/sync.ts +0 -121
- package/core/types/task.ts +0 -72
- package/core/types/template.ts +0 -24
- package/core/types/utils.ts +0 -92
- package/core/types/workflow.ts +0 -23
- package/core/utils/agent-stream.ts +0 -140
- package/core/utils/animations.ts +0 -251
- package/core/utils/branding.ts +0 -88
- package/core/utils/cache.ts +0 -187
- package/core/utils/citations.ts +0 -39
- package/core/utils/collection-filters.ts +0 -209
- package/core/utils/date-helper.ts +0 -176
- package/core/utils/error-messages.ts +0 -38
- package/core/utils/file-helper.ts +0 -277
- package/core/utils/fs-helpers.ts +0 -14
- package/core/utils/help.ts +0 -314
- package/core/utils/jsonl-helper.ts +0 -290
- package/core/utils/keychain.ts +0 -127
- package/core/utils/logger.ts +0 -77
- package/core/utils/markdown-builder.ts +0 -280
- package/core/utils/next-steps.ts +0 -95
- package/core/utils/output.ts +0 -403
- package/core/utils/preserve-sections.ts +0 -218
- package/core/utils/project-commands.ts +0 -126
- package/core/utils/project-credentials.ts +0 -143
- package/core/utils/provider-cache.ts +0 -49
- package/core/utils/retry.ts +0 -318
- package/core/utils/runtime.ts +0 -108
- package/core/utils/session-helper.ts +0 -278
- package/core/utils/subtask-table.ts +0 -227
- package/core/utils/version.ts +0 -128
- package/core/wizard/index.ts +0 -13
- package/core/wizard/onboarding.ts +0 -633
- package/core/workflow/index.ts +0 -7
- package/core/workflow/state-machine.ts +0 -198
- package/core/workflow/workflow-preferences.ts +0 -294
- package/dist/core/infrastructure/command-installer.js +0 -1141
- package/dist/core/infrastructure/editors-config.js +0 -177
- package/dist/core/infrastructure/setup.js +0 -2244
- package/dist/core/utils/version.js +0 -141
- package/templates/agentic/agent-routing.md +0 -45
- package/templates/agentic/agents/uxui.md +0 -63
- package/templates/agentic/checklist-routing.md +0 -98
- package/templates/agentic/orchestrator.md +0 -68
- package/templates/agentic/task-fragmentation.md +0 -89
- package/templates/agents/AGENTS.md +0 -68
- package/templates/analysis/analyze.md +0 -84
- package/templates/analysis/patterns.md +0 -60
- package/templates/antigravity/SKILL.md +0 -39
- package/templates/architect/discovery.md +0 -67
- package/templates/architect/phases.md +0 -59
- package/templates/checklists/architecture.md +0 -28
- package/templates/checklists/code-quality.md +0 -28
- package/templates/checklists/data.md +0 -33
- package/templates/checklists/documentation.md +0 -33
- package/templates/checklists/infrastructure.md +0 -33
- package/templates/checklists/performance.md +0 -33
- package/templates/checklists/security.md +0 -33
- package/templates/checklists/testing.md +0 -33
- package/templates/checklists/ux-ui.md +0 -37
- package/templates/commands/analyze.md +0 -56
- package/templates/commands/auth.md +0 -234
- package/templates/commands/bug.md +0 -163
- package/templates/commands/cleanup.md +0 -19
- package/templates/commands/dash.md +0 -99
- package/templates/commands/design.md +0 -15
- package/templates/commands/done.md +0 -291
- package/templates/commands/enrich.md +0 -174
- package/templates/commands/git.md +0 -295
- package/templates/commands/history.md +0 -389
- package/templates/commands/idea.md +0 -88
- package/templates/commands/impact.md +0 -864
- package/templates/commands/init.md +0 -54
- package/templates/commands/jira.md +0 -278
- package/templates/commands/linear.md +0 -288
- package/templates/commands/merge.md +0 -206
- package/templates/commands/next.md +0 -80
- package/templates/commands/p.md +0 -67
- package/templates/commands/p.toml +0 -37
- package/templates/commands/pause.md +0 -136
- package/templates/commands/plan.md +0 -696
- package/templates/commands/prd.md +0 -356
- package/templates/commands/resume.md +0 -171
- package/templates/commands/review.md +0 -276
- package/templates/commands/serve.md +0 -118
- package/templates/commands/setup.md +0 -91
- package/templates/commands/ship.md +0 -475
- package/templates/commands/skill.md +0 -259
- package/templates/commands/spec.md +0 -218
- package/templates/commands/status.md +0 -207
- package/templates/commands/sync.md +0 -104
- package/templates/commands/task.md +0 -312
- package/templates/commands/test.md +0 -93
- package/templates/commands/update.md +0 -63
- package/templates/commands/verify.md +0 -204
- package/templates/commands/workflow.md +0 -150
- package/templates/config/skill-mappings.json +0 -82
- package/templates/context/dashboard.md +0 -256
- package/templates/context/roadmap.md +0 -221
- package/templates/cursor/commands/bug.md +0 -8
- package/templates/cursor/commands/done.md +0 -4
- package/templates/cursor/commands/pause.md +0 -6
- package/templates/cursor/commands/resume.md +0 -4
- package/templates/cursor/commands/ship.md +0 -8
- package/templates/cursor/commands/sync.md +0 -4
- package/templates/cursor/commands/task.md +0 -8
- package/templates/cursor/p.md +0 -29
- package/templates/cursor/router.mdc +0 -28
- package/templates/design/api.md +0 -95
- package/templates/design/architecture.md +0 -77
- package/templates/design/component.md +0 -89
- package/templates/design/database.md +0 -78
- package/templates/design/flow.md +0 -94
- package/templates/global/ANTIGRAVITY.md +0 -254
- package/templates/global/CLAUDE.md +0 -497
- package/templates/global/CURSOR.mdc +0 -266
- package/templates/global/GEMINI.md +0 -293
- package/templates/global/STORAGE-SPEC.md +0 -391
- package/templates/global/WINDSURF.md +0 -266
- package/templates/global/modules/CLAUDE-commands.md +0 -70
- package/templates/global/modules/CLAUDE-core.md +0 -105
- package/templates/global/modules/CLAUDE-git.md +0 -50
- package/templates/global/modules/CLAUDE-intelligence.md +0 -92
- package/templates/global/modules/CLAUDE-storage.md +0 -50
- package/templates/global/modules/module-config.json +0 -36
- package/templates/mcp-config.json +0 -19
- package/templates/permissions/default.jsonc +0 -60
- package/templates/permissions/permissive.jsonc +0 -49
- package/templates/permissions/strict.jsonc +0 -58
- package/templates/planning-methodology.md +0 -195
- package/templates/skills/code-review.md +0 -47
- package/templates/skills/debug.md +0 -61
- package/templates/skills/refactor.md +0 -47
- package/templates/subagents/agent-base.md +0 -20
- package/templates/subagents/domain/backend.md +0 -109
- package/templates/subagents/domain/database.md +0 -121
- package/templates/subagents/domain/devops.md +0 -152
- package/templates/subagents/domain/frontend.md +0 -103
- package/templates/subagents/domain/testing.md +0 -169
- package/templates/subagents/pm-expert.md +0 -366
- package/templates/subagents/workflow/chief-architect.md +0 -657
- package/templates/subagents/workflow/prjct-planner.md +0 -159
- package/templates/subagents/workflow/prjct-shipper.md +0 -188
- package/templates/subagents/workflow/prjct-workflow.md +0 -98
- package/templates/tools/bash.txt +0 -22
- package/templates/tools/edit.txt +0 -18
- package/templates/tools/glob.txt +0 -19
- package/templates/tools/grep.txt +0 -21
- package/templates/tools/read.txt +0 -14
- package/templates/tools/task.txt +0 -20
- package/templates/tools/webfetch.txt +0 -16
- package/templates/tools/websearch.txt +0 -18
- package/templates/tools/write.txt +0 -17
- package/templates/windsurf/router.md +0 -28
- package/templates/windsurf/workflows/bug.md +0 -8
- package/templates/windsurf/workflows/done.md +0 -4
- package/templates/windsurf/workflows/pause.md +0 -4
- package/templates/windsurf/workflows/resume.md +0 -4
- package/templates/windsurf/workflows/ship.md +0 -8
- package/templates/windsurf/workflows/sync.md +0 -4
- package/templates/windsurf/workflows/task.md +0 -8
|
@@ -1,377 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Analysis Injection Tests (PRJ-260)
|
|
3
|
-
*
|
|
4
|
-
* Tests for injecting sealed analysis data into task context:
|
|
5
|
-
* - Prompt builder renders analysis in ground_truth section
|
|
6
|
-
* - Anti-hallucination block enriched with analysis data
|
|
7
|
-
* - Graceful degradation when no analysis available
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
import { beforeEach, describe, expect, it } from 'bun:test'
|
|
11
|
-
import {
|
|
12
|
-
buildAntiHallucinationBlock,
|
|
13
|
-
type ProjectGroundTruth,
|
|
14
|
-
} from '../../agentic/anti-hallucination'
|
|
15
|
-
import promptBuilder from '../../agentic/prompt-builder'
|
|
16
|
-
import type { OrchestratorContext, SealedAnalysisContext } from '../../types'
|
|
17
|
-
|
|
18
|
-
// =============================================================================
|
|
19
|
-
// Test Fixtures
|
|
20
|
-
// =============================================================================
|
|
21
|
-
|
|
22
|
-
const mockSealedAnalysis: SealedAnalysisContext = {
|
|
23
|
-
languages: ['TypeScript', 'JavaScript'],
|
|
24
|
-
frameworks: ['Hono', 'Zod'],
|
|
25
|
-
packageManager: 'bun',
|
|
26
|
-
sourceDir: 'core/',
|
|
27
|
-
testDir: 'core/__tests__/',
|
|
28
|
-
fileCount: 295,
|
|
29
|
-
patterns: [
|
|
30
|
-
{
|
|
31
|
-
name: 'StorageManager pattern',
|
|
32
|
-
description: 'All storage uses StorageManager base class with read/write/update',
|
|
33
|
-
location: 'core/storage/',
|
|
34
|
-
},
|
|
35
|
-
{
|
|
36
|
-
name: 'Zod schemas',
|
|
37
|
-
description: 'Runtime validation with Zod for all data structures',
|
|
38
|
-
},
|
|
39
|
-
],
|
|
40
|
-
antiPatterns: [
|
|
41
|
-
{
|
|
42
|
-
issue: 'Direct fs.writeFile without StorageManager',
|
|
43
|
-
file: 'core/storage/legacy.ts',
|
|
44
|
-
suggestion: 'Use StorageManager.write() instead',
|
|
45
|
-
},
|
|
46
|
-
],
|
|
47
|
-
status: 'sealed',
|
|
48
|
-
commitHash: 'abc123def456',
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
function makeOrchestratorContext(
|
|
52
|
-
sealedAnalysis: SealedAnalysisContext | null = mockSealedAnalysis
|
|
53
|
-
): OrchestratorContext {
|
|
54
|
-
return {
|
|
55
|
-
detectedDomains: ['backend', 'testing'],
|
|
56
|
-
primaryDomain: 'backend',
|
|
57
|
-
agents: [],
|
|
58
|
-
skills: [],
|
|
59
|
-
requiresFragmentation: false,
|
|
60
|
-
subtasks: null,
|
|
61
|
-
project: {
|
|
62
|
-
id: 'test-project',
|
|
63
|
-
ecosystem: 'TypeScript',
|
|
64
|
-
conventions: ['Hono', 'Zod'],
|
|
65
|
-
},
|
|
66
|
-
sealedAnalysis,
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
// =============================================================================
|
|
71
|
-
// Prompt Builder — Ground Truth Section
|
|
72
|
-
// =============================================================================
|
|
73
|
-
|
|
74
|
-
describe('Analysis Injection in Prompt Builder (PRJ-260)', () => {
|
|
75
|
-
beforeEach(() => {
|
|
76
|
-
promptBuilder.resetContext()
|
|
77
|
-
})
|
|
78
|
-
|
|
79
|
-
it('should render sealed analysis in ground truth section', async () => {
|
|
80
|
-
const template = {
|
|
81
|
-
frontmatter: { description: 'Test task' },
|
|
82
|
-
content: '## Instructions\nDo the work',
|
|
83
|
-
}
|
|
84
|
-
const context = { projectPath: '/test/project', files: ['a.ts'] }
|
|
85
|
-
const orcCtx = makeOrchestratorContext()
|
|
86
|
-
|
|
87
|
-
const prompt = await promptBuilder.build(
|
|
88
|
-
template,
|
|
89
|
-
context,
|
|
90
|
-
{},
|
|
91
|
-
null,
|
|
92
|
-
null,
|
|
93
|
-
null,
|
|
94
|
-
null,
|
|
95
|
-
null,
|
|
96
|
-
orcCtx
|
|
97
|
-
)
|
|
98
|
-
|
|
99
|
-
// Should contain analysis data
|
|
100
|
-
expect(prompt).toContain('Languages**: TypeScript, JavaScript')
|
|
101
|
-
expect(prompt).toContain('Frameworks**: Hono, Zod')
|
|
102
|
-
expect(prompt).toContain('Package Manager**: bun')
|
|
103
|
-
expect(prompt).toContain('Source Dir**: core/')
|
|
104
|
-
expect(prompt).toContain('Test Dir**: core/__tests__/')
|
|
105
|
-
expect(prompt).toContain('Files Analyzed**: 295')
|
|
106
|
-
expect(prompt).toContain('Analysis Status**: sealed')
|
|
107
|
-
expect(prompt).toContain('abc123de') // truncated commit hash
|
|
108
|
-
})
|
|
109
|
-
|
|
110
|
-
it('should render code patterns from sealed analysis', async () => {
|
|
111
|
-
const template = {
|
|
112
|
-
frontmatter: { description: 'Test' },
|
|
113
|
-
content: '## Do',
|
|
114
|
-
}
|
|
115
|
-
const context = { projectPath: '/test', files: [] }
|
|
116
|
-
const orcCtx = makeOrchestratorContext()
|
|
117
|
-
|
|
118
|
-
const prompt = await promptBuilder.build(
|
|
119
|
-
template,
|
|
120
|
-
context,
|
|
121
|
-
{},
|
|
122
|
-
null,
|
|
123
|
-
null,
|
|
124
|
-
null,
|
|
125
|
-
null,
|
|
126
|
-
null,
|
|
127
|
-
orcCtx
|
|
128
|
-
)
|
|
129
|
-
|
|
130
|
-
expect(prompt).toContain('Code Patterns (Follow These)')
|
|
131
|
-
expect(prompt).toContain('StorageManager pattern')
|
|
132
|
-
expect(prompt).toContain('All storage uses StorageManager base class')
|
|
133
|
-
expect(prompt).toContain('core/storage/')
|
|
134
|
-
expect(prompt).toContain('Zod schemas')
|
|
135
|
-
})
|
|
136
|
-
|
|
137
|
-
it('should render anti-patterns from sealed analysis', async () => {
|
|
138
|
-
const template = {
|
|
139
|
-
frontmatter: { description: 'Test' },
|
|
140
|
-
content: '## Do',
|
|
141
|
-
}
|
|
142
|
-
const context = { projectPath: '/test', files: [] }
|
|
143
|
-
const orcCtx = makeOrchestratorContext()
|
|
144
|
-
|
|
145
|
-
const prompt = await promptBuilder.build(
|
|
146
|
-
template,
|
|
147
|
-
context,
|
|
148
|
-
{},
|
|
149
|
-
null,
|
|
150
|
-
null,
|
|
151
|
-
null,
|
|
152
|
-
null,
|
|
153
|
-
null,
|
|
154
|
-
orcCtx
|
|
155
|
-
)
|
|
156
|
-
|
|
157
|
-
expect(prompt).toContain('Anti-Patterns (Avoid These)')
|
|
158
|
-
expect(prompt).toContain('Direct fs.writeFile without StorageManager')
|
|
159
|
-
expect(prompt).toContain('core/storage/legacy.ts')
|
|
160
|
-
expect(prompt).toContain('Use StorageManager.write() instead')
|
|
161
|
-
})
|
|
162
|
-
|
|
163
|
-
it('should gracefully handle null sealed analysis', async () => {
|
|
164
|
-
const template = {
|
|
165
|
-
frontmatter: { description: 'Test' },
|
|
166
|
-
content: '## Do',
|
|
167
|
-
}
|
|
168
|
-
const context = { projectPath: '/test', files: [] }
|
|
169
|
-
const orcCtx = makeOrchestratorContext(null)
|
|
170
|
-
|
|
171
|
-
const prompt = await promptBuilder.build(
|
|
172
|
-
template,
|
|
173
|
-
context,
|
|
174
|
-
{},
|
|
175
|
-
null,
|
|
176
|
-
null,
|
|
177
|
-
null,
|
|
178
|
-
null,
|
|
179
|
-
null,
|
|
180
|
-
orcCtx
|
|
181
|
-
)
|
|
182
|
-
|
|
183
|
-
// Should still have basic project analysis
|
|
184
|
-
expect(prompt).toContain('PROJECT ANALYSIS')
|
|
185
|
-
expect(prompt).toContain('Ecosystem**: TypeScript')
|
|
186
|
-
// Should NOT have analysis-specific fields
|
|
187
|
-
expect(prompt).not.toContain('Languages**:')
|
|
188
|
-
expect(prompt).not.toContain('Code Patterns (Follow These)')
|
|
189
|
-
expect(prompt).not.toContain('Anti-Patterns (Avoid These)')
|
|
190
|
-
})
|
|
191
|
-
|
|
192
|
-
it('should handle empty patterns and anti-patterns', async () => {
|
|
193
|
-
const template = {
|
|
194
|
-
frontmatter: { description: 'Test' },
|
|
195
|
-
content: '## Do',
|
|
196
|
-
}
|
|
197
|
-
const context = { projectPath: '/test', files: [] }
|
|
198
|
-
const emptyAnalysis: SealedAnalysisContext = {
|
|
199
|
-
...mockSealedAnalysis,
|
|
200
|
-
patterns: [],
|
|
201
|
-
antiPatterns: [],
|
|
202
|
-
}
|
|
203
|
-
const orcCtx = makeOrchestratorContext(emptyAnalysis)
|
|
204
|
-
|
|
205
|
-
const prompt = await promptBuilder.build(
|
|
206
|
-
template,
|
|
207
|
-
context,
|
|
208
|
-
{},
|
|
209
|
-
null,
|
|
210
|
-
null,
|
|
211
|
-
null,
|
|
212
|
-
null,
|
|
213
|
-
null,
|
|
214
|
-
orcCtx
|
|
215
|
-
)
|
|
216
|
-
|
|
217
|
-
expect(prompt).toContain('Languages**: TypeScript, JavaScript')
|
|
218
|
-
expect(prompt).not.toContain('Code Patterns (Follow These)')
|
|
219
|
-
expect(prompt).not.toContain('Anti-Patterns (Avoid These)')
|
|
220
|
-
})
|
|
221
|
-
|
|
222
|
-
it('should handle no orchestrator context at all', async () => {
|
|
223
|
-
const template = {
|
|
224
|
-
frontmatter: { description: 'Test' },
|
|
225
|
-
content: '## Do',
|
|
226
|
-
}
|
|
227
|
-
const context = { projectPath: '/test', files: [] }
|
|
228
|
-
|
|
229
|
-
const prompt = await promptBuilder.build(
|
|
230
|
-
template,
|
|
231
|
-
context,
|
|
232
|
-
{},
|
|
233
|
-
null,
|
|
234
|
-
null,
|
|
235
|
-
null,
|
|
236
|
-
null,
|
|
237
|
-
null,
|
|
238
|
-
null
|
|
239
|
-
)
|
|
240
|
-
|
|
241
|
-
// Should not crash, should have fallback rules
|
|
242
|
-
expect(prompt).toContain('CONSTRAINTS')
|
|
243
|
-
expect(prompt).not.toContain('PROJECT ANALYSIS')
|
|
244
|
-
})
|
|
245
|
-
})
|
|
246
|
-
|
|
247
|
-
// =============================================================================
|
|
248
|
-
// Anti-Hallucination Block — Analysis Enrichment
|
|
249
|
-
// =============================================================================
|
|
250
|
-
|
|
251
|
-
describe('Anti-Hallucination Block with Analysis (PRJ-260)', () => {
|
|
252
|
-
it('should include analysis languages in AVAILABLE list', () => {
|
|
253
|
-
const truth: ProjectGroundTruth = {
|
|
254
|
-
projectPath: '/test',
|
|
255
|
-
language: 'TypeScript',
|
|
256
|
-
techStack: ['Hono'],
|
|
257
|
-
analysisLanguages: ['TypeScript', 'JavaScript', 'Shell'],
|
|
258
|
-
analysisFrameworks: ['Hono', 'Vitest'],
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
const block = buildAntiHallucinationBlock(truth)
|
|
262
|
-
|
|
263
|
-
expect(block).toContain('AVAILABLE in this project:')
|
|
264
|
-
// TypeScript and Hono should not be duplicated
|
|
265
|
-
expect(block).toContain('JavaScript')
|
|
266
|
-
expect(block).toContain('Shell')
|
|
267
|
-
expect(block).toContain('Vitest')
|
|
268
|
-
})
|
|
269
|
-
|
|
270
|
-
it('should deduplicate analysis data with existing tech stack (case-insensitive)', () => {
|
|
271
|
-
const truth: ProjectGroundTruth = {
|
|
272
|
-
projectPath: '/test',
|
|
273
|
-
language: 'TypeScript',
|
|
274
|
-
framework: 'Hono',
|
|
275
|
-
techStack: ['Zod'],
|
|
276
|
-
analysisLanguages: ['typescript'], // lowercase duplicate
|
|
277
|
-
analysisFrameworks: ['hono', 'Zod'], // lowercase duplicates
|
|
278
|
-
}
|
|
279
|
-
|
|
280
|
-
const block = buildAntiHallucinationBlock(truth)
|
|
281
|
-
|
|
282
|
-
// Count occurrences — each should appear exactly once
|
|
283
|
-
const typescriptMatches = block.match(/typescript/gi)
|
|
284
|
-
expect(typescriptMatches?.length).toBe(1)
|
|
285
|
-
|
|
286
|
-
const honoMatches = block.match(/hono/gi)
|
|
287
|
-
expect(honoMatches?.length).toBe(1)
|
|
288
|
-
})
|
|
289
|
-
|
|
290
|
-
it('should show package manager from analysis', () => {
|
|
291
|
-
const truth: ProjectGroundTruth = {
|
|
292
|
-
projectPath: '/test',
|
|
293
|
-
analysisPackageManager: 'bun',
|
|
294
|
-
}
|
|
295
|
-
|
|
296
|
-
const block = buildAntiHallucinationBlock(truth)
|
|
297
|
-
|
|
298
|
-
expect(block).toContain('PACKAGE MANAGER: bun')
|
|
299
|
-
})
|
|
300
|
-
|
|
301
|
-
it('should work without any analysis data', () => {
|
|
302
|
-
const truth: ProjectGroundTruth = {
|
|
303
|
-
projectPath: '/test',
|
|
304
|
-
language: 'Python',
|
|
305
|
-
}
|
|
306
|
-
|
|
307
|
-
const block = buildAntiHallucinationBlock(truth)
|
|
308
|
-
|
|
309
|
-
expect(block).toContain('AVAILABLE in this project: Python')
|
|
310
|
-
expect(block).not.toContain('PACKAGE MANAGER:')
|
|
311
|
-
})
|
|
312
|
-
|
|
313
|
-
it('should handle empty analysis arrays gracefully', () => {
|
|
314
|
-
const truth: ProjectGroundTruth = {
|
|
315
|
-
projectPath: '/test',
|
|
316
|
-
analysisLanguages: [],
|
|
317
|
-
analysisFrameworks: [],
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
const block = buildAntiHallucinationBlock(truth)
|
|
321
|
-
|
|
322
|
-
// Should not contain AVAILABLE line (no tech to show)
|
|
323
|
-
expect(block).not.toContain('AVAILABLE in this project:')
|
|
324
|
-
expect(block).toContain('CONSTRAINTS')
|
|
325
|
-
})
|
|
326
|
-
})
|
|
327
|
-
|
|
328
|
-
// =============================================================================
|
|
329
|
-
// SealedAnalysisContext Type
|
|
330
|
-
// =============================================================================
|
|
331
|
-
|
|
332
|
-
describe('SealedAnalysisContext type (PRJ-260)', () => {
|
|
333
|
-
it('should accept valid sealed analysis data', () => {
|
|
334
|
-
const analysis: SealedAnalysisContext = {
|
|
335
|
-
languages: ['TypeScript'],
|
|
336
|
-
frameworks: ['Hono'],
|
|
337
|
-
fileCount: 100,
|
|
338
|
-
patterns: [{ name: 'test', description: 'test pattern' }],
|
|
339
|
-
antiPatterns: [{ issue: 'test', file: 'test.ts', suggestion: 'fix it' }],
|
|
340
|
-
status: 'sealed',
|
|
341
|
-
}
|
|
342
|
-
|
|
343
|
-
expect(analysis.languages).toEqual(['TypeScript'])
|
|
344
|
-
expect(analysis.status).toBe('sealed')
|
|
345
|
-
})
|
|
346
|
-
|
|
347
|
-
it('should accept draft status', () => {
|
|
348
|
-
const analysis: SealedAnalysisContext = {
|
|
349
|
-
languages: [],
|
|
350
|
-
frameworks: [],
|
|
351
|
-
fileCount: 0,
|
|
352
|
-
patterns: [],
|
|
353
|
-
antiPatterns: [],
|
|
354
|
-
status: 'draft',
|
|
355
|
-
}
|
|
356
|
-
|
|
357
|
-
expect(analysis.status).toBe('draft')
|
|
358
|
-
})
|
|
359
|
-
|
|
360
|
-
it('should accept optional fields', () => {
|
|
361
|
-
const analysis: SealedAnalysisContext = {
|
|
362
|
-
languages: ['Python'],
|
|
363
|
-
frameworks: [],
|
|
364
|
-
fileCount: 50,
|
|
365
|
-
patterns: [],
|
|
366
|
-
antiPatterns: [],
|
|
367
|
-
status: 'sealed',
|
|
368
|
-
packageManager: 'pip',
|
|
369
|
-
sourceDir: 'src/',
|
|
370
|
-
testDir: 'tests/',
|
|
371
|
-
commitHash: 'abc123',
|
|
372
|
-
}
|
|
373
|
-
|
|
374
|
-
expect(analysis.packageManager).toBe('pip')
|
|
375
|
-
expect(analysis.commitHash).toBe('abc123')
|
|
376
|
-
})
|
|
377
|
-
})
|
|
@@ -1,294 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Cache Eviction Policy Tests (PRJ-288)
|
|
3
|
-
* Verifies TTLCache integration, context caps, and archival.
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { afterAll, beforeAll, beforeEach, describe, expect, it } from 'bun:test'
|
|
7
|
-
import fs from 'node:fs/promises'
|
|
8
|
-
import path from 'node:path'
|
|
9
|
-
import { ContextBuilder } from '../../agentic/context-builder'
|
|
10
|
-
import { MemorySystem, PatternStore } from '../../agentic/memory-system'
|
|
11
|
-
import pathManager from '../../infrastructure/path-manager'
|
|
12
|
-
import { SessionLogManager } from '../../session/session-log-manager'
|
|
13
|
-
import { TTLCache } from '../../utils/cache'
|
|
14
|
-
|
|
15
|
-
const TEST_GLOBAL_BASE_DIR = path.join(process.cwd(), '.tmp', 'prjct-cli-cache-tests')
|
|
16
|
-
let testCounter = 0
|
|
17
|
-
const getTestProjectId = () => `test-cache-${Date.now()}-${++testCounter}`
|
|
18
|
-
|
|
19
|
-
describe('Cache Eviction Policies (PRJ-288)', () => {
|
|
20
|
-
beforeAll(async () => {
|
|
21
|
-
pathManager.setGlobalBaseDir(TEST_GLOBAL_BASE_DIR)
|
|
22
|
-
await fs.mkdir(TEST_GLOBAL_BASE_DIR, { recursive: true })
|
|
23
|
-
})
|
|
24
|
-
|
|
25
|
-
afterAll(async () => {
|
|
26
|
-
await fs.rm(TEST_GLOBAL_BASE_DIR, { recursive: true, force: true })
|
|
27
|
-
})
|
|
28
|
-
|
|
29
|
-
// ===========================================================================
|
|
30
|
-
// TTLCache unit tests
|
|
31
|
-
// ===========================================================================
|
|
32
|
-
|
|
33
|
-
describe('TTLCache', () => {
|
|
34
|
-
it('should evict oldest entries when maxSize exceeded', () => {
|
|
35
|
-
const cache = new TTLCache<string>({ maxSize: 3, ttl: 60_000 })
|
|
36
|
-
|
|
37
|
-
cache.set('a', 'val-a')
|
|
38
|
-
cache.set('b', 'val-b')
|
|
39
|
-
cache.set('c', 'val-c')
|
|
40
|
-
expect(cache.size).toBe(3)
|
|
41
|
-
|
|
42
|
-
cache.set('d', 'val-d')
|
|
43
|
-
expect(cache.size).toBe(3)
|
|
44
|
-
// 'a' was oldest, should be evicted
|
|
45
|
-
expect(cache.get('a')).toBeNull()
|
|
46
|
-
expect(cache.get('d')).toBe('val-d')
|
|
47
|
-
})
|
|
48
|
-
|
|
49
|
-
it('should expire entries after TTL', async () => {
|
|
50
|
-
const cache = new TTLCache<string>({ maxSize: 10, ttl: 50 })
|
|
51
|
-
|
|
52
|
-
cache.set('key', 'value')
|
|
53
|
-
expect(cache.get('key')).toBe('value')
|
|
54
|
-
|
|
55
|
-
await new Promise((r) => setTimeout(r, 80))
|
|
56
|
-
expect(cache.get('key')).toBeNull()
|
|
57
|
-
})
|
|
58
|
-
|
|
59
|
-
it('should prune expired entries', async () => {
|
|
60
|
-
const cache = new TTLCache<string>({ maxSize: 10, ttl: 50 })
|
|
61
|
-
|
|
62
|
-
cache.set('a', '1')
|
|
63
|
-
cache.set('b', '2')
|
|
64
|
-
await new Promise((r) => setTimeout(r, 80))
|
|
65
|
-
|
|
66
|
-
cache.set('c', '3') // fresh
|
|
67
|
-
const pruned = cache.prune()
|
|
68
|
-
expect(pruned).toBe(2)
|
|
69
|
-
expect(cache.size).toBe(1)
|
|
70
|
-
expect(cache.get('c')).toBe('3')
|
|
71
|
-
})
|
|
72
|
-
})
|
|
73
|
-
|
|
74
|
-
// ===========================================================================
|
|
75
|
-
// SessionLogManager
|
|
76
|
-
// ===========================================================================
|
|
77
|
-
|
|
78
|
-
describe('SessionLogManager', () => {
|
|
79
|
-
it('should use TTLCache with LRU eviction at 50 entries', () => {
|
|
80
|
-
const manager = new SessionLogManager()
|
|
81
|
-
// Verify clearCache works (exercises TTLCache.clear())
|
|
82
|
-
manager.clearCache()
|
|
83
|
-
})
|
|
84
|
-
})
|
|
85
|
-
|
|
86
|
-
// ===========================================================================
|
|
87
|
-
// ContextBuilder
|
|
88
|
-
// ===========================================================================
|
|
89
|
-
|
|
90
|
-
describe('ContextBuilder', () => {
|
|
91
|
-
it('should report stats from TTLCache', () => {
|
|
92
|
-
const builder = new ContextBuilder()
|
|
93
|
-
const stats = builder.getCacheStats()
|
|
94
|
-
expect(stats).toHaveProperty('size')
|
|
95
|
-
expect(stats).toHaveProperty('maxSize')
|
|
96
|
-
expect(stats).toHaveProperty('ttl')
|
|
97
|
-
expect(stats.maxSize).toBe(200)
|
|
98
|
-
expect(stats.ttl).toBe(5000)
|
|
99
|
-
})
|
|
100
|
-
|
|
101
|
-
it('should clear cache and reset projectId', () => {
|
|
102
|
-
const builder = new ContextBuilder()
|
|
103
|
-
builder.clearCache()
|
|
104
|
-
const stats = builder.getCacheStats()
|
|
105
|
-
expect(stats.size).toBe(0)
|
|
106
|
-
})
|
|
107
|
-
|
|
108
|
-
it('should invalidate specific cache entries', async () => {
|
|
109
|
-
const builder = new ContextBuilder()
|
|
110
|
-
|
|
111
|
-
// Write a temp file, batchRead to populate cache, then invalidate
|
|
112
|
-
const tmpDir = path.join(TEST_GLOBAL_BASE_DIR, 'ctx-test')
|
|
113
|
-
await fs.mkdir(tmpDir, { recursive: true })
|
|
114
|
-
const tmpFile = path.join(tmpDir, 'test.txt')
|
|
115
|
-
await fs.writeFile(tmpFile, 'hello')
|
|
116
|
-
|
|
117
|
-
const result = await builder.batchRead([tmpFile])
|
|
118
|
-
expect(result.get(tmpFile)).toBe('hello')
|
|
119
|
-
|
|
120
|
-
// Invalidate and verify
|
|
121
|
-
builder.invalidateCache(tmpFile)
|
|
122
|
-
// After invalidation, a fresh batchRead should re-read from disk
|
|
123
|
-
await fs.writeFile(tmpFile, 'updated')
|
|
124
|
-
const result2 = await builder.batchRead([tmpFile])
|
|
125
|
-
expect(result2.get(tmpFile)).toBe('updated')
|
|
126
|
-
})
|
|
127
|
-
})
|
|
128
|
-
|
|
129
|
-
// ===========================================================================
|
|
130
|
-
// PatternStore - contexts cap
|
|
131
|
-
// ===========================================================================
|
|
132
|
-
|
|
133
|
-
describe('PatternStore contexts cap', () => {
|
|
134
|
-
let TEST_PROJECT_ID: string
|
|
135
|
-
|
|
136
|
-
beforeEach(() => {
|
|
137
|
-
TEST_PROJECT_ID = getTestProjectId()
|
|
138
|
-
})
|
|
139
|
-
|
|
140
|
-
it('should cap decision contexts at 20 (FIFO)', async () => {
|
|
141
|
-
const store = new PatternStore()
|
|
142
|
-
|
|
143
|
-
// Record a decision with many unique contexts
|
|
144
|
-
for (let i = 0; i < 25; i++) {
|
|
145
|
-
await store.recordDecision(TEST_PROJECT_ID, 'test-key', 'test-value', `context-${i}`)
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
const patterns = await store.load(TEST_PROJECT_ID)
|
|
149
|
-
const decision = patterns.decisions['test-key']
|
|
150
|
-
expect(decision.contexts.length).toBeLessThanOrEqual(20)
|
|
151
|
-
// Should keep the latest contexts
|
|
152
|
-
expect(decision.contexts).toContain('context-24')
|
|
153
|
-
expect(decision.contexts).toContain('context-5')
|
|
154
|
-
// Oldest should be evicted
|
|
155
|
-
expect(decision.contexts).not.toContain('context-0')
|
|
156
|
-
})
|
|
157
|
-
|
|
158
|
-
it('should truncate oversized contexts on afterLoad', async () => {
|
|
159
|
-
const store = new PatternStore()
|
|
160
|
-
const projectId = getTestProjectId()
|
|
161
|
-
|
|
162
|
-
// Manually write a patterns file with oversized contexts
|
|
163
|
-
const filePath = path.join(
|
|
164
|
-
pathManager.getGlobalProjectPath(projectId),
|
|
165
|
-
'memory',
|
|
166
|
-
'patterns.json'
|
|
167
|
-
)
|
|
168
|
-
await fs.mkdir(path.dirname(filePath), { recursive: true })
|
|
169
|
-
|
|
170
|
-
const oversizedPatterns = {
|
|
171
|
-
version: 1,
|
|
172
|
-
decisions: {
|
|
173
|
-
'big-key': {
|
|
174
|
-
value: 'v',
|
|
175
|
-
count: 30,
|
|
176
|
-
firstSeen: '2025-01-01T00:00:00.000Z',
|
|
177
|
-
lastSeen: '2025-06-01T00:00:00.000Z',
|
|
178
|
-
confidence: 'high',
|
|
179
|
-
contexts: Array.from({ length: 50 }, (_, i) => `ctx-${i}`),
|
|
180
|
-
userConfirmed: true,
|
|
181
|
-
},
|
|
182
|
-
},
|
|
183
|
-
preferences: {},
|
|
184
|
-
workflows: {},
|
|
185
|
-
counters: {},
|
|
186
|
-
}
|
|
187
|
-
await fs.writeFile(filePath, JSON.stringify(oversizedPatterns, null, 2), 'utf-8')
|
|
188
|
-
|
|
189
|
-
// Load triggers afterLoad which should truncate
|
|
190
|
-
const patterns = await store.load(projectId)
|
|
191
|
-
expect(patterns.decisions['big-key'].contexts.length).toBe(20)
|
|
192
|
-
// Should keep the latest 20 (indices 30-49)
|
|
193
|
-
expect(patterns.decisions['big-key'].contexts[19]).toBe('ctx-49')
|
|
194
|
-
})
|
|
195
|
-
})
|
|
196
|
-
|
|
197
|
-
// ===========================================================================
|
|
198
|
-
// PatternStore - archival
|
|
199
|
-
// ===========================================================================
|
|
200
|
-
|
|
201
|
-
describe('PatternStore archival', () => {
|
|
202
|
-
it('should archive decisions older than 90 days', async () => {
|
|
203
|
-
const store = new PatternStore()
|
|
204
|
-
const projectId = getTestProjectId()
|
|
205
|
-
|
|
206
|
-
// Write patterns with a stale and an active decision
|
|
207
|
-
const filePath = path.join(
|
|
208
|
-
pathManager.getGlobalProjectPath(projectId),
|
|
209
|
-
'memory',
|
|
210
|
-
'patterns.json'
|
|
211
|
-
)
|
|
212
|
-
await fs.mkdir(path.dirname(filePath), { recursive: true })
|
|
213
|
-
|
|
214
|
-
const now = new Date()
|
|
215
|
-
const staleDate = new Date(now.getTime() - 100 * 24 * 60 * 60 * 1000) // 100 days ago
|
|
216
|
-
const recentDate = new Date(now.getTime() - 5 * 24 * 60 * 60 * 1000) // 5 days ago
|
|
217
|
-
|
|
218
|
-
const patterns = {
|
|
219
|
-
version: 1,
|
|
220
|
-
decisions: {
|
|
221
|
-
'stale-key': {
|
|
222
|
-
value: 'old',
|
|
223
|
-
count: 3,
|
|
224
|
-
firstSeen: staleDate.toISOString(),
|
|
225
|
-
lastSeen: staleDate.toISOString(),
|
|
226
|
-
confidence: 'medium',
|
|
227
|
-
contexts: ['ctx1'],
|
|
228
|
-
userConfirmed: false,
|
|
229
|
-
},
|
|
230
|
-
'active-key': {
|
|
231
|
-
value: 'new',
|
|
232
|
-
count: 5,
|
|
233
|
-
firstSeen: recentDate.toISOString(),
|
|
234
|
-
lastSeen: recentDate.toISOString(),
|
|
235
|
-
confidence: 'high',
|
|
236
|
-
contexts: ['ctx2'],
|
|
237
|
-
userConfirmed: true,
|
|
238
|
-
},
|
|
239
|
-
},
|
|
240
|
-
preferences: {},
|
|
241
|
-
workflows: {},
|
|
242
|
-
counters: {},
|
|
243
|
-
}
|
|
244
|
-
await fs.writeFile(filePath, JSON.stringify(patterns, null, 2), 'utf-8')
|
|
245
|
-
|
|
246
|
-
// Reset store cache to force disk read
|
|
247
|
-
store.reset()
|
|
248
|
-
const archived = await store.archiveStaleDecisions(projectId)
|
|
249
|
-
expect(archived).toBe(1)
|
|
250
|
-
|
|
251
|
-
// Verify active decision remains
|
|
252
|
-
const updated = await store.load(projectId)
|
|
253
|
-
expect(updated.decisions['active-key']).toBeDefined()
|
|
254
|
-
expect(updated.decisions['stale-key']).toBeUndefined()
|
|
255
|
-
|
|
256
|
-
// Verify archive file was created
|
|
257
|
-
const archivePath = path.join(
|
|
258
|
-
pathManager.getGlobalProjectPath(projectId),
|
|
259
|
-
'memory',
|
|
260
|
-
'patterns-archive.json'
|
|
261
|
-
)
|
|
262
|
-
const archiveContent = JSON.parse(await fs.readFile(archivePath, 'utf-8'))
|
|
263
|
-
expect(archiveContent['stale-key']).toBeDefined()
|
|
264
|
-
expect(archiveContent['stale-key'].value).toBe('old')
|
|
265
|
-
})
|
|
266
|
-
|
|
267
|
-
it('should return 0 when no stale decisions exist', async () => {
|
|
268
|
-
const store = new PatternStore()
|
|
269
|
-
const projectId = getTestProjectId()
|
|
270
|
-
|
|
271
|
-
// Record a fresh decision
|
|
272
|
-
await store.recordDecision(projectId, 'fresh-key', 'fresh-val', 'ctx')
|
|
273
|
-
const archived = await store.archiveStaleDecisions(projectId)
|
|
274
|
-
expect(archived).toBe(0)
|
|
275
|
-
})
|
|
276
|
-
})
|
|
277
|
-
|
|
278
|
-
// ===========================================================================
|
|
279
|
-
// MemorySystem delegation
|
|
280
|
-
// ===========================================================================
|
|
281
|
-
|
|
282
|
-
describe('MemorySystem.archiveStaleDecisions', () => {
|
|
283
|
-
it('should delegate to PatternStore', async () => {
|
|
284
|
-
const ms = new MemorySystem()
|
|
285
|
-
const projectId = getTestProjectId()
|
|
286
|
-
|
|
287
|
-
// Just verify no crash - no stale data to archive
|
|
288
|
-
ms.resetState()
|
|
289
|
-
await ms.recordDecision(projectId, 'key', 'val', 'ctx')
|
|
290
|
-
const count = await ms.archiveStaleDecisions(projectId)
|
|
291
|
-
expect(count).toBe(0)
|
|
292
|
-
})
|
|
293
|
-
})
|
|
294
|
-
})
|