claude-memory-layer 1.0.31 → 1.0.32
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/README.md +9 -2
- package/dist/cli/index.js +1 -1
- package/package.json +11 -2
- package/scripts/postinstall-embedding-backend.cjs +16 -12
- package/AGENTS.md +0 -71
- package/CLAUDE.md +0 -30
- package/HANDOFF.md +0 -92
- package/Memo.txt +0 -558
- package/benchmarks/replay/anonymized-real-sessions.json +0 -48
- package/config/kpi-thresholds.json +0 -7
- package/context.md +0 -636
- package/docs/ARCHITECTURE_COMPARISON_AND_RECOMMENDATIONS.md +0 -627
- package/docs/HERMES_MEMORY_INGESTION_ANALYSIS.md +0 -440
- package/docs/MCP_MEMORY_SERVICE_COMPARATIVE_REVIEW.md +0 -271
- package/docs/MEMORY_USEFULNESS_AUDIT.md +0 -371
- package/docs/MEMORY_USEFULNESS_AUDIT_RAW.json +0 -80
- package/docs/MEMSEARCH_PROJECT_STRUCTURE_ANALYSIS.md +0 -333
- package/docs/MEMU_ADOPTION.md +0 -40
- package/docs/OPERATIONS.md +0 -18
- package/docs/PRODUCT_VALIDATION_MATRIX.md +0 -82
- package/docs/PROJECT_STRUCTURE_ANALYSIS.md +0 -421
- package/docs/REFACTORING_MILESTONES_AND_ISSUES.md +0 -501
- package/docs/REFACTORING_PLAN_THIN_CORE.md +0 -414
- package/docs/REFERENCE_PROJECT_ANALYSES.md +0 -25
- package/docs/SUPERLOCALMEMORY_PROJECT_STRUCTURE_ANALYSIS.md +0 -452
- package/docs/TARGET_ARCHITECTURE_AND_FOLDER_STRUCTURE.md +0 -446
- package/docs/architecture/comparison-index.md +0 -47
- package/docs/reports/codex-real-data-validation-20260505T040447Z.md +0 -46
- package/plan.md +0 -1642
- package/scripts/build.ts +0 -159
- package/scripts/bump-patch-version.sh +0 -18
- package/scripts/delete-unknown-projects.js +0 -154
- package/scripts/fix-sync-gap.js +0 -32
- package/scripts/generate-session-qrels.ts +0 -126
- package/scripts/heartbeat-memory-orchestrator.sh +0 -28
- package/scripts/replay-retrieval-benchmark.ts +0 -69
- package/scripts/report-sync-gap.js +0 -26
- package/scripts/review-queue-auto-resolve.js +0 -21
- package/scripts/sync-gap-auto-heal.sh +0 -17
- package/spec.md +0 -624
- package/specs/20260207-dashboard-upgrade/context.md +0 -38
- package/specs/20260207-dashboard-upgrade/spec.md +0 -96
- package/specs/citations-system/context.md +0 -243
- package/specs/citations-system/plan.md +0 -495
- package/specs/citations-system/spec.md +0 -371
- package/specs/endless-mode/context.md +0 -305
- package/specs/endless-mode/plan.md +0 -620
- package/specs/endless-mode/spec.md +0 -455
- package/specs/entity-edge-model/context.md +0 -401
- package/specs/entity-edge-model/plan.md +0 -459
- package/specs/entity-edge-model/spec.md +0 -391
- package/specs/evidence-aligner-v2/context.md +0 -401
- package/specs/evidence-aligner-v2/plan.md +0 -303
- package/specs/evidence-aligner-v2/spec.md +0 -312
- package/specs/mcp-desktop-integration/context.md +0 -278
- package/specs/mcp-desktop-integration/plan.md +0 -550
- package/specs/mcp-desktop-integration/spec.md +0 -494
- package/specs/memory-utilization-improvements/context.md +0 -145
- package/specs/memory-utilization-improvements/plan.md +0 -361
- package/specs/memory-utilization-improvements/spec.md +0 -361
- package/specs/post-tool-use-hook/context.md +0 -319
- package/specs/post-tool-use-hook/plan.md +0 -469
- package/specs/post-tool-use-hook/spec.md +0 -364
- package/specs/private-tags/context.md +0 -288
- package/specs/private-tags/plan.md +0 -412
- package/specs/private-tags/spec.md +0 -345
- package/specs/progressive-disclosure/context.md +0 -346
- package/specs/progressive-disclosure/plan.md +0 -663
- package/specs/progressive-disclosure/spec.md +0 -415
- package/specs/selective-tool-observation/context.md +0 -100
- package/specs/selective-tool-observation/plan.md +0 -158
- package/specs/selective-tool-observation/spec.md +0 -127
- package/specs/task-entity-system/context.md +0 -297
- package/specs/task-entity-system/plan.md +0 -301
- package/specs/task-entity-system/spec.md +0 -314
- package/specs/thin-core-refactor/context.md +0 -275
- package/specs/thin-core-refactor/plan.md +0 -536
- package/specs/thin-core-refactor/spec.md +0 -465
- package/specs/vector-outbox-v2/context.md +0 -470
- package/specs/vector-outbox-v2/plan.md +0 -562
- package/specs/vector-outbox-v2/spec.md +0 -466
- package/specs/web-viewer-ui/context.md +0 -384
- package/specs/web-viewer-ui/plan.md +0 -797
- package/specs/web-viewer-ui/spec.md +0 -516
- package/src/adapters/claude/capture/index.ts +0 -3
- package/src/adapters/claude/context/index.ts +0 -3
- package/src/adapters/claude/hooks/index.ts +0 -21
- package/src/adapters/claude/hooks/post-tool-use.ts +0 -239
- package/src/adapters/claude/hooks/prompt-injection-policy.ts +0 -104
- package/src/adapters/claude/hooks/semantic-daemon-client.ts +0 -209
- package/src/adapters/claude/hooks/semantic-daemon.ts +0 -283
- package/src/adapters/claude/hooks/session-end.ts +0 -59
- package/src/adapters/claude/hooks/session-start.ts +0 -73
- package/src/adapters/claude/hooks/stop.ts +0 -128
- package/src/adapters/claude/hooks/user-prompt-submit.ts +0 -361
- package/src/adapters/claude/index.ts +0 -4
- package/src/adapters/claude/transcript/index.ts +0 -4
- package/src/adapters/claude/transcript/transcript-reader.ts +0 -57
- package/src/adapters/claude/transcript/turn-reconstructor.ts +0 -65
- package/src/apps/cli/claude-settings-hooks.ts +0 -138
- package/src/apps/cli/codex-import-runner.ts +0 -125
- package/src/apps/cli/codex-validation-output.ts +0 -95
- package/src/apps/cli/hermes-import-runner.ts +0 -130
- package/src/apps/cli/hermes-validation-output.ts +0 -91
- package/src/apps/cli/index.ts +0 -1735
- package/src/apps/cli/mcp-install.ts +0 -106
- package/src/apps/cli/retrieval-disclosure-output.ts +0 -196
- package/src/apps/dashboard/assets/js/bootstrap.js +0 -244
- package/src/apps/dashboard/assets/js/chat.js +0 -373
- package/src/apps/dashboard/assets/js/disclosure.js +0 -232
- package/src/apps/dashboard/assets/js/modals.js +0 -298
- package/src/apps/dashboard/assets/js/overview.js +0 -655
- package/src/apps/dashboard/assets/js/state.js +0 -72
- package/src/apps/dashboard/assets/js/views.js +0 -468
- package/src/apps/dashboard/index.html +0 -543
- package/src/apps/dashboard/index.ts +0 -3
- package/src/apps/dashboard/style.css +0 -1750
- package/src/apps/index.ts +0 -5
- package/src/apps/server/api/chat.ts +0 -244
- package/src/apps/server/api/citations.ts +0 -105
- package/src/apps/server/api/events.ts +0 -137
- package/src/apps/server/api/health.ts +0 -53
- package/src/apps/server/api/index.ts +0 -26
- package/src/apps/server/api/projects.ts +0 -74
- package/src/apps/server/api/search.ts +0 -184
- package/src/apps/server/api/sessions.ts +0 -115
- package/src/apps/server/api/stats.ts +0 -723
- package/src/apps/server/api/turns.ts +0 -143
- package/src/apps/server/api/utils.ts +0 -65
- package/src/apps/server/index.ts +0 -111
- package/src/cli/index.ts +0 -3
- package/src/cli/retrieval-disclosure-output.ts +0 -2
- package/src/compat/index.ts +0 -5
- package/src/core/canonical-key.ts +0 -186
- package/src/core/citation-generator.ts +0 -63
- package/src/core/consolidated-store.ts +0 -356
- package/src/core/consolidation-worker.ts +0 -493
- package/src/core/context-formatter.ts +0 -276
- package/src/core/continuity-manager.ts +0 -341
- package/src/core/db-wrapper.ts +0 -64
- package/src/core/derive/fact-deriver.ts +0 -170
- package/src/core/derive/index.ts +0 -2
- package/src/core/derive/summary-deriver.ts +0 -76
- package/src/core/edge-repo.ts +0 -333
- package/src/core/embedder.ts +0 -4
- package/src/core/engine/embedding-maintenance-service.ts +0 -187
- package/src/core/engine/endless-memory-services.ts +0 -4
- package/src/core/engine/index.ts +0 -19
- package/src/core/engine/memory-engine-services.ts +0 -170
- package/src/core/engine/memory-ingest-service.ts +0 -317
- package/src/core/engine/memory-query-service.ts +0 -173
- package/src/core/engine/memory-runtime-service.ts +0 -162
- package/src/core/engine/memory-service-composition.ts +0 -231
- package/src/core/engine/retrieval-analytics-service.ts +0 -181
- package/src/core/engine/retrieval-disclosure-service.ts +0 -420
- package/src/core/engine/retrieval-orchestrator.ts +0 -377
- package/src/core/engine/retrieval-services.ts +0 -176
- package/src/core/engine/shared-memory-services.ts +0 -4
- package/src/core/entity-repo.ts +0 -349
- package/src/core/event-store.ts +0 -779
- package/src/core/evidence-aligner.ts +0 -635
- package/src/core/external-market-context.ts +0 -582
- package/src/core/graduation-worker.ts +0 -171
- package/src/core/graduation.ts +0 -377
- package/src/core/index.ts +0 -64
- package/src/core/ingest-interceptor.ts +0 -80
- package/src/core/markdown-mirror.ts +0 -70
- package/src/core/matcher.ts +0 -208
- package/src/core/md-mirror.ts +0 -92
- package/src/core/metadata-extractor.ts +0 -203
- package/src/core/model/memory-fact.ts +0 -30
- package/src/core/model/memory-rule.ts +0 -14
- package/src/core/model/memory-summary.ts +0 -21
- package/src/core/model/raw-event.ts +0 -28
- package/src/core/model/retrieval-result.ts +0 -35
- package/src/core/mongo-sync-config.ts +0 -165
- package/src/core/mongo-sync-worker.ts +0 -381
- package/src/core/privacy/filter.ts +0 -190
- package/src/core/privacy/index.ts +0 -20
- package/src/core/privacy/tag-parser.ts +0 -145
- package/src/core/product-validation-matrix.ts +0 -314
- package/src/core/progressive-retriever.ts +0 -414
- package/src/core/registry/project-path.ts +0 -54
- package/src/core/registry/session-registry.ts +0 -69
- package/src/core/replay-evaluator.ts +0 -625
- package/src/core/retrieval-benchmark.ts +0 -117
- package/src/core/retrieval-quality.ts +0 -109
- package/src/core/retriever.ts +0 -800
- package/src/core/session-qrels.ts +0 -360
- package/src/core/shared-event-store.ts +0 -114
- package/src/core/shared-promoter.ts +0 -249
- package/src/core/shared-store.ts +0 -289
- package/src/core/shared-vector-store.ts +0 -203
- package/src/core/sqlite-event-store.ts +0 -1846
- package/src/core/sqlite-wrapper.ts +0 -116
- package/src/core/sync-worker.ts +0 -228
- package/src/core/tag-taxonomy.ts +0 -51
- package/src/core/task/blocker-resolver.ts +0 -333
- package/src/core/task/index.ts +0 -9
- package/src/core/task/task-matcher.ts +0 -240
- package/src/core/task/task-projector.ts +0 -358
- package/src/core/task/task-resolver.ts +0 -421
- package/src/core/turn-state.ts +0 -207
- package/src/core/types.ts +0 -952
- package/src/core/vector-outbox.ts +0 -299
- package/src/core/vector-store.ts +0 -231
- package/src/core/vector-worker.ts +0 -521
- package/src/core/working-set-store.ts +0 -257
- package/src/extensions/endless-memory/endless-memory-services.ts +0 -350
- package/src/extensions/endless-memory/index.ts +0 -1
- package/src/extensions/index.ts +0 -5
- package/src/extensions/mcp/handlers.ts +0 -960
- package/src/extensions/mcp/index.ts +0 -48
- package/src/extensions/mcp/tools.ts +0 -252
- package/src/extensions/shared-memory/index.ts +0 -1
- package/src/extensions/shared-memory/shared-memory-services.ts +0 -211
- package/src/extensions/vector/embedder.ts +0 -234
- package/src/extensions/vector/index.ts +0 -1
- package/src/hooks/post-tool-use.ts +0 -9
- package/src/hooks/semantic-daemon-client.ts +0 -1
- package/src/hooks/semantic-daemon.ts +0 -11
- package/src/hooks/session-end.ts +0 -9
- package/src/hooks/session-start.ts +0 -9
- package/src/hooks/stop.ts +0 -9
- package/src/hooks/user-prompt-submit.ts +0 -9
- package/src/index.ts +0 -13
- package/src/mcp/handlers.ts +0 -2
- package/src/mcp/index.ts +0 -4
- package/src/mcp/tools.ts +0 -2
- package/src/server/api/chat.ts +0 -2
- package/src/server/api/citations.ts +0 -2
- package/src/server/api/events.ts +0 -2
- package/src/server/api/health.ts +0 -2
- package/src/server/api/index.ts +0 -2
- package/src/server/api/projects.ts +0 -2
- package/src/server/api/search.ts +0 -2
- package/src/server/api/sessions.ts +0 -2
- package/src/server/api/stats.ts +0 -2
- package/src/server/api/turns.ts +0 -2
- package/src/server/api/utils.ts +0 -2
- package/src/server/index.ts +0 -2
- package/src/services/bootstrap-organizer.ts +0 -463
- package/src/services/codex-session-history-importer.ts +0 -966
- package/src/services/hermes-session-history-importer.ts +0 -733
- package/src/services/memory-service-config.ts +0 -36
- package/src/services/memory-service-registry.ts +0 -150
- package/src/services/memory-service.ts +0 -688
- package/src/services/session-history-importer.ts +0 -629
- package/tests/README.md +0 -23
- package/tests/adapters/claude/claude-semantic-daemon-adapter.test.ts +0 -54
- package/tests/adapters/claude/claude-transcript-reconstructor.test.ts +0 -98
- package/tests/adapters/claude-hook-prompt-injection-policy.test.ts +0 -99
- package/tests/apps/app-layer-boundary.test.ts +0 -48
- package/tests/apps/claude-settings-hooks.test.ts +0 -107
- package/tests/apps/cli-disclosure-output.test.ts +0 -212
- package/tests/apps/codex-import-runner.test.ts +0 -99
- package/tests/apps/codex-validation-output.test.ts +0 -100
- package/tests/apps/hermes-import-runner.test.ts +0 -99
- package/tests/apps/mcp-install-command.test.ts +0 -59
- package/tests/apps/package-build-entrypoints.test.ts +0 -30
- package/tests/apps/postinstall-embedding-backend.test.ts +0 -185
- package/tests/apps/search-api-disclosure.test.ts +0 -162
- package/tests/apps/stats-api-lightweight.test.ts +0 -67
- package/tests/apps/ui-disclosure-output.test.ts +0 -140
- package/tests/core/bootstrap-organizer.test.ts +0 -111
- package/tests/core/canonical-key.test.ts +0 -101
- package/tests/core/codex-session-history-importer-validation.test.ts +0 -185
- package/tests/core/consolidation-worker.test.ts +0 -75
- package/tests/core/embedding-maintenance-service.test.ts +0 -282
- package/tests/core/evidence-aligner.test.ts +0 -152
- package/tests/core/external-market-context.test.ts +0 -209
- package/tests/core/fact-deriver.test.ts +0 -79
- package/tests/core/hermes-session-history-importer-validation.test.ts +0 -609
- package/tests/core/ingest-interceptor.test.ts +0 -38
- package/tests/core/markdown-mirror.test.ts +0 -85
- package/tests/core/matcher.test.ts +0 -112
- package/tests/core/md-mirror.test.ts +0 -50
- package/tests/core/memory-engine-services.test.ts +0 -240
- package/tests/core/memory-ingest-service.test.ts +0 -296
- package/tests/core/memory-query-service.test.ts +0 -129
- package/tests/core/memory-runtime-service.test.ts +0 -201
- package/tests/core/memory-service-composition.test.ts +0 -192
- package/tests/core/memory-service-config.test.ts +0 -41
- package/tests/core/memory-service-facade.test.ts +0 -30
- package/tests/core/memory-service-registry.test.ts +0 -206
- package/tests/core/product-validation-matrix.test.ts +0 -61
- package/tests/core/project-registry.test.ts +0 -78
- package/tests/core/replay-evaluator.test.ts +0 -181
- package/tests/core/retrieval-analytics-service.test.ts +0 -210
- package/tests/core/retrieval-benchmark.test.ts +0 -93
- package/tests/core/retrieval-disclosure-service.test.ts +0 -264
- package/tests/core/retrieval-orchestrator.test.ts +0 -403
- package/tests/core/retrieval-quality.test.ts +0 -31
- package/tests/core/retrieval-services.test.ts +0 -185
- package/tests/core/retriever-fallback-chain.test.ts +0 -223
- package/tests/core/retriever-strategy-scope.test.ts +0 -164
- package/tests/core/retriever.memu-adoption.test.ts +0 -122
- package/tests/core/session-history-importer-filter.test.ts +0 -78
- package/tests/core/session-qrels.test.ts +0 -250
- package/tests/core/sqlite-event-store-replication.test.ts +0 -127
- package/tests/core/summary-deriver.test.ts +0 -66
- package/tests/extensions/embedder-warning-suppression.test.ts +0 -84
- package/tests/extensions/endless-memory-extension-boundary.test.ts +0 -17
- package/tests/extensions/endless-memory-services.test.ts +0 -325
- package/tests/extensions/mcp-context-tools.test.ts +0 -905
- package/tests/extensions/mcp-extension-boundary.test.ts +0 -21
- package/tests/extensions/mcp-package-build.test.ts +0 -22
- package/tests/extensions/mcp-project-aware-tools.test.ts +0 -102
- package/tests/extensions/shared-memory-extension-boundary.test.ts +0 -24
- package/tests/extensions/shared-memory-services.test.ts +0 -309
- package/tests/extensions/vector-extension-boundary.test.ts +0 -21
- package/tsconfig.json +0 -24
- package/vitest.config.ts +0 -15
|
@@ -1,78 +0,0 @@
|
|
|
1
|
-
import { mkdtempSync, rmSync, utimesSync, writeFileSync } from 'node:fs';
|
|
2
|
-
import { tmpdir } from 'node:os';
|
|
3
|
-
import { join } from 'node:path';
|
|
4
|
-
import { afterEach, describe, expect, it, vi } from 'vitest';
|
|
5
|
-
|
|
6
|
-
import {
|
|
7
|
-
isClaudeLocalCommandArtifact,
|
|
8
|
-
isWorthStoringPrompt,
|
|
9
|
-
SessionHistoryImporter,
|
|
10
|
-
type ImportResult
|
|
11
|
-
} from '../../src/services/session-history-importer.js';
|
|
12
|
-
|
|
13
|
-
const tempDirs: string[] = [];
|
|
14
|
-
|
|
15
|
-
function tempDir() {
|
|
16
|
-
const dir = mkdtempSync(join(tmpdir(), 'cml-session-importer-'));
|
|
17
|
-
tempDirs.push(dir);
|
|
18
|
-
return dir;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
function makeImportResult(overrides: Partial<ImportResult> = {}): ImportResult {
|
|
22
|
-
return {
|
|
23
|
-
totalSessions: 1,
|
|
24
|
-
totalMessages: 1,
|
|
25
|
-
importedPrompts: 1,
|
|
26
|
-
importedResponses: 0,
|
|
27
|
-
skippedDuplicates: 0,
|
|
28
|
-
errors: [],
|
|
29
|
-
...overrides
|
|
30
|
-
};
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
afterEach(() => {
|
|
34
|
-
for (const dir of tempDirs.splice(0)) {
|
|
35
|
-
rmSync(dir, { recursive: true, force: true });
|
|
36
|
-
}
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
describe('session history importer prompt filtering', () => {
|
|
40
|
-
it('drops Claude local-command artifacts that dilute retrieval quality', () => {
|
|
41
|
-
const artifact = `<command-name>/model</command-name>\n<local-command-stdout>Using model opus</local-command-stdout>`;
|
|
42
|
-
|
|
43
|
-
expect(isClaudeLocalCommandArtifact(artifact)).toBe(true);
|
|
44
|
-
expect(isWorthStoringPrompt(artifact)).toBe(false);
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
it('keeps substantive imported user prompts', () => {
|
|
48
|
-
expect(isWorthStoringPrompt('이 프로젝트에서 memory retrieval 구조를 더 가볍게 개선해줘')).toBe(true);
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
it('imports only the most recently modified matching Claude session when sessionLimit is set', async () => {
|
|
52
|
-
const dir = tempDir();
|
|
53
|
-
const older = join(dir, 'older.jsonl');
|
|
54
|
-
const newest = join(dir, 'newest.jsonl');
|
|
55
|
-
const oldest = join(dir, 'oldest.jsonl');
|
|
56
|
-
for (const filePath of [older, newest, oldest]) {
|
|
57
|
-
writeFileSync(filePath, '{}\n', 'utf8');
|
|
58
|
-
}
|
|
59
|
-
utimesSync(oldest, new Date('2026-05-01T00:00:00.000Z'), new Date('2026-05-01T00:00:00.000Z'));
|
|
60
|
-
utimesSync(older, new Date('2026-05-02T00:00:00.000Z'), new Date('2026-05-02T00:00:00.000Z'));
|
|
61
|
-
utimesSync(newest, new Date('2026-05-03T00:00:00.000Z'), new Date('2026-05-03T00:00:00.000Z'));
|
|
62
|
-
|
|
63
|
-
const importer = new SessionHistoryImporter({} as never) as SessionHistoryImporter & {
|
|
64
|
-
findProjectDirs: ReturnType<typeof vi.fn>;
|
|
65
|
-
findSessionFiles: ReturnType<typeof vi.fn>;
|
|
66
|
-
importSessionFile: ReturnType<typeof vi.fn>;
|
|
67
|
-
};
|
|
68
|
-
importer.findProjectDirs = vi.fn(async () => [dir]);
|
|
69
|
-
importer.findSessionFiles = vi.fn(async () => [older, newest, oldest]);
|
|
70
|
-
importer.importSessionFile = vi.fn(async () => makeImportResult());
|
|
71
|
-
|
|
72
|
-
const result = await importer.importProject('/repo/current', { sessionLimit: 1 });
|
|
73
|
-
|
|
74
|
-
expect(result.totalSessions).toBe(1);
|
|
75
|
-
expect(importer.importSessionFile).toHaveBeenCalledTimes(1);
|
|
76
|
-
expect(importer.importSessionFile).toHaveBeenCalledWith(newest, expect.objectContaining({ sessionLimit: 1 }));
|
|
77
|
-
});
|
|
78
|
-
});
|
|
@@ -1,250 +0,0 @@
|
|
|
1
|
-
import { mkdir, writeFile } from 'node:fs/promises';
|
|
2
|
-
import * as path from 'node:path';
|
|
3
|
-
import { tmpdir } from 'node:os';
|
|
4
|
-
|
|
5
|
-
import { describe, expect, it } from 'vitest';
|
|
6
|
-
|
|
7
|
-
import {
|
|
8
|
-
buildSessionQrelsFixtureFromJsonl,
|
|
9
|
-
collectClaudeSessionJsonlFiles,
|
|
10
|
-
summarizeSessionQrelsFixture
|
|
11
|
-
} from '../../src/core/session-qrels.js';
|
|
12
|
-
|
|
13
|
-
describe('session qrels fixture generation', () => {
|
|
14
|
-
it('turns Claude-style user/assistant session turns into replay qrels', () => {
|
|
15
|
-
const jsonl = [
|
|
16
|
-
JSON.stringify({
|
|
17
|
-
type: 'user',
|
|
18
|
-
sessionId: 's1',
|
|
19
|
-
message: { role: 'user', content: 'retrieval benchmark should report nDCG and precision together' },
|
|
20
|
-
timestamp: '2026-05-05T00:00:00.000Z'
|
|
21
|
-
}),
|
|
22
|
-
JSON.stringify({
|
|
23
|
-
type: 'assistant',
|
|
24
|
-
sessionId: 's1',
|
|
25
|
-
message: { role: 'assistant', content: [{ type: 'text', text: 'Use graded relevance qrels and report nDCG@k beside Precision@k and Recall@k.' }] },
|
|
26
|
-
timestamp: '2026-05-05T00:01:00.000Z'
|
|
27
|
-
}),
|
|
28
|
-
JSON.stringify({
|
|
29
|
-
type: 'user',
|
|
30
|
-
sessionId: 's1',
|
|
31
|
-
message: { role: 'user', content: '<command-name>/model</command-name>\n<local-command-stdout>opus</local-command-stdout>' }
|
|
32
|
-
}),
|
|
33
|
-
JSON.stringify({
|
|
34
|
-
type: 'assistant',
|
|
35
|
-
sessionId: 's1',
|
|
36
|
-
message: { role: 'assistant', content: 'This local command result must not become a benchmark qrel.' }
|
|
37
|
-
}),
|
|
38
|
-
JSON.stringify({
|
|
39
|
-
type: 'user',
|
|
40
|
-
sessionId: 's1',
|
|
41
|
-
message: { role: 'user', content: [{ type: 'text', text: 'fast search CLI should avoid embedding startup for qrels smoke tests' }] }
|
|
42
|
-
}),
|
|
43
|
-
JSON.stringify({
|
|
44
|
-
type: 'assistant',
|
|
45
|
-
sessionId: 's1',
|
|
46
|
-
message: { role: 'assistant', content: 'Fast search benchmark fixtures should stay lightweight and deterministic.' }
|
|
47
|
-
})
|
|
48
|
-
].join('\n');
|
|
49
|
-
|
|
50
|
-
const fixture = buildSessionQrelsFixtureFromJsonl(jsonl, {
|
|
51
|
-
name: 'session-qrels-test',
|
|
52
|
-
ks: [1, 3]
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
expect(fixture.name).toBe('session-qrels-test');
|
|
56
|
-
expect(fixture.ks).toEqual([1, 3]);
|
|
57
|
-
expect(fixture.queries).toHaveLength(2);
|
|
58
|
-
expect(fixture.memories).toHaveLength(2);
|
|
59
|
-
expect(fixture.queries[0]).toMatchObject({
|
|
60
|
-
queryId: 'q-s1-1',
|
|
61
|
-
query: 'retrieval benchmark should report nDCG and precision together',
|
|
62
|
-
expectedIds: ['m-s1-1'],
|
|
63
|
-
expectedRelevance: { 'm-s1-1': 2 }
|
|
64
|
-
});
|
|
65
|
-
expect(fixture.memories[0]).toMatchObject({
|
|
66
|
-
id: 'm-s1-1',
|
|
67
|
-
content: 'Use graded relevance qrels and report nDCG@k beside Precision@k and Recall@k.',
|
|
68
|
-
sourceSessionId: 's1'
|
|
69
|
-
});
|
|
70
|
-
expect(fixture.queries.map((query) => query.query)).not.toContain('opus');
|
|
71
|
-
});
|
|
72
|
-
|
|
73
|
-
it('pairs pending prompts independently per session id', () => {
|
|
74
|
-
const jsonl = [
|
|
75
|
-
JSON.stringify({
|
|
76
|
-
type: 'user',
|
|
77
|
-
sessionId: 's1',
|
|
78
|
-
message: { role: 'user', content: 'session one asks about retrieval replay metrics' }
|
|
79
|
-
}),
|
|
80
|
-
JSON.stringify({
|
|
81
|
-
type: 'user',
|
|
82
|
-
sessionId: 's2',
|
|
83
|
-
message: { role: 'user', content: 'session two asks about qrels generation safety' }
|
|
84
|
-
}),
|
|
85
|
-
JSON.stringify({
|
|
86
|
-
type: 'assistant',
|
|
87
|
-
sessionId: 's1',
|
|
88
|
-
message: { role: 'assistant', content: 'Session one answer explains nDCG replay metrics.' }
|
|
89
|
-
}),
|
|
90
|
-
JSON.stringify({
|
|
91
|
-
type: 'assistant',
|
|
92
|
-
sessionId: 's2',
|
|
93
|
-
message: { role: 'assistant', content: 'Session two answer explains qrels generation safety.' }
|
|
94
|
-
})
|
|
95
|
-
].join('\n');
|
|
96
|
-
|
|
97
|
-
const fixture = buildSessionQrelsFixtureFromJsonl(jsonl);
|
|
98
|
-
|
|
99
|
-
expect(fixture.queries.map((query) => [query.queryId, query.query])).toEqual([
|
|
100
|
-
['q-s1-1', 'session one asks about retrieval replay metrics'],
|
|
101
|
-
['q-s2-1', 'session two asks about qrels generation safety']
|
|
102
|
-
]);
|
|
103
|
-
expect(fixture.memories.map((memory) => [memory.id, memory.content])).toEqual([
|
|
104
|
-
['m-s1-1', 'Session one answer explains nDCG replay metrics.'],
|
|
105
|
-
['m-s2-1', 'Session two answer explains qrels generation safety.']
|
|
106
|
-
]);
|
|
107
|
-
});
|
|
108
|
-
|
|
109
|
-
it('adds known-answer labels and explicit no-match qrels', () => {
|
|
110
|
-
const jsonl = [
|
|
111
|
-
JSON.stringify({
|
|
112
|
-
type: 'user',
|
|
113
|
-
sessionId: 's-known',
|
|
114
|
-
message: { role: 'user', content: 'how should replay qrels record known answer fixtures' }
|
|
115
|
-
}),
|
|
116
|
-
JSON.stringify({
|
|
117
|
-
type: 'assistant',
|
|
118
|
-
sessionId: 's-known',
|
|
119
|
-
message: { role: 'assistant', content: 'Known-answer qrels should keep the expected assistant answer tied to the expected memory id.' }
|
|
120
|
-
})
|
|
121
|
-
].join('\n');
|
|
122
|
-
|
|
123
|
-
const fixture = buildSessionQrelsFixtureFromJsonl(jsonl, {
|
|
124
|
-
noMatchQueries: [
|
|
125
|
-
{
|
|
126
|
-
queryId: 'q-negative-empty-result',
|
|
127
|
-
query: 'unanswered moon cheese deployment question',
|
|
128
|
-
forbiddenIds: ['m-s-known-1'],
|
|
129
|
-
sourceSessionId: 's-known'
|
|
130
|
-
}
|
|
131
|
-
]
|
|
132
|
-
});
|
|
133
|
-
|
|
134
|
-
expect(fixture.queries).toHaveLength(2);
|
|
135
|
-
expect(fixture.queries[0]).toMatchObject({
|
|
136
|
-
queryId: 'q-s-known-1',
|
|
137
|
-
expectation: 'match',
|
|
138
|
-
expectedIds: ['m-s-known-1'],
|
|
139
|
-
knownAnswer: 'Known-answer qrels should keep the expected assistant answer tied to the expected memory id.'
|
|
140
|
-
});
|
|
141
|
-
expect(fixture.queries[1]).toMatchObject({
|
|
142
|
-
queryId: 'q-negative-empty-result',
|
|
143
|
-
query: 'unanswered moon cheese deployment question',
|
|
144
|
-
expectation: 'no_match',
|
|
145
|
-
expectedIds: [],
|
|
146
|
-
expectedRelevance: {},
|
|
147
|
-
forbiddenIds: ['m-s-known-1'],
|
|
148
|
-
sourceSessionId: 's-known'
|
|
149
|
-
});
|
|
150
|
-
|
|
151
|
-
const summary = summarizeSessionQrelsFixture(fixture);
|
|
152
|
-
expect(summary).toMatchObject({
|
|
153
|
-
queryCount: 2,
|
|
154
|
-
positiveQueryCount: 1,
|
|
155
|
-
noMatchQueryCount: 1,
|
|
156
|
-
knownAnswerCount: 1
|
|
157
|
-
});
|
|
158
|
-
});
|
|
159
|
-
|
|
160
|
-
it('can redact raw session text when generating shareable qrels metadata', () => {
|
|
161
|
-
const jsonl = [
|
|
162
|
-
JSON.stringify({
|
|
163
|
-
type: 'user',
|
|
164
|
-
sessionId: 'sensitive',
|
|
165
|
-
message: { role: 'user', content: 'SECRET customer project prompt should not leak' }
|
|
166
|
-
}),
|
|
167
|
-
JSON.stringify({
|
|
168
|
-
type: 'assistant',
|
|
169
|
-
sessionId: 'sensitive',
|
|
170
|
-
message: { role: 'assistant', content: 'SECRET implementation answer should not leak' }
|
|
171
|
-
})
|
|
172
|
-
].join('\n');
|
|
173
|
-
|
|
174
|
-
const fixture = buildSessionQrelsFixtureFromJsonl(jsonl, { redactContent: true });
|
|
175
|
-
const serialized = JSON.stringify(fixture);
|
|
176
|
-
|
|
177
|
-
expect(serialized).not.toContain('SECRET');
|
|
178
|
-
expect(fixture.queries[0]).toMatchObject({
|
|
179
|
-
queryId: 'q-sensitive-1',
|
|
180
|
-
query: '[redacted query q-sensitive-1]',
|
|
181
|
-
expectedIds: ['m-sensitive-1']
|
|
182
|
-
});
|
|
183
|
-
expect(fixture.memories[0]).toMatchObject({
|
|
184
|
-
id: 'm-sensitive-1',
|
|
185
|
-
content: '[redacted memory m-sensitive-1]'
|
|
186
|
-
});
|
|
187
|
-
});
|
|
188
|
-
|
|
189
|
-
it('summarizes generated qrels without leaking raw prompt or memory content', () => {
|
|
190
|
-
const jsonl = [
|
|
191
|
-
JSON.stringify({
|
|
192
|
-
type: 'user',
|
|
193
|
-
sessionId: 'sensitive',
|
|
194
|
-
message: { role: 'user', content: 'SECRET customer retrieval question with enough words' }
|
|
195
|
-
}),
|
|
196
|
-
JSON.stringify({
|
|
197
|
-
type: 'assistant',
|
|
198
|
-
sessionId: 'sensitive',
|
|
199
|
-
message: { role: 'assistant', content: 'SECRET assistant answer that should stay out of reports' }
|
|
200
|
-
})
|
|
201
|
-
].join('\n');
|
|
202
|
-
const fixture = buildSessionQrelsFixtureFromJsonl(jsonl, {
|
|
203
|
-
name: 'private-real-session-qrels',
|
|
204
|
-
ks: [1, 3, 10],
|
|
205
|
-
sourceFileCount: 2,
|
|
206
|
-
rawContentIncluded: true
|
|
207
|
-
});
|
|
208
|
-
|
|
209
|
-
const summary = summarizeSessionQrelsFixture(fixture);
|
|
210
|
-
const serialized = JSON.stringify(summary);
|
|
211
|
-
|
|
212
|
-
expect(summary).toMatchObject({
|
|
213
|
-
name: 'private-real-session-qrels',
|
|
214
|
-
queryCount: 1,
|
|
215
|
-
memoryCount: 1,
|
|
216
|
-
sourceSessionCount: 1,
|
|
217
|
-
sourceFileCount: 2,
|
|
218
|
-
rawContentIncluded: true,
|
|
219
|
-
ks: [1, 3, 10]
|
|
220
|
-
});
|
|
221
|
-
expect(summary.perSession).toEqual([
|
|
222
|
-
{
|
|
223
|
-
sourceSessionId: 'sensitive',
|
|
224
|
-
queryCount: 1,
|
|
225
|
-
memoryCount: 1,
|
|
226
|
-
firstTurnIndex: 1,
|
|
227
|
-
lastTurnIndex: 1
|
|
228
|
-
}
|
|
229
|
-
]);
|
|
230
|
-
expect(serialized).not.toContain('SECRET');
|
|
231
|
-
});
|
|
232
|
-
|
|
233
|
-
it('collects Claude JSONL sessions recursively while excluding subagents by default', async () => {
|
|
234
|
-
const root = path.join(tmpdir(), `session-qrels-${process.pid}-${Date.now()}`);
|
|
235
|
-
const projectDir = path.join(root, 'project-a');
|
|
236
|
-
const subagentDir = path.join(projectDir, 'subagents');
|
|
237
|
-
await mkdir(subagentDir, { recursive: true });
|
|
238
|
-
await writeFile(path.join(projectDir, 'top.jsonl'), '{}\n', 'utf8');
|
|
239
|
-
await writeFile(path.join(subagentDir, 'agent.jsonl'), '{}\n', 'utf8');
|
|
240
|
-
await writeFile(path.join(projectDir, 'notes.txt'), 'ignore', 'utf8');
|
|
241
|
-
|
|
242
|
-
await expect(collectClaudeSessionJsonlFiles(root)).resolves.toEqual([
|
|
243
|
-
path.join(projectDir, 'top.jsonl')
|
|
244
|
-
]);
|
|
245
|
-
await expect(collectClaudeSessionJsonlFiles(root, { includeSubagents: true })).resolves.toEqual([
|
|
246
|
-
path.join(subagentDir, 'agent.jsonl'),
|
|
247
|
-
path.join(projectDir, 'top.jsonl')
|
|
248
|
-
]);
|
|
249
|
-
});
|
|
250
|
-
});
|
|
@@ -1,127 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tests for SQLiteEventStore replication helpers used by Mongo sync.
|
|
3
|
-
*/
|
|
4
|
-
|
|
5
|
-
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
6
|
-
import * as fs from 'fs';
|
|
7
|
-
import * as path from 'path';
|
|
8
|
-
import * as os from 'os';
|
|
9
|
-
|
|
10
|
-
import { SQLiteEventStore } from '../../src/core/sqlite-event-store.js';
|
|
11
|
-
|
|
12
|
-
describe('SQLiteEventStore replication helpers', () => {
|
|
13
|
-
let tempDir: string;
|
|
14
|
-
let storeA: SQLiteEventStore;
|
|
15
|
-
let storeB: SQLiteEventStore;
|
|
16
|
-
|
|
17
|
-
beforeEach(async () => {
|
|
18
|
-
tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'claude-memory-layer-test-'));
|
|
19
|
-
storeA = new SQLiteEventStore(path.join(tempDir, 'a.sqlite'));
|
|
20
|
-
storeB = new SQLiteEventStore(path.join(tempDir, 'b.sqlite'));
|
|
21
|
-
await storeA.initialize();
|
|
22
|
-
await storeB.initialize();
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
afterEach(() => {
|
|
26
|
-
try { storeA.close(); } catch {}
|
|
27
|
-
try { storeB.close(); } catch {}
|
|
28
|
-
fs.rmSync(tempDir, { recursive: true, force: true });
|
|
29
|
-
});
|
|
30
|
-
|
|
31
|
-
it('getEventsSinceRowid returns incremental batches in insertion order', async () => {
|
|
32
|
-
const sessionId = 'session-1';
|
|
33
|
-
|
|
34
|
-
await storeA.append({ eventType: 'user_prompt', sessionId, timestamp: new Date(), content: 'a' });
|
|
35
|
-
await storeA.append({ eventType: 'user_prompt', sessionId, timestamp: new Date(), content: 'b' });
|
|
36
|
-
await storeA.append({ eventType: 'user_prompt', sessionId, timestamp: new Date(), content: 'c' });
|
|
37
|
-
|
|
38
|
-
const batch1 = await storeA.getEventsSinceRowid(0, 10);
|
|
39
|
-
expect(batch1).toHaveLength(3);
|
|
40
|
-
expect(batch1.map(x => x.event.content)).toEqual(['a', 'b', 'c']);
|
|
41
|
-
|
|
42
|
-
// rowid should be strictly increasing
|
|
43
|
-
expect(batch1[0].rowid).toBeLessThan(batch1[1].rowid);
|
|
44
|
-
expect(batch1[1].rowid).toBeLessThan(batch1[2].rowid);
|
|
45
|
-
|
|
46
|
-
const lastRowid = batch1[2].rowid;
|
|
47
|
-
|
|
48
|
-
await storeA.append({ eventType: 'user_prompt', sessionId, timestamp: new Date(), content: 'd' });
|
|
49
|
-
|
|
50
|
-
const batch2 = await storeA.getEventsSinceRowid(lastRowid, 10);
|
|
51
|
-
expect(batch2).toHaveLength(1);
|
|
52
|
-
expect(batch2[0].event.content).toBe('d');
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
it('importEvents preserves stable IDs and is idempotent via dedupeKey', async () => {
|
|
56
|
-
const sessionId = 'session-2';
|
|
57
|
-
const appendRes = await storeA.append({
|
|
58
|
-
eventType: 'user_prompt',
|
|
59
|
-
sessionId,
|
|
60
|
-
timestamp: new Date(),
|
|
61
|
-
content: 'hello world'
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
expect(appendRes.success).toBe(true);
|
|
65
|
-
const sourceEvent = await storeA.getEvent(appendRes.eventId!);
|
|
66
|
-
expect(sourceEvent).not.toBeNull();
|
|
67
|
-
|
|
68
|
-
const imported1 = await storeB.importEvents([sourceEvent!]);
|
|
69
|
-
expect(imported1.inserted).toBe(1);
|
|
70
|
-
expect(imported1.skipped).toBe(0);
|
|
71
|
-
|
|
72
|
-
const importedEvent = await storeB.getEvent(sourceEvent!.id);
|
|
73
|
-
expect(importedEvent?.content).toBe('hello world');
|
|
74
|
-
|
|
75
|
-
// Importing again should be a no-op
|
|
76
|
-
const imported2 = await storeB.importEvents([sourceEvent!]);
|
|
77
|
-
expect(imported2.inserted).toBe(0);
|
|
78
|
-
expect(imported2.skipped).toBe(1);
|
|
79
|
-
|
|
80
|
-
// append() should treat it as duplicate due to event_dedup entry
|
|
81
|
-
const dup = await storeB.append({
|
|
82
|
-
eventType: 'user_prompt',
|
|
83
|
-
sessionId,
|
|
84
|
-
timestamp: new Date(),
|
|
85
|
-
content: 'hello world'
|
|
86
|
-
});
|
|
87
|
-
expect(dup.success).toBe(true);
|
|
88
|
-
expect(dup.isDuplicate).toBe(true);
|
|
89
|
-
expect(dup.eventId).toBe(sourceEvent!.id);
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
it('keeps FTS event_id populated after deleteSessionEvents recreates triggers', async () => {
|
|
93
|
-
await storeA.append({ eventType: 'user_prompt', sessionId: 'delete-me', timestamp: new Date(), content: 'temporary kiwi memory' });
|
|
94
|
-
await storeA.append({ eventType: 'user_prompt', sessionId: 'keep-me', timestamp: new Date(), content: 'persistent pineapple memory' });
|
|
95
|
-
|
|
96
|
-
await expect(storeA.deleteSessionEvents('delete-me')).resolves.toBe(1);
|
|
97
|
-
|
|
98
|
-
const appendAfterTriggerRecreation = await storeA.append({
|
|
99
|
-
eventType: 'user_prompt',
|
|
100
|
-
sessionId: 'keep-me',
|
|
101
|
-
timestamp: new Date(),
|
|
102
|
-
content: 'fresh dragonfruit memory'
|
|
103
|
-
});
|
|
104
|
-
expect(appendAfterTriggerRecreation.success).toBe(true);
|
|
105
|
-
|
|
106
|
-
await expect(storeA.rebuildFtsIndex()).resolves.toBe(2);
|
|
107
|
-
const results = await storeA.keywordSearch('dragonfruit', 5);
|
|
108
|
-
|
|
109
|
-
expect(results.map((result) => result.event.content)).toEqual(['fresh dragonfruit memory']);
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
it('supports event updates and deletes on a fresh FTS table before manual rebuild', async () => {
|
|
113
|
-
const appendResult = await storeA.append({
|
|
114
|
-
eventType: 'user_prompt',
|
|
115
|
-
sessionId: 'mutable-session',
|
|
116
|
-
timestamp: new Date(),
|
|
117
|
-
content: 'mutable mango memory'
|
|
118
|
-
});
|
|
119
|
-
expect(appendResult.success).toBe(true);
|
|
120
|
-
if (!appendResult.success) return;
|
|
121
|
-
|
|
122
|
-
await expect(storeA.incrementAccessCount([appendResult.eventId])).resolves.toBeUndefined();
|
|
123
|
-
await expect(storeA.deleteSessionEvents('mutable-session')).resolves.toBe(1);
|
|
124
|
-
await expect(storeA.keywordSearch('mango', 5)).resolves.toEqual([]);
|
|
125
|
-
});
|
|
126
|
-
});
|
|
127
|
-
|
|
@@ -1,66 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it } from 'vitest';
|
|
2
|
-
|
|
3
|
-
import { SummaryDeriver } from '../../src/core/derive/summary-deriver.js';
|
|
4
|
-
import type { MemoryEvent } from '../../src/core/types.js';
|
|
5
|
-
|
|
6
|
-
function event(overrides: Partial<MemoryEvent>): MemoryEvent {
|
|
7
|
-
return {
|
|
8
|
-
id: '11111111-1111-4111-8111-111111111111',
|
|
9
|
-
eventType: 'user_prompt',
|
|
10
|
-
sessionId: 'session-1',
|
|
11
|
-
timestamp: new Date('2026-04-30T00:00:00.000Z'),
|
|
12
|
-
content: 'default content',
|
|
13
|
-
canonicalKey: 'default-content',
|
|
14
|
-
dedupeKey: 'session-1:default-content',
|
|
15
|
-
metadata: {},
|
|
16
|
-
...overrides
|
|
17
|
-
};
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
describe('SummaryDeriver', () => {
|
|
21
|
-
it('derives the legacy rule-based session summary from events', () => {
|
|
22
|
-
const deriver = new SummaryDeriver();
|
|
23
|
-
|
|
24
|
-
const result = deriver.deriveSessionSummary([
|
|
25
|
-
event({ id: '11111111-1111-4111-8111-111111111111', eventType: 'user_prompt', content: '첫 번째 작업 요청\n자세한 설명', timestamp: new Date('2026-04-30T01:00:00.000Z') }),
|
|
26
|
-
event({ id: '22222222-2222-4222-8222-222222222222', eventType: 'agent_response', content: '응답', timestamp: new Date('2026-04-30T01:01:00.000Z') }),
|
|
27
|
-
event({ id: '33333333-3333-4333-8333-333333333333', eventType: 'tool_observation', content: '{}', timestamp: new Date('2026-04-30T01:02:00.000Z'), metadata: { toolName: 'terminal', exitCode: 1 } }),
|
|
28
|
-
event({ id: '44444444-4444-4444-8444-444444444444', eventType: 'tool_observation', content: '{}', timestamp: new Date('2026-04-30T01:03:00.000Z'), metadata: { toolName: 'terminal', exitCode: 0 } }),
|
|
29
|
-
event({ id: '55555555-5555-4555-8555-555555555555', eventType: 'tool_observation', content: '{}', timestamp: new Date('2026-04-30T01:04:00.000Z'), metadata: { toolName: 'read_file', success: false } }),
|
|
30
|
-
event({ id: '66666666-6666-4666-8666-666666666666', eventType: 'user_prompt', content: '두 번째 요청', timestamp: new Date('2026-04-30T01:05:00.000Z') })
|
|
31
|
-
]);
|
|
32
|
-
|
|
33
|
-
expect(result).toEqual({
|
|
34
|
-
text: '[2026-04-30] 2턴 세션. 주요 작업: 첫 번째 작업 요청 자세한 설명. 사용 툴: terminal, read_file. 오류 2건 발생',
|
|
35
|
-
metadata: { generated: 'rule-based', eventCount: 6 }
|
|
36
|
-
});
|
|
37
|
-
});
|
|
38
|
-
|
|
39
|
-
it('returns null for too-short sessions and sessions that already have summaries', () => {
|
|
40
|
-
const deriver = new SummaryDeriver();
|
|
41
|
-
|
|
42
|
-
expect(deriver.deriveSessionSummary([
|
|
43
|
-
event({ id: '11111111-1111-4111-8111-111111111111' }),
|
|
44
|
-
event({ id: '22222222-2222-4222-8222-222222222222', eventType: 'agent_response' })
|
|
45
|
-
])).toBeNull();
|
|
46
|
-
|
|
47
|
-
expect(deriver.deriveSessionSummary([
|
|
48
|
-
event({ id: '33333333-3333-4333-8333-333333333333' }),
|
|
49
|
-
event({ id: '44444444-4444-4444-8444-444444444444', eventType: 'agent_response' }),
|
|
50
|
-
event({ id: '55555555-5555-4555-8555-555555555555', eventType: 'session_summary', content: 'already summarized' })
|
|
51
|
-
])).toBeNull();
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('limits the first prompt preview to keep summaries compact', () => {
|
|
55
|
-
const deriver = new SummaryDeriver();
|
|
56
|
-
const longPrompt = `${'a'.repeat(140)}\nextra`;
|
|
57
|
-
|
|
58
|
-
const result = deriver.deriveSessionSummary([
|
|
59
|
-
event({ id: '11111111-1111-4111-8111-111111111111', content: longPrompt }),
|
|
60
|
-
event({ id: '22222222-2222-4222-8222-222222222222', eventType: 'agent_response', content: '응답' }),
|
|
61
|
-
event({ id: '33333333-3333-4333-8333-333333333333', eventType: 'tool_observation', content: '{}', metadata: { toolName: 'terminal' } })
|
|
62
|
-
]);
|
|
63
|
-
|
|
64
|
-
expect(result?.text).toBe(`[2026-04-30] 1턴 세션. 주요 작업: ${'a'.repeat(120)}. 사용 툴: terminal`);
|
|
65
|
-
});
|
|
66
|
-
});
|
|
@@ -1,84 +0,0 @@
|
|
|
1
|
-
import { describe, expect, it, vi } from 'vitest';
|
|
2
|
-
|
|
3
|
-
import {
|
|
4
|
-
DEFAULT_EMBEDDING_MODEL,
|
|
5
|
-
Embedder,
|
|
6
|
-
createEmbeddingBackendUnavailableError,
|
|
7
|
-
isKnownBenignTransformersWarning,
|
|
8
|
-
isMissingTransformersDependencyError,
|
|
9
|
-
withSuppressedKnownTransformersWarnings
|
|
10
|
-
} from '../../src/extensions/vector/embedder.js';
|
|
11
|
-
import { ConfigSchema } from '../../src/core/types.js';
|
|
12
|
-
|
|
13
|
-
describe('Embedder warning suppression', () => {
|
|
14
|
-
it('uses a CPU-friendly multilingual Korean-capable default embedding model', () => {
|
|
15
|
-
expect(DEFAULT_EMBEDDING_MODEL).toBe('Xenova/multilingual-e5-small');
|
|
16
|
-
expect(new Embedder().getModelName()).toBe(DEFAULT_EMBEDDING_MODEL);
|
|
17
|
-
|
|
18
|
-
const parsedConfig = ConfigSchema.parse({});
|
|
19
|
-
expect(parsedConfig.embedding.model).toBe(DEFAULT_EMBEDDING_MODEL);
|
|
20
|
-
expect(parsedConfig.embedding.openaiModel).toBe(DEFAULT_EMBEDDING_MODEL);
|
|
21
|
-
});
|
|
22
|
-
|
|
23
|
-
it('turns missing required @huggingface/transformers errors into actionable install guidance', () => {
|
|
24
|
-
const missingBackendError = Object.assign(
|
|
25
|
-
new Error("Cannot find package '@huggingface/transformers' imported from /tmp/dist/cli/index.js"),
|
|
26
|
-
{ code: 'ERR_MODULE_NOT_FOUND' }
|
|
27
|
-
);
|
|
28
|
-
|
|
29
|
-
expect(isMissingTransformersDependencyError(missingBackendError)).toBe(true);
|
|
30
|
-
expect(isMissingTransformersDependencyError(new Error('network failure'))).toBe(false);
|
|
31
|
-
|
|
32
|
-
const friendly = createEmbeddingBackendUnavailableError(missingBackendError);
|
|
33
|
-
expect(friendly.message).toContain('Required embedding backend is not installed');
|
|
34
|
-
expect(friendly.message).toContain('Claude Memory Layer requires @huggingface/transformers');
|
|
35
|
-
expect(friendly.message).toContain('ONNXRUNTIME_NODE_INSTALL_CUDA=skip npm install -g claude-memory-layer@latest');
|
|
36
|
-
expect(friendly.message).toContain('ONNXRUNTIME_NODE_INSTALL_CUDA=skip npm install --no-save --no-package-lock --omit=dev @huggingface/transformers@3.8.1');
|
|
37
|
-
expect(friendly.cause).toBe(missingBackendError);
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
it('recognizes known benign transformer warnings', () => {
|
|
41
|
-
expect(isKnownBenignTransformersWarning('Unknown model class "eurobert", attempting to construct from base class.')).toBe(true);
|
|
42
|
-
expect(isKnownBenignTransformersWarning('dtype not specified for "model". Using the default dtype (fp32).')).toBe(true);
|
|
43
|
-
expect(isKnownBenignTransformersWarning('serious model load failure')).toBe(false);
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
it('suppresses known transformer warnings but keeps unrelated warnings', async () => {
|
|
47
|
-
const warn = vi.spyOn(console, 'warn').mockImplementation(() => undefined);
|
|
48
|
-
|
|
49
|
-
await withSuppressedKnownTransformersWarnings(async () => {
|
|
50
|
-
console.warn('Unknown model class "eurobert", attempting to construct from base class.');
|
|
51
|
-
console.warn('dtype not specified for "model". Using the default dtype (fp32).');
|
|
52
|
-
console.warn('serious model load failure');
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
expect(warn).toHaveBeenCalledTimes(1);
|
|
56
|
-
expect(warn).toHaveBeenCalledWith('serious model load failure');
|
|
57
|
-
warn.mockRestore();
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
it('restores console.warn only after overlapping suppressions complete', async () => {
|
|
61
|
-
const warn = vi.spyOn(console, 'warn').mockImplementation(() => undefined);
|
|
62
|
-
const originalWarn = console.warn;
|
|
63
|
-
let releaseFirst!: () => void;
|
|
64
|
-
let releaseSecond!: () => void;
|
|
65
|
-
|
|
66
|
-
const first = withSuppressedKnownTransformersWarnings(async () => {
|
|
67
|
-
await new Promise<void>((resolve) => { releaseFirst = resolve; });
|
|
68
|
-
});
|
|
69
|
-
const suppressedWarn = console.warn;
|
|
70
|
-
const second = withSuppressedKnownTransformersWarnings(async () => {
|
|
71
|
-
await new Promise<void>((resolve) => { releaseSecond = resolve; });
|
|
72
|
-
});
|
|
73
|
-
|
|
74
|
-
expect(console.warn).toBe(suppressedWarn);
|
|
75
|
-
releaseFirst();
|
|
76
|
-
await first;
|
|
77
|
-
expect(console.warn).toBe(suppressedWarn);
|
|
78
|
-
|
|
79
|
-
releaseSecond();
|
|
80
|
-
await second;
|
|
81
|
-
expect(console.warn).toBe(originalWarn);
|
|
82
|
-
warn.mockRestore();
|
|
83
|
-
});
|
|
84
|
-
});
|
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
import { readFileSync } from 'node:fs';
|
|
2
|
-
import { describe, expect, it } from 'vitest';
|
|
3
|
-
|
|
4
|
-
import { createEndlessMemoryServices as createFromExtension } from '../../src/extensions/endless-memory/index.js';
|
|
5
|
-
import { createEndlessMemoryServices as createFromCompat } from '../../src/core/engine/endless-memory-services.js';
|
|
6
|
-
|
|
7
|
-
describe('endless-memory extension boundary', () => {
|
|
8
|
-
it('keeps endless memory implementation under extensions with an engine compatibility re-export', () => {
|
|
9
|
-
const compatSource = readFileSync('src/core/engine/endless-memory-services.ts', 'utf8');
|
|
10
|
-
const memoryServiceSource = readFileSync('src/services/memory-service.ts', 'utf8');
|
|
11
|
-
|
|
12
|
-
expect(createFromCompat).toBe(createFromExtension);
|
|
13
|
-
expect(compatSource).toContain("../../extensions/endless-memory/index.js");
|
|
14
|
-
expect(memoryServiceSource).not.toContain("../core/engine/endless-memory-services.js");
|
|
15
|
-
expect(memoryServiceSource).toContain("../extensions/endless-memory/index.js");
|
|
16
|
-
});
|
|
17
|
-
});
|