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.
Files changed (299) hide show
  1. package/README.md +1 -1
  2. package/bin/nexo-brain.js +92 -9
  3. package/bin/postinstall.js +22 -15
  4. package/package.json +7 -4
  5. package/src/auto_update.py +194 -5
  6. package/src/crons/sync.py +6 -2
  7. package/src/db/_core.py +1 -0
  8. package/src/db/_entities.py +1 -0
  9. package/src/db/_episodic.py +1 -0
  10. package/src/db/_learnings.py +1 -0
  11. package/src/db/_reminders.py +1 -0
  12. package/src/db/_schema.py +11 -1
  13. package/src/db/_sessions.py +1 -0
  14. package/src/db/_skills.py +1 -0
  15. package/src/hooks/capture-tool-logs.sh +23 -6
  16. package/src/hooks/session-start.sh +4 -3
  17. package/src/plugin_loader.py +1 -0
  18. package/src/plugins/update.py +377 -26
  19. package/src/scripts/deep-sleep/apply_findings.py +1 -0
  20. package/src/scripts/deep-sleep/collect.py +1 -0
  21. package/src/scripts/deep-sleep/extract.py +1 -0
  22. package/src/scripts/deep-sleep/synthesize.py +1 -0
  23. package/src/scripts/nexo-catchup.py +29 -4
  24. package/src/scripts/nexo-daily-self-audit.py +21 -1
  25. package/src/scripts/nexo-evolution-run.py +21 -1
  26. package/src/scripts/nexo-learning-housekeep.py +1 -0
  27. package/src/scripts/nexo-postmortem-consolidator.py +34 -9
  28. package/src/scripts/nexo-sleep.py +32 -10
  29. package/src/scripts/nexo-synthesis.py +29 -9
  30. package/src/scripts/nexo-update.sh +109 -7
  31. package/src/scripts/nexo-watchdog.sh +122 -58
  32. package/src/server.py +66 -1
  33. package/src/tools_coordination.py +1 -0
  34. package/src/tools_sessions.py +1 -0
  35. package/scripts/migrate-to-unified 2.sh +0 -813
  36. package/scripts/migrate-to-unified.sh +0 -813
  37. package/scripts/migrate-v1.5-to-v1.6 2.py +0 -778
  38. package/scripts/migrate-v1.5-to-v1.6.py +0 -778
  39. package/scripts/migrate-v1.7-to-v1.8 2.py +0 -214
  40. package/scripts/migrate-v1.7-to-v1.8.py +0 -214
  41. package/scripts/nexo-preflight.sh +0 -236
  42. package/scripts/pre-commit-check 2.sh +0 -55
  43. package/scripts/pre-commit-check.sh +0 -55
  44. package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
  45. package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
  46. package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
  47. package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
  48. package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
  49. package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
  50. package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
  51. package/src/__pycache__/plugin_loader.cpython-314.pyc +0 -0
  52. package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
  53. package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
  54. package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
  55. package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
  56. package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
  57. package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
  58. package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
  59. package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
  60. package/src/auto_close_sessions 2.py +0 -159
  61. package/src/auto_update 2.py +0 -634
  62. package/src/claim_graph 2.py +0 -323
  63. package/src/cognitive/__init__ 2.py +0 -62
  64. package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
  65. package/src/cognitive/__pycache__/__init__.cpython-312.pyc +0 -0
  66. package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
  67. package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
  68. package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
  69. package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
  70. package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
  71. package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
  72. package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
  73. package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
  74. package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
  75. package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
  76. package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
  77. package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
  78. package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
  79. package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
  80. package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
  81. package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
  82. package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
  83. package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
  84. package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
  85. package/src/cognitive/_core 2.py +0 -567
  86. package/src/cognitive/_decay 2.py +0 -382
  87. package/src/cognitive/_ingest 2.py +0 -892
  88. package/src/cognitive/_memory 2.py +0 -912
  89. package/src/cognitive/_search 2.py +0 -949
  90. package/src/cognitive/_trust 2.py +0 -464
  91. package/src/crons/__pycache__/sync.cpython-314.pyc +0 -0
  92. package/src/crons/manifest 2.json +0 -106
  93. package/src/crons/sync 2.py +0 -217
  94. package/src/dashboard/__init__ 2.py +0 -0
  95. package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
  96. package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
  97. package/src/dashboard/app 2.py +0 -789
  98. package/src/db/__init__ 2.py +0 -89
  99. package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
  100. package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
  101. package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
  102. package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
  103. package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
  104. package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
  105. package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
  106. package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
  107. package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
  108. package/src/db/__pycache__/_cron_runs.cpython-310.pyc +0 -0
  109. package/src/db/__pycache__/_cron_runs.cpython-314.pyc +0 -0
  110. package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
  111. package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
  112. package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
  113. package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
  114. package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
  115. package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
  116. package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
  117. package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
  118. package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
  119. package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
  120. package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
  121. package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
  122. package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
  123. package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
  124. package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
  125. package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
  126. package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
  127. package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
  128. package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
  129. package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
  130. package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
  131. package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
  132. package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
  133. package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
  134. package/src/db/__pycache__/_skills.cpython-310.pyc +0 -0
  135. package/src/db/__pycache__/_skills.cpython-312.pyc +0 -0
  136. package/src/db/__pycache__/_skills.cpython-314.pyc +0 -0
  137. package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
  138. package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
  139. package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
  140. package/src/db/_core 2.py +0 -417
  141. package/src/db/_credentials 2.py +0 -124
  142. package/src/db/_entities 2.py +0 -178
  143. package/src/db/_episodic 2.py +0 -738
  144. package/src/db/_evolution 2.py +0 -54
  145. package/src/db/_fts 2.py +0 -406
  146. package/src/db/_learnings 2.py +0 -168
  147. package/src/db/_reminders 2.py +0 -338
  148. package/src/db/_schema 2.py +0 -364
  149. package/src/db/_sessions 2.py +0 -300
  150. package/src/db/_tasks 2.py +0 -91
  151. package/src/evolution_cycle 2.py +0 -266
  152. package/src/hnsw_index 2.py +0 -254
  153. package/src/hooks/auto_capture 2.py +0 -208
  154. package/src/hooks/caffeinate-guard 2.sh +0 -8
  155. package/src/hooks/capture-session 2.sh +0 -21
  156. package/src/hooks/capture-tool-logs 2.sh +0 -127
  157. package/src/hooks/daily-briefing-check 2.sh +0 -33
  158. package/src/hooks/inbox-hook 2.sh +0 -76
  159. package/src/hooks/post-compact 2.sh +0 -148
  160. package/src/hooks/pre-compact 2.sh +0 -151
  161. package/src/hooks/session-start 2.sh +0 -268
  162. package/src/hooks/session-stop 2.sh +0 -140
  163. package/src/kg_populate 2.py +0 -290
  164. package/src/knowledge_graph 2.py +0 -257
  165. package/src/maintenance 2.py +0 -59
  166. package/src/migrate_embeddings 2.py +0 -122
  167. package/src/plugin_loader 2.py +0 -202
  168. package/src/plugins/__init__ 2.py +0 -0
  169. package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
  170. package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
  171. package/src/plugins/__pycache__/__init__.cpython-314.pyc +0 -0
  172. package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
  173. package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
  174. package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
  175. package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
  176. package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
  177. package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
  178. package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
  179. package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
  180. package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
  181. package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
  182. package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
  183. package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
  184. package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
  185. package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
  186. package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
  187. package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
  188. package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
  189. package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
  190. package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
  191. package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
  192. package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
  193. package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
  194. package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
  195. package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
  196. package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
  197. package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
  198. package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
  199. package/src/plugins/__pycache__/schedule.cpython-310.pyc +0 -0
  200. package/src/plugins/__pycache__/schedule.cpython-314.pyc +0 -0
  201. package/src/plugins/__pycache__/skills.cpython-310.pyc +0 -0
  202. package/src/plugins/__pycache__/skills.cpython-314.pyc +0 -0
  203. package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
  204. package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
  205. package/src/plugins/adaptive_mode 2.py +0 -805
  206. package/src/plugins/agents 2.py +0 -52
  207. package/src/plugins/artifact_registry 2.py +0 -450
  208. package/src/plugins/backup 2.py +0 -104
  209. package/src/plugins/cognitive_memory 2.py +0 -564
  210. package/src/plugins/core_rules 2.py +0 -252
  211. package/src/plugins/cortex 2.py +0 -299
  212. package/src/plugins/entities 2.py +0 -67
  213. package/src/plugins/episodic_memory 2.py +0 -533
  214. package/src/plugins/evolution 2.py +0 -115
  215. package/src/plugins/guard 2.py +0 -746
  216. package/src/plugins/knowledge_graph_tools 2.py +0 -105
  217. package/src/plugins/preferences 2.py +0 -47
  218. package/src/plugins/update 2.py +0 -256
  219. package/src/requirements 2.txt +0 -12
  220. package/src/rules/__init__ 2.py +0 -0
  221. package/src/rules/core-rules 2.json +0 -331
  222. package/src/rules/migrate 2.py +0 -207
  223. package/src/scripts/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
  224. package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
  225. package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
  226. package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
  227. package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
  228. package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
  229. package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
  230. package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
  231. package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
  232. package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
  233. package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
  234. package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
  235. package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
  236. package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
  237. package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
  238. package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
  239. package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
  240. package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
  241. package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
  242. package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
  243. package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
  244. package/src/scripts/check-context 2.py +0 -264
  245. package/src/scripts/nexo-auto-update 2.py +0 -6
  246. package/src/scripts/nexo-backup 2.sh +0 -25
  247. package/src/scripts/nexo-brain-activation 2.sh +0 -140
  248. package/src/scripts/nexo-catchup 2.py +0 -242
  249. package/src/scripts/nexo-cognitive-decay 2.py +0 -182
  250. package/src/scripts/nexo-daily-self-audit 2.py +0 -552
  251. package/src/scripts/nexo-deep-sleep 2.sh +0 -97
  252. package/src/scripts/nexo-evolution-run 2.py +0 -597
  253. package/src/scripts/nexo-followup-hygiene 2.py +0 -112
  254. package/src/scripts/nexo-github-monitor 2.py +0 -256
  255. package/src/scripts/nexo-immune 2.py +0 -927
  256. package/src/scripts/nexo-inbox-hook 2.sh +0 -74
  257. package/src/scripts/nexo-install 2.py +0 -6
  258. package/src/scripts/nexo-learning-housekeep 2.py +0 -245
  259. package/src/scripts/nexo-learning-validator 2.py +0 -207
  260. package/src/scripts/nexo-migrate 2.py +0 -232
  261. package/src/scripts/nexo-postmortem-consolidator 2.py +0 -421
  262. package/src/scripts/nexo-pre-commit 2.py +0 -120
  263. package/src/scripts/nexo-prevent-sleep 2.sh +0 -29
  264. package/src/scripts/nexo-proactive-dashboard 2.py +0 -345
  265. package/src/scripts/nexo-reflection 2.py +0 -253
  266. package/src/scripts/nexo-runtime-preflight 2.py +0 -274
  267. package/src/scripts/nexo-send-email 2.py +0 -25
  268. package/src/scripts/nexo-send-email.py +0 -25
  269. package/src/scripts/nexo-send-reply 2.py +0 -178
  270. package/src/scripts/nexo-send-reply.py +0 -178
  271. package/src/scripts/nexo-sleep 2.py +0 -592
  272. package/src/scripts/nexo-snapshot-restore 2.sh +0 -35
  273. package/src/scripts/nexo-synthesis 2.py +0 -253
  274. package/src/scripts/nexo-tcc-approve 2.sh +0 -79
  275. package/src/scripts/nexo-update 2.sh +0 -161
  276. package/src/scripts/nexo-watchdog 2.sh +0 -878
  277. package/src/scripts/nexo-watchdog-smoke 2.py +0 -119
  278. package/src/server 2.py +0 -733
  279. package/src/storage_router 2.py +0 -32
  280. package/src/tools_coordination 2.py +0 -102
  281. package/src/tools_credentials 2.py +0 -68
  282. package/src/tools_learnings 2.py +0 -220
  283. package/src/tools_menu 2.py +0 -227
  284. package/src/tools_reminders 2.py +0 -86
  285. package/src/tools_reminders_crud 2.py +0 -159
  286. package/src/tools_sessions 2.py +0 -476
  287. package/src/tools_task_history 2.py +0 -57
  288. package/templates/CLAUDE.md 2.template +0 -63
  289. package/templates/openclaw 2.json +0 -13
  290. package/tests/__init__ 2.py +0 -0
  291. package/tests/__init__.py +0 -0
  292. package/tests/conftest 2.py +0 -71
  293. package/tests/conftest.py +0 -71
  294. package/tests/test_cognitive 2.py +0 -205
  295. package/tests/test_cognitive.py +0 -205
  296. package/tests/test_knowledge_graph 2.py +0 -140
  297. package/tests/test_knowledge_graph.py +0 -140
  298. package/tests/test_migrations 2.py +0 -137
  299. 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
@@ -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