cognitive-core 0.2.1 → 0.2.2
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/dist/atlas.d.ts +10 -0
- package/dist/atlas.d.ts.map +1 -1
- package/dist/atlas.js +65 -0
- package/dist/atlas.js.map +1 -1
- package/dist/learning/pipeline.d.ts +4 -31
- package/dist/learning/pipeline.d.ts.map +1 -1
- package/dist/learning/pipeline.js +12 -64
- package/dist/learning/pipeline.js.map +1 -1
- package/dist/memory/curated-loader.d.ts +21 -4
- package/dist/memory/curated-loader.d.ts.map +1 -1
- package/dist/memory/curated-loader.js +53 -16
- package/dist/memory/curated-loader.js.map +1 -1
- package/dist/memory/index.d.ts +2 -1
- package/dist/memory/index.d.ts.map +1 -1
- package/dist/memory/index.js +3 -1
- package/dist/memory/index.js.map +1 -1
- package/dist/memory/playbook.d.ts +6 -0
- package/dist/memory/playbook.d.ts.map +1 -1
- package/dist/memory/playbook.js +15 -0
- package/dist/memory/playbook.js.map +1 -1
- package/dist/memory/source-resolver.d.ts +120 -0
- package/dist/memory/source-resolver.d.ts.map +1 -0
- package/dist/memory/source-resolver.js +300 -0
- package/dist/memory/source-resolver.js.map +1 -0
- package/dist/types/config.d.ts +141 -0
- package/dist/types/config.d.ts.map +1 -1
- package/dist/types/config.js +40 -0
- package/dist/types/config.js.map +1 -1
- package/dist/types/index.d.ts +1 -1
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/index.js +1 -1
- package/dist/types/index.js.map +1 -1
- package/dist/workspace/types.d.ts +12 -54
- package/dist/workspace/types.d.ts.map +1 -1
- package/dist/workspace/types.js.map +1 -1
- package/package.json +2 -2
- package/playbooks/compound-engineering/adversarial-review.json +51 -0
- package/playbooks/compound-engineering/agent-native-architecture.json +59 -0
- package/playbooks/compound-engineering/agent-native-review.json +54 -0
- package/playbooks/compound-engineering/api-contract-review.json +52 -0
- package/playbooks/compound-engineering/brainstorm-requirements.json +55 -0
- package/playbooks/compound-engineering/bug-reproduction.json +62 -0
- package/playbooks/compound-engineering/confidence-calibration.json +49 -0
- package/playbooks/compound-engineering/correctness-review.json +49 -0
- package/playbooks/compound-engineering/data-migration-safety.json +59 -0
- package/playbooks/compound-engineering/deployment-verification.json +63 -0
- package/playbooks/compound-engineering/error-recovery-patterns.json +53 -0
- package/playbooks/compound-engineering/implementation-planning.json +64 -0
- package/playbooks/compound-engineering/issue-pattern-analysis.json +53 -0
- package/playbooks/compound-engineering/knowledge-compounding.json +63 -0
- package/playbooks/compound-engineering/learnings-research.json +54 -0
- package/playbooks/compound-engineering/maintainability-review.json +49 -0
- package/playbooks/compound-engineering/performance-review.json +54 -0
- package/playbooks/compound-engineering/plan-adversarial-review.json +56 -0
- package/playbooks/compound-engineering/plan-feasibility-review.json +56 -0
- package/playbooks/compound-engineering/project-standards-review.json +52 -0
- package/playbooks/compound-engineering/reliability-review.json +53 -0
- package/playbooks/compound-engineering/review-orchestration.json +64 -0
- package/playbooks/compound-engineering/security-review.json +54 -0
- package/playbooks/compound-engineering/systematic-execution.json +64 -0
- package/playbooks/compound-engineering/testing-review.json +50 -0
- package/src/atlas.ts +96 -0
- package/src/memory/curated-loader.ts +69 -16
- package/src/memory/index.ts +16 -0
- package/src/memory/playbook.ts +19 -0
- package/src/memory/source-resolver.ts +422 -0
- package/src/types/config.ts +46 -0
- package/src/types/index.ts +4 -0
- package/src/workspace/types.ts +22 -78
- package/tests/integration/curated-sources-e2e.test.ts +502 -0
- package/tests/memory/compound-engineering-seed.test.ts +338 -0
- package/tests/memory/curated-loader-extended.test.ts +225 -0
- package/tests/memory/playbook-quality-validation.test.ts +430 -0
- package/tests/memory/source-resolver.test.ts +700 -0
- package/.claude/settings.local.json +0 -11
- package/dist/learning/llm-extractor.d.ts +0 -88
- package/dist/learning/llm-extractor.d.ts.map +0 -1
- package/dist/learning/llm-extractor.js +0 -372
- package/dist/learning/llm-extractor.js.map +0 -1
- package/dist/learning/loop-coordinator.d.ts +0 -61
- package/dist/learning/loop-coordinator.d.ts.map +0 -1
- package/dist/learning/loop-coordinator.js +0 -96
- package/dist/learning/loop-coordinator.js.map +0 -1
- package/references/agent-workspace/CLAUDE.md +0 -74
- package/references/agent-workspace/README.md +0 -587
- package/references/agent-workspace/media/banner.png +0 -0
- package/references/agent-workspace/package-lock.json +0 -2061
- package/references/agent-workspace/package.json +0 -54
- package/references/agent-workspace/src/handle.ts +0 -122
- package/references/agent-workspace/src/index.ts +0 -32
- package/references/agent-workspace/src/manager.ts +0 -102
- package/references/agent-workspace/src/readers/json.ts +0 -71
- package/references/agent-workspace/src/readers/markdown.ts +0 -37
- package/references/agent-workspace/src/readers/raw.ts +0 -27
- package/references/agent-workspace/src/types.ts +0 -68
- package/references/agent-workspace/src/validation.ts +0 -93
- package/references/agent-workspace/src/writers/json.ts +0 -17
- package/references/agent-workspace/src/writers/markdown.ts +0 -27
- package/references/agent-workspace/src/writers/raw.ts +0 -22
- package/references/agent-workspace/tests/errors.test.ts +0 -652
- package/references/agent-workspace/tests/handle.test.ts +0 -144
- package/references/agent-workspace/tests/manager.test.ts +0 -124
- package/references/agent-workspace/tests/readers.test.ts +0 -205
- package/references/agent-workspace/tests/validation.test.ts +0 -196
- package/references/agent-workspace/tests/writers.test.ts +0 -108
- package/references/agent-workspace/tsconfig.json +0 -20
- package/references/agent-workspace/tsup.config.ts +0 -9
- package/references/minimem/.claude/settings.json +0 -7
- package/references/minimem/.sudocode/issues.jsonl +0 -18
- package/references/minimem/.sudocode/specs.jsonl +0 -1
- package/references/minimem/CLAUDE.md +0 -310
- package/references/minimem/README.md +0 -556
- package/references/minimem/claude-plugin/.claude-plugin/plugin.json +0 -10
- package/references/minimem/claude-plugin/.mcp.json +0 -7
- package/references/minimem/claude-plugin/README.md +0 -158
- package/references/minimem/claude-plugin/commands/recall.md +0 -47
- package/references/minimem/claude-plugin/commands/remember.md +0 -41
- package/references/minimem/claude-plugin/hooks/__tests__/hooks.test.ts +0 -272
- package/references/minimem/claude-plugin/hooks/hooks.json +0 -27
- package/references/minimem/claude-plugin/hooks/session-end.sh +0 -86
- package/references/minimem/claude-plugin/hooks/session-start.sh +0 -85
- package/references/minimem/claude-plugin/skills/memory/SKILL.md +0 -108
- package/references/minimem/package-lock.json +0 -5373
- package/references/minimem/package.json +0 -60
- package/references/minimem/scripts/postbuild.js +0 -35
- package/references/minimem/src/__tests__/edge-cases.test.ts +0 -371
- package/references/minimem/src/__tests__/errors.test.ts +0 -265
- package/references/minimem/src/__tests__/helpers.ts +0 -199
- package/references/minimem/src/__tests__/internal.test.ts +0 -407
- package/references/minimem/src/__tests__/knowledge.test.ts +0 -287
- package/references/minimem/src/__tests__/minimem.integration.test.ts +0 -1127
- package/references/minimem/src/__tests__/session.test.ts +0 -190
- package/references/minimem/src/cli/__tests__/commands.test.ts +0 -759
- package/references/minimem/src/cli/commands/__tests__/conflicts.test.ts +0 -141
- package/references/minimem/src/cli/commands/append.ts +0 -76
- package/references/minimem/src/cli/commands/config.ts +0 -262
- package/references/minimem/src/cli/commands/conflicts.ts +0 -413
- package/references/minimem/src/cli/commands/daemon.ts +0 -169
- package/references/minimem/src/cli/commands/index.ts +0 -12
- package/references/minimem/src/cli/commands/init.ts +0 -88
- package/references/minimem/src/cli/commands/mcp.ts +0 -177
- package/references/minimem/src/cli/commands/push-pull.ts +0 -213
- package/references/minimem/src/cli/commands/search.ts +0 -158
- package/references/minimem/src/cli/commands/status.ts +0 -84
- package/references/minimem/src/cli/commands/sync-init.ts +0 -290
- package/references/minimem/src/cli/commands/sync.ts +0 -70
- package/references/minimem/src/cli/commands/upsert.ts +0 -197
- package/references/minimem/src/cli/config.ts +0 -584
- package/references/minimem/src/cli/index.ts +0 -264
- package/references/minimem/src/cli/shared.ts +0 -161
- package/references/minimem/src/cli/sync/__tests__/central.test.ts +0 -152
- package/references/minimem/src/cli/sync/__tests__/conflicts.test.ts +0 -209
- package/references/minimem/src/cli/sync/__tests__/daemon.test.ts +0 -118
- package/references/minimem/src/cli/sync/__tests__/detection.test.ts +0 -207
- package/references/minimem/src/cli/sync/__tests__/integration.test.ts +0 -476
- package/references/minimem/src/cli/sync/__tests__/registry.test.ts +0 -363
- package/references/minimem/src/cli/sync/__tests__/state.test.ts +0 -255
- package/references/minimem/src/cli/sync/__tests__/validation.test.ts +0 -193
- package/references/minimem/src/cli/sync/__tests__/watcher.test.ts +0 -178
- package/references/minimem/src/cli/sync/central.ts +0 -292
- package/references/minimem/src/cli/sync/conflicts.ts +0 -204
- package/references/minimem/src/cli/sync/daemon.ts +0 -407
- package/references/minimem/src/cli/sync/detection.ts +0 -138
- package/references/minimem/src/cli/sync/index.ts +0 -107
- package/references/minimem/src/cli/sync/operations.ts +0 -373
- package/references/minimem/src/cli/sync/registry.ts +0 -279
- package/references/minimem/src/cli/sync/state.ts +0 -355
- package/references/minimem/src/cli/sync/validation.ts +0 -206
- package/references/minimem/src/cli/sync/watcher.ts +0 -234
- package/references/minimem/src/cli/version.ts +0 -34
- package/references/minimem/src/core/index.ts +0 -9
- package/references/minimem/src/core/indexer.ts +0 -628
- package/references/minimem/src/core/searcher.ts +0 -221
- package/references/minimem/src/db/schema.ts +0 -183
- package/references/minimem/src/db/sqlite-vec.ts +0 -24
- package/references/minimem/src/embeddings/__tests__/embeddings.test.ts +0 -431
- package/references/minimem/src/embeddings/batch-gemini.ts +0 -392
- package/references/minimem/src/embeddings/batch-openai.ts +0 -409
- package/references/minimem/src/embeddings/embeddings.ts +0 -434
- package/references/minimem/src/index.ts +0 -109
- package/references/minimem/src/internal.ts +0 -299
- package/references/minimem/src/minimem.ts +0 -1276
- package/references/minimem/src/search/__tests__/hybrid.test.ts +0 -247
- package/references/minimem/src/search/graph.ts +0 -234
- package/references/minimem/src/search/hybrid.ts +0 -151
- package/references/minimem/src/search/search.ts +0 -256
- package/references/minimem/src/server/__tests__/mcp.test.ts +0 -341
- package/references/minimem/src/server/__tests__/tools.test.ts +0 -364
- package/references/minimem/src/server/mcp.ts +0 -326
- package/references/minimem/src/server/tools.ts +0 -720
- package/references/minimem/src/session.ts +0 -460
- package/references/minimem/tsconfig.json +0 -19
- package/references/minimem/tsup.config.ts +0 -26
- package/references/minimem/vitest.config.ts +0 -24
- package/references/sessionlog/.husky/pre-commit +0 -1
- package/references/sessionlog/.lintstagedrc.json +0 -4
- package/references/sessionlog/.prettierignore +0 -4
- package/references/sessionlog/.prettierrc.json +0 -11
- package/references/sessionlog/LICENSE +0 -21
- package/references/sessionlog/README.md +0 -453
- package/references/sessionlog/eslint.config.js +0 -58
- package/references/sessionlog/package-lock.json +0 -3672
- package/references/sessionlog/package.json +0 -65
- package/references/sessionlog/src/__tests__/agent-hooks.test.ts +0 -570
- package/references/sessionlog/src/__tests__/agent-registry.test.ts +0 -127
- package/references/sessionlog/src/__tests__/claude-code-hooks.test.ts +0 -225
- package/references/sessionlog/src/__tests__/claude-generator.test.ts +0 -46
- package/references/sessionlog/src/__tests__/commit-msg.test.ts +0 -86
- package/references/sessionlog/src/__tests__/cursor-agent.test.ts +0 -224
- package/references/sessionlog/src/__tests__/e2e-live.test.ts +0 -890
- package/references/sessionlog/src/__tests__/event-log.test.ts +0 -183
- package/references/sessionlog/src/__tests__/flush-sentinel.test.ts +0 -105
- package/references/sessionlog/src/__tests__/gemini-agent.test.ts +0 -375
- package/references/sessionlog/src/__tests__/git-hooks.test.ts +0 -78
- package/references/sessionlog/src/__tests__/hook-managers.test.ts +0 -121
- package/references/sessionlog/src/__tests__/lifecycle-tasks.test.ts +0 -759
- package/references/sessionlog/src/__tests__/opencode-agent.test.ts +0 -338
- package/references/sessionlog/src/__tests__/redaction.test.ts +0 -136
- package/references/sessionlog/src/__tests__/session-repo.test.ts +0 -353
- package/references/sessionlog/src/__tests__/session-store.test.ts +0 -166
- package/references/sessionlog/src/__tests__/setup-ccweb.test.ts +0 -466
- package/references/sessionlog/src/__tests__/skill-live.test.ts +0 -461
- package/references/sessionlog/src/__tests__/summarize.test.ts +0 -348
- package/references/sessionlog/src/__tests__/task-plan-e2e.test.ts +0 -610
- package/references/sessionlog/src/__tests__/task-plan-live.test.ts +0 -632
- package/references/sessionlog/src/__tests__/transcript-timestamp.test.ts +0 -121
- package/references/sessionlog/src/__tests__/types.test.ts +0 -166
- package/references/sessionlog/src/__tests__/utils.test.ts +0 -333
- package/references/sessionlog/src/__tests__/validation.test.ts +0 -103
- package/references/sessionlog/src/__tests__/worktree.test.ts +0 -57
- package/references/sessionlog/src/agent/agents/claude-code.ts +0 -1089
- package/references/sessionlog/src/agent/agents/cursor.ts +0 -361
- package/references/sessionlog/src/agent/agents/gemini-cli.ts +0 -632
- package/references/sessionlog/src/agent/agents/opencode.ts +0 -540
- package/references/sessionlog/src/agent/registry.ts +0 -143
- package/references/sessionlog/src/agent/session-types.ts +0 -113
- package/references/sessionlog/src/agent/types.ts +0 -220
- package/references/sessionlog/src/cli.ts +0 -597
- package/references/sessionlog/src/commands/clean.ts +0 -133
- package/references/sessionlog/src/commands/disable.ts +0 -84
- package/references/sessionlog/src/commands/doctor.ts +0 -145
- package/references/sessionlog/src/commands/enable.ts +0 -202
- package/references/sessionlog/src/commands/explain.ts +0 -261
- package/references/sessionlog/src/commands/reset.ts +0 -105
- package/references/sessionlog/src/commands/resume.ts +0 -180
- package/references/sessionlog/src/commands/rewind.ts +0 -195
- package/references/sessionlog/src/commands/setup-ccweb.ts +0 -275
- package/references/sessionlog/src/commands/status.ts +0 -172
- package/references/sessionlog/src/config.ts +0 -165
- package/references/sessionlog/src/events/event-log.ts +0 -126
- package/references/sessionlog/src/git-operations.ts +0 -558
- package/references/sessionlog/src/hooks/git-hooks.ts +0 -165
- package/references/sessionlog/src/hooks/lifecycle.ts +0 -391
- package/references/sessionlog/src/index.ts +0 -650
- package/references/sessionlog/src/security/redaction.ts +0 -283
- package/references/sessionlog/src/session/state-machine.ts +0 -452
- package/references/sessionlog/src/store/checkpoint-store.ts +0 -509
- package/references/sessionlog/src/store/native-store.ts +0 -173
- package/references/sessionlog/src/store/provider-types.ts +0 -99
- package/references/sessionlog/src/store/session-store.ts +0 -266
- package/references/sessionlog/src/strategy/attribution.ts +0 -296
- package/references/sessionlog/src/strategy/common.ts +0 -207
- package/references/sessionlog/src/strategy/content-overlap.ts +0 -228
- package/references/sessionlog/src/strategy/manual-commit.ts +0 -988
- package/references/sessionlog/src/strategy/types.ts +0 -279
- package/references/sessionlog/src/summarize/claude-generator.ts +0 -115
- package/references/sessionlog/src/summarize/summarize.ts +0 -432
- package/references/sessionlog/src/types.ts +0 -508
- package/references/sessionlog/src/utils/chunk-files.ts +0 -49
- package/references/sessionlog/src/utils/commit-message.ts +0 -65
- package/references/sessionlog/src/utils/detect-agent.ts +0 -36
- package/references/sessionlog/src/utils/hook-managers.ts +0 -125
- package/references/sessionlog/src/utils/ide-tags.ts +0 -32
- package/references/sessionlog/src/utils/paths.ts +0 -79
- package/references/sessionlog/src/utils/preview-rewind.ts +0 -80
- package/references/sessionlog/src/utils/rewind-conflict.ts +0 -121
- package/references/sessionlog/src/utils/shadow-branch.ts +0 -109
- package/references/sessionlog/src/utils/string-utils.ts +0 -46
- package/references/sessionlog/src/utils/todo-extract.ts +0 -188
- package/references/sessionlog/src/utils/trailers.ts +0 -187
- package/references/sessionlog/src/utils/transcript-parse.ts +0 -177
- package/references/sessionlog/src/utils/transcript-timestamp.ts +0 -59
- package/references/sessionlog/src/utils/tree-ops.ts +0 -219
- package/references/sessionlog/src/utils/tty.ts +0 -72
- package/references/sessionlog/src/utils/validation.ts +0 -65
- package/references/sessionlog/src/utils/worktree.ts +0 -58
- package/references/sessionlog/src/wire-types.ts +0 -59
- package/references/sessionlog/templates/setup-env.sh +0 -153
- package/references/sessionlog/tsconfig.json +0 -18
- package/references/sessionlog/vitest.config.ts +0 -12
- package/references/skill-tree/.claude/settings.json +0 -6
- package/references/skill-tree/.sudocode/issues.jsonl +0 -19
- package/references/skill-tree/.sudocode/specs.jsonl +0 -3
- package/references/skill-tree/CLAUDE.md +0 -126
- package/references/skill-tree/README.md +0 -372
- package/references/skill-tree/docs/GAPS_v1.md +0 -221
- package/references/skill-tree/docs/INTEGRATION_PLAN.md +0 -467
- package/references/skill-tree/docs/TODOS.md +0 -91
- package/references/skill-tree/docs/anthropic_skill_guide.md +0 -1364
- package/references/skill-tree/docs/design/federated-skill-trees.md +0 -524
- package/references/skill-tree/docs/design/multi-agent-sync.md +0 -759
- package/references/skill-tree/docs/scraper/BRAINSTORM.md +0 -583
- package/references/skill-tree/docs/scraper/POC_PLAN.md +0 -420
- package/references/skill-tree/docs/scraper/README.md +0 -170
- package/references/skill-tree/examples/basic-usage.ts +0 -164
- package/references/skill-tree/package-lock.json +0 -1852
- package/references/skill-tree/package.json +0 -66
- package/references/skill-tree/scraper/README.md +0 -123
- package/references/skill-tree/scraper/docs/DESIGN.md +0 -683
- package/references/skill-tree/scraper/docs/PLAN.md +0 -336
- package/references/skill-tree/scraper/drizzle.config.ts +0 -10
- package/references/skill-tree/scraper/package-lock.json +0 -6329
- package/references/skill-tree/scraper/package.json +0 -68
- package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-description.md +0 -7
- package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-name.md +0 -7
- package/references/skill-tree/scraper/test/fixtures/minimal-skill/SKILL.md +0 -27
- package/references/skill-tree/scraper/test/fixtures/skill-json/SKILL.json +0 -21
- package/references/skill-tree/scraper/test/fixtures/skill-with-meta/SKILL.md +0 -54
- package/references/skill-tree/scraper/test/fixtures/skill-with-meta/_meta.json +0 -24
- package/references/skill-tree/scraper/test/fixtures/valid-skill/SKILL.md +0 -93
- package/references/skill-tree/scraper/test/fixtures/valid-skill/_meta.json +0 -22
- package/references/skill-tree/scraper/tsup.config.ts +0 -14
- package/references/skill-tree/scraper/vitest.config.ts +0 -17
- package/references/skill-tree/scripts/convert-to-vitest.ts +0 -166
- package/references/skill-tree/skills/skill-writer/SKILL.md +0 -339
- package/references/skill-tree/skills/skill-writer/references/examples.md +0 -326
- package/references/skill-tree/skills/skill-writer/references/patterns.md +0 -210
- package/references/skill-tree/skills/skill-writer/references/quality-checklist.md +0 -123
- package/references/skill-tree/test/run-all.ts +0 -106
- package/references/skill-tree/test/utils.ts +0 -128
- package/references/skill-tree/vitest.config.ts +0 -16
|
@@ -1,99 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Sessionlog Provider Types
|
|
3
|
-
*
|
|
4
|
-
* These types define the public interface for accessing Sessionlog session
|
|
5
|
-
* and checkpoint data. They are the contract between the Sessionlog module
|
|
6
|
-
* and external consumers (providers, daemon, etc.).
|
|
7
|
-
*
|
|
8
|
-
* This file is the canonical source of truth — external code should
|
|
9
|
-
* import these types from the sessionlog package rather than redefining them.
|
|
10
|
-
*/
|
|
11
|
-
|
|
12
|
-
// ============================================================================
|
|
13
|
-
// Session
|
|
14
|
-
// ============================================================================
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Sessionlog session state (from .git/sessionlog-sessions/<id>.json)
|
|
18
|
-
*/
|
|
19
|
-
export interface SessionlogSession {
|
|
20
|
-
id: string;
|
|
21
|
-
agent: string;
|
|
22
|
-
phase: 'ACTIVE' | 'IDLE' | 'ENDED';
|
|
23
|
-
baseCommit?: string;
|
|
24
|
-
branch?: string;
|
|
25
|
-
startedAt?: string;
|
|
26
|
-
endedAt?: string;
|
|
27
|
-
checkpoints?: string[];
|
|
28
|
-
tokenUsage?: SessionlogTokenUsage;
|
|
29
|
-
filesTouched?: string[];
|
|
30
|
-
summary?: string;
|
|
31
|
-
|
|
32
|
-
/** Skills used during this session (populated by SkillTracker) */
|
|
33
|
-
skillsUsed?: SessionlogSkillUsage;
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
/**
|
|
37
|
-
* Skill usage data embedded in session metadata
|
|
38
|
-
*/
|
|
39
|
-
export interface SessionlogSkillUsage {
|
|
40
|
-
/** Distinct skill names used */
|
|
41
|
-
skills: string[];
|
|
42
|
-
|
|
43
|
-
/** Total invocation count across all skills */
|
|
44
|
-
totalInvocations: number;
|
|
45
|
-
|
|
46
|
-
/** Per-skill invocation counts */
|
|
47
|
-
counts: Record<string, number>;
|
|
48
|
-
|
|
49
|
-
/** Per-skill success/failure counts */
|
|
50
|
-
outcomes: Record<string, { success: number; failure: number }>;
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
// ============================================================================
|
|
54
|
-
// Checkpoint
|
|
55
|
-
// ============================================================================
|
|
56
|
-
|
|
57
|
-
/**
|
|
58
|
-
* Sessionlog checkpoint metadata
|
|
59
|
-
*/
|
|
60
|
-
export interface SessionlogCheckpoint {
|
|
61
|
-
id: string;
|
|
62
|
-
sessionId?: string;
|
|
63
|
-
commitHash?: string;
|
|
64
|
-
commitMessage?: string;
|
|
65
|
-
promptCount?: number;
|
|
66
|
-
filesModified?: string[];
|
|
67
|
-
filesNew?: string[];
|
|
68
|
-
filesDeleted?: string[];
|
|
69
|
-
tokenUsage?: SessionlogTokenUsage;
|
|
70
|
-
context?: string;
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
// ============================================================================
|
|
74
|
-
// Token Usage
|
|
75
|
-
// ============================================================================
|
|
76
|
-
|
|
77
|
-
/**
|
|
78
|
-
* Token usage statistics (provider-facing, simplified)
|
|
79
|
-
*/
|
|
80
|
-
export interface SessionlogTokenUsage {
|
|
81
|
-
input?: number;
|
|
82
|
-
output?: number;
|
|
83
|
-
cache?: number;
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
// ============================================================================
|
|
87
|
-
// Store Interface
|
|
88
|
-
// ============================================================================
|
|
89
|
-
|
|
90
|
-
/**
|
|
91
|
-
* Interface for accessing Sessionlog data (CLI or direct reads)
|
|
92
|
-
*/
|
|
93
|
-
export interface SessionlogStore {
|
|
94
|
-
getSession(id: string): Promise<SessionlogSession | null>;
|
|
95
|
-
listSessions(): Promise<SessionlogSession[]>;
|
|
96
|
-
getCheckpoint(id: string): Promise<SessionlogCheckpoint | null>;
|
|
97
|
-
listCheckpoints(): Promise<SessionlogCheckpoint[]>;
|
|
98
|
-
search(query: string): Promise<Array<SessionlogSession | SessionlogCheckpoint>>;
|
|
99
|
-
}
|
|
@@ -1,266 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Session Store
|
|
3
|
-
*
|
|
4
|
-
* Manages session state files in .git/sessionlog-sessions/.
|
|
5
|
-
* Each active session has a JSON state file that tracks
|
|
6
|
-
* its lifecycle, files touched, checkpoints, and token usage.
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import * as fs from 'node:fs';
|
|
10
|
-
import * as path from 'node:path';
|
|
11
|
-
import {
|
|
12
|
-
type SessionState,
|
|
13
|
-
type SessionPhase,
|
|
14
|
-
type TokenUsage,
|
|
15
|
-
type PlanEntry,
|
|
16
|
-
STALE_SESSION_DAYS,
|
|
17
|
-
} from '../types.js';
|
|
18
|
-
import { getSessionsDir, atomicWriteFile } from '../git-operations.js';
|
|
19
|
-
|
|
20
|
-
// ============================================================================
|
|
21
|
-
// Session Store Interface
|
|
22
|
-
// ============================================================================
|
|
23
|
-
|
|
24
|
-
export interface SessionStore {
|
|
25
|
-
/** Load a session state by ID */
|
|
26
|
-
load(sessionID: string): Promise<SessionState | null>;
|
|
27
|
-
|
|
28
|
-
/** List all session states */
|
|
29
|
-
list(): Promise<SessionState[]>;
|
|
30
|
-
|
|
31
|
-
/** Save a session state */
|
|
32
|
-
save(state: SessionState): Promise<void>;
|
|
33
|
-
|
|
34
|
-
/** Delete a session state */
|
|
35
|
-
delete(sessionID: string): Promise<void>;
|
|
36
|
-
|
|
37
|
-
/** Get the sessions directory path */
|
|
38
|
-
getDir(): Promise<string>;
|
|
39
|
-
|
|
40
|
-
/** Check if a session exists */
|
|
41
|
-
exists(sessionID: string): Promise<boolean>;
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
// ============================================================================
|
|
45
|
-
// Filesystem Session Store
|
|
46
|
-
// ============================================================================
|
|
47
|
-
|
|
48
|
-
/**
|
|
49
|
-
* Create a session store.
|
|
50
|
-
*
|
|
51
|
-
* @param cwd - The project working directory (used to locate .git/sessionlog-sessions/)
|
|
52
|
-
* @param sessionsDir - Optional explicit directory for session state files.
|
|
53
|
-
* When provided, session files are stored here instead of .git/sessionlog-sessions/.
|
|
54
|
-
* This is used when a separate session repo is configured.
|
|
55
|
-
*/
|
|
56
|
-
export function createSessionStore(cwd?: string, sessionsDir?: string): SessionStore {
|
|
57
|
-
let sessionsDirCache: string | null = sessionsDir ?? null;
|
|
58
|
-
|
|
59
|
-
async function getDir(): Promise<string> {
|
|
60
|
-
if (sessionsDirCache) return sessionsDirCache;
|
|
61
|
-
sessionsDirCache = await getSessionsDir(cwd);
|
|
62
|
-
return sessionsDirCache;
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
function sessionFilePath(dir: string, sessionID: string): string {
|
|
66
|
-
return path.join(dir, `${sessionID}.json`);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
function parseSessionFile(filePath: string): SessionState | null {
|
|
70
|
-
try {
|
|
71
|
-
const content = fs.readFileSync(filePath, 'utf-8');
|
|
72
|
-
const data = JSON.parse(content) as Record<string, unknown>;
|
|
73
|
-
const id = path.basename(filePath, '.json');
|
|
74
|
-
|
|
75
|
-
return normalizeSessionState(id, data);
|
|
76
|
-
} catch {
|
|
77
|
-
return null;
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
function isStale(state: SessionState): boolean {
|
|
82
|
-
if (state.phase !== 'ended' || !state.endedAt) return false;
|
|
83
|
-
const endedAt = new Date(state.endedAt);
|
|
84
|
-
const staleThreshold = Date.now() - STALE_SESSION_DAYS * 24 * 60 * 60 * 1000;
|
|
85
|
-
return endedAt.getTime() < staleThreshold;
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
return {
|
|
89
|
-
async load(sessionID: string): Promise<SessionState | null> {
|
|
90
|
-
const dir = await getDir();
|
|
91
|
-
const filePath = sessionFilePath(dir, sessionID);
|
|
92
|
-
|
|
93
|
-
const state = parseSessionFile(filePath);
|
|
94
|
-
if (!state) return null;
|
|
95
|
-
|
|
96
|
-
// Auto-delete stale sessions
|
|
97
|
-
if (isStale(state)) {
|
|
98
|
-
try {
|
|
99
|
-
await fs.promises.unlink(filePath);
|
|
100
|
-
} catch {
|
|
101
|
-
// Ignore deletion errors
|
|
102
|
-
}
|
|
103
|
-
return null;
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
return state;
|
|
107
|
-
},
|
|
108
|
-
|
|
109
|
-
async list(): Promise<SessionState[]> {
|
|
110
|
-
const dir = await getDir();
|
|
111
|
-
|
|
112
|
-
try {
|
|
113
|
-
const files = await fs.promises.readdir(dir);
|
|
114
|
-
const states: SessionState[] = [];
|
|
115
|
-
|
|
116
|
-
for (const file of files) {
|
|
117
|
-
if (!file.endsWith('.json')) continue;
|
|
118
|
-
const filePath = path.join(dir, file);
|
|
119
|
-
const state = parseSessionFile(filePath);
|
|
120
|
-
|
|
121
|
-
if (!state) continue;
|
|
122
|
-
|
|
123
|
-
// Auto-delete stale sessions
|
|
124
|
-
if (isStale(state)) {
|
|
125
|
-
try {
|
|
126
|
-
await fs.promises.unlink(filePath);
|
|
127
|
-
} catch {
|
|
128
|
-
// Ignore
|
|
129
|
-
}
|
|
130
|
-
continue;
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
states.push(state);
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
return states;
|
|
137
|
-
} catch {
|
|
138
|
-
return [];
|
|
139
|
-
}
|
|
140
|
-
},
|
|
141
|
-
|
|
142
|
-
async save(state: SessionState): Promise<void> {
|
|
143
|
-
const dir = await getDir();
|
|
144
|
-
const filePath = sessionFilePath(dir, state.sessionID);
|
|
145
|
-
const content = JSON.stringify(serializeSessionState(state), null, 2);
|
|
146
|
-
await atomicWriteFile(filePath, content);
|
|
147
|
-
},
|
|
148
|
-
|
|
149
|
-
async delete(sessionID: string): Promise<void> {
|
|
150
|
-
const dir = await getDir();
|
|
151
|
-
const filePath = sessionFilePath(dir, sessionID);
|
|
152
|
-
try {
|
|
153
|
-
await fs.promises.unlink(filePath);
|
|
154
|
-
} catch {
|
|
155
|
-
// Ignore if already deleted
|
|
156
|
-
}
|
|
157
|
-
},
|
|
158
|
-
|
|
159
|
-
async getDir(): Promise<string> {
|
|
160
|
-
return getDir();
|
|
161
|
-
},
|
|
162
|
-
|
|
163
|
-
async exists(sessionID: string): Promise<boolean> {
|
|
164
|
-
const dir = await getDir();
|
|
165
|
-
const filePath = sessionFilePath(dir, sessionID);
|
|
166
|
-
try {
|
|
167
|
-
await fs.promises.access(filePath);
|
|
168
|
-
return true;
|
|
169
|
-
} catch {
|
|
170
|
-
return false;
|
|
171
|
-
}
|
|
172
|
-
},
|
|
173
|
-
};
|
|
174
|
-
}
|
|
175
|
-
|
|
176
|
-
// ============================================================================
|
|
177
|
-
// Normalization
|
|
178
|
-
// ============================================================================
|
|
179
|
-
|
|
180
|
-
/**
|
|
181
|
-
* Normalize raw JSON data into a SessionState, handling field name variations
|
|
182
|
-
*/
|
|
183
|
-
export function normalizeSessionState(id: string, data: Record<string, unknown>): SessionState {
|
|
184
|
-
return {
|
|
185
|
-
sessionID: String(data.sessionID ?? data.session_id ?? id),
|
|
186
|
-
cliVersion: data.cliVersion as string | undefined,
|
|
187
|
-
baseCommit: String(data.baseCommit ?? data.base_commit ?? ''),
|
|
188
|
-
attributionBaseCommit: data.attributionBaseCommit as string | undefined,
|
|
189
|
-
worktreePath: data.worktreePath as string | undefined,
|
|
190
|
-
worktreeID: data.worktreeID as string | undefined,
|
|
191
|
-
startedAt: String(data.startedAt ?? data.started_at ?? new Date().toISOString()),
|
|
192
|
-
endedAt: (data.endedAt ?? data.ended_at) as string | undefined,
|
|
193
|
-
phase: normalizePhase(String(data.phase ?? data.state ?? 'idle')),
|
|
194
|
-
turnID: data.turnID as string | undefined,
|
|
195
|
-
turnCheckpointIDs: Array.isArray(data.turnCheckpointIDs)
|
|
196
|
-
? data.turnCheckpointIDs.map(String)
|
|
197
|
-
: [],
|
|
198
|
-
lastInteractionTime: (data.lastInteractionTime ?? data.last_interaction_time) as
|
|
199
|
-
| string
|
|
200
|
-
| undefined,
|
|
201
|
-
stepCount: Number(data.stepCount ?? data.step_count ?? 0),
|
|
202
|
-
checkpointTranscriptStart: Number(data.checkpointTranscriptStart ?? 0),
|
|
203
|
-
untrackedFilesAtStart: Array.isArray(data.untrackedFilesAtStart)
|
|
204
|
-
? data.untrackedFilesAtStart.map(String)
|
|
205
|
-
: [],
|
|
206
|
-
filesTouched: Array.isArray(data.filesTouched) ? data.filesTouched.map(String) : [],
|
|
207
|
-
lastCheckpointID: data.lastCheckpointID as string | undefined,
|
|
208
|
-
agentType: String(data.agentType ?? data.agent ?? data.agent_type ?? 'Agent'),
|
|
209
|
-
tokenUsage: data.tokenUsage as TokenUsage | undefined,
|
|
210
|
-
transcriptIdentifierAtStart: data.transcriptIdentifierAtStart as string | undefined,
|
|
211
|
-
transcriptPath: data.transcriptPath as string | undefined,
|
|
212
|
-
firstPrompt: data.firstPrompt as string | undefined,
|
|
213
|
-
promptAttributions: data.promptAttributions as SessionState['promptAttributions'],
|
|
214
|
-
pendingPromptAttribution:
|
|
215
|
-
data.pendingPromptAttribution as SessionState['pendingPromptAttribution'],
|
|
216
|
-
tasks: data.tasks as SessionState['tasks'],
|
|
217
|
-
inPlanMode: data.inPlanMode as boolean | undefined,
|
|
218
|
-
planModeEntries: data.planModeEntries as number | undefined,
|
|
219
|
-
planEntries: normalizePlanEntries(data),
|
|
220
|
-
skillsUsed: Array.isArray(data.skillsUsed)
|
|
221
|
-
? (data.skillsUsed as SessionState['skillsUsed'])
|
|
222
|
-
: undefined,
|
|
223
|
-
};
|
|
224
|
-
}
|
|
225
|
-
|
|
226
|
-
function normalizePhase(phase: string): SessionPhase {
|
|
227
|
-
const lower = phase.toLowerCase();
|
|
228
|
-
if (lower === 'active') return 'active';
|
|
229
|
-
if (lower === 'idle') return 'idle';
|
|
230
|
-
if (lower === 'ended') return 'ended';
|
|
231
|
-
return 'idle';
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
function normalizePlanEntries(data: Record<string, unknown>): PlanEntry[] | undefined {
|
|
235
|
-
if (Array.isArray(data.planEntries)) {
|
|
236
|
-
return data.planEntries as PlanEntry[];
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
// Migration: synthesize from old planFilePath/planContent fields
|
|
240
|
-
const filePath = data.planFilePath as string | undefined;
|
|
241
|
-
const content = data.planContent as string | undefined;
|
|
242
|
-
if (filePath || content) {
|
|
243
|
-
return [
|
|
244
|
-
{
|
|
245
|
-
enteredAt: (data.startedAt as string) ?? new Date().toISOString(),
|
|
246
|
-
exitedAt: (data.startedAt as string) ?? new Date().toISOString(),
|
|
247
|
-
filePath,
|
|
248
|
-
content,
|
|
249
|
-
},
|
|
250
|
-
];
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
return undefined;
|
|
254
|
-
}
|
|
255
|
-
|
|
256
|
-
/**
|
|
257
|
-
* Serialize a SessionState for writing to disk
|
|
258
|
-
*/
|
|
259
|
-
function serializeSessionState(state: SessionState): Record<string, unknown> {
|
|
260
|
-
const result: Record<string, unknown> = { ...state };
|
|
261
|
-
// Remove undefined values
|
|
262
|
-
for (const [key, value] of Object.entries(result)) {
|
|
263
|
-
if (value === undefined) delete result[key];
|
|
264
|
-
}
|
|
265
|
-
return result;
|
|
266
|
-
}
|
|
@@ -1,296 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Attribution Logic
|
|
3
|
-
*
|
|
4
|
-
* Computes line-level attribution between agent-generated and human-edited code.
|
|
5
|
-
* Uses diff-based analysis to track who wrote which lines.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import type { InitialAttribution } from '../types.js';
|
|
9
|
-
import { countLines } from '../utils/string-utils.js';
|
|
10
|
-
import type { PromptAttribution } from './types.js';
|
|
11
|
-
|
|
12
|
-
// ============================================================================
|
|
13
|
-
// Diff-based Attribution
|
|
14
|
-
// ============================================================================
|
|
15
|
-
|
|
16
|
-
/**
|
|
17
|
-
* Compare two strings and return line-level diff stats.
|
|
18
|
-
* Returns [unchanged, added, removed] line counts.
|
|
19
|
-
*
|
|
20
|
-
* Uses a simple line-by-line comparison. For production-grade diffing,
|
|
21
|
-
* we compare line sets since we only need counts, not actual patches.
|
|
22
|
-
*/
|
|
23
|
-
export function diffLines(
|
|
24
|
-
oldContent: string,
|
|
25
|
-
newContent: string,
|
|
26
|
-
): [unchanged: number, added: number, removed: number] {
|
|
27
|
-
if (oldContent === newContent) {
|
|
28
|
-
return [countLines(newContent), 0, 0];
|
|
29
|
-
}
|
|
30
|
-
if (oldContent === '') {
|
|
31
|
-
return [0, countLines(newContent), 0];
|
|
32
|
-
}
|
|
33
|
-
if (newContent === '') {
|
|
34
|
-
return [0, 0, countLines(oldContent)];
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
// Line-by-line LCS-based diff for accurate counting
|
|
38
|
-
const oldLines = oldContent.split('\n');
|
|
39
|
-
const newLines = newContent.split('\n');
|
|
40
|
-
|
|
41
|
-
// Remove trailing empty string from split if content ends with newline
|
|
42
|
-
if (oldLines[oldLines.length - 1] === '' && oldContent.endsWith('\n')) {
|
|
43
|
-
oldLines.pop();
|
|
44
|
-
}
|
|
45
|
-
if (newLines[newLines.length - 1] === '' && newContent.endsWith('\n')) {
|
|
46
|
-
newLines.pop();
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
// Build a set-based approximation for line counts
|
|
50
|
-
// This is simpler than full LCS but gives good accuracy for attribution
|
|
51
|
-
const oldSet = new Map<string, number>();
|
|
52
|
-
for (const line of oldLines) {
|
|
53
|
-
oldSet.set(line, (oldSet.get(line) ?? 0) + 1);
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
let unchanged = 0;
|
|
57
|
-
const matchedOld = new Map<string, number>();
|
|
58
|
-
|
|
59
|
-
for (const line of newLines) {
|
|
60
|
-
const available = (oldSet.get(line) ?? 0) - (matchedOld.get(line) ?? 0);
|
|
61
|
-
if (available > 0) {
|
|
62
|
-
unchanged++;
|
|
63
|
-
matchedOld.set(line, (matchedOld.get(line) ?? 0) + 1);
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
const removed = oldLines.length - unchanged;
|
|
68
|
-
const added = newLines.length - unchanged;
|
|
69
|
-
|
|
70
|
-
return [unchanged, added, removed];
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
/**
|
|
74
|
-
* Get all changed files between two sets of file contents.
|
|
75
|
-
*/
|
|
76
|
-
export function getAllChangedFiles(
|
|
77
|
-
tree1: Map<string, string>,
|
|
78
|
-
tree2: Map<string, string>,
|
|
79
|
-
): string[] {
|
|
80
|
-
const changed = new Set<string>();
|
|
81
|
-
|
|
82
|
-
for (const [path, hash1] of tree1) {
|
|
83
|
-
const hash2 = tree2.get(path);
|
|
84
|
-
if (hash2 === undefined || hash1 !== hash2) {
|
|
85
|
-
changed.add(path);
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
for (const path of tree2.keys()) {
|
|
90
|
-
if (!tree1.has(path)) {
|
|
91
|
-
changed.add(path);
|
|
92
|
-
}
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
return Array.from(changed);
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
/**
|
|
99
|
-
* Calculate attribution with accumulated prompt data.
|
|
100
|
-
*
|
|
101
|
-
* This provides more accurate attribution than tree-only comparison because
|
|
102
|
-
* it captures user edits between checkpoints.
|
|
103
|
-
*
|
|
104
|
-
* @param baseFiles - File contents at session start
|
|
105
|
-
* @param shadowFiles - File contents at last checkpoint
|
|
106
|
-
* @param headFiles - File contents at commit time (HEAD)
|
|
107
|
-
* @param filesTouched - Files the agent modified
|
|
108
|
-
* @param promptAttributions - Per-prompt user edit tracking
|
|
109
|
-
*/
|
|
110
|
-
export function calculateAttributionWithAccumulated(
|
|
111
|
-
baseFiles: Map<string, string>,
|
|
112
|
-
shadowFiles: Map<string, string>,
|
|
113
|
-
headFiles: Map<string, string>,
|
|
114
|
-
filesTouched: string[],
|
|
115
|
-
promptAttributions: PromptAttribution[],
|
|
116
|
-
): InitialAttribution | null {
|
|
117
|
-
if (filesTouched.length === 0) return null;
|
|
118
|
-
|
|
119
|
-
// Sum accumulated user lines from prompt attributions
|
|
120
|
-
let accumulatedUserRemoved = 0;
|
|
121
|
-
const accumulatedUserAddedPerFile = new Map<string, number>();
|
|
122
|
-
|
|
123
|
-
for (const pa of promptAttributions) {
|
|
124
|
-
accumulatedUserRemoved += pa.userLinesRemoved;
|
|
125
|
-
for (const [filePath, added] of Object.entries(pa.userAddedPerFile)) {
|
|
126
|
-
accumulatedUserAddedPerFile.set(
|
|
127
|
-
filePath,
|
|
128
|
-
(accumulatedUserAddedPerFile.get(filePath) ?? 0) + added,
|
|
129
|
-
);
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
// Calculate for agent-touched files
|
|
134
|
-
let totalAgentAndUserWork = 0;
|
|
135
|
-
let postCheckpointUserAdded = 0;
|
|
136
|
-
let postCheckpointUserRemoved = 0;
|
|
137
|
-
const postCheckpointUserRemovedPerFile = new Map<string, number>();
|
|
138
|
-
|
|
139
|
-
for (const filePath of filesTouched) {
|
|
140
|
-
const baseContent = baseFiles.get(filePath) ?? '';
|
|
141
|
-
const shadowContent = shadowFiles.get(filePath) ?? '';
|
|
142
|
-
const headContent = headFiles.get(filePath) ?? '';
|
|
143
|
-
|
|
144
|
-
// Total work in shadow: base → shadow
|
|
145
|
-
const [, workAdded] = diffLines(baseContent, shadowContent);
|
|
146
|
-
totalAgentAndUserWork += workAdded;
|
|
147
|
-
|
|
148
|
-
// Post-checkpoint user edits: shadow → head
|
|
149
|
-
const [, postUserAdded, postUserRemoved] = diffLines(shadowContent, headContent);
|
|
150
|
-
postCheckpointUserAdded += postUserAdded;
|
|
151
|
-
postCheckpointUserRemoved += postUserRemoved;
|
|
152
|
-
|
|
153
|
-
if (postUserRemoved > 0) {
|
|
154
|
-
postCheckpointUserRemovedPerFile.set(filePath, postUserRemoved);
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
// User edits to non-agent files
|
|
159
|
-
const nonAgentFiles = getAllChangedFiles(baseFiles, headFiles);
|
|
160
|
-
const touchedSet = new Set(filesTouched);
|
|
161
|
-
let allUserEditsToNonAgentFiles = 0;
|
|
162
|
-
|
|
163
|
-
for (const filePath of nonAgentFiles) {
|
|
164
|
-
if (touchedSet.has(filePath)) continue;
|
|
165
|
-
const baseContent = baseFiles.get(filePath) ?? '';
|
|
166
|
-
const headContent = headFiles.get(filePath) ?? '';
|
|
167
|
-
const [, userAdded] = diffLines(baseContent, headContent);
|
|
168
|
-
allUserEditsToNonAgentFiles += userAdded;
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
// Separate accumulated edits by file type
|
|
172
|
-
const committedNonAgentSet = new Set(nonAgentFiles.filter((f) => !touchedSet.has(f)));
|
|
173
|
-
|
|
174
|
-
let accumulatedToAgentFiles = 0;
|
|
175
|
-
let accumulatedToCommittedNonAgentFiles = 0;
|
|
176
|
-
for (const [filePath, added] of accumulatedUserAddedPerFile) {
|
|
177
|
-
if (touchedSet.has(filePath)) {
|
|
178
|
-
accumulatedToAgentFiles += added;
|
|
179
|
-
} else if (committedNonAgentSet.has(filePath)) {
|
|
180
|
-
accumulatedToCommittedNonAgentFiles += added;
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
// Agent work = (base→shadow for agent files) - (accumulated user edits to agent files)
|
|
185
|
-
const totalAgentAdded = Math.max(0, totalAgentAndUserWork - accumulatedToAgentFiles);
|
|
186
|
-
|
|
187
|
-
// Post-checkpoint edits to non-agent files
|
|
188
|
-
const postToNonAgentFiles = Math.max(
|
|
189
|
-
0,
|
|
190
|
-
allUserEditsToNonAgentFiles - accumulatedToCommittedNonAgentFiles,
|
|
191
|
-
);
|
|
192
|
-
|
|
193
|
-
// Total user contribution
|
|
194
|
-
const relevantAccumulatedUser = accumulatedToAgentFiles + accumulatedToCommittedNonAgentFiles;
|
|
195
|
-
const totalUserAdded = relevantAccumulatedUser + postCheckpointUserAdded + postToNonAgentFiles;
|
|
196
|
-
const totalUserRemoved = accumulatedUserRemoved + postCheckpointUserRemoved;
|
|
197
|
-
|
|
198
|
-
// Estimate modified lines
|
|
199
|
-
const totalHumanModified = Math.min(totalUserAdded, totalUserRemoved);
|
|
200
|
-
|
|
201
|
-
// Estimate user self-modifications
|
|
202
|
-
const userSelfModified = estimateUserSelfModifications(
|
|
203
|
-
accumulatedUserAddedPerFile,
|
|
204
|
-
postCheckpointUserRemovedPerFile,
|
|
205
|
-
);
|
|
206
|
-
|
|
207
|
-
const humanModifiedAgent = Math.max(0, totalHumanModified - userSelfModified);
|
|
208
|
-
const pureUserAdded = totalUserAdded - totalHumanModified;
|
|
209
|
-
const pureUserRemoved = totalUserRemoved - totalHumanModified;
|
|
210
|
-
|
|
211
|
-
let totalCommitted = totalAgentAdded + pureUserAdded - pureUserRemoved;
|
|
212
|
-
if (totalCommitted <= 0) {
|
|
213
|
-
totalCommitted = Math.max(0, totalAgentAdded);
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
const agentLinesInCommit = Math.max(0, totalAgentAdded - pureUserRemoved - humanModifiedAgent);
|
|
217
|
-
|
|
218
|
-
let agentPercentage = 0;
|
|
219
|
-
if (totalCommitted > 0) {
|
|
220
|
-
agentPercentage = (agentLinesInCommit / totalCommitted) * 100;
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
return {
|
|
224
|
-
calculatedAt: new Date().toISOString(),
|
|
225
|
-
agentLines: agentLinesInCommit,
|
|
226
|
-
humanAdded: pureUserAdded,
|
|
227
|
-
humanModified: totalHumanModified,
|
|
228
|
-
humanRemoved: pureUserRemoved,
|
|
229
|
-
totalCommitted,
|
|
230
|
-
agentPercentage,
|
|
231
|
-
};
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
/**
|
|
235
|
-
* Estimate how many removed lines were the user's own additions.
|
|
236
|
-
* Uses LIFO assumption: user removes their own recent additions first.
|
|
237
|
-
*/
|
|
238
|
-
function estimateUserSelfModifications(
|
|
239
|
-
accumulatedUserAddedPerFile: Map<string, number>,
|
|
240
|
-
postCheckpointUserRemovedPerFile: Map<string, number>,
|
|
241
|
-
): number {
|
|
242
|
-
let selfModified = 0;
|
|
243
|
-
for (const [filePath, removed] of postCheckpointUserRemovedPerFile) {
|
|
244
|
-
const userAddedToFile = accumulatedUserAddedPerFile.get(filePath) ?? 0;
|
|
245
|
-
selfModified += Math.min(removed, userAddedToFile);
|
|
246
|
-
}
|
|
247
|
-
return selfModified;
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
/**
|
|
251
|
-
* Calculate prompt attribution at the start of a prompt.
|
|
252
|
-
* Captures user edits since the last checkpoint BEFORE the agent makes changes.
|
|
253
|
-
*/
|
|
254
|
-
export function calculatePromptAttribution(
|
|
255
|
-
baseFiles: Map<string, string>,
|
|
256
|
-
lastCheckpointFiles: Map<string, string> | null,
|
|
257
|
-
worktreeFiles: Map<string, string>,
|
|
258
|
-
checkpointNumber: number,
|
|
259
|
-
): PromptAttribution {
|
|
260
|
-
const result: PromptAttribution = {
|
|
261
|
-
checkpointNumber,
|
|
262
|
-
userLinesAdded: 0,
|
|
263
|
-
userLinesRemoved: 0,
|
|
264
|
-
agentLinesAdded: 0,
|
|
265
|
-
agentLinesRemoved: 0,
|
|
266
|
-
userAddedPerFile: {},
|
|
267
|
-
};
|
|
268
|
-
|
|
269
|
-
if (worktreeFiles.size === 0) return result;
|
|
270
|
-
|
|
271
|
-
const referenceFiles = lastCheckpointFiles ?? baseFiles;
|
|
272
|
-
|
|
273
|
-
for (const [filePath, worktreeContent] of worktreeFiles) {
|
|
274
|
-
const referenceContent = referenceFiles.get(filePath) ?? '';
|
|
275
|
-
const baseContent = baseFiles.get(filePath) ?? '';
|
|
276
|
-
|
|
277
|
-
// User changes: diff(reference, worktree)
|
|
278
|
-
const [, userAdded, userRemoved] = diffLines(referenceContent, worktreeContent);
|
|
279
|
-
result.userLinesAdded += userAdded;
|
|
280
|
-
result.userLinesRemoved += userRemoved;
|
|
281
|
-
|
|
282
|
-
if (userAdded > 0) {
|
|
283
|
-
result.userAddedPerFile[filePath] = userAdded;
|
|
284
|
-
}
|
|
285
|
-
|
|
286
|
-
// Agent lines so far: diff(base, lastCheckpoint)
|
|
287
|
-
if (lastCheckpointFiles) {
|
|
288
|
-
const checkpointContent = lastCheckpointFiles.get(filePath) ?? '';
|
|
289
|
-
const [, agentAdded, agentRemoved] = diffLines(baseContent, checkpointContent);
|
|
290
|
-
result.agentLinesAdded += agentAdded;
|
|
291
|
-
result.agentLinesRemoved += agentRemoved;
|
|
292
|
-
}
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
return result;
|
|
296
|
-
}
|