nexo-brain 2.3.0 → 2.3.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/bin/nexo-brain.js +92 -9
- package/bin/postinstall.js +22 -15
- package/package.json +7 -4
- package/src/auto_update.py +194 -5
- package/src/crons/sync.py +6 -2
- package/src/db/_core.py +1 -0
- package/src/db/_entities.py +1 -0
- package/src/db/_episodic.py +1 -0
- package/src/db/_learnings.py +1 -0
- package/src/db/_reminders.py +1 -0
- package/src/db/_schema.py +11 -1
- package/src/db/_sessions.py +1 -0
- package/src/db/_skills.py +1 -0
- package/src/hooks/capture-tool-logs.sh +23 -6
- package/src/hooks/session-start.sh +4 -3
- package/src/plugin_loader.py +1 -0
- package/src/plugins/update.py +377 -26
- package/src/scripts/deep-sleep/apply_findings.py +1 -0
- package/src/scripts/deep-sleep/collect.py +1 -0
- package/src/scripts/deep-sleep/extract.py +1 -0
- package/src/scripts/deep-sleep/synthesize.py +1 -0
- package/src/scripts/nexo-catchup.py +29 -4
- package/src/scripts/nexo-daily-self-audit.py +21 -1
- package/src/scripts/nexo-evolution-run.py +21 -1
- package/src/scripts/nexo-learning-housekeep.py +1 -0
- package/src/scripts/nexo-postmortem-consolidator.py +34 -9
- package/src/scripts/nexo-sleep.py +32 -10
- package/src/scripts/nexo-synthesis.py +29 -9
- package/src/scripts/nexo-update.sh +109 -7
- package/src/scripts/nexo-watchdog.sh +122 -58
- package/src/server.py +66 -1
- package/src/tools_coordination.py +1 -0
- package/src/tools_sessions.py +1 -0
- package/scripts/migrate-to-unified 2.sh +0 -813
- package/scripts/migrate-to-unified.sh +0 -813
- package/scripts/migrate-v1.5-to-v1.6 2.py +0 -778
- package/scripts/migrate-v1.5-to-v1.6.py +0 -778
- package/scripts/migrate-v1.7-to-v1.8 2.py +0 -214
- package/scripts/migrate-v1.7-to-v1.8.py +0 -214
- package/scripts/nexo-preflight.sh +0 -236
- package/scripts/pre-commit-check 2.sh +0 -55
- package/scripts/pre-commit-check.sh +0 -55
- package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
- package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
- package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
- package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
- package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
- package/src/__pycache__/plugin_loader.cpython-314.pyc +0 -0
- package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
- package/src/auto_close_sessions 2.py +0 -159
- package/src/auto_update 2.py +0 -634
- package/src/claim_graph 2.py +0 -323
- package/src/cognitive/__init__ 2.py +0 -62
- package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
- package/src/cognitive/_core 2.py +0 -567
- package/src/cognitive/_decay 2.py +0 -382
- package/src/cognitive/_ingest 2.py +0 -892
- package/src/cognitive/_memory 2.py +0 -912
- package/src/cognitive/_search 2.py +0 -949
- package/src/cognitive/_trust 2.py +0 -464
- package/src/crons/__pycache__/sync.cpython-314.pyc +0 -0
- package/src/crons/manifest 2.json +0 -106
- package/src/crons/sync 2.py +0 -217
- package/src/dashboard/__init__ 2.py +0 -0
- package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
- package/src/dashboard/app 2.py +0 -789
- package/src/db/__init__ 2.py +0 -89
- package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
- package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_cron_runs.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_cron_runs.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_skills.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_skills.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_skills.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
- package/src/db/_core 2.py +0 -417
- package/src/db/_credentials 2.py +0 -124
- package/src/db/_entities 2.py +0 -178
- package/src/db/_episodic 2.py +0 -738
- package/src/db/_evolution 2.py +0 -54
- package/src/db/_fts 2.py +0 -406
- package/src/db/_learnings 2.py +0 -168
- package/src/db/_reminders 2.py +0 -338
- package/src/db/_schema 2.py +0 -364
- package/src/db/_sessions 2.py +0 -300
- package/src/db/_tasks 2.py +0 -91
- package/src/evolution_cycle 2.py +0 -266
- package/src/hnsw_index 2.py +0 -254
- package/src/hooks/auto_capture 2.py +0 -208
- package/src/hooks/caffeinate-guard 2.sh +0 -8
- package/src/hooks/capture-session 2.sh +0 -21
- package/src/hooks/capture-tool-logs 2.sh +0 -127
- package/src/hooks/daily-briefing-check 2.sh +0 -33
- package/src/hooks/inbox-hook 2.sh +0 -76
- package/src/hooks/post-compact 2.sh +0 -148
- package/src/hooks/pre-compact 2.sh +0 -151
- package/src/hooks/session-start 2.sh +0 -268
- package/src/hooks/session-stop 2.sh +0 -140
- package/src/kg_populate 2.py +0 -290
- package/src/knowledge_graph 2.py +0 -257
- package/src/maintenance 2.py +0 -59
- package/src/migrate_embeddings 2.py +0 -122
- package/src/plugin_loader 2.py +0 -202
- package/src/plugins/__init__ 2.py +0 -0
- package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/schedule.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/schedule.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/skills.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/skills.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
- package/src/plugins/adaptive_mode 2.py +0 -805
- package/src/plugins/agents 2.py +0 -52
- package/src/plugins/artifact_registry 2.py +0 -450
- package/src/plugins/backup 2.py +0 -104
- package/src/plugins/cognitive_memory 2.py +0 -564
- package/src/plugins/core_rules 2.py +0 -252
- package/src/plugins/cortex 2.py +0 -299
- package/src/plugins/entities 2.py +0 -67
- package/src/plugins/episodic_memory 2.py +0 -533
- package/src/plugins/evolution 2.py +0 -115
- package/src/plugins/guard 2.py +0 -746
- package/src/plugins/knowledge_graph_tools 2.py +0 -105
- package/src/plugins/preferences 2.py +0 -47
- package/src/plugins/update 2.py +0 -256
- package/src/requirements 2.txt +0 -12
- package/src/rules/__init__ 2.py +0 -0
- package/src/rules/core-rules 2.json +0 -331
- package/src/rules/migrate 2.py +0 -207
- package/src/scripts/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
- package/src/scripts/check-context 2.py +0 -264
- package/src/scripts/nexo-auto-update 2.py +0 -6
- package/src/scripts/nexo-backup 2.sh +0 -25
- package/src/scripts/nexo-brain-activation 2.sh +0 -140
- package/src/scripts/nexo-catchup 2.py +0 -242
- package/src/scripts/nexo-cognitive-decay 2.py +0 -182
- package/src/scripts/nexo-daily-self-audit 2.py +0 -552
- package/src/scripts/nexo-deep-sleep 2.sh +0 -97
- package/src/scripts/nexo-evolution-run 2.py +0 -597
- package/src/scripts/nexo-followup-hygiene 2.py +0 -112
- package/src/scripts/nexo-github-monitor 2.py +0 -256
- package/src/scripts/nexo-immune 2.py +0 -927
- package/src/scripts/nexo-inbox-hook 2.sh +0 -74
- package/src/scripts/nexo-install 2.py +0 -6
- package/src/scripts/nexo-learning-housekeep 2.py +0 -245
- package/src/scripts/nexo-learning-validator 2.py +0 -207
- package/src/scripts/nexo-migrate 2.py +0 -232
- package/src/scripts/nexo-postmortem-consolidator 2.py +0 -421
- package/src/scripts/nexo-pre-commit 2.py +0 -120
- package/src/scripts/nexo-prevent-sleep 2.sh +0 -29
- package/src/scripts/nexo-proactive-dashboard 2.py +0 -345
- package/src/scripts/nexo-reflection 2.py +0 -253
- package/src/scripts/nexo-runtime-preflight 2.py +0 -274
- package/src/scripts/nexo-send-email 2.py +0 -25
- package/src/scripts/nexo-send-email.py +0 -25
- package/src/scripts/nexo-send-reply 2.py +0 -178
- package/src/scripts/nexo-send-reply.py +0 -178
- package/src/scripts/nexo-sleep 2.py +0 -592
- package/src/scripts/nexo-snapshot-restore 2.sh +0 -35
- package/src/scripts/nexo-synthesis 2.py +0 -253
- package/src/scripts/nexo-tcc-approve 2.sh +0 -79
- package/src/scripts/nexo-update 2.sh +0 -161
- package/src/scripts/nexo-watchdog 2.sh +0 -878
- package/src/scripts/nexo-watchdog-smoke 2.py +0 -119
- package/src/server 2.py +0 -733
- package/src/storage_router 2.py +0 -32
- package/src/tools_coordination 2.py +0 -102
- package/src/tools_credentials 2.py +0 -68
- package/src/tools_learnings 2.py +0 -220
- package/src/tools_menu 2.py +0 -227
- package/src/tools_reminders 2.py +0 -86
- package/src/tools_reminders_crud 2.py +0 -159
- package/src/tools_sessions 2.py +0 -476
- package/src/tools_task_history 2.py +0 -57
- package/templates/CLAUDE.md 2.template +0 -63
- package/templates/openclaw 2.json +0 -13
- package/tests/__init__ 2.py +0 -0
- package/tests/__init__.py +0 -0
- package/tests/conftest 2.py +0 -71
- package/tests/conftest.py +0 -71
- package/tests/test_cognitive 2.py +0 -205
- package/tests/test_cognitive.py +0 -205
- package/tests/test_knowledge_graph 2.py +0 -140
- package/tests/test_knowledge_graph.py +0 -140
- package/tests/test_migrations 2.py +0 -137
- package/tests/test_migrations.py +0 -137
|
@@ -1,140 +0,0 @@
|
|
|
1
|
-
"""Tests for Knowledge Graph operations."""
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
def test_upsert_and_get_node():
|
|
5
|
-
"""Create a node and retrieve it."""
|
|
6
|
-
import cognitive
|
|
7
|
-
cognitive._get_db() # Ensure KG tables exist
|
|
8
|
-
|
|
9
|
-
import knowledge_graph as kg
|
|
10
|
-
node_id = kg.upsert_node("area", "area:test", "Test Area", {"color": "blue"})
|
|
11
|
-
assert node_id > 0
|
|
12
|
-
|
|
13
|
-
node = kg.get_node("area", "area:test")
|
|
14
|
-
assert node is not None
|
|
15
|
-
assert node["label"] == "Test Area"
|
|
16
|
-
|
|
17
|
-
# Upsert same node should return same id (update)
|
|
18
|
-
node_id2 = kg.upsert_node("area", "area:test", "Test Area Updated")
|
|
19
|
-
assert node_id2 == node_id
|
|
20
|
-
|
|
21
|
-
node2 = kg.get_node("area", "area:test")
|
|
22
|
-
assert node2["label"] == "Test Area Updated"
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
def test_upsert_and_get_edge():
|
|
26
|
-
"""Create nodes and edges, verify traversal."""
|
|
27
|
-
import cognitive
|
|
28
|
-
cognitive._get_db()
|
|
29
|
-
|
|
30
|
-
import knowledge_graph as kg
|
|
31
|
-
kg.upsert_node("learning", "learning:1", "L1")
|
|
32
|
-
kg.upsert_node("file", "file:foo.py", "foo.py")
|
|
33
|
-
|
|
34
|
-
result = kg.upsert_edge(
|
|
35
|
-
"learning", "learning:1", "touched",
|
|
36
|
-
"file", "file:foo.py", weight=1.0,
|
|
37
|
-
)
|
|
38
|
-
assert result["action"] == "ADD"
|
|
39
|
-
|
|
40
|
-
# Same edge again → NOOP
|
|
41
|
-
result2 = kg.upsert_edge(
|
|
42
|
-
"learning", "learning:1", "touched",
|
|
43
|
-
"file", "file:foo.py", weight=1.0,
|
|
44
|
-
)
|
|
45
|
-
assert result2["action"] == "NOOP"
|
|
46
|
-
|
|
47
|
-
# Different weight → UPDATE (closes old, creates new)
|
|
48
|
-
result3 = kg.upsert_edge(
|
|
49
|
-
"learning", "learning:1", "touched",
|
|
50
|
-
"file", "file:foo.py", weight=0.5,
|
|
51
|
-
)
|
|
52
|
-
assert result3["action"] == "UPDATE"
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
def test_neighbors():
|
|
56
|
-
"""Get direct neighbors of a node."""
|
|
57
|
-
import cognitive
|
|
58
|
-
cognitive._get_db()
|
|
59
|
-
|
|
60
|
-
import knowledge_graph as kg
|
|
61
|
-
kg.upsert_node("area", "area:myproject", "MyProject")
|
|
62
|
-
kg.upsert_node("learning", "learning:10", "L10")
|
|
63
|
-
kg.upsert_node("learning", "learning:11", "L11")
|
|
64
|
-
kg.upsert_edge("learning", "learning:10", "belongs_to", "area", "area:myproject")
|
|
65
|
-
kg.upsert_edge("learning", "learning:11", "belongs_to", "area", "area:myproject")
|
|
66
|
-
|
|
67
|
-
area_node = kg.get_node("area", "area:myproject")
|
|
68
|
-
neighbors = kg.get_neighbors(area_node["id"])
|
|
69
|
-
assert len(neighbors) == 2
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
def test_traverse():
|
|
73
|
-
"""BFS traversal from a node."""
|
|
74
|
-
import cognitive
|
|
75
|
-
cognitive._get_db()
|
|
76
|
-
|
|
77
|
-
import knowledge_graph as kg
|
|
78
|
-
# A → B → C chain
|
|
79
|
-
kg.upsert_node("area", "area:a", "A")
|
|
80
|
-
kg.upsert_node("file", "file:b", "B")
|
|
81
|
-
kg.upsert_node("change", "change:c", "C")
|
|
82
|
-
kg.upsert_edge("area", "area:a", "has", "file", "file:b")
|
|
83
|
-
kg.upsert_edge("file", "file:b", "modified_by", "change", "change:c")
|
|
84
|
-
|
|
85
|
-
a_node = kg.get_node("area", "area:a")
|
|
86
|
-
result = kg.traverse(a_node["id"], max_depth=2)
|
|
87
|
-
node_ids = {n["id"] for n in result["nodes"]}
|
|
88
|
-
# Should reach A, B, and C within depth 2
|
|
89
|
-
assert len(node_ids) == 3
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
def test_shortest_path():
|
|
93
|
-
"""Find shortest path between two nodes."""
|
|
94
|
-
import cognitive
|
|
95
|
-
cognitive._get_db()
|
|
96
|
-
|
|
97
|
-
import knowledge_graph as kg
|
|
98
|
-
kg.upsert_node("area", "area:x", "X")
|
|
99
|
-
kg.upsert_node("file", "file:y", "Y")
|
|
100
|
-
kg.upsert_node("change", "change:z", "Z")
|
|
101
|
-
kg.upsert_edge("area", "area:x", "has", "file", "file:y")
|
|
102
|
-
kg.upsert_edge("file", "file:y", "modified_by", "change", "change:z")
|
|
103
|
-
|
|
104
|
-
x = kg.get_node("area", "area:x")
|
|
105
|
-
z = kg.get_node("change", "change:z")
|
|
106
|
-
path = kg.shortest_path(x["id"], z["id"])
|
|
107
|
-
assert path is not None
|
|
108
|
-
assert len(path) == 3 # X → Y → Z
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
def test_delete_edge():
|
|
112
|
-
"""Soft-delete an edge (set valid_until)."""
|
|
113
|
-
import cognitive
|
|
114
|
-
cognitive._get_db()
|
|
115
|
-
|
|
116
|
-
import knowledge_graph as kg
|
|
117
|
-
kg.upsert_node("area", "area:del", "Del")
|
|
118
|
-
kg.upsert_node("file", "file:del", "Del File")
|
|
119
|
-
kg.upsert_edge("area", "area:del", "has", "file", "file:del")
|
|
120
|
-
|
|
121
|
-
deleted = kg.delete_edge("area", "area:del", "has", "file", "file:del")
|
|
122
|
-
assert deleted is True
|
|
123
|
-
|
|
124
|
-
# After deletion, no active neighbors
|
|
125
|
-
node = kg.get_node("area", "area:del")
|
|
126
|
-
neighbors = kg.get_neighbors(node["id"], active_only=True)
|
|
127
|
-
assert len(neighbors) == 0
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
def test_stats():
|
|
131
|
-
"""Stats should return valid counts."""
|
|
132
|
-
import cognitive
|
|
133
|
-
cognitive._get_db()
|
|
134
|
-
|
|
135
|
-
import knowledge_graph as kg
|
|
136
|
-
kg.upsert_node("area", "area:stats", "Stats Test")
|
|
137
|
-
s = kg.stats()
|
|
138
|
-
assert s["nodes"] >= 1
|
|
139
|
-
assert isinstance(s["edges_active"], int)
|
|
140
|
-
assert isinstance(s["node_types"], dict)
|
|
@@ -1,137 +0,0 @@
|
|
|
1
|
-
"""Tests for database schema and migrations."""
|
|
2
|
-
|
|
3
|
-
import db as db_mod
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
def test_init_db_creates_core_tables():
|
|
7
|
-
"""All core tables should exist after init_db."""
|
|
8
|
-
conn = db_mod.get_db()
|
|
9
|
-
rows = conn.execute(
|
|
10
|
-
"SELECT name FROM sqlite_master WHERE type='table' ORDER BY name"
|
|
11
|
-
).fetchall()
|
|
12
|
-
tables = {r["name"] for r in rows}
|
|
13
|
-
|
|
14
|
-
expected = {
|
|
15
|
-
"sessions", "tracked_files", "messages", "message_reads",
|
|
16
|
-
"questions", "reminders", "followups", "learnings", "credentials",
|
|
17
|
-
"task_history", "task_frequencies", "plugins", "entities",
|
|
18
|
-
"preferences", "agents", "change_log", "decisions",
|
|
19
|
-
}
|
|
20
|
-
assert expected.issubset(tables), f"Missing tables: {expected - tables}"
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
def test_migrations_idempotent():
|
|
24
|
-
"""Running migrations twice should not raise."""
|
|
25
|
-
db_mod.run_migrations()
|
|
26
|
-
db_mod.run_migrations()
|
|
27
|
-
version = db_mod.get_schema_version()
|
|
28
|
-
assert version >= 1
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
def test_session_crud():
|
|
32
|
-
"""Register, update, and clean sessions."""
|
|
33
|
-
info = db_mod.register_session("test-sid-1", "test task")
|
|
34
|
-
assert info["sid"] == "test-sid-1"
|
|
35
|
-
|
|
36
|
-
active = db_mod.get_active_sessions()
|
|
37
|
-
sids = [s["sid"] for s in active]
|
|
38
|
-
assert "test-sid-1" in sids
|
|
39
|
-
|
|
40
|
-
db_mod.update_session("test-sid-1", "updated task")
|
|
41
|
-
|
|
42
|
-
db_mod.complete_session("test-sid-1")
|
|
43
|
-
active2 = db_mod.get_active_sessions()
|
|
44
|
-
sids2 = [s["sid"] for s in active2]
|
|
45
|
-
assert "test-sid-1" not in sids2
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
def test_learning_crud():
|
|
49
|
-
"""Create, search, update, and delete learnings."""
|
|
50
|
-
result = db_mod.create_learning(
|
|
51
|
-
category="test-cat",
|
|
52
|
-
title="Test Learning Title",
|
|
53
|
-
content="Some content about testing patterns.",
|
|
54
|
-
)
|
|
55
|
-
learning_id = result["id"]
|
|
56
|
-
assert learning_id > 0
|
|
57
|
-
|
|
58
|
-
found = db_mod.search_learnings("testing patterns")
|
|
59
|
-
assert any(l["id"] == learning_id for l in found)
|
|
60
|
-
|
|
61
|
-
db_mod.update_learning(learning_id, title="Updated Title")
|
|
62
|
-
found2 = db_mod.search_learnings("Updated Title")
|
|
63
|
-
assert any(l["id"] == learning_id for l in found2)
|
|
64
|
-
|
|
65
|
-
db_mod.delete_learning(learning_id)
|
|
66
|
-
found3 = db_mod.search_learnings("Updated Title")
|
|
67
|
-
assert not any(l["id"] == learning_id for l in found3)
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
def test_reminder_followup_crud():
|
|
71
|
-
"""Create and complete reminders and followups."""
|
|
72
|
-
db_mod.create_reminder("R-TEST1", "Test reminder", date="2026-12-31")
|
|
73
|
-
reminder = db_mod.get_reminder("R-TEST1")
|
|
74
|
-
assert reminder is not None
|
|
75
|
-
assert reminder["status"] == "PENDING"
|
|
76
|
-
|
|
77
|
-
db_mod.complete_reminder("R-TEST1")
|
|
78
|
-
reminder2 = db_mod.get_reminder("R-TEST1")
|
|
79
|
-
assert reminder2["status"] == "COMPLETED"
|
|
80
|
-
|
|
81
|
-
db_mod.create_followup("NF-TEST1", "Test followup", date="2026-12-31")
|
|
82
|
-
followup = db_mod.get_followup("NF-TEST1")
|
|
83
|
-
assert followup is not None
|
|
84
|
-
|
|
85
|
-
db_mod.complete_followup("NF-TEST1", result="done")
|
|
86
|
-
followup2 = db_mod.get_followup("NF-TEST1")
|
|
87
|
-
assert followup2["status"] == "COMPLETED"
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
def test_recurring_followup():
|
|
91
|
-
"""Recurring followup: complete archives with date suffix, creates new pending, returns correct IDs."""
|
|
92
|
-
db_mod.create_followup("NF-REC1", "Recurring test", date="2026-03-31", recurrence="weekly:monday")
|
|
93
|
-
followup = db_mod.get_followup("NF-REC1")
|
|
94
|
-
assert followup is not None
|
|
95
|
-
assert followup["recurrence"] == "weekly:monday"
|
|
96
|
-
|
|
97
|
-
result = db_mod.complete_followup("NF-REC1", result="done weekly")
|
|
98
|
-
|
|
99
|
-
# Result should reference the archived ID, not the recycled NF-REC1
|
|
100
|
-
assert result["status"] == "COMPLETED"
|
|
101
|
-
assert result["id"].startswith("NF-REC1-") # archived with date suffix
|
|
102
|
-
assert result["next_id"] == "NF-REC1"
|
|
103
|
-
assert result["next_date"] is not None
|
|
104
|
-
|
|
105
|
-
# The new NF-REC1 should be PENDING (not the completed one)
|
|
106
|
-
new_followup = db_mod.get_followup("NF-REC1")
|
|
107
|
-
assert new_followup is not None
|
|
108
|
-
assert new_followup["status"] == "PENDING"
|
|
109
|
-
|
|
110
|
-
# The archived one should exist with date suffix
|
|
111
|
-
archived = db_mod.get_followup(result["id"])
|
|
112
|
-
assert archived is not None
|
|
113
|
-
assert archived["status"] == "COMPLETED"
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
def test_credential_crud():
|
|
117
|
-
"""Create, get, and delete credentials."""
|
|
118
|
-
db_mod.create_credential("test-service", "api_key", "secret123", notes="test")
|
|
119
|
-
creds = db_mod.get_credential("test-service", "api_key")
|
|
120
|
-
assert len(creds) == 1
|
|
121
|
-
assert creds[0]["value"] == "secret123"
|
|
122
|
-
|
|
123
|
-
db_mod.delete_credential("test-service", "api_key")
|
|
124
|
-
creds2 = db_mod.get_credential("test-service", "api_key")
|
|
125
|
-
assert len(creds2) == 0
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
def test_fts_tables_created():
|
|
129
|
-
"""FTS5 virtual tables should exist after init + migrations."""
|
|
130
|
-
conn = db_mod.get_db()
|
|
131
|
-
rows = conn.execute(
|
|
132
|
-
"SELECT name FROM sqlite_master WHERE type='table' AND name LIKE '%fts%'"
|
|
133
|
-
).fetchall()
|
|
134
|
-
# At minimum the learnings FTS should exist (created in init or migration)
|
|
135
|
-
table_names = {r["name"] for r in rows}
|
|
136
|
-
# nexo_fts is the main FTS table
|
|
137
|
-
assert "nexo_fts" in table_names or len(table_names) > 0
|
package/tests/test_migrations.py
DELETED
|
@@ -1,137 +0,0 @@
|
|
|
1
|
-
"""Tests for database schema and migrations."""
|
|
2
|
-
|
|
3
|
-
import db as db_mod
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
def test_init_db_creates_core_tables():
|
|
7
|
-
"""All core tables should exist after init_db."""
|
|
8
|
-
conn = db_mod.get_db()
|
|
9
|
-
rows = conn.execute(
|
|
10
|
-
"SELECT name FROM sqlite_master WHERE type='table' ORDER BY name"
|
|
11
|
-
).fetchall()
|
|
12
|
-
tables = {r["name"] for r in rows}
|
|
13
|
-
|
|
14
|
-
expected = {
|
|
15
|
-
"sessions", "tracked_files", "messages", "message_reads",
|
|
16
|
-
"questions", "reminders", "followups", "learnings", "credentials",
|
|
17
|
-
"task_history", "task_frequencies", "plugins", "entities",
|
|
18
|
-
"preferences", "agents", "change_log", "decisions",
|
|
19
|
-
}
|
|
20
|
-
assert expected.issubset(tables), f"Missing tables: {expected - tables}"
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
def test_migrations_idempotent():
|
|
24
|
-
"""Running migrations twice should not raise."""
|
|
25
|
-
db_mod.run_migrations()
|
|
26
|
-
db_mod.run_migrations()
|
|
27
|
-
version = db_mod.get_schema_version()
|
|
28
|
-
assert version >= 1
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
def test_session_crud():
|
|
32
|
-
"""Register, update, and clean sessions."""
|
|
33
|
-
info = db_mod.register_session("test-sid-1", "test task")
|
|
34
|
-
assert info["sid"] == "test-sid-1"
|
|
35
|
-
|
|
36
|
-
active = db_mod.get_active_sessions()
|
|
37
|
-
sids = [s["sid"] for s in active]
|
|
38
|
-
assert "test-sid-1" in sids
|
|
39
|
-
|
|
40
|
-
db_mod.update_session("test-sid-1", "updated task")
|
|
41
|
-
|
|
42
|
-
db_mod.complete_session("test-sid-1")
|
|
43
|
-
active2 = db_mod.get_active_sessions()
|
|
44
|
-
sids2 = [s["sid"] for s in active2]
|
|
45
|
-
assert "test-sid-1" not in sids2
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
def test_learning_crud():
|
|
49
|
-
"""Create, search, update, and delete learnings."""
|
|
50
|
-
result = db_mod.create_learning(
|
|
51
|
-
category="test-cat",
|
|
52
|
-
title="Test Learning Title",
|
|
53
|
-
content="Some content about testing patterns.",
|
|
54
|
-
)
|
|
55
|
-
learning_id = result["id"]
|
|
56
|
-
assert learning_id > 0
|
|
57
|
-
|
|
58
|
-
found = db_mod.search_learnings("testing patterns")
|
|
59
|
-
assert any(l["id"] == learning_id for l in found)
|
|
60
|
-
|
|
61
|
-
db_mod.update_learning(learning_id, title="Updated Title")
|
|
62
|
-
found2 = db_mod.search_learnings("Updated Title")
|
|
63
|
-
assert any(l["id"] == learning_id for l in found2)
|
|
64
|
-
|
|
65
|
-
db_mod.delete_learning(learning_id)
|
|
66
|
-
found3 = db_mod.search_learnings("Updated Title")
|
|
67
|
-
assert not any(l["id"] == learning_id for l in found3)
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
def test_reminder_followup_crud():
|
|
71
|
-
"""Create and complete reminders and followups."""
|
|
72
|
-
db_mod.create_reminder("R-TEST1", "Test reminder", date="2026-12-31")
|
|
73
|
-
reminder = db_mod.get_reminder("R-TEST1")
|
|
74
|
-
assert reminder is not None
|
|
75
|
-
assert reminder["status"] == "PENDING"
|
|
76
|
-
|
|
77
|
-
db_mod.complete_reminder("R-TEST1")
|
|
78
|
-
reminder2 = db_mod.get_reminder("R-TEST1")
|
|
79
|
-
assert reminder2["status"] == "COMPLETED"
|
|
80
|
-
|
|
81
|
-
db_mod.create_followup("NF-TEST1", "Test followup", date="2026-12-31")
|
|
82
|
-
followup = db_mod.get_followup("NF-TEST1")
|
|
83
|
-
assert followup is not None
|
|
84
|
-
|
|
85
|
-
db_mod.complete_followup("NF-TEST1", result="done")
|
|
86
|
-
followup2 = db_mod.get_followup("NF-TEST1")
|
|
87
|
-
assert followup2["status"] == "COMPLETED"
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
def test_recurring_followup():
|
|
91
|
-
"""Recurring followup: complete archives with date suffix, creates new pending, returns correct IDs."""
|
|
92
|
-
db_mod.create_followup("NF-REC1", "Recurring test", date="2026-03-31", recurrence="weekly:monday")
|
|
93
|
-
followup = db_mod.get_followup("NF-REC1")
|
|
94
|
-
assert followup is not None
|
|
95
|
-
assert followup["recurrence"] == "weekly:monday"
|
|
96
|
-
|
|
97
|
-
result = db_mod.complete_followup("NF-REC1", result="done weekly")
|
|
98
|
-
|
|
99
|
-
# Result should reference the archived ID, not the recycled NF-REC1
|
|
100
|
-
assert result["status"] == "COMPLETED"
|
|
101
|
-
assert result["id"].startswith("NF-REC1-") # archived with date suffix
|
|
102
|
-
assert result["next_id"] == "NF-REC1"
|
|
103
|
-
assert result["next_date"] is not None
|
|
104
|
-
|
|
105
|
-
# The new NF-REC1 should be PENDING (not the completed one)
|
|
106
|
-
new_followup = db_mod.get_followup("NF-REC1")
|
|
107
|
-
assert new_followup is not None
|
|
108
|
-
assert new_followup["status"] == "PENDING"
|
|
109
|
-
|
|
110
|
-
# The archived one should exist with date suffix
|
|
111
|
-
archived = db_mod.get_followup(result["id"])
|
|
112
|
-
assert archived is not None
|
|
113
|
-
assert archived["status"] == "COMPLETED"
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
def test_credential_crud():
|
|
117
|
-
"""Create, get, and delete credentials."""
|
|
118
|
-
db_mod.create_credential("test-service", "api_key", "secret123", notes="test")
|
|
119
|
-
creds = db_mod.get_credential("test-service", "api_key")
|
|
120
|
-
assert len(creds) == 1
|
|
121
|
-
assert creds[0]["value"] == "secret123"
|
|
122
|
-
|
|
123
|
-
db_mod.delete_credential("test-service", "api_key")
|
|
124
|
-
creds2 = db_mod.get_credential("test-service", "api_key")
|
|
125
|
-
assert len(creds2) == 0
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
def test_fts_tables_created():
|
|
129
|
-
"""FTS5 virtual tables should exist after init + migrations."""
|
|
130
|
-
conn = db_mod.get_db()
|
|
131
|
-
rows = conn.execute(
|
|
132
|
-
"SELECT name FROM sqlite_master WHERE type='table' AND name LIKE '%fts%'"
|
|
133
|
-
).fetchall()
|
|
134
|
-
# At minimum the learnings FTS should exist (created in init or migration)
|
|
135
|
-
table_names = {r["name"] for r in rows}
|
|
136
|
-
# nexo_fts is the main FTS table
|
|
137
|
-
assert "nexo_fts" in table_names or len(table_names) > 0
|