stellavault 0.2.0 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +1 -1
- package/packages/core/dist/api/dashboard.d.ts +3 -0
- package/packages/core/{src/api/dashboard.ts → dist/api/dashboard.js} +8 -11
- package/packages/core/dist/api/graph-data.d.ts +11 -0
- package/packages/core/dist/api/graph-data.js +255 -0
- package/packages/core/dist/api/pwa.d.ts +3 -0
- package/packages/core/{src/api/pwa.ts → dist/api/pwa.js} +27 -32
- package/packages/core/dist/api/server.d.ts +16 -0
- package/packages/core/dist/api/server.js +718 -0
- package/packages/core/dist/capture/voice.d.ts +24 -0
- package/packages/core/dist/capture/voice.js +135 -0
- package/packages/core/{src/cloud/index.ts → dist/cloud/index.d.ts} +1 -0
- package/packages/core/dist/cloud/index.js +2 -0
- package/packages/core/dist/cloud/sync.d.ts +29 -0
- package/packages/core/dist/cloud/sync.js +137 -0
- package/packages/core/dist/config.d.ts +27 -0
- package/packages/core/dist/config.js +55 -0
- package/packages/core/dist/federation/credits.d.ts +26 -0
- package/packages/core/dist/federation/credits.js +56 -0
- package/packages/core/dist/federation/identity.d.ts +14 -0
- package/packages/core/dist/federation/identity.js +74 -0
- package/packages/core/{src/federation/index.ts → dist/federation/index.d.ts} +1 -2
- package/packages/core/dist/federation/index.js +5 -0
- package/packages/core/dist/federation/node.d.ts +31 -0
- package/packages/core/dist/federation/node.js +216 -0
- package/packages/core/dist/federation/privacy.d.ts +8 -0
- package/packages/core/dist/federation/privacy.js +40 -0
- package/packages/core/dist/federation/reputation.d.ts +37 -0
- package/packages/core/dist/federation/reputation.js +139 -0
- package/packages/core/dist/federation/search.d.ts +19 -0
- package/packages/core/dist/federation/search.js +101 -0
- package/packages/core/dist/federation/sharing.d.ts +72 -0
- package/packages/core/dist/federation/sharing.js +246 -0
- package/packages/core/dist/federation/trust.d.ts +15 -0
- package/packages/core/dist/federation/trust.js +60 -0
- package/packages/core/dist/federation/types.d.ts +40 -0
- package/packages/core/dist/federation/types.js +3 -0
- package/packages/core/dist/i18n/index.d.ts +6 -0
- package/packages/core/dist/i18n/index.js +81 -0
- package/packages/core/{src/index.ts → dist/index.d.ts} +46 -65
- package/packages/core/dist/index.js +69 -0
- package/packages/core/dist/indexer/chunker.d.ts +14 -0
- package/packages/core/dist/indexer/chunker.js +148 -0
- package/packages/core/dist/indexer/embedder.d.ts +8 -0
- package/packages/core/dist/indexer/embedder.js +3 -0
- package/packages/core/dist/indexer/index.d.ts +28 -0
- package/packages/core/dist/indexer/index.js +74 -0
- package/packages/core/dist/indexer/local-embedder.d.ts +3 -0
- package/packages/core/dist/indexer/local-embedder.js +29 -0
- package/packages/core/dist/indexer/scanner.d.ts +11 -0
- package/packages/core/dist/indexer/scanner.js +137 -0
- package/packages/core/dist/indexer/watcher.d.ts +19 -0
- package/packages/core/dist/indexer/watcher.js +49 -0
- package/packages/core/dist/intelligence/code-linker.d.ts +20 -0
- package/packages/core/dist/intelligence/code-linker.js +88 -0
- package/packages/core/dist/intelligence/contradiction-detector.d.ts +20 -0
- package/packages/core/dist/intelligence/contradiction-detector.js +115 -0
- package/packages/core/dist/intelligence/decay-engine.d.ts +27 -0
- package/packages/core/dist/intelligence/decay-engine.js +190 -0
- package/packages/core/dist/intelligence/duplicate-detector.d.ts +20 -0
- package/packages/core/dist/intelligence/duplicate-detector.js +55 -0
- package/packages/core/dist/intelligence/fsrs.d.ts +43 -0
- package/packages/core/dist/intelligence/fsrs.js +70 -0
- package/packages/core/dist/intelligence/gap-detector.d.ts +25 -0
- package/packages/core/dist/intelligence/gap-detector.js +78 -0
- package/packages/core/dist/intelligence/learning-path.d.ts +31 -0
- package/packages/core/dist/intelligence/learning-path.js +53 -0
- package/packages/core/dist/intelligence/notifications.d.ts +31 -0
- package/packages/core/dist/intelligence/notifications.js +65 -0
- package/packages/core/dist/intelligence/predictive-gaps.d.ts +14 -0
- package/packages/core/dist/intelligence/predictive-gaps.js +74 -0
- package/packages/core/dist/intelligence/semantic-versioning.d.ts +37 -0
- package/packages/core/dist/intelligence/semantic-versioning.js +68 -0
- package/packages/core/dist/intelligence/types.d.ts +28 -0
- package/packages/core/dist/intelligence/types.js +3 -0
- package/packages/core/dist/mcp/custom-tools.d.ts +29 -0
- package/packages/core/dist/mcp/custom-tools.js +70 -0
- package/packages/core/{src/mcp/index.ts → dist/mcp/index.d.ts} +1 -0
- package/packages/core/dist/mcp/index.js +2 -0
- package/packages/core/dist/mcp/server.d.ts +49 -0
- package/packages/core/dist/mcp/server.js +151 -0
- package/packages/core/dist/mcp/tools/agentic-graph.d.ts +87 -0
- package/packages/core/dist/mcp/tools/agentic-graph.js +88 -0
- package/packages/core/dist/mcp/tools/brief.d.ts +31 -0
- package/packages/core/dist/mcp/tools/brief.js +39 -0
- package/packages/core/dist/mcp/tools/decay.d.ts +33 -0
- package/packages/core/dist/mcp/tools/decay.js +32 -0
- package/packages/core/dist/mcp/tools/decision-journal.d.ts +78 -0
- package/packages/core/dist/mcp/tools/decision-journal.js +79 -0
- package/packages/core/dist/mcp/tools/detect-gaps.d.ts +24 -0
- package/packages/core/dist/mcp/tools/detect-gaps.js +47 -0
- package/packages/core/dist/mcp/tools/export.d.ts +29 -0
- package/packages/core/dist/mcp/tools/export.js +60 -0
- package/packages/core/dist/mcp/tools/federated-search.d.ts +29 -0
- package/packages/core/dist/mcp/tools/federated-search.js +36 -0
- package/packages/core/dist/mcp/tools/generate-claude-md.d.ts +35 -0
- package/packages/core/dist/mcp/tools/generate-claude-md.js +107 -0
- package/packages/core/dist/mcp/tools/get-document.d.ts +35 -0
- package/packages/core/dist/mcp/tools/get-document.js +25 -0
- package/packages/core/dist/mcp/tools/get-evolution.d.ts +28 -0
- package/packages/core/dist/mcp/tools/get-evolution.js +70 -0
- package/packages/core/dist/mcp/tools/get-related.d.ts +32 -0
- package/packages/core/dist/mcp/tools/get-related.js +33 -0
- package/packages/core/dist/mcp/tools/learning-path.d.ts +23 -0
- package/packages/core/dist/mcp/tools/learning-path.js +45 -0
- package/packages/core/dist/mcp/tools/link-code.d.ts +34 -0
- package/packages/core/dist/mcp/tools/link-code.js +44 -0
- package/packages/core/dist/mcp/tools/list-topics.d.ts +15 -0
- package/packages/core/dist/mcp/tools/list-topics.js +18 -0
- package/packages/core/dist/mcp/tools/search.d.ts +39 -0
- package/packages/core/dist/mcp/tools/search.js +29 -0
- package/packages/core/dist/mcp/tools/snapshot.d.ts +47 -0
- package/packages/core/dist/mcp/tools/snapshot.js +84 -0
- package/packages/core/dist/multi-vault/index.d.ts +26 -0
- package/packages/core/dist/multi-vault/index.js +80 -0
- package/packages/core/dist/pack/creator.d.ts +21 -0
- package/packages/core/dist/pack/creator.js +105 -0
- package/packages/core/dist/pack/exporter.d.ts +4 -0
- package/packages/core/dist/pack/exporter.js +18 -0
- package/packages/core/dist/pack/importer.d.ts +10 -0
- package/packages/core/dist/pack/importer.js +55 -0
- package/packages/core/{src/pack/index.ts → dist/pack/index.d.ts} +1 -0
- package/packages/core/dist/pack/index.js +5 -0
- package/packages/core/dist/pack/marketplace.d.ts +14 -0
- package/packages/core/dist/pack/marketplace.js +90 -0
- package/packages/core/dist/pack/pii-masker.d.ts +7 -0
- package/packages/core/dist/pack/pii-masker.js +29 -0
- package/packages/core/dist/pack/types.d.ts +36 -0
- package/packages/core/dist/pack/types.js +3 -0
- package/packages/core/dist/plugins/index.d.ts +35 -0
- package/packages/core/dist/plugins/index.js +57 -0
- package/packages/core/dist/plugins/webhooks.d.ts +30 -0
- package/packages/core/dist/plugins/webhooks.js +79 -0
- package/packages/core/dist/search/adaptive.d.ts +16 -0
- package/packages/core/dist/search/adaptive.js +67 -0
- package/packages/core/dist/search/bm25.d.ts +4 -0
- package/packages/core/dist/search/bm25.js +10 -0
- package/packages/core/dist/search/index.d.ts +15 -0
- package/packages/core/dist/search/index.js +64 -0
- package/packages/core/dist/search/rrf.d.ts +7 -0
- package/packages/core/dist/search/rrf.js +21 -0
- package/packages/core/dist/search/semantic.d.ts +5 -0
- package/packages/core/dist/search/semantic.js +6 -0
- package/packages/core/{src/store/index.ts → dist/store/index.d.ts} +1 -0
- package/packages/core/dist/store/index.js +2 -0
- package/packages/core/dist/store/sqlite-vec.d.ts +6 -0
- package/packages/core/dist/store/sqlite-vec.js +251 -0
- package/packages/core/dist/store/types.d.ts +20 -0
- package/packages/core/dist/store/types.js +3 -0
- package/packages/core/dist/team/index.d.ts +25 -0
- package/packages/core/dist/team/index.js +97 -0
- package/packages/core/dist/types/chunk.d.ts +23 -0
- package/packages/core/dist/types/chunk.js +3 -0
- package/packages/core/dist/types/document.d.ts +23 -0
- package/packages/core/dist/types/document.js +3 -0
- package/packages/core/dist/types/graph.d.ts +39 -0
- package/packages/core/dist/types/graph.js +3 -0
- package/packages/core/dist/types/index.d.ts +5 -0
- package/packages/core/dist/types/index.js +2 -0
- package/packages/core/dist/types/search.d.ts +39 -0
- package/packages/core/dist/types/search.js +3 -0
- package/packages/core/dist/utils/retry.d.ts +25 -0
- package/packages/core/dist/utils/retry.js +59 -0
- package/.github/workflows/pages.yml +0 -37
- package/memory/MEMORY.md +0 -25
- package/packages/cli/dist/commands/brief-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/brief-cmd.js.map +0 -1
- package/packages/cli/dist/commands/capture-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/capture-cmd.js.map +0 -1
- package/packages/cli/dist/commands/card-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/card-cmd.js.map +0 -1
- package/packages/cli/dist/commands/clip-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/clip-cmd.js.map +0 -1
- package/packages/cli/dist/commands/cloud-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/cloud-cmd.js.map +0 -1
- package/packages/cli/dist/commands/contradictions-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/contradictions-cmd.js.map +0 -1
- package/packages/cli/dist/commands/decay-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/decay-cmd.js.map +0 -1
- package/packages/cli/dist/commands/digest-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/digest-cmd.js.map +0 -1
- package/packages/cli/dist/commands/duplicates-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/duplicates-cmd.js.map +0 -1
- package/packages/cli/dist/commands/federate-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/federate-cmd.js.map +0 -1
- package/packages/cli/dist/commands/gaps-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/gaps-cmd.js.map +0 -1
- package/packages/cli/dist/commands/graph-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/graph-cmd.js.map +0 -1
- package/packages/cli/dist/commands/index-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/index-cmd.js.map +0 -1
- package/packages/cli/dist/commands/init-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/init-cmd.js.map +0 -1
- package/packages/cli/dist/commands/learn-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/learn-cmd.js.map +0 -1
- package/packages/cli/dist/commands/pack-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/pack-cmd.js.map +0 -1
- package/packages/cli/dist/commands/review-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/review-cmd.js.map +0 -1
- package/packages/cli/dist/commands/search-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/search-cmd.js.map +0 -1
- package/packages/cli/dist/commands/serve-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/serve-cmd.js.map +0 -1
- package/packages/cli/dist/commands/status-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/status-cmd.js.map +0 -1
- package/packages/cli/dist/commands/sync-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/sync-cmd.js.map +0 -1
- package/packages/cli/dist/commands/vault-cmd.d.ts.map +0 -1
- package/packages/cli/dist/commands/vault-cmd.js.map +0 -1
- package/packages/cli/dist/index.d.ts.map +0 -1
- package/packages/cli/dist/index.js.map +0 -1
- package/packages/cli/src/commands/brief-cmd.ts +0 -87
- package/packages/cli/src/commands/capture-cmd.ts +0 -34
- package/packages/cli/src/commands/card-cmd.ts +0 -29
- package/packages/cli/src/commands/clip-cmd.ts +0 -172
- package/packages/cli/src/commands/cloud-cmd.ts +0 -75
- package/packages/cli/src/commands/contradictions-cmd.ts +0 -41
- package/packages/cli/src/commands/decay-cmd.ts +0 -57
- package/packages/cli/src/commands/digest-cmd.ts +0 -89
- package/packages/cli/src/commands/duplicates-cmd.ts +0 -38
- package/packages/cli/src/commands/federate-cmd.ts +0 -256
- package/packages/cli/src/commands/gaps-cmd.ts +0 -40
- package/packages/cli/src/commands/graph-cmd.ts +0 -88
- package/packages/cli/src/commands/index-cmd.ts +0 -65
- package/packages/cli/src/commands/init-cmd.ts +0 -145
- package/packages/cli/src/commands/learn-cmd.ts +0 -56
- package/packages/cli/src/commands/pack-cmd.ts +0 -121
- package/packages/cli/src/commands/review-cmd.ts +0 -125
- package/packages/cli/src/commands/search-cmd.ts +0 -45
- package/packages/cli/src/commands/serve-cmd.ts +0 -17
- package/packages/cli/src/commands/status-cmd.ts +0 -37
- package/packages/cli/src/commands/sync-cmd.ts +0 -68
- package/packages/cli/src/commands/vault-cmd.ts +0 -64
- package/packages/cli/src/index.ts +0 -187
- package/packages/core/src/api/graph-data.ts +0 -286
- package/packages/core/src/api/server.ts +0 -660
- package/packages/core/src/capture/voice.ts +0 -168
- package/packages/core/src/cloud/sync.ts +0 -167
- package/packages/core/src/config.ts +0 -82
- package/packages/core/src/federation/credits.ts +0 -80
- package/packages/core/src/federation/hyperswarm.d.ts +0 -19
- package/packages/core/src/federation/identity.ts +0 -90
- package/packages/core/src/federation/node.ts +0 -235
- package/packages/core/src/federation/privacy.ts +0 -52
- package/packages/core/src/federation/reputation.ts +0 -202
- package/packages/core/src/federation/search.ts +0 -129
- package/packages/core/src/federation/sharing.ts +0 -315
- package/packages/core/src/federation/trust.ts +0 -76
- package/packages/core/src/federation/types.ts +0 -25
- package/packages/core/src/i18n/index.ts +0 -85
- package/packages/core/src/indexer/chunker.ts +0 -180
- package/packages/core/src/indexer/embedder.ts +0 -9
- package/packages/core/src/indexer/index.ts +0 -113
- package/packages/core/src/indexer/local-embedder.ts +0 -35
- package/packages/core/src/indexer/scanner.ts +0 -142
- package/packages/core/src/indexer/watcher.ts +0 -62
- package/packages/core/src/intelligence/contradiction-detector.ts +0 -134
- package/packages/core/src/intelligence/decay-engine.ts +0 -229
- package/packages/core/src/intelligence/duplicate-detector.ts +0 -71
- package/packages/core/src/intelligence/fsrs.ts +0 -79
- package/packages/core/src/intelligence/gap-detector.ts +0 -109
- package/packages/core/src/intelligence/learning-path.ts +0 -86
- package/packages/core/src/intelligence/notifications.ts +0 -106
- package/packages/core/src/intelligence/predictive-gaps.ts +0 -94
- package/packages/core/src/intelligence/semantic-versioning.ts +0 -97
- package/packages/core/src/intelligence/types.ts +0 -28
- package/packages/core/src/mcp/custom-tools.ts +0 -97
- package/packages/core/src/mcp/server.ts +0 -142
- package/packages/core/src/mcp/tools/agentic-graph.ts +0 -96
- package/packages/core/src/mcp/tools/brief.ts +0 -49
- package/packages/core/src/mcp/tools/decay.ts +0 -40
- package/packages/core/src/mcp/tools/decision-journal.ts +0 -95
- package/packages/core/src/mcp/tools/export.ts +0 -72
- package/packages/core/src/mcp/tools/federated-search.ts +0 -43
- package/packages/core/src/mcp/tools/generate-claude-md.ts +0 -130
- package/packages/core/src/mcp/tools/get-document.ts +0 -26
- package/packages/core/src/mcp/tools/get-related.ts +0 -41
- package/packages/core/src/mcp/tools/learning-path.ts +0 -52
- package/packages/core/src/mcp/tools/list-topics.ts +0 -20
- package/packages/core/src/mcp/tools/search.ts +0 -35
- package/packages/core/src/mcp/tools/snapshot.ts +0 -98
- package/packages/core/src/multi-vault/index.ts +0 -118
- package/packages/core/src/pack/creator.ts +0 -127
- package/packages/core/src/pack/exporter.ts +0 -21
- package/packages/core/src/pack/importer.ts +0 -82
- package/packages/core/src/pack/marketplace.ts +0 -103
- package/packages/core/src/pack/pii-masker.ts +0 -38
- package/packages/core/src/pack/types.ts +0 -39
- package/packages/core/src/plugins/index.ts +0 -100
- package/packages/core/src/plugins/webhooks.ts +0 -110
- package/packages/core/src/search/bm25.ts +0 -16
- package/packages/core/src/search/index.ts +0 -83
- package/packages/core/src/search/rrf.ts +0 -31
- package/packages/core/src/search/semantic.ts +0 -15
- package/packages/core/src/store/sqlite-vec.ts +0 -290
- package/packages/core/src/store/types.ts +0 -22
- package/packages/core/src/team/index.ts +0 -126
- package/packages/core/src/types/chunk.ts +0 -25
- package/packages/core/src/types/document.ts +0 -24
- package/packages/core/src/types/graph.ts +0 -44
- package/packages/core/src/types/index.ts +0 -15
- package/packages/core/src/types/search.ts +0 -38
- package/packages/core/src/utils/retry.ts +0 -85
- package/packages/core/tests/api-card.test.ts +0 -60
- package/packages/core/tests/api-routes.test.ts +0 -98
- package/packages/core/tests/bm25.test.ts +0 -87
- package/packages/core/tests/chunker.test.ts +0 -48
- package/packages/core/tests/cluster.test.ts +0 -75
- package/packages/core/tests/constellation.test.ts +0 -77
- package/packages/core/tests/export-utils.test.ts +0 -97
- package/packages/core/tests/fsrs.test.ts +0 -96
- package/packages/core/tests/gesture-detector.test.ts +0 -45
- package/packages/core/tests/graph-data.test.ts +0 -87
- package/packages/core/tests/layout.test.ts +0 -83
- package/packages/core/tests/mcp.test.ts +0 -148
- package/packages/core/tests/pack.test.ts +0 -127
- package/packages/core/tests/pii-masker.test.ts +0 -42
- package/packages/core/tests/profile-card.test.ts +0 -62
- package/packages/core/tests/rrf.test.ts +0 -29
- package/packages/core/tests/search-integration.test.ts +0 -139
- package/packages/core/tests/store.test.ts +0 -80
- package/packages/graph/click-result.png +0 -0
- package/packages/graph/dist/assets/camera_utils-BMxqtvoZ.js +0 -1
- package/packages/graph/dist/assets/hands-DXA01_mx.js +0 -18
- package/packages/graph/dist/assets/index-DMEe2diW.js +0 -4192
- package/packages/graph/dist/assets/layout.worker-DbKCEFTz.js +0 -1
- package/packages/graph/dist/index.html +0 -17
- package/packages/graph/index.html +0 -17
- package/packages/graph/package.json +0 -32
- package/packages/graph/src/App.tsx +0 -7
- package/packages/graph/src/api/client.ts +0 -39
- package/packages/graph/src/components/ClusterFilter.tsx +0 -73
- package/packages/graph/src/components/ConstellationView.tsx +0 -232
- package/packages/graph/src/components/ExportPanel.tsx +0 -177
- package/packages/graph/src/components/Graph3D.tsx +0 -230
- package/packages/graph/src/components/GraphEdges.tsx +0 -100
- package/packages/graph/src/components/GraphNodes.tsx +0 -386
- package/packages/graph/src/components/HealthDashboard.tsx +0 -173
- package/packages/graph/src/components/Layout.tsx +0 -214
- package/packages/graph/src/components/MotionOverlay.tsx +0 -81
- package/packages/graph/src/components/MotionToggle.tsx +0 -33
- package/packages/graph/src/components/MultiverseView.tsx +0 -286
- package/packages/graph/src/components/NodeDetail.tsx +0 -232
- package/packages/graph/src/components/PulseParticle.tsx +0 -232
- package/packages/graph/src/components/SearchBar.tsx +0 -107
- package/packages/graph/src/components/StarField.tsx +0 -197
- package/packages/graph/src/components/StatusBar.tsx +0 -53
- package/packages/graph/src/components/Timeline.tsx +0 -148
- package/packages/graph/src/components/ToolsPanel.tsx +0 -512
- package/packages/graph/src/components/Tooltip.tsx +0 -100
- package/packages/graph/src/components/TypeFilter.tsx +0 -131
- package/packages/graph/src/embed/EmbedGraph.tsx +0 -144
- package/packages/graph/src/hooks/useConstellationLOD.ts +0 -76
- package/packages/graph/src/hooks/useDecay.ts +0 -37
- package/packages/graph/src/hooks/useExport.ts +0 -165
- package/packages/graph/src/hooks/useGraph.ts +0 -69
- package/packages/graph/src/hooks/useKeyboardNav.ts +0 -122
- package/packages/graph/src/hooks/useLayout.ts +0 -45
- package/packages/graph/src/hooks/useMotion.ts +0 -120
- package/packages/graph/src/hooks/usePulse.ts +0 -58
- package/packages/graph/src/hooks/useSearch.ts +0 -71
- package/packages/graph/src/lib/constellation.ts +0 -107
- package/packages/graph/src/lib/export-utils.ts +0 -48
- package/packages/graph/src/lib/gesture-detector.ts +0 -123
- package/packages/graph/src/lib/layout.worker.ts +0 -153
- package/packages/graph/src/lib/motion-controller.ts +0 -83
- package/packages/graph/src/lib/profile-card.ts +0 -122
- package/packages/graph/src/main.tsx +0 -4
- package/packages/graph/src/stores/graph-store.ts +0 -155
- package/packages/graph/success.png +0 -0
- package/packages/graph/test-click.mjs +0 -49
- package/packages/graph/test-explore.mjs +0 -102
- package/packages/graph/test-final.mjs +0 -61
- package/packages/graph/test-graph.mjs +0 -139
- package/packages/graph/test-hover.mjs +0 -48
- package/packages/graph/test-pulse.mjs +0 -68
- package/packages/graph/test-screenshot.mjs +0 -56
- package/packages/graph/test-v2.mjs +0 -97
- package/packages/graph/tsconfig.tsbuildinfo +0 -1
- package/packages/graph/vite.config.ts +0 -15
- package/packages/sync/.env.example +0 -11
- package/packages/sync/.sync-state.json +0 -317
- package/packages/sync/.upload-state.json +0 -1009
- package/packages/sync/create-stella-network-notion.mjs +0 -151
- package/packages/sync/create-stellavault-project-notion.mjs +0 -322
- package/packages/sync/logs/sync-2026-03-28.log +0 -6
- package/packages/sync/logs/sync-2026-03-29.log +0 -12
- package/packages/sync/logs/sync-2026-03-30.log +0 -6
- package/packages/sync/logs/sync-2026-03-31.log +0 -6
- package/packages/sync/logs/sync-2026-04-01.log +0 -6
- package/packages/sync/logs/sync-2026-04-02.log +0 -6
- package/packages/sync/package-lock.json +0 -373
- package/packages/sync/package.json +0 -16
- package/packages/sync/run-sync.bat +0 -18
- package/packages/sync/run-sync.mjs +0 -46
- package/packages/sync/setup-scheduler.mjs +0 -119
- package/packages/sync/structured-sync.mjs +0 -187
- package/packages/sync/sync-to-obsidian.mjs +0 -264
- package/packages/sync/upload-pdca-to-notion.mjs +0 -495
- package/tsconfig.base.json +0 -18
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
// Design Ref: §3.1 — Core Types (Chunk)
|
|
2
|
-
|
|
3
|
-
export interface Chunk {
|
|
4
|
-
/** document.id + "#" + chunkIndex */
|
|
5
|
-
id: string;
|
|
6
|
-
/** 소속 문서 ID */
|
|
7
|
-
documentId: string;
|
|
8
|
-
/** 청크 텍스트 */
|
|
9
|
-
content: string;
|
|
10
|
-
/** 소속 heading (## 제목) */
|
|
11
|
-
heading: string;
|
|
12
|
-
/** 원문 시작 줄 번호 */
|
|
13
|
-
startLine: number;
|
|
14
|
-
/** 원문 끝 줄 번호 */
|
|
15
|
-
endLine: number;
|
|
16
|
-
/** 토큰 수 */
|
|
17
|
-
tokenCount: number;
|
|
18
|
-
/** 768차원 벡터 (로딩 시 선택적) */
|
|
19
|
-
embedding?: number[];
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
export interface ScoredChunk {
|
|
23
|
-
chunkId: string;
|
|
24
|
-
score: number;
|
|
25
|
-
}
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
// Design Ref: §3.1 — Core Types (Document)
|
|
2
|
-
|
|
3
|
-
export interface Document {
|
|
4
|
-
/** SHA-256 hash of file path */
|
|
5
|
-
id: string;
|
|
6
|
-
/** vault 기준 상대 경로 */
|
|
7
|
-
filePath: string;
|
|
8
|
-
/** frontmatter title 또는 첫 heading */
|
|
9
|
-
title: string;
|
|
10
|
-
/** 전체 마크다운 텍스트 */
|
|
11
|
-
content: string;
|
|
12
|
-
/** YAML frontmatter */
|
|
13
|
-
frontmatter: Record<string, unknown>;
|
|
14
|
-
/** #태그 목록 */
|
|
15
|
-
tags: string[];
|
|
16
|
-
/** ISO 8601 */
|
|
17
|
-
lastModified: string;
|
|
18
|
-
/** SHA-256 of content (증분 인덱싱용) */
|
|
19
|
-
contentHash: string;
|
|
20
|
-
/** 출처: local | notion | clip | bridge | pack */
|
|
21
|
-
source?: string;
|
|
22
|
-
/** 노트 유형: note | clip | sync | bridge | decision | snapshot */
|
|
23
|
-
type?: string;
|
|
24
|
-
}
|
|
@@ -1,44 +0,0 @@
|
|
|
1
|
-
// Design Ref: §3.1 — Graph Data Types
|
|
2
|
-
|
|
3
|
-
export interface GraphNode {
|
|
4
|
-
id: string;
|
|
5
|
-
label: string;
|
|
6
|
-
filePath: string;
|
|
7
|
-
tags: string[];
|
|
8
|
-
clusterId: number;
|
|
9
|
-
position?: [number, number, number];
|
|
10
|
-
size: number;
|
|
11
|
-
source?: string;
|
|
12
|
-
type?: string;
|
|
13
|
-
lastModified?: string;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
export interface GraphEdge {
|
|
17
|
-
source: string;
|
|
18
|
-
target: string;
|
|
19
|
-
weight: number;
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
export interface Cluster {
|
|
23
|
-
id: number;
|
|
24
|
-
label: string;
|
|
25
|
-
color: string;
|
|
26
|
-
nodeCount: number;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
export interface GraphData {
|
|
30
|
-
nodes: GraphNode[];
|
|
31
|
-
edges: GraphEdge[];
|
|
32
|
-
clusters: Cluster[];
|
|
33
|
-
stats: {
|
|
34
|
-
nodeCount: number;
|
|
35
|
-
edgeCount: number;
|
|
36
|
-
clusterCount: number;
|
|
37
|
-
};
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
export interface GraphResponse {
|
|
41
|
-
data: GraphData;
|
|
42
|
-
generatedAt: string;
|
|
43
|
-
cacheKey: string;
|
|
44
|
-
}
|
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
export type { Document } from './document.js';
|
|
2
|
-
export type { Chunk, ScoredChunk } from './chunk.js';
|
|
3
|
-
export type {
|
|
4
|
-
SearchResult,
|
|
5
|
-
SearchOptions,
|
|
6
|
-
TopicInfo,
|
|
7
|
-
StoreStats,
|
|
8
|
-
} from './search.js';
|
|
9
|
-
export type {
|
|
10
|
-
GraphNode,
|
|
11
|
-
GraphEdge,
|
|
12
|
-
Cluster,
|
|
13
|
-
GraphData,
|
|
14
|
-
GraphResponse,
|
|
15
|
-
} from './graph.js';
|
|
@@ -1,38 +0,0 @@
|
|
|
1
|
-
// Design Ref: §3.1 — Core Types (Search)
|
|
2
|
-
|
|
3
|
-
import type { Chunk } from './chunk.js';
|
|
4
|
-
import type { Document } from './document.js';
|
|
5
|
-
|
|
6
|
-
export interface SearchResult {
|
|
7
|
-
chunk: Chunk;
|
|
8
|
-
document: Document;
|
|
9
|
-
/** RRF 통합 점수 (0~1) */
|
|
10
|
-
score: number;
|
|
11
|
-
/** 매칭 부분 하이라이트 */
|
|
12
|
-
highlights: string[];
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export interface SearchOptions {
|
|
16
|
-
query: string;
|
|
17
|
-
/** default: 10 */
|
|
18
|
-
limit?: number;
|
|
19
|
-
/** minimum score, default: 0.1 */
|
|
20
|
-
threshold?: number;
|
|
21
|
-
/** 태그 필터 */
|
|
22
|
-
tags?: string[];
|
|
23
|
-
/** 날짜 범위 필터 */
|
|
24
|
-
dateRange?: { from?: string; to?: string };
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
export interface TopicInfo {
|
|
28
|
-
topic: string;
|
|
29
|
-
count: number;
|
|
30
|
-
recentDocuments: Array<{ id: string; title: string }>;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
export interface StoreStats {
|
|
34
|
-
documentCount: number;
|
|
35
|
-
chunkCount: number;
|
|
36
|
-
dbSizeBytes: number;
|
|
37
|
-
lastIndexed: string | null;
|
|
38
|
-
}
|
|
@@ -1,85 +0,0 @@
|
|
|
1
|
-
// Error Recovery System (F-A02) — retry with exponential backoff + meaningful messages
|
|
2
|
-
|
|
3
|
-
export interface RetryOptions {
|
|
4
|
-
maxRetries?: number;
|
|
5
|
-
baseDelayMs?: number;
|
|
6
|
-
maxDelayMs?: number;
|
|
7
|
-
onRetry?: (error: Error, attempt: number, maxRetries: number) => void;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
const DEFAULT_OPTIONS: Required<Pick<RetryOptions, 'maxRetries' | 'baseDelayMs' | 'maxDelayMs'>> = {
|
|
11
|
-
maxRetries: 3,
|
|
12
|
-
baseDelayMs: 500,
|
|
13
|
-
maxDelayMs: 10000,
|
|
14
|
-
};
|
|
15
|
-
|
|
16
|
-
export async function withRetry<T>(
|
|
17
|
-
fn: () => Promise<T>,
|
|
18
|
-
options: RetryOptions = {},
|
|
19
|
-
): Promise<T> {
|
|
20
|
-
const { maxRetries, baseDelayMs, maxDelayMs } = { ...DEFAULT_OPTIONS, ...options };
|
|
21
|
-
|
|
22
|
-
let lastError: Error | undefined;
|
|
23
|
-
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
24
|
-
try {
|
|
25
|
-
return await fn();
|
|
26
|
-
} catch (err) {
|
|
27
|
-
lastError = err instanceof Error ? err : new Error(String(err));
|
|
28
|
-
if (attempt >= maxRetries) break;
|
|
29
|
-
|
|
30
|
-
const delay = Math.min(baseDelayMs * Math.pow(2, attempt), maxDelayMs);
|
|
31
|
-
options.onRetry?.(lastError, attempt + 1, maxRetries);
|
|
32
|
-
await new Promise((r) => setTimeout(r, delay));
|
|
33
|
-
}
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
throw lastError!;
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
export class StellavaultError extends Error {
|
|
40
|
-
constructor(
|
|
41
|
-
message: string,
|
|
42
|
-
public readonly code: string,
|
|
43
|
-
public readonly suggestion?: string,
|
|
44
|
-
public readonly cause?: Error,
|
|
45
|
-
) {
|
|
46
|
-
super(message);
|
|
47
|
-
this.name = 'StellavaultError';
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
format(): string {
|
|
51
|
-
const lines = [`Error [${this.code}]: ${this.message}`];
|
|
52
|
-
if (this.suggestion) lines.push(` Fix: ${this.suggestion}`);
|
|
53
|
-
if (this.cause) lines.push(` Cause: ${this.cause.message}`);
|
|
54
|
-
return lines.join('\n');
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
export function wrapError(err: unknown, code: string, suggestion?: string): StellavaultError {
|
|
59
|
-
const cause = err instanceof Error ? err : new Error(String(err));
|
|
60
|
-
return new StellavaultError(cause.message, code, suggestion, cause);
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
// Common error factories
|
|
64
|
-
export const errors = {
|
|
65
|
-
vaultNotFound: (path: string) =>
|
|
66
|
-
new StellavaultError(`Vault not found: ${path}`, 'VAULT_NOT_FOUND', 'Check the path exists and contains .md files'),
|
|
67
|
-
|
|
68
|
-
dbInitFailed: (err: unknown) =>
|
|
69
|
-
wrapError(err, 'DB_INIT_FAILED', 'Delete ~/.stellavault/index.db and re-index'),
|
|
70
|
-
|
|
71
|
-
embedderFailed: (err: unknown) =>
|
|
72
|
-
wrapError(err, 'EMBEDDER_FAILED', 'Check disk space and try again. The model downloads on first run (~80MB)'),
|
|
73
|
-
|
|
74
|
-
indexingFailed: (file: string, err: unknown) =>
|
|
75
|
-
wrapError(err, 'INDEX_FAILED', `Skipping "${file}". Re-run indexing to retry failed files`),
|
|
76
|
-
|
|
77
|
-
searchFailed: (err: unknown) =>
|
|
78
|
-
wrapError(err, 'SEARCH_FAILED', 'Re-index your vault: stellavault index <path>'),
|
|
79
|
-
|
|
80
|
-
configInvalid: (field: string) =>
|
|
81
|
-
new StellavaultError(`Invalid config: ${field}`, 'CONFIG_INVALID', 'Check ~/.stellavault.json format'),
|
|
82
|
-
|
|
83
|
-
apiServerFailed: (port: number, err: unknown) =>
|
|
84
|
-
wrapError(err, 'API_SERVER_FAILED', `Port ${port} may be in use. Try: stellavault graph --port ${port + 1}`),
|
|
85
|
-
};
|
|
@@ -1,60 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeAll, afterAll } from 'vitest';
|
|
2
|
-
import { createSqliteVecStore } from '../src/store/sqlite-vec.js';
|
|
3
|
-
import { createSearchEngine } from '../src/search/index.js';
|
|
4
|
-
import { createApiServer } from '../src/api/server.js';
|
|
5
|
-
import type { VectorStore } from '../src/store/types.js';
|
|
6
|
-
|
|
7
|
-
const DIMS = 4;
|
|
8
|
-
const PORT = 13334;
|
|
9
|
-
let store: VectorStore;
|
|
10
|
-
|
|
11
|
-
beforeAll(async () => {
|
|
12
|
-
store = createSqliteVecStore(':memory:', DIMS);
|
|
13
|
-
await store.initialize();
|
|
14
|
-
await store.upsertDocument({
|
|
15
|
-
id: 'doc1', filePath: 'test.md', title: 'Test',
|
|
16
|
-
content: 'Content', frontmatter: {}, tags: ['test'],
|
|
17
|
-
lastModified: '2026-01-01', contentHash: 'h1',
|
|
18
|
-
});
|
|
19
|
-
await store.upsertChunks([{
|
|
20
|
-
id: 'doc1#0', documentId: 'doc1', content: 'Content',
|
|
21
|
-
heading: 'Test', startLine: 1, endLine: 1, tokenCount: 2,
|
|
22
|
-
embedding: [1, 0, 0, 0],
|
|
23
|
-
}]);
|
|
24
|
-
|
|
25
|
-
const embedder = {
|
|
26
|
-
dimensions: DIMS, modelName: 'test',
|
|
27
|
-
initialize: async () => {},
|
|
28
|
-
embed: async () => [0.5, 0.5, 0.5, 0.5],
|
|
29
|
-
embedBatch: async (t: string[]) => t.map(() => [0.5, 0.5, 0.5, 0.5]),
|
|
30
|
-
};
|
|
31
|
-
const searchEngine = createSearchEngine({ store, embedder });
|
|
32
|
-
const server = createApiServer({ store, searchEngine, port: PORT });
|
|
33
|
-
await server.start();
|
|
34
|
-
});
|
|
35
|
-
|
|
36
|
-
afterAll(async () => { await store.close(); });
|
|
37
|
-
|
|
38
|
-
describe('GET /api/profile-card', () => {
|
|
39
|
-
it('SVG 반환', async () => {
|
|
40
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/profile-card`);
|
|
41
|
-
expect(res.ok).toBe(true);
|
|
42
|
-
expect(res.headers.get('content-type')).toContain('image/svg+xml');
|
|
43
|
-
const text = await res.text();
|
|
44
|
-
expect(text).toContain('<svg');
|
|
45
|
-
expect(text).toContain('</svg>');
|
|
46
|
-
});
|
|
47
|
-
|
|
48
|
-
it('문서 수 포함', async () => {
|
|
49
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/profile-card`);
|
|
50
|
-
const text = await res.text();
|
|
51
|
-
expect(text).toContain('1 docs');
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('mode=folder 지원', async () => {
|
|
55
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/profile-card?mode=folder`);
|
|
56
|
-
expect(res.ok).toBe(true);
|
|
57
|
-
const text = await res.text();
|
|
58
|
-
expect(text).toContain('<svg');
|
|
59
|
-
});
|
|
60
|
-
});
|
|
@@ -1,98 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect, beforeAll, afterAll } from 'vitest';
|
|
2
|
-
import { createSqliteVecStore } from '../src/store/sqlite-vec.js';
|
|
3
|
-
import { createSearchEngine } from '../src/search/index.js';
|
|
4
|
-
import { createApiServer } from '../src/api/server.js';
|
|
5
|
-
import type { VectorStore } from '../src/store/types.js';
|
|
6
|
-
import type { Embedder } from '../src/indexer/embedder.js';
|
|
7
|
-
|
|
8
|
-
const DIMS = 4;
|
|
9
|
-
let store: VectorStore;
|
|
10
|
-
let server: ReturnType<typeof createApiServer>;
|
|
11
|
-
const PORT = 13333; // 테스트용 포트
|
|
12
|
-
|
|
13
|
-
function mockEmbedder(): Embedder {
|
|
14
|
-
return {
|
|
15
|
-
dimensions: DIMS, modelName: 'test',
|
|
16
|
-
initialize: async () => {},
|
|
17
|
-
embed: async () => [0.5, 0.5, 0.5, 0.5],
|
|
18
|
-
embedBatch: async (texts) => texts.map(() => [0.5, 0.5, 0.5, 0.5]),
|
|
19
|
-
};
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
beforeAll(async () => {
|
|
23
|
-
store = createSqliteVecStore(':memory:', DIMS);
|
|
24
|
-
await store.initialize();
|
|
25
|
-
|
|
26
|
-
await store.upsertDocument({
|
|
27
|
-
id: 'doc1', filePath: 'test.md', title: 'Test Doc',
|
|
28
|
-
content: 'OAuth authentication patterns', frontmatter: {}, tags: ['auth'],
|
|
29
|
-
lastModified: '2026-01-01', contentHash: 'h1',
|
|
30
|
-
});
|
|
31
|
-
await store.upsertChunks([{
|
|
32
|
-
id: 'doc1#0', documentId: 'doc1', content: 'OAuth authentication patterns',
|
|
33
|
-
heading: 'Auth', startLine: 1, endLine: 1, tokenCount: 3,
|
|
34
|
-
embedding: [1, 0, 0, 0],
|
|
35
|
-
}]);
|
|
36
|
-
|
|
37
|
-
const embedder = mockEmbedder();
|
|
38
|
-
const searchEngine = createSearchEngine({ store, embedder });
|
|
39
|
-
server = createApiServer({ store, searchEngine, port: PORT });
|
|
40
|
-
await server.start();
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
afterAll(async () => { await store.close(); });
|
|
44
|
-
|
|
45
|
-
describe('API Routes', () => {
|
|
46
|
-
it('GET /api/stats', async () => {
|
|
47
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/stats`);
|
|
48
|
-
expect(res.ok).toBe(true);
|
|
49
|
-
const data = await res.json();
|
|
50
|
-
expect(data.documentCount).toBe(1);
|
|
51
|
-
expect(data.chunkCount).toBe(1);
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('GET /api/graph', async () => {
|
|
55
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/graph`);
|
|
56
|
-
expect(res.ok).toBe(true);
|
|
57
|
-
const data = await res.json();
|
|
58
|
-
expect(data.data.nodes.length).toBe(1);
|
|
59
|
-
expect(data.data.stats.nodeCount).toBe(1);
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
it('GET /api/graph?mode=folder', async () => {
|
|
63
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/graph?mode=folder`);
|
|
64
|
-
expect(res.ok).toBe(true);
|
|
65
|
-
const data = await res.json();
|
|
66
|
-
expect(data.data.clusters.length).toBeGreaterThan(0);
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
it('GET /api/search?q=OAuth', async () => {
|
|
70
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/search?q=OAuth`);
|
|
71
|
-
expect(res.ok).toBe(true);
|
|
72
|
-
const data = await res.json();
|
|
73
|
-
expect(data.query).toBe('OAuth');
|
|
74
|
-
expect(data.results.length).toBeGreaterThan(0);
|
|
75
|
-
expect(data.results[0].documentId).toBe('doc1');
|
|
76
|
-
});
|
|
77
|
-
|
|
78
|
-
it('GET /api/search 빈 쿼리', async () => {
|
|
79
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/search?q=`);
|
|
80
|
-
expect(res.ok).toBe(true);
|
|
81
|
-
const data = await res.json();
|
|
82
|
-
expect(data.results).toEqual([]);
|
|
83
|
-
});
|
|
84
|
-
|
|
85
|
-
it('GET /api/document/:id 존재', async () => {
|
|
86
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/document/doc1`);
|
|
87
|
-
expect(res.ok).toBe(true);
|
|
88
|
-
const data = await res.json();
|
|
89
|
-
expect(data.title).toBe('Test Doc');
|
|
90
|
-
expect(data.content).toContain('OAuth');
|
|
91
|
-
expect(data.related).toBeDefined();
|
|
92
|
-
});
|
|
93
|
-
|
|
94
|
-
it('GET /api/document/:id 미존재 → 404', async () => {
|
|
95
|
-
const res = await fetch(`http://127.0.0.1:${PORT}/api/document/nonexistent`);
|
|
96
|
-
expect(res.status).toBe(404);
|
|
97
|
-
});
|
|
98
|
-
});
|
|
@@ -1,87 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { searchBm25 } from '../src/search/bm25.js';
|
|
3
|
-
import type { VectorStore } from '../src/store/types.js';
|
|
4
|
-
import type { ScoredChunk } from '../src/types/chunk.js';
|
|
5
|
-
|
|
6
|
-
// FTS5 검색을 시뮬레이션하는 mock store
|
|
7
|
-
function createMockStore(keywordResults: ScoredChunk[] = []): VectorStore {
|
|
8
|
-
return {
|
|
9
|
-
searchKeyword: async (_query: string, _limit: number) => keywordResults,
|
|
10
|
-
// 사용하지 않는 메서드 stub
|
|
11
|
-
initialize: async () => {},
|
|
12
|
-
close: async () => {},
|
|
13
|
-
upsertDocument: async () => {},
|
|
14
|
-
upsertChunks: async () => {},
|
|
15
|
-
deleteByDocumentId: async () => {},
|
|
16
|
-
getDocument: async () => null,
|
|
17
|
-
getChunk: async () => null,
|
|
18
|
-
searchSemantic: async () => [],
|
|
19
|
-
getTopics: async () => [],
|
|
20
|
-
getStats: async () => ({ documentCount: 0, chunkCount: 0, dbSizeBytes: 0, lastIndexed: null }),
|
|
21
|
-
getAllDocumentHashes: async () => new Map(),
|
|
22
|
-
} as VectorStore;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
describe('searchBm25', () => {
|
|
26
|
-
it('빈 쿼리는 빈 결과 반환', async () => {
|
|
27
|
-
const store = createMockStore();
|
|
28
|
-
const results = await searchBm25(store, '', 10);
|
|
29
|
-
expect(results).toEqual([]);
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
it('특수문자만 있는 쿼리는 빈 결과 반환', async () => {
|
|
33
|
-
const store = createMockStore();
|
|
34
|
-
const results = await searchBm25(store, '!@#$%^&*()', 10);
|
|
35
|
-
expect(results).toEqual([]);
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
it('일반 영문 쿼리 OR 조합 생성', async () => {
|
|
39
|
-
let capturedQuery = '';
|
|
40
|
-
const store = createMockStore();
|
|
41
|
-
store.searchKeyword = async (query: string, _limit: number) => {
|
|
42
|
-
capturedQuery = query;
|
|
43
|
-
return [];
|
|
44
|
-
};
|
|
45
|
-
|
|
46
|
-
await searchBm25(store, 'React state management', 10);
|
|
47
|
-
expect(capturedQuery).toBe('React OR state OR management');
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
it('한국어 쿼리 처리', async () => {
|
|
51
|
-
let capturedQuery = '';
|
|
52
|
-
const store = createMockStore();
|
|
53
|
-
store.searchKeyword = async (query: string, _limit: number) => {
|
|
54
|
-
capturedQuery = query;
|
|
55
|
-
return [];
|
|
56
|
-
};
|
|
57
|
-
|
|
58
|
-
await searchBm25(store, '리액트 상태관리', 10);
|
|
59
|
-
expect(capturedQuery).toBe('리액트 OR 상태관리');
|
|
60
|
-
});
|
|
61
|
-
|
|
62
|
-
it('특수문자 제거 후 정상 검색', async () => {
|
|
63
|
-
let capturedQuery = '';
|
|
64
|
-
const store = createMockStore();
|
|
65
|
-
store.searchKeyword = async (query: string, _limit: number) => {
|
|
66
|
-
capturedQuery = query;
|
|
67
|
-
return [];
|
|
68
|
-
};
|
|
69
|
-
|
|
70
|
-
await searchBm25(store, 'OAuth2.0 인증/설계', 10);
|
|
71
|
-
// 특수문자(., /) 제거 후 단어 추출
|
|
72
|
-
expect(capturedQuery).toContain('OR');
|
|
73
|
-
expect(capturedQuery).not.toContain('.');
|
|
74
|
-
expect(capturedQuery).not.toContain('/');
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it('결과를 그대로 반환', async () => {
|
|
78
|
-
const mockResults: ScoredChunk[] = [
|
|
79
|
-
{ chunkId: 'doc1#0', score: 0.9 },
|
|
80
|
-
{ chunkId: 'doc2#1', score: 0.5 },
|
|
81
|
-
];
|
|
82
|
-
const store = createMockStore(mockResults);
|
|
83
|
-
const results = await searchBm25(store, 'test query', 10);
|
|
84
|
-
expect(results).toEqual(mockResults);
|
|
85
|
-
expect(results.length).toBe(2);
|
|
86
|
-
});
|
|
87
|
-
});
|
|
@@ -1,48 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
import { chunkDocument, estimateTokens } from '../src/indexer/chunker.js';
|
|
3
|
-
|
|
4
|
-
describe('estimateTokens', () => {
|
|
5
|
-
it('영문 토큰 추정', () => {
|
|
6
|
-
expect(estimateTokens('hello world')).toBe(3);
|
|
7
|
-
});
|
|
8
|
-
|
|
9
|
-
it('한국어 토큰 추정', () => {
|
|
10
|
-
expect(estimateTokens('안녕하세요')).toBe(3);
|
|
11
|
-
});
|
|
12
|
-
|
|
13
|
-
it('빈 문자열', () => {
|
|
14
|
-
expect(estimateTokens('')).toBe(0);
|
|
15
|
-
});
|
|
16
|
-
});
|
|
17
|
-
|
|
18
|
-
describe('chunkDocument', () => {
|
|
19
|
-
it('짧은 문서는 1개 청크', () => {
|
|
20
|
-
const chunks = chunkDocument('doc1', '# Title\n\nShort content.');
|
|
21
|
-
expect(chunks.length).toBe(1);
|
|
22
|
-
expect(chunks[0].documentId).toBe('doc1');
|
|
23
|
-
expect(chunks[0].id).toBe('doc1#0');
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
it('heading으로 분할', () => {
|
|
27
|
-
const md = '# Title\n\nIntro.\n\n## A\n\n' + 'Content A. '.repeat(50) + '\n\n## B\n\n' + 'Content B. '.repeat(50);
|
|
28
|
-
const chunks = chunkDocument('doc2', md);
|
|
29
|
-
expect(chunks.length).toBeGreaterThanOrEqual(2);
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
it('빈 문서는 빈 배열', () => {
|
|
33
|
-
const chunks = chunkDocument('doc3', '');
|
|
34
|
-
expect(chunks).toEqual([]);
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
it('heading 없는 문서도 처리', () => {
|
|
38
|
-
const chunks = chunkDocument('doc4', 'Just plain text without headings.');
|
|
39
|
-
expect(chunks.length).toBe(1);
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
it('짧은 청크 병합', () => {
|
|
43
|
-
const md = '## A\n\nHi\n\n## B\n\nBye';
|
|
44
|
-
const chunks = chunkDocument('doc5', md, { minTokens: 50 });
|
|
45
|
-
// 둘 다 50 토큰 미만이므로 병합됨
|
|
46
|
-
expect(chunks.length).toBe(1);
|
|
47
|
-
});
|
|
48
|
-
});
|
|
@@ -1,75 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
|
|
3
|
-
// graph-data.ts에서 K-means 내부 함수는 export 안 되므로, buildGraphData로 간접 테스트
|
|
4
|
-
import { buildGraphData } from '../src/api/graph-data.js';
|
|
5
|
-
import { createSqliteVecStore } from '../src/store/sqlite-vec.js';
|
|
6
|
-
import type { VectorStore } from '../src/store/types.js';
|
|
7
|
-
|
|
8
|
-
const DIMS = 4;
|
|
9
|
-
|
|
10
|
-
async function setupStore(docCount: number): Promise<VectorStore> {
|
|
11
|
-
const store = createSqliteVecStore(':memory:', DIMS);
|
|
12
|
-
await store.initialize();
|
|
13
|
-
|
|
14
|
-
for (let i = 0; i < docCount; i++) {
|
|
15
|
-
// 3그룹으로 나뉘는 벡터: [1,0,0,0], [0,1,0,0], [0,0,1,0]
|
|
16
|
-
const group = i % 3;
|
|
17
|
-
const vec = [0, 0, 0, 0.1];
|
|
18
|
-
vec[group] = 1;
|
|
19
|
-
|
|
20
|
-
await store.upsertDocument({
|
|
21
|
-
id: `d${i}`, filePath: `folder${group}/doc${i}.md`, title: `Doc ${i} Group ${group}`,
|
|
22
|
-
content: `Content ${i}`, frontmatter: {}, tags: [],
|
|
23
|
-
lastModified: '2026-01-01', contentHash: `h${i}`,
|
|
24
|
-
});
|
|
25
|
-
await store.upsertChunks([{
|
|
26
|
-
id: `d${i}#0`, documentId: `d${i}`, content: `Content ${i}`,
|
|
27
|
-
heading: `Doc ${i}`, startLine: 1, endLine: 1, tokenCount: 2,
|
|
28
|
-
embedding: vec,
|
|
29
|
-
}]);
|
|
30
|
-
}
|
|
31
|
-
return store;
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
describe('K-Means Clustering (via buildGraphData)', () => {
|
|
35
|
-
it('클러스터 수가 5~10 범위', async () => {
|
|
36
|
-
const store = await setupStore(30);
|
|
37
|
-
const data = await buildGraphData(store, { mode: 'semantic' });
|
|
38
|
-
expect(data.clusters.length).toBeGreaterThanOrEqual(3);
|
|
39
|
-
expect(data.clusters.length).toBeLessThanOrEqual(10);
|
|
40
|
-
await store.close();
|
|
41
|
-
});
|
|
42
|
-
|
|
43
|
-
it('모든 노드에 clusterId 할당', async () => {
|
|
44
|
-
const store = await setupStore(15);
|
|
45
|
-
const data = await buildGraphData(store);
|
|
46
|
-
for (const node of data.nodes) {
|
|
47
|
-
expect(node.clusterId).toBeTypeOf('number');
|
|
48
|
-
}
|
|
49
|
-
await store.close();
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
it('클러스터 라벨이 비어있지 않음', async () => {
|
|
53
|
-
const store = await setupStore(15);
|
|
54
|
-
const data = await buildGraphData(store);
|
|
55
|
-
for (const c of data.clusters) {
|
|
56
|
-
expect(c.label.length).toBeGreaterThan(0);
|
|
57
|
-
}
|
|
58
|
-
await store.close();
|
|
59
|
-
});
|
|
60
|
-
|
|
61
|
-
it('클러스터 nodeCount 합 = 전체 노드 수', async () => {
|
|
62
|
-
const store = await setupStore(12);
|
|
63
|
-
const data = await buildGraphData(store);
|
|
64
|
-
const totalFromClusters = data.clusters.reduce((sum, c) => sum + c.nodeCount, 0);
|
|
65
|
-
expect(totalFromClusters).toBe(data.nodes.length);
|
|
66
|
-
await store.close();
|
|
67
|
-
});
|
|
68
|
-
|
|
69
|
-
it('folder 모드: 폴더 수만큼 클러스터', async () => {
|
|
70
|
-
const store = await setupStore(9);
|
|
71
|
-
const data = await buildGraphData(store, { mode: 'folder' });
|
|
72
|
-
expect(data.clusters.length).toBe(3); // folder0, folder1, folder2
|
|
73
|
-
await store.close();
|
|
74
|
-
});
|
|
75
|
-
});
|
|
@@ -1,77 +0,0 @@
|
|
|
1
|
-
import { describe, it, expect } from 'vitest';
|
|
2
|
-
|
|
3
|
-
// constellation.ts는 graph 패키지에 있으므로 MST 알고리즘 로직만 테스트
|
|
4
|
-
// Prim's MST: 노드를 최소 거리로 연결
|
|
5
|
-
|
|
6
|
-
function primMST(positions: Array<[number, number, number]>): Array<[number, number]> {
|
|
7
|
-
const n = positions.length;
|
|
8
|
-
if (n < 2) return [];
|
|
9
|
-
|
|
10
|
-
const inMST = new Array(n).fill(false);
|
|
11
|
-
const minEdge = new Array(n).fill(Infinity);
|
|
12
|
-
const parent = new Array(n).fill(-1);
|
|
13
|
-
minEdge[0] = 0;
|
|
14
|
-
const edges: Array<[number, number]> = [];
|
|
15
|
-
|
|
16
|
-
for (let iter = 0; iter < n; iter++) {
|
|
17
|
-
let u = -1;
|
|
18
|
-
for (let i = 0; i < n; i++) {
|
|
19
|
-
if (!inMST[i] && (u === -1 || minEdge[i] < minEdge[u])) u = i;
|
|
20
|
-
}
|
|
21
|
-
if (u === -1) break;
|
|
22
|
-
inMST[u] = true;
|
|
23
|
-
if (parent[u] !== -1) edges.push([parent[u], u]);
|
|
24
|
-
|
|
25
|
-
for (let v = 0; v < n; v++) {
|
|
26
|
-
if (inMST[v]) continue;
|
|
27
|
-
const d = Math.sqrt(
|
|
28
|
-
(positions[u][0] - positions[v][0]) ** 2 +
|
|
29
|
-
(positions[u][1] - positions[v][1]) ** 2 +
|
|
30
|
-
(positions[u][2] - positions[v][2]) ** 2
|
|
31
|
-
);
|
|
32
|
-
if (d < minEdge[v]) { minEdge[v] = d; parent[v] = u; }
|
|
33
|
-
}
|
|
34
|
-
}
|
|
35
|
-
return edges;
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
describe('Constellation MST', () => {
|
|
39
|
-
it('2개 노드 → 1개 엣지', () => {
|
|
40
|
-
const edges = primMST([[0, 0, 0], [10, 0, 0]]);
|
|
41
|
-
expect(edges.length).toBe(1);
|
|
42
|
-
});
|
|
43
|
-
|
|
44
|
-
it('3개 노드 → 2개 엣지', () => {
|
|
45
|
-
const edges = primMST([[0, 0, 0], [10, 0, 0], [5, 10, 0]]);
|
|
46
|
-
expect(edges.length).toBe(2);
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
it('1개 노드 → 엣지 없음', () => {
|
|
50
|
-
const edges = primMST([[0, 0, 0]]);
|
|
51
|
-
expect(edges.length).toBe(0);
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('빈 배열 → 엣지 없음', () => {
|
|
55
|
-
const edges = primMST([]);
|
|
56
|
-
expect(edges.length).toBe(0);
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
it('MST는 N-1개 엣지 (트리)', () => {
|
|
60
|
-
const positions: Array<[number, number, number]> = [];
|
|
61
|
-
for (let i = 0; i < 10; i++) positions.push([i * 10, Math.random() * 10, 0]);
|
|
62
|
-
const edges = primMST(positions);
|
|
63
|
-
expect(edges.length).toBe(9);
|
|
64
|
-
});
|
|
65
|
-
|
|
66
|
-
it('모든 노드가 연결됨', () => {
|
|
67
|
-
const positions: Array<[number, number, number]> = [[0, 0, 0], [100, 0, 0], [50, 100, 0], [50, 50, 50]];
|
|
68
|
-
const edges = primMST(positions);
|
|
69
|
-
const connected = new Set<number>();
|
|
70
|
-
connected.add(0);
|
|
71
|
-
for (const [a, b] of edges) {
|
|
72
|
-
connected.add(a);
|
|
73
|
-
connected.add(b);
|
|
74
|
-
}
|
|
75
|
-
expect(connected.size).toBe(4);
|
|
76
|
-
});
|
|
77
|
-
});
|