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,373 +0,0 @@
|
|
|
1
|
-
function loadChatHistory() {
|
|
2
|
-
try {
|
|
3
|
-
const raw = localStorage.getItem(CHAT_STORAGE_KEY);
|
|
4
|
-
return raw ? JSON.parse(raw) : [];
|
|
5
|
-
} catch { return []; }
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
function saveChatHistory(conversations) {
|
|
9
|
-
try {
|
|
10
|
-
// Keep last 50 conversations max
|
|
11
|
-
const trimmed = conversations.slice(-50);
|
|
12
|
-
localStorage.setItem(CHAT_STORAGE_KEY, JSON.stringify(trimmed));
|
|
13
|
-
} catch { /* storage full or unavailable */ }
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
function saveCurrentConversation() {
|
|
17
|
-
if (state.chatMessages.length === 0) return;
|
|
18
|
-
const conversations = loadChatHistory();
|
|
19
|
-
const firstUserMsg = state.chatMessages.find(m => m.role === 'user');
|
|
20
|
-
const title = firstUserMsg ? firstUserMsg.content.slice(0, 80) : 'Untitled';
|
|
21
|
-
|
|
22
|
-
if (state.chatConversationId) {
|
|
23
|
-
// Update existing
|
|
24
|
-
const idx = conversations.findIndex(c => c.id === state.chatConversationId);
|
|
25
|
-
if (idx >= 0) {
|
|
26
|
-
conversations[idx].messages = [...state.chatMessages];
|
|
27
|
-
conversations[idx].updatedAt = new Date().toISOString();
|
|
28
|
-
conversations[idx].title = title;
|
|
29
|
-
}
|
|
30
|
-
} else {
|
|
31
|
-
// Create new
|
|
32
|
-
state.chatConversationId = 'chat-' + Date.now();
|
|
33
|
-
conversations.push({
|
|
34
|
-
id: state.chatConversationId,
|
|
35
|
-
title,
|
|
36
|
-
messages: [...state.chatMessages],
|
|
37
|
-
createdAt: new Date().toISOString(),
|
|
38
|
-
updatedAt: new Date().toISOString(),
|
|
39
|
-
project: state.currentProject || 'global'
|
|
40
|
-
});
|
|
41
|
-
}
|
|
42
|
-
saveChatHistory(conversations);
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
function startNewConversation() {
|
|
46
|
-
saveCurrentConversation();
|
|
47
|
-
state.chatMessages = [];
|
|
48
|
-
state.chatConversationId = null;
|
|
49
|
-
|
|
50
|
-
const container = document.getElementById('chat-messages');
|
|
51
|
-
container.innerHTML = `
|
|
52
|
-
<div class="chat-welcome">
|
|
53
|
-
<div class="chat-welcome-icon">🧠</div>
|
|
54
|
-
<div class="chat-welcome-title">Ask about your memories</div>
|
|
55
|
-
<div class="chat-welcome-text">
|
|
56
|
-
I can search through your coding sessions, tool usage, and stored knowledge to answer questions.
|
|
57
|
-
</div>
|
|
58
|
-
</div>
|
|
59
|
-
`;
|
|
60
|
-
switchChatTab('chat');
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
function loadConversation(id) {
|
|
64
|
-
const conversations = loadChatHistory();
|
|
65
|
-
const conv = conversations.find(c => c.id === id);
|
|
66
|
-
if (!conv) return;
|
|
67
|
-
|
|
68
|
-
// Save current first
|
|
69
|
-
if (state.chatMessages.length > 0 && state.chatConversationId !== id) {
|
|
70
|
-
saveCurrentConversation();
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
state.chatConversationId = conv.id;
|
|
74
|
-
state.chatMessages = [...conv.messages];
|
|
75
|
-
|
|
76
|
-
// Render messages
|
|
77
|
-
const container = document.getElementById('chat-messages');
|
|
78
|
-
container.innerHTML = '';
|
|
79
|
-
for (const msg of conv.messages) {
|
|
80
|
-
appendChatMessage(msg.role, msg.content);
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
switchChatTab('chat');
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
function deleteConversation(id, evt) {
|
|
87
|
-
evt.stopPropagation();
|
|
88
|
-
const conversations = loadChatHistory().filter(c => c.id !== id);
|
|
89
|
-
saveChatHistory(conversations);
|
|
90
|
-
if (state.chatConversationId === id) {
|
|
91
|
-
state.chatMessages = [];
|
|
92
|
-
state.chatConversationId = null;
|
|
93
|
-
const container = document.getElementById('chat-messages');
|
|
94
|
-
container.innerHTML = `
|
|
95
|
-
<div class="chat-welcome">
|
|
96
|
-
<div class="chat-welcome-icon">🧠</div>
|
|
97
|
-
<div class="chat-welcome-title">Ask about your memories</div>
|
|
98
|
-
<div class="chat-welcome-text">
|
|
99
|
-
I can search through your coding sessions, tool usage, and stored knowledge to answer questions.
|
|
100
|
-
</div>
|
|
101
|
-
</div>
|
|
102
|
-
`;
|
|
103
|
-
}
|
|
104
|
-
renderHistoryList();
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
function renderHistoryList() {
|
|
108
|
-
const container = document.getElementById('chat-history-view');
|
|
109
|
-
const conversations = loadChatHistory().reverse(); // newest first
|
|
110
|
-
|
|
111
|
-
if (conversations.length === 0) {
|
|
112
|
-
container.innerHTML = '<div class="chat-history-empty">No conversation history yet.</div>';
|
|
113
|
-
return;
|
|
114
|
-
}
|
|
115
|
-
|
|
116
|
-
container.innerHTML = conversations.map(conv => {
|
|
117
|
-
const date = new Date(conv.updatedAt || conv.createdAt);
|
|
118
|
-
const dateStr = date.toLocaleDateString() + ' ' + date.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' });
|
|
119
|
-
const msgCount = conv.messages.length;
|
|
120
|
-
const isActive = conv.id === state.chatConversationId;
|
|
121
|
-
return `
|
|
122
|
-
<div class="chat-history-item${isActive ? ' active' : ''}" onclick="loadConversation('${conv.id}')"
|
|
123
|
-
style="${isActive ? 'border-color:var(--accent-primary);background:rgba(123,97,255,0.08);' : ''}">
|
|
124
|
-
<div class="chat-history-item-title">${escapeHtml(conv.title)}</div>
|
|
125
|
-
<div class="chat-history-item-meta">
|
|
126
|
-
<span>${dateStr} · ${msgCount} messages</span>
|
|
127
|
-
<button class="chat-history-item-delete" onclick="deleteConversation('${conv.id}', event)" title="Delete">
|
|
128
|
-
<i class="ri-delete-bin-line"></i>
|
|
129
|
-
</button>
|
|
130
|
-
</div>
|
|
131
|
-
</div>
|
|
132
|
-
`;
|
|
133
|
-
}).join('');
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
function switchChatTab(tab) {
|
|
137
|
-
const msgContainer = document.getElementById('chat-messages');
|
|
138
|
-
const historyContainer = document.getElementById('chat-history-view');
|
|
139
|
-
const inputArea = document.querySelector('.chat-input-area');
|
|
140
|
-
|
|
141
|
-
document.querySelectorAll('.chat-header-tab').forEach(t => {
|
|
142
|
-
t.classList.toggle('active', t.dataset.chatTab === tab);
|
|
143
|
-
});
|
|
144
|
-
|
|
145
|
-
if (tab === 'chat') {
|
|
146
|
-
msgContainer.classList.remove('hidden');
|
|
147
|
-
historyContainer.classList.remove('active');
|
|
148
|
-
if (inputArea) inputArea.style.display = '';
|
|
149
|
-
} else {
|
|
150
|
-
msgContainer.classList.add('hidden');
|
|
151
|
-
historyContainer.classList.add('active');
|
|
152
|
-
if (inputArea) inputArea.style.display = 'none';
|
|
153
|
-
renderHistoryList();
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
state.chatCurrentTab = tab;
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
function toggleChatPanel() {
|
|
160
|
-
if (state.isChatOpen) {
|
|
161
|
-
closeChatPanel();
|
|
162
|
-
} else {
|
|
163
|
-
openChatPanel();
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
function openChatPanel() {
|
|
168
|
-
const panel = document.getElementById('chat-panel');
|
|
169
|
-
if (panel) {
|
|
170
|
-
panel.classList.add('open');
|
|
171
|
-
state.isChatOpen = true;
|
|
172
|
-
updateChatProjectScope();
|
|
173
|
-
setTimeout(() => {
|
|
174
|
-
document.getElementById('chat-input')?.focus();
|
|
175
|
-
}, 300);
|
|
176
|
-
}
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
function closeChatPanel() {
|
|
180
|
-
const panel = document.getElementById('chat-panel');
|
|
181
|
-
if (panel) {
|
|
182
|
-
panel.classList.remove('open');
|
|
183
|
-
state.isChatOpen = false;
|
|
184
|
-
}
|
|
185
|
-
if (state.chatAbortController) {
|
|
186
|
-
state.chatAbortController.abort();
|
|
187
|
-
state.chatAbortController = null;
|
|
188
|
-
state.isChatStreaming = false;
|
|
189
|
-
}
|
|
190
|
-
// Auto-save on close
|
|
191
|
-
saveCurrentConversation();
|
|
192
|
-
}
|
|
193
|
-
|
|
194
|
-
function updateChatProjectScope() {
|
|
195
|
-
const el = document.getElementById('chat-project-scope');
|
|
196
|
-
if (!el) return;
|
|
197
|
-
if (state.currentProject) {
|
|
198
|
-
const proj = state.projects.find(p => p.hash === state.currentProject);
|
|
199
|
-
el.textContent = `Scope: ${proj?.projectName || state.currentProject}`;
|
|
200
|
-
} else {
|
|
201
|
-
el.textContent = 'Scope: All (Global)';
|
|
202
|
-
}
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
async function sendChatMessage() {
|
|
206
|
-
const input = document.getElementById('chat-input');
|
|
207
|
-
const message = input.value.trim();
|
|
208
|
-
if (!message) return;
|
|
209
|
-
|
|
210
|
-
input.value = '';
|
|
211
|
-
input.style.height = 'auto';
|
|
212
|
-
document.getElementById('chat-send-btn').disabled = true;
|
|
213
|
-
|
|
214
|
-
// Add user message
|
|
215
|
-
state.chatMessages.push({ role: 'user', content: message });
|
|
216
|
-
appendChatMessage('user', message);
|
|
217
|
-
|
|
218
|
-
// Remove welcome
|
|
219
|
-
const welcome = document.querySelector('.chat-welcome');
|
|
220
|
-
if (welcome) welcome.remove();
|
|
221
|
-
|
|
222
|
-
// Show loading
|
|
223
|
-
const loadingEl = appendChatLoading();
|
|
224
|
-
|
|
225
|
-
state.isChatStreaming = true;
|
|
226
|
-
state.chatAbortController = new AbortController();
|
|
227
|
-
|
|
228
|
-
try {
|
|
229
|
-
const response = await fetch(apiUrl(`${API_BASE}/chat`), {
|
|
230
|
-
method: 'POST',
|
|
231
|
-
headers: { 'Content-Type': 'application/json' },
|
|
232
|
-
body: JSON.stringify({
|
|
233
|
-
message,
|
|
234
|
-
history: state.chatMessages.slice(-10)
|
|
235
|
-
}),
|
|
236
|
-
signal: state.chatAbortController.signal
|
|
237
|
-
});
|
|
238
|
-
|
|
239
|
-
if (!response.ok) {
|
|
240
|
-
const err = await response.json().catch(() => ({ error: `HTTP ${response.status}` }));
|
|
241
|
-
throw new Error(err.error || `Request failed: ${response.status}`);
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
loadingEl.remove();
|
|
245
|
-
const msgEl = appendChatMessage('assistant', '', true);
|
|
246
|
-
let fullContent = '';
|
|
247
|
-
|
|
248
|
-
const reader = response.body.getReader();
|
|
249
|
-
const decoder = new TextDecoder();
|
|
250
|
-
let sseBuffer = '';
|
|
251
|
-
|
|
252
|
-
while (true) {
|
|
253
|
-
const { done, value } = await reader.read();
|
|
254
|
-
if (done) break;
|
|
255
|
-
|
|
256
|
-
sseBuffer += decoder.decode(value, { stream: true });
|
|
257
|
-
const lines = sseBuffer.split('\n');
|
|
258
|
-
sseBuffer = lines.pop() || '';
|
|
259
|
-
|
|
260
|
-
for (const line of lines) {
|
|
261
|
-
if (line.startsWith('data: ')) {
|
|
262
|
-
const dataStr = line.slice(6);
|
|
263
|
-
try {
|
|
264
|
-
const data = JSON.parse(dataStr);
|
|
265
|
-
if (data.content) {
|
|
266
|
-
fullContent += data.content;
|
|
267
|
-
updateChatMessageContent(msgEl, fullContent);
|
|
268
|
-
scrollChatToBottom();
|
|
269
|
-
}
|
|
270
|
-
if (data.error) {
|
|
271
|
-
fullContent += `\n\n**Error:** ${data.error}`;
|
|
272
|
-
updateChatMessageContent(msgEl, fullContent);
|
|
273
|
-
}
|
|
274
|
-
} catch { /* skip */ }
|
|
275
|
-
}
|
|
276
|
-
}
|
|
277
|
-
}
|
|
278
|
-
|
|
279
|
-
msgEl.classList.remove('streaming');
|
|
280
|
-
if (fullContent) {
|
|
281
|
-
state.chatMessages.push({ role: 'assistant', content: fullContent });
|
|
282
|
-
}
|
|
283
|
-
|
|
284
|
-
// Auto-save after each response
|
|
285
|
-
saveCurrentConversation();
|
|
286
|
-
|
|
287
|
-
} catch (err) {
|
|
288
|
-
if (loadingEl.parentNode) loadingEl.remove();
|
|
289
|
-
if (err.name !== 'AbortError') {
|
|
290
|
-
appendChatMessage('assistant',
|
|
291
|
-
`**Error:** ${err.message}\n\nMake sure the Claude CLI is installed and authenticated.`
|
|
292
|
-
);
|
|
293
|
-
}
|
|
294
|
-
} finally {
|
|
295
|
-
state.isChatStreaming = false;
|
|
296
|
-
state.chatAbortController = null;
|
|
297
|
-
const sendBtn = document.getElementById('chat-send-btn');
|
|
298
|
-
const chatInput = document.getElementById('chat-input');
|
|
299
|
-
if (sendBtn && chatInput) {
|
|
300
|
-
sendBtn.disabled = !chatInput.value.trim();
|
|
301
|
-
}
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
function appendChatMessage(role, content, streaming = false) {
|
|
306
|
-
const container = document.getElementById('chat-messages');
|
|
307
|
-
const el = document.createElement('div');
|
|
308
|
-
el.className = `chat-msg ${role}${streaming ? ' streaming' : ''}`;
|
|
309
|
-
|
|
310
|
-
if (role === 'assistant') {
|
|
311
|
-
el.innerHTML = renderMarkdown(content);
|
|
312
|
-
} else {
|
|
313
|
-
el.textContent = content;
|
|
314
|
-
}
|
|
315
|
-
|
|
316
|
-
container.appendChild(el);
|
|
317
|
-
scrollChatToBottom();
|
|
318
|
-
return el;
|
|
319
|
-
}
|
|
320
|
-
|
|
321
|
-
function appendChatLoading() {
|
|
322
|
-
const container = document.getElementById('chat-messages');
|
|
323
|
-
const el = document.createElement('div');
|
|
324
|
-
el.className = 'chat-loading';
|
|
325
|
-
el.innerHTML = `
|
|
326
|
-
<div class="chat-loading-dot"></div>
|
|
327
|
-
<div class="chat-loading-dot"></div>
|
|
328
|
-
<div class="chat-loading-dot"></div>
|
|
329
|
-
`;
|
|
330
|
-
container.appendChild(el);
|
|
331
|
-
scrollChatToBottom();
|
|
332
|
-
return el;
|
|
333
|
-
}
|
|
334
|
-
|
|
335
|
-
function updateChatMessageContent(el, content) {
|
|
336
|
-
el.innerHTML = renderMarkdown(content);
|
|
337
|
-
}
|
|
338
|
-
|
|
339
|
-
function scrollChatToBottom() {
|
|
340
|
-
const container = document.getElementById('chat-messages');
|
|
341
|
-
if (container) container.scrollTop = container.scrollHeight;
|
|
342
|
-
}
|
|
343
|
-
|
|
344
|
-
function renderMarkdown(text) {
|
|
345
|
-
if (!text) return '';
|
|
346
|
-
|
|
347
|
-
let html = escapeHtml(text);
|
|
348
|
-
|
|
349
|
-
// Code blocks
|
|
350
|
-
html = html.replace(/```(\w*)\n([\s\S]*?)```/g, '<pre><code>$2</code></pre>');
|
|
351
|
-
|
|
352
|
-
// Inline code
|
|
353
|
-
html = html.replace(/`([^`]+)`/g, '<code>$1</code>');
|
|
354
|
-
|
|
355
|
-
// Bold
|
|
356
|
-
html = html.replace(/\*\*(.+?)\*\*/g, '<strong>$1</strong>');
|
|
357
|
-
|
|
358
|
-
// Italic
|
|
359
|
-
html = html.replace(/\*(.+?)\*/g, '<em>$1</em>');
|
|
360
|
-
|
|
361
|
-
// Headers
|
|
362
|
-
html = html.replace(/^### (.+)$/gm, '<div style="font-weight:600;color:var(--text-primary);margin:12px 0 4px;">$1</div>');
|
|
363
|
-
html = html.replace(/^## (.+)$/gm, '<div style="font-size:15px;font-weight:600;color:var(--text-primary);margin:12px 0 4px;">$1</div>');
|
|
364
|
-
|
|
365
|
-
// Lists
|
|
366
|
-
html = html.replace(/^- (.+)$/gm, '<div style="padding-left:16px;">• $1</div>');
|
|
367
|
-
|
|
368
|
-
// Line breaks
|
|
369
|
-
html = html.replace(/\n/g, '<br>');
|
|
370
|
-
|
|
371
|
-
return html;
|
|
372
|
-
}
|
|
373
|
-
|
|
@@ -1,232 +0,0 @@
|
|
|
1
|
-
|
|
2
|
-
// --- Progressive Disclosure Dashboard ---
|
|
3
|
-
|
|
4
|
-
function setupDisclosureSearchListeners() {
|
|
5
|
-
const input = document.getElementById('disclosure-search-input');
|
|
6
|
-
const button = document.getElementById('disclosure-search-btn');
|
|
7
|
-
if (button) button.addEventListener('click', handleDisclosureSearch);
|
|
8
|
-
if (input) {
|
|
9
|
-
input.addEventListener('keydown', (e) => {
|
|
10
|
-
if (e.key === 'Enter') {
|
|
11
|
-
e.preventDefault();
|
|
12
|
-
handleDisclosureSearch();
|
|
13
|
-
}
|
|
14
|
-
});
|
|
15
|
-
}
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
function renderDisclosureStatus(message, isError = false) {
|
|
19
|
-
const status = document.getElementById('disclosure-status');
|
|
20
|
-
if (!status) return;
|
|
21
|
-
status.textContent = message || '';
|
|
22
|
-
status.style.color = isError ? 'var(--error)' : 'var(--text-muted)';
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
async function handleDisclosureSearch() {
|
|
26
|
-
const input = document.getElementById('disclosure-search-input');
|
|
27
|
-
const query = (input?.value || '').trim();
|
|
28
|
-
if (!query || state.isDisclosureLoading) return;
|
|
29
|
-
|
|
30
|
-
const button = document.getElementById('disclosure-search-btn');
|
|
31
|
-
const includeShared = document.getElementById('disclosure-include-shared')?.checked || false;
|
|
32
|
-
const strategy = document.getElementById('disclosure-strategy')?.value || 'auto';
|
|
33
|
-
const topK = parseInt(document.getElementById('disclosure-topk')?.value || '8', 10);
|
|
34
|
-
|
|
35
|
-
state.isDisclosureLoading = true;
|
|
36
|
-
state.disclosureResults = [];
|
|
37
|
-
state.disclosureMeta = null;
|
|
38
|
-
state.disclosureSelectedId = null;
|
|
39
|
-
state.disclosureExpansion = null;
|
|
40
|
-
state.disclosureSource = null;
|
|
41
|
-
if (button) button.disabled = true;
|
|
42
|
-
renderDisclosureStatus('Searching compact retrieval envelopes...');
|
|
43
|
-
renderDisclosureResults();
|
|
44
|
-
renderDisclosureDrilldown();
|
|
45
|
-
|
|
46
|
-
try {
|
|
47
|
-
const res = await fetch(apiUrl(`${API_BASE}/search/disclosure`), {
|
|
48
|
-
method: 'POST',
|
|
49
|
-
headers: { 'Content-Type': 'application/json' },
|
|
50
|
-
body: JSON.stringify({
|
|
51
|
-
query,
|
|
52
|
-
options: {
|
|
53
|
-
topK: Number.isFinite(topK) ? topK : 8,
|
|
54
|
-
includeShared,
|
|
55
|
-
strategy
|
|
56
|
-
}
|
|
57
|
-
})
|
|
58
|
-
});
|
|
59
|
-
const data = await res.json();
|
|
60
|
-
if (!res.ok) throw new Error(data.error || 'Disclosure search failed');
|
|
61
|
-
|
|
62
|
-
state.disclosureResults = data.results || [];
|
|
63
|
-
state.disclosureMeta = data.meta || null;
|
|
64
|
-
renderDisclosureStatus(`Search layer returned ${state.disclosureResults.length} result(s). Click a result to expand/source.`);
|
|
65
|
-
renderDisclosureResults();
|
|
66
|
-
} catch (error) {
|
|
67
|
-
state.disclosureResults = [];
|
|
68
|
-
renderDisclosureStatus(error.message || 'Disclosure search failed', true);
|
|
69
|
-
renderDisclosureResults();
|
|
70
|
-
} finally {
|
|
71
|
-
state.isDisclosureLoading = false;
|
|
72
|
-
if (button) button.disabled = false;
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
function renderDisclosureResults() {
|
|
77
|
-
const container = document.getElementById('disclosure-results');
|
|
78
|
-
if (!container) return;
|
|
79
|
-
|
|
80
|
-
if (state.isDisclosureLoading) {
|
|
81
|
-
container.innerHTML = '<div class="disclosure-empty">Searching...</div>';
|
|
82
|
-
return;
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
if (!state.disclosureResults.length) {
|
|
86
|
-
container.innerHTML = '<div class="disclosure-empty">Search memory to inspect compact envelopes, expansion context, and raw sources.</div>';
|
|
87
|
-
return;
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
const meta = state.disclosureMeta || {};
|
|
91
|
-
const metaHtml = `
|
|
92
|
-
<div class="disclosure-meta">
|
|
93
|
-
total=${escapeHtml(String(meta.total ?? state.disclosureResults.length))}
|
|
94
|
-
· vector=${meta.usedVector ? 'yes' : 'no'}
|
|
95
|
-
· keyword=${meta.usedKeyword ? 'yes' : 'no'}
|
|
96
|
-
· fallback=${meta.fallbackApplied ? 'yes' : 'no'}
|
|
97
|
-
</div>`;
|
|
98
|
-
|
|
99
|
-
const resultHtml = state.disclosureResults.map((r, idx) => {
|
|
100
|
-
const active = r.id === state.disclosureSelectedId ? ' active' : '';
|
|
101
|
-
const score = Number(r.score || 0).toFixed(3);
|
|
102
|
-
const reasons = (r.reasons || []).map(reason => `<span class="disclosure-chip">${escapeHtml(reason)}</span>`).join('');
|
|
103
|
-
const provenance = renderDisclosureProvenance(r.metadata, ['sourceProjectHash', 'sourceEntryId', 'topics']);
|
|
104
|
-
return `
|
|
105
|
-
<button class="disclosure-result${active}" data-result-id="${escapeHtml(r.id)}">
|
|
106
|
-
<div class="disclosure-result-head">
|
|
107
|
-
<span class="event-type-badge">#${idx + 1} ${escapeHtml(r.resultType || 'result')}</span>
|
|
108
|
-
<span class="disclosure-score">score ${score}</span>
|
|
109
|
-
</div>
|
|
110
|
-
<div class="disclosure-title">${escapeHtml(r.title || r.sourceRef || r.id)}</div>
|
|
111
|
-
<div class="disclosure-snippet">${escapeHtml(r.snippet || '(no snippet)')}</div>
|
|
112
|
-
${provenance}
|
|
113
|
-
<div class="disclosure-reasons">${reasons || '<span class="disclosure-chip">no_reason</span>'}</div>
|
|
114
|
-
</button>`;
|
|
115
|
-
}).join('');
|
|
116
|
-
|
|
117
|
-
container.innerHTML = metaHtml + resultHtml;
|
|
118
|
-
container.querySelectorAll('.disclosure-result').forEach(btn => {
|
|
119
|
-
btn.addEventListener('click', () => loadDisclosureDrilldown(btn.dataset.resultId));
|
|
120
|
-
});
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
async function loadDisclosureDrilldown(resultId) {
|
|
124
|
-
if (!resultId) return;
|
|
125
|
-
state.disclosureSelectedId = resultId;
|
|
126
|
-
state.disclosureExpansion = null;
|
|
127
|
-
state.disclosureSource = null;
|
|
128
|
-
renderDisclosureResults();
|
|
129
|
-
renderDisclosureDrilldown(true);
|
|
130
|
-
|
|
131
|
-
try {
|
|
132
|
-
const encodedId = encodeURIComponent(resultId);
|
|
133
|
-
const [expandRes, sourceRes] = await Promise.all([
|
|
134
|
-
fetch(apiUrl(`${API_BASE}/search/disclosure/${encodedId}/expand`, { windowSize: 3 })),
|
|
135
|
-
fetch(apiUrl(`${API_BASE}/search/disclosure/${encodedId}/source`))
|
|
136
|
-
]);
|
|
137
|
-
const expansion = await expandRes.json();
|
|
138
|
-
const source = await sourceRes.json();
|
|
139
|
-
if (!expandRes.ok) throw new Error(expansion.error || 'Expand failed');
|
|
140
|
-
state.disclosureExpansion = expansion;
|
|
141
|
-
state.disclosureSource = sourceRes.ok ? source : null;
|
|
142
|
-
renderDisclosureDrilldown(false, sourceRes.ok ? null : (source.error || 'Source not found'));
|
|
143
|
-
} catch (error) {
|
|
144
|
-
renderDisclosureDrilldown(false, error.message || 'Failed to load drilldown', true);
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
function renderDisclosureDrilldown(isLoading = false, message = null, isError = false) {
|
|
149
|
-
const container = document.getElementById('disclosure-drilldown');
|
|
150
|
-
if (!container) return;
|
|
151
|
-
if (isLoading) {
|
|
152
|
-
container.innerHTML = '<div class="disclosure-empty">Loading expand/source layers...</div>';
|
|
153
|
-
return;
|
|
154
|
-
}
|
|
155
|
-
if (message && isError) {
|
|
156
|
-
container.innerHTML = `<div class="disclosure-empty" style="color:var(--error);">${escapeHtml(message)}</div>`;
|
|
157
|
-
return;
|
|
158
|
-
}
|
|
159
|
-
if (!state.disclosureSelectedId) {
|
|
160
|
-
container.innerHTML = '<div class="disclosure-empty">No result selected.</div>';
|
|
161
|
-
return;
|
|
162
|
-
}
|
|
163
|
-
|
|
164
|
-
const expansion = state.disclosureExpansion;
|
|
165
|
-
const source = state.disclosureSource;
|
|
166
|
-
if (!expansion) {
|
|
167
|
-
container.innerHTML = '<div class="disclosure-empty">Expansion not loaded yet.</div>';
|
|
168
|
-
return;
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
const surrounding = (expansion.surroundingFacts || []).map(f => `
|
|
172
|
-
<div class="disclosure-context-item">
|
|
173
|
-
<span class="event-type-badge">${escapeHtml(f.resultType || 'fact')}</span>
|
|
174
|
-
<span>${escapeHtml(f.snippet || f.title || f.id || '')}</span>
|
|
175
|
-
</div>`).join('') || '<div class="disclosure-empty compact">No surrounding facts.</div>';
|
|
176
|
-
|
|
177
|
-
const related = (expansion.relatedSources || []).map(s => `
|
|
178
|
-
<div class="disclosure-source-ref">
|
|
179
|
-
<span class="disclosure-chip">${escapeHtml(s.sourceRef)} · ${escapeHtml(s.sourceType || 'source')}</span>
|
|
180
|
-
${renderDisclosureProvenance(s.metadata, ['sourceProjectHash', 'sourceEntryId', 'topics'])}
|
|
181
|
-
</div>`).join('') || '<span class="disclosure-chip">no source refs</span>';
|
|
182
|
-
|
|
183
|
-
const rawEvent = source?.primaryEvent || source?.rawEvents?.[0] || source?.rawEvent || source?.event || null;
|
|
184
|
-
const isSharedSource = source?.sourceType === 'shared_troubleshooting';
|
|
185
|
-
const sourcePreview = rawEvent
|
|
186
|
-
? `${rawEvent.eventType || rawEvent.type || 'event'} · ${rawEvent.timestamp || ''}\n${rawEvent.content || rawEvent.preview || ''}`
|
|
187
|
-
: isSharedSource
|
|
188
|
-
? 'No local raw events for this shared source.'
|
|
189
|
-
: (source ? JSON.stringify(source, null, 2) : (message || 'Source layer returned no raw event.'));
|
|
190
|
-
const sourceProvenance = source?.metadata
|
|
191
|
-
? renderDisclosureProvenance(source.metadata, ['sourceProjectHash', 'sourceEntryId', 'topics', 'rootCause', 'solution', 'symptoms', 'confidence', 'usageCount'])
|
|
192
|
-
: '';
|
|
193
|
-
const sourceProvenanceBlock = sourceProvenance
|
|
194
|
-
? `<div class="modal-section-title">${isSharedSource ? 'Shared source provenance' : 'Source metadata'}</div>${sourceProvenance}`
|
|
195
|
-
: '';
|
|
196
|
-
|
|
197
|
-
container.innerHTML = `
|
|
198
|
-
<div class="disclosure-layer">
|
|
199
|
-
<div class="modal-section-title">Expand layer</div>
|
|
200
|
-
<div class="modal-content-block">${escapeHtml(expansion.expandedContext || expansion.target?.snippet || '')}</div>
|
|
201
|
-
<div class="modal-section-title">Surrounding context</div>
|
|
202
|
-
<div class="disclosure-context-list">${surrounding}</div>
|
|
203
|
-
<div class="modal-section-title">Related sources</div>
|
|
204
|
-
<div class="disclosure-reasons">${related}</div>
|
|
205
|
-
</div>
|
|
206
|
-
<div class="disclosure-layer">
|
|
207
|
-
<div class="modal-section-title">Source layer</div>
|
|
208
|
-
<div class="modal-content-block">${escapeHtml(sourcePreview)}</div>
|
|
209
|
-
${sourceProvenanceBlock}
|
|
210
|
-
</div>`;
|
|
211
|
-
}
|
|
212
|
-
|
|
213
|
-
function renderDisclosureProvenance(metadata, allowedKeys = null) {
|
|
214
|
-
if (!metadata) return '';
|
|
215
|
-
const entries = Object.entries(metadata)
|
|
216
|
-
.filter(([key, value]) => value !== undefined && value !== null && (!allowedKeys || allowedKeys.includes(key)));
|
|
217
|
-
if (!entries.length) return '';
|
|
218
|
-
return `
|
|
219
|
-
<div class="disclosure-provenance">
|
|
220
|
-
${entries.map(([key, value]) => `
|
|
221
|
-
<div class="disclosure-provenance-row">
|
|
222
|
-
<span class="disclosure-provenance-key">${escapeHtml(key)}</span>
|
|
223
|
-
<span class="disclosure-provenance-value">${escapeHtml(formatDisclosureMetadataValue(value))}</span>
|
|
224
|
-
</div>`).join('')}
|
|
225
|
-
</div>`;
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
function formatDisclosureMetadataValue(value) {
|
|
229
|
-
if (Array.isArray(value)) return value.join(', ');
|
|
230
|
-
if (value && typeof value === 'object') return JSON.stringify(value);
|
|
231
|
-
return String(value);
|
|
232
|
-
}
|