nexo-brain 2.3.0 → 2.3.1

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 (287) hide show
  1. package/README.md +1 -1
  2. package/package.json +6 -3
  3. package/src/auto_update.py +1 -0
  4. package/src/crons/sync.py +1 -2
  5. package/src/db/_core.py +1 -0
  6. package/src/db/_entities.py +1 -0
  7. package/src/db/_episodic.py +1 -0
  8. package/src/db/_learnings.py +1 -0
  9. package/src/db/_reminders.py +1 -0
  10. package/src/db/_sessions.py +1 -0
  11. package/src/db/_skills.py +1 -0
  12. package/src/plugin_loader.py +1 -0
  13. package/src/plugins/update.py +1 -0
  14. package/src/scripts/deep-sleep/apply_findings.py +1 -0
  15. package/src/scripts/deep-sleep/collect.py +1 -0
  16. package/src/scripts/deep-sleep/extract.py +1 -0
  17. package/src/scripts/deep-sleep/synthesize.py +1 -0
  18. package/src/scripts/nexo-learning-housekeep.py +1 -0
  19. package/src/scripts/nexo-watchdog.sh +19 -11
  20. package/src/server.py +1 -0
  21. package/src/tools_coordination.py +1 -0
  22. package/src/tools_sessions.py +1 -0
  23. package/scripts/migrate-to-unified 2.sh +0 -813
  24. package/scripts/migrate-to-unified.sh +0 -813
  25. package/scripts/migrate-v1.5-to-v1.6 2.py +0 -778
  26. package/scripts/migrate-v1.5-to-v1.6.py +0 -778
  27. package/scripts/migrate-v1.7-to-v1.8 2.py +0 -214
  28. package/scripts/migrate-v1.7-to-v1.8.py +0 -214
  29. package/scripts/nexo-preflight.sh +0 -236
  30. package/scripts/pre-commit-check 2.sh +0 -55
  31. package/scripts/pre-commit-check.sh +0 -55
  32. package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
  33. package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
  34. package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
  35. package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
  36. package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
  37. package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
  38. package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
  39. package/src/__pycache__/plugin_loader.cpython-314.pyc +0 -0
  40. package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
  41. package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
  42. package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
  43. package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
  44. package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
  45. package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
  46. package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
  47. package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
  48. package/src/auto_close_sessions 2.py +0 -159
  49. package/src/auto_update 2.py +0 -634
  50. package/src/claim_graph 2.py +0 -323
  51. package/src/cognitive/__init__ 2.py +0 -62
  52. package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
  53. package/src/cognitive/__pycache__/__init__.cpython-312.pyc +0 -0
  54. package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
  55. package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
  56. package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
  57. package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
  58. package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
  59. package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
  60. package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
  61. package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
  62. package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
  63. package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
  64. package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
  65. package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
  66. package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
  67. package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
  68. package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
  69. package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
  70. package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
  71. package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
  72. package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
  73. package/src/cognitive/_core 2.py +0 -567
  74. package/src/cognitive/_decay 2.py +0 -382
  75. package/src/cognitive/_ingest 2.py +0 -892
  76. package/src/cognitive/_memory 2.py +0 -912
  77. package/src/cognitive/_search 2.py +0 -949
  78. package/src/cognitive/_trust 2.py +0 -464
  79. package/src/crons/__pycache__/sync.cpython-314.pyc +0 -0
  80. package/src/crons/manifest 2.json +0 -106
  81. package/src/crons/sync 2.py +0 -217
  82. package/src/dashboard/__init__ 2.py +0 -0
  83. package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
  84. package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
  85. package/src/dashboard/app 2.py +0 -789
  86. package/src/db/__init__ 2.py +0 -89
  87. package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
  88. package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
  89. package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
  90. package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
  91. package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
  92. package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
  93. package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
  94. package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
  95. package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
  96. package/src/db/__pycache__/_cron_runs.cpython-310.pyc +0 -0
  97. package/src/db/__pycache__/_cron_runs.cpython-314.pyc +0 -0
  98. package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
  99. package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
  100. package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
  101. package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
  102. package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
  103. package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
  104. package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
  105. package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
  106. package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
  107. package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
  108. package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
  109. package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
  110. package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
  111. package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
  112. package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
  113. package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
  114. package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
  115. package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
  116. package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
  117. package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
  118. package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
  119. package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
  120. package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
  121. package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
  122. package/src/db/__pycache__/_skills.cpython-310.pyc +0 -0
  123. package/src/db/__pycache__/_skills.cpython-312.pyc +0 -0
  124. package/src/db/__pycache__/_skills.cpython-314.pyc +0 -0
  125. package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
  126. package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
  127. package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
  128. package/src/db/_core 2.py +0 -417
  129. package/src/db/_credentials 2.py +0 -124
  130. package/src/db/_entities 2.py +0 -178
  131. package/src/db/_episodic 2.py +0 -738
  132. package/src/db/_evolution 2.py +0 -54
  133. package/src/db/_fts 2.py +0 -406
  134. package/src/db/_learnings 2.py +0 -168
  135. package/src/db/_reminders 2.py +0 -338
  136. package/src/db/_schema 2.py +0 -364
  137. package/src/db/_sessions 2.py +0 -300
  138. package/src/db/_tasks 2.py +0 -91
  139. package/src/evolution_cycle 2.py +0 -266
  140. package/src/hnsw_index 2.py +0 -254
  141. package/src/hooks/auto_capture 2.py +0 -208
  142. package/src/hooks/caffeinate-guard 2.sh +0 -8
  143. package/src/hooks/capture-session 2.sh +0 -21
  144. package/src/hooks/capture-tool-logs 2.sh +0 -127
  145. package/src/hooks/daily-briefing-check 2.sh +0 -33
  146. package/src/hooks/inbox-hook 2.sh +0 -76
  147. package/src/hooks/post-compact 2.sh +0 -148
  148. package/src/hooks/pre-compact 2.sh +0 -151
  149. package/src/hooks/session-start 2.sh +0 -268
  150. package/src/hooks/session-stop 2.sh +0 -140
  151. package/src/kg_populate 2.py +0 -290
  152. package/src/knowledge_graph 2.py +0 -257
  153. package/src/maintenance 2.py +0 -59
  154. package/src/migrate_embeddings 2.py +0 -122
  155. package/src/plugin_loader 2.py +0 -202
  156. package/src/plugins/__init__ 2.py +0 -0
  157. package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
  158. package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
  159. package/src/plugins/__pycache__/__init__.cpython-314.pyc +0 -0
  160. package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
  161. package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
  162. package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
  163. package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
  164. package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
  165. package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
  166. package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
  167. package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
  168. package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
  169. package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
  170. package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
  171. package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
  172. package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
  173. package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
  174. package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
  175. package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
  176. package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
  177. package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
  178. package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
  179. package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
  180. package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
  181. package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
  182. package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
  183. package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
  184. package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
  185. package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
  186. package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
  187. package/src/plugins/__pycache__/schedule.cpython-310.pyc +0 -0
  188. package/src/plugins/__pycache__/schedule.cpython-314.pyc +0 -0
  189. package/src/plugins/__pycache__/skills.cpython-310.pyc +0 -0
  190. package/src/plugins/__pycache__/skills.cpython-314.pyc +0 -0
  191. package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
  192. package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
  193. package/src/plugins/adaptive_mode 2.py +0 -805
  194. package/src/plugins/agents 2.py +0 -52
  195. package/src/plugins/artifact_registry 2.py +0 -450
  196. package/src/plugins/backup 2.py +0 -104
  197. package/src/plugins/cognitive_memory 2.py +0 -564
  198. package/src/plugins/core_rules 2.py +0 -252
  199. package/src/plugins/cortex 2.py +0 -299
  200. package/src/plugins/entities 2.py +0 -67
  201. package/src/plugins/episodic_memory 2.py +0 -533
  202. package/src/plugins/evolution 2.py +0 -115
  203. package/src/plugins/guard 2.py +0 -746
  204. package/src/plugins/knowledge_graph_tools 2.py +0 -105
  205. package/src/plugins/preferences 2.py +0 -47
  206. package/src/plugins/update 2.py +0 -256
  207. package/src/requirements 2.txt +0 -12
  208. package/src/rules/__init__ 2.py +0 -0
  209. package/src/rules/core-rules 2.json +0 -331
  210. package/src/rules/migrate 2.py +0 -207
  211. package/src/scripts/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
  212. package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
  213. package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
  214. package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
  215. package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
  216. package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
  217. package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
  218. package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
  219. package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
  220. package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
  221. package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
  222. package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
  223. package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
  224. package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
  225. package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
  226. package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
  227. package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
  228. package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
  229. package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
  230. package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
  231. package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
  232. package/src/scripts/check-context 2.py +0 -264
  233. package/src/scripts/nexo-auto-update 2.py +0 -6
  234. package/src/scripts/nexo-backup 2.sh +0 -25
  235. package/src/scripts/nexo-brain-activation 2.sh +0 -140
  236. package/src/scripts/nexo-catchup 2.py +0 -242
  237. package/src/scripts/nexo-cognitive-decay 2.py +0 -182
  238. package/src/scripts/nexo-daily-self-audit 2.py +0 -552
  239. package/src/scripts/nexo-deep-sleep 2.sh +0 -97
  240. package/src/scripts/nexo-evolution-run 2.py +0 -597
  241. package/src/scripts/nexo-followup-hygiene 2.py +0 -112
  242. package/src/scripts/nexo-github-monitor 2.py +0 -256
  243. package/src/scripts/nexo-immune 2.py +0 -927
  244. package/src/scripts/nexo-inbox-hook 2.sh +0 -74
  245. package/src/scripts/nexo-install 2.py +0 -6
  246. package/src/scripts/nexo-learning-housekeep 2.py +0 -245
  247. package/src/scripts/nexo-learning-validator 2.py +0 -207
  248. package/src/scripts/nexo-migrate 2.py +0 -232
  249. package/src/scripts/nexo-postmortem-consolidator 2.py +0 -421
  250. package/src/scripts/nexo-pre-commit 2.py +0 -120
  251. package/src/scripts/nexo-prevent-sleep 2.sh +0 -29
  252. package/src/scripts/nexo-proactive-dashboard 2.py +0 -345
  253. package/src/scripts/nexo-reflection 2.py +0 -253
  254. package/src/scripts/nexo-runtime-preflight 2.py +0 -274
  255. package/src/scripts/nexo-send-email 2.py +0 -25
  256. package/src/scripts/nexo-send-email.py +0 -25
  257. package/src/scripts/nexo-send-reply 2.py +0 -178
  258. package/src/scripts/nexo-send-reply.py +0 -178
  259. package/src/scripts/nexo-sleep 2.py +0 -592
  260. package/src/scripts/nexo-snapshot-restore 2.sh +0 -35
  261. package/src/scripts/nexo-synthesis 2.py +0 -253
  262. package/src/scripts/nexo-tcc-approve 2.sh +0 -79
  263. package/src/scripts/nexo-update 2.sh +0 -161
  264. package/src/scripts/nexo-watchdog 2.sh +0 -878
  265. package/src/scripts/nexo-watchdog-smoke 2.py +0 -119
  266. package/src/server 2.py +0 -733
  267. package/src/storage_router 2.py +0 -32
  268. package/src/tools_coordination 2.py +0 -102
  269. package/src/tools_credentials 2.py +0 -68
  270. package/src/tools_learnings 2.py +0 -220
  271. package/src/tools_menu 2.py +0 -227
  272. package/src/tools_reminders 2.py +0 -86
  273. package/src/tools_reminders_crud 2.py +0 -159
  274. package/src/tools_sessions 2.py +0 -476
  275. package/src/tools_task_history 2.py +0 -57
  276. package/templates/CLAUDE.md 2.template +0 -63
  277. package/templates/openclaw 2.json +0 -13
  278. package/tests/__init__ 2.py +0 -0
  279. package/tests/__init__.py +0 -0
  280. package/tests/conftest 2.py +0 -71
  281. package/tests/conftest.py +0 -71
  282. package/tests/test_cognitive 2.py +0 -205
  283. package/tests/test_cognitive.py +0 -205
  284. package/tests/test_knowledge_graph 2.py +0 -140
  285. package/tests/test_knowledge_graph.py +0 -140
  286. package/tests/test_migrations 2.py +0 -137
  287. package/tests/test_migrations.py +0 -137
package/tests/conftest.py DELETED
@@ -1,71 +0,0 @@
1
- """Shared fixtures for NEXO test suite.
2
-
3
- Uses isolated temp databases so tests never touch production data.
4
- """
5
-
6
- import os
7
- import sys
8
- import sqlite3
9
-
10
- import pytest
11
-
12
- # Add src/ to path so we can import nexo modules
13
- sys.path.insert(0, os.path.join(os.path.dirname(__file__), "..", "src"))
14
-
15
-
16
- @pytest.fixture(autouse=True)
17
- def isolated_db(tmp_path, monkeypatch):
18
- """Redirect both nexo.db and cognitive.db to temp files per test."""
19
- test_db = str(tmp_path / "test_nexo.db")
20
- test_cog_db = str(tmp_path / "test_cognitive.db")
21
-
22
- monkeypatch.setenv("NEXO_TEST_DB", test_db)
23
- monkeypatch.setenv("NEXO_COGNITIVE_DB", test_cog_db)
24
- monkeypatch.setenv("NEXO_SKIP_FS_INDEX", "1")
25
-
26
- import db._core as db_core
27
- import cognitive._core as cog_core
28
-
29
- # Close existing connections
30
- db_core.close_db()
31
- if cog_core._conn is not None:
32
- try:
33
- cog_core._conn.close()
34
- except Exception:
35
- pass
36
- cog_core._conn = None
37
-
38
- # Point to temp paths
39
- db_core.DB_PATH = test_db
40
- cog_core.COGNITIVE_DB = test_cog_db
41
-
42
- # Create a fresh raw connection
43
- raw = sqlite3.connect(test_db, timeout=30, check_same_thread=False,
44
- isolation_level=None)
45
- raw.execute("PRAGMA journal_mode=WAL")
46
- raw.execute("PRAGMA busy_timeout=30000")
47
- raw.execute("PRAGMA foreign_keys=ON")
48
- raw.row_factory = sqlite3.Row
49
-
50
- wrapped = db_core._SerializedConnection(raw)
51
- db_core._shared_conn = wrapped
52
-
53
- # Initialize schemas
54
- from db._core import init_db
55
- from db._schema import run_migrations
56
- init_db()
57
- run_migrations()
58
-
59
- yield {
60
- "nexo_db": test_db,
61
- "cognitive_db": test_cog_db,
62
- }
63
-
64
- # Cleanup
65
- db_core.close_db()
66
- if cog_core._conn is not None:
67
- try:
68
- cog_core._conn.close()
69
- except Exception:
70
- pass
71
- cog_core._conn = None
@@ -1,205 +0,0 @@
1
- """Tests for cognitive engine: embeddings, cosine similarity, search, KG boost, decay."""
2
-
3
- import math
4
- import numpy as np
5
-
6
-
7
- def test_cosine_similarity_identical():
8
- """Identical vectors should have similarity 1.0."""
9
- import cognitive
10
- a = np.array([1.0, 2.0, 3.0], dtype=np.float32)
11
- assert abs(cognitive.cosine_similarity(a, a) - 1.0) < 1e-6
12
-
13
-
14
- def test_cosine_similarity_orthogonal():
15
- """Orthogonal vectors should have similarity 0.0."""
16
- import cognitive
17
- a = np.array([1.0, 0.0, 0.0], dtype=np.float32)
18
- b = np.array([0.0, 1.0, 0.0], dtype=np.float32)
19
- assert abs(cognitive.cosine_similarity(a, b)) < 1e-6
20
-
21
-
22
- def test_cosine_similarity_zero_norm():
23
- """Zero vector should return 0.0 (not NaN)."""
24
- import cognitive
25
- a = np.zeros(3, dtype=np.float32)
26
- b = np.array([1.0, 2.0, 3.0], dtype=np.float32)
27
- assert cognitive.cosine_similarity(a, b) == 0.0
28
- assert cognitive.cosine_similarity(b, a) == 0.0
29
-
30
-
31
- def test_blob_roundtrip():
32
- """Array → blob → array should be lossless."""
33
- import cognitive
34
- arr = np.random.randn(768).astype(np.float32)
35
- blob = cognitive._array_to_blob(arr)
36
- recovered = cognitive._blob_to_array(blob)
37
- np.testing.assert_array_equal(arr, recovered)
38
-
39
-
40
- def test_kg_boost_results_no_kg_data():
41
- """KG boost should be a no-op when there are no KG nodes."""
42
- import cognitive
43
- results = [
44
- {"source_type": "learning", "source_id": "L999", "score": 0.7},
45
- {"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
46
- ]
47
- boosted = cognitive._kg_boost_results(results)
48
- # No KG nodes in test DB → no boost applied
49
- assert boosted[0]["score"] == 0.7
50
- assert boosted[1]["score"] == 0.6
51
- assert "kg_boost" not in boosted[0]
52
-
53
-
54
- def test_kg_boost_results_with_connections():
55
- """KG boost should increase scores for connected nodes."""
56
- import cognitive
57
- import knowledge_graph as kg
58
-
59
- db = cognitive._get_db()
60
- # Initialize KG tables
61
- db.executescript("""
62
- CREATE TABLE IF NOT EXISTS kg_nodes (
63
- id INTEGER PRIMARY KEY AUTOINCREMENT,
64
- node_type TEXT NOT NULL,
65
- node_ref TEXT NOT NULL UNIQUE,
66
- label TEXT NOT NULL DEFAULT '',
67
- properties TEXT DEFAULT '{}'
68
- );
69
- CREATE TABLE IF NOT EXISTS kg_edges (
70
- id INTEGER PRIMARY KEY AUTOINCREMENT,
71
- source_id INTEGER NOT NULL,
72
- target_id INTEGER NOT NULL,
73
- relation TEXT NOT NULL DEFAULT '',
74
- weight REAL DEFAULT 1.0,
75
- confidence REAL DEFAULT 1.0,
76
- valid_from TEXT,
77
- valid_until TEXT,
78
- source_memory_id TEXT DEFAULT '',
79
- properties TEXT DEFAULT '{}'
80
- );
81
- """)
82
-
83
- # Create a learning node with 8 connections
84
- node_id = db.execute(
85
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
86
- ("learning", "learning:42", "Test Learning")
87
- ).lastrowid
88
-
89
- for i in range(8):
90
- file_id = db.execute(
91
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
92
- ("file", f"file:test{i}.py", f"test{i}.py")
93
- ).lastrowid
94
- db.execute(
95
- "INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
96
- (node_id, file_id, "touched", 1.0)
97
- )
98
- db.commit()
99
-
100
- results = [
101
- {"source_type": "learning", "source_id": "L42", "score": 0.6},
102
- {"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
103
- ]
104
- boosted = cognitive._kg_boost_results(results)
105
-
106
- # Learning with 8 edges should get boost
107
- assert boosted[0].get("kg_boost") is not None
108
- assert boosted[0]["score"] > 0.6
109
- expected_boost = min(0.08, 0.015 * math.log2(8 + 1))
110
- assert abs(boosted[0]["kg_boost"] - round(expected_boost, 4)) < 0.001
111
-
112
- # Sensory with no KG node should NOT get boost
113
- assert "kg_boost" not in boosted[1]
114
-
115
-
116
- def test_kg_boost_relevance_gate():
117
- """KG boost should not apply to low-relevance results (score < 0.45)."""
118
- import cognitive
119
-
120
- db = cognitive._get_db()
121
- db.executescript("""
122
- CREATE TABLE IF NOT EXISTS kg_nodes (
123
- id INTEGER PRIMARY KEY AUTOINCREMENT,
124
- node_type TEXT NOT NULL,
125
- node_ref TEXT NOT NULL UNIQUE,
126
- label TEXT NOT NULL DEFAULT '',
127
- properties TEXT DEFAULT '{}'
128
- );
129
- CREATE TABLE IF NOT EXISTS kg_edges (
130
- id INTEGER PRIMARY KEY AUTOINCREMENT,
131
- source_id INTEGER NOT NULL,
132
- target_id INTEGER NOT NULL,
133
- relation TEXT NOT NULL DEFAULT '',
134
- weight REAL DEFAULT 1.0,
135
- confidence REAL DEFAULT 1.0,
136
- valid_from TEXT,
137
- valid_until TEXT,
138
- source_memory_id TEXT DEFAULT '',
139
- properties TEXT DEFAULT '{}'
140
- );
141
- """)
142
-
143
- node_id = db.execute(
144
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
145
- ("learning", "learning:99", "Low Score Learning")
146
- ).lastrowid
147
- for i in range(20):
148
- fid = db.execute(
149
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
150
- ("file", f"file:low{i}.py", f"low{i}.py")
151
- ).lastrowid
152
- db.execute(
153
- "INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
154
- (node_id, fid, "touched", 1.0)
155
- )
156
- db.commit()
157
-
158
- results = [
159
- {"source_type": "learning", "source_id": "L99", "score": 0.3},
160
- ]
161
- boosted = cognitive._kg_boost_results(results)
162
- # Score below 0.45 gate → no boost
163
- assert boosted[0]["score"] == 0.3
164
- assert "kg_boost" not in boosted[0]
165
-
166
-
167
- def test_decay_formula():
168
- """Ebbinghaus decay should reduce strength over time (lambda operates on HOURS)."""
169
- import cognitive
170
- # STM decay: lambda=0.004126, after 7 days (168h) strength should be ~0.5
171
- initial = 1.0
172
- hours_7d = 7 * 24
173
- decayed = initial * math.exp(-cognitive.LAMBDA_STM * hours_7d)
174
- assert 0.45 < decayed < 0.55
175
-
176
- # LTM decay: lambda=0.000481, after 60 days (1440h) strength should be ~0.5
177
- hours_60d = 60 * 24
178
- decayed_ltm = initial * math.exp(-cognitive.LAMBDA_LTM * hours_60d)
179
- assert 0.45 < decayed_ltm < 0.55
180
-
181
-
182
- def test_apply_temporal_boost_historical():
183
- """Historical queries should get no temporal boost."""
184
- import cognitive
185
- results = [
186
- {"source_type": "learning", "source_id": "L1", "score": 0.7,
187
- "created_at": "2026-03-28T10:00:00"},
188
- ]
189
- boosted = cognitive._apply_temporal_boost(results, "what happened months ago")
190
- # Historical cue "months" should disable boost
191
- assert boosted[0]["score"] == 0.7
192
-
193
-
194
- def test_apply_temporal_boost_operational():
195
- """Operational queries should get a higher temporal boost."""
196
- import cognitive
197
- from datetime import datetime
198
- now_str = datetime.now().strftime("%Y-%m-%dT%H:%M:%S")
199
- results = [
200
- {"source_type": "learning", "source_id": "L1", "score": 0.7,
201
- "created_at": now_str},
202
- ]
203
- boosted = cognitive._apply_temporal_boost(results, "active backend issues today")
204
- # Very recent + operational → should get noticeable boost
205
- assert boosted[0]["score"] > 0.7
@@ -1,205 +0,0 @@
1
- """Tests for cognitive engine: embeddings, cosine similarity, search, KG boost, decay."""
2
-
3
- import math
4
- import numpy as np
5
-
6
-
7
- def test_cosine_similarity_identical():
8
- """Identical vectors should have similarity 1.0."""
9
- import cognitive
10
- a = np.array([1.0, 2.0, 3.0], dtype=np.float32)
11
- assert abs(cognitive.cosine_similarity(a, a) - 1.0) < 1e-6
12
-
13
-
14
- def test_cosine_similarity_orthogonal():
15
- """Orthogonal vectors should have similarity 0.0."""
16
- import cognitive
17
- a = np.array([1.0, 0.0, 0.0], dtype=np.float32)
18
- b = np.array([0.0, 1.0, 0.0], dtype=np.float32)
19
- assert abs(cognitive.cosine_similarity(a, b)) < 1e-6
20
-
21
-
22
- def test_cosine_similarity_zero_norm():
23
- """Zero vector should return 0.0 (not NaN)."""
24
- import cognitive
25
- a = np.zeros(3, dtype=np.float32)
26
- b = np.array([1.0, 2.0, 3.0], dtype=np.float32)
27
- assert cognitive.cosine_similarity(a, b) == 0.0
28
- assert cognitive.cosine_similarity(b, a) == 0.0
29
-
30
-
31
- def test_blob_roundtrip():
32
- """Array → blob → array should be lossless."""
33
- import cognitive
34
- arr = np.random.randn(768).astype(np.float32)
35
- blob = cognitive._array_to_blob(arr)
36
- recovered = cognitive._blob_to_array(blob)
37
- np.testing.assert_array_equal(arr, recovered)
38
-
39
-
40
- def test_kg_boost_results_no_kg_data():
41
- """KG boost should be a no-op when there are no KG nodes."""
42
- import cognitive
43
- results = [
44
- {"source_type": "learning", "source_id": "L999", "score": 0.7},
45
- {"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
46
- ]
47
- boosted = cognitive._kg_boost_results(results)
48
- # No KG nodes in test DB → no boost applied
49
- assert boosted[0]["score"] == 0.7
50
- assert boosted[1]["score"] == 0.6
51
- assert "kg_boost" not in boosted[0]
52
-
53
-
54
- def test_kg_boost_results_with_connections():
55
- """KG boost should increase scores for connected nodes."""
56
- import cognitive
57
- import knowledge_graph as kg
58
-
59
- db = cognitive._get_db()
60
- # Initialize KG tables
61
- db.executescript("""
62
- CREATE TABLE IF NOT EXISTS kg_nodes (
63
- id INTEGER PRIMARY KEY AUTOINCREMENT,
64
- node_type TEXT NOT NULL,
65
- node_ref TEXT NOT NULL UNIQUE,
66
- label TEXT NOT NULL DEFAULT '',
67
- properties TEXT DEFAULT '{}'
68
- );
69
- CREATE TABLE IF NOT EXISTS kg_edges (
70
- id INTEGER PRIMARY KEY AUTOINCREMENT,
71
- source_id INTEGER NOT NULL,
72
- target_id INTEGER NOT NULL,
73
- relation TEXT NOT NULL DEFAULT '',
74
- weight REAL DEFAULT 1.0,
75
- confidence REAL DEFAULT 1.0,
76
- valid_from TEXT,
77
- valid_until TEXT,
78
- source_memory_id TEXT DEFAULT '',
79
- properties TEXT DEFAULT '{}'
80
- );
81
- """)
82
-
83
- # Create a learning node with 8 connections
84
- node_id = db.execute(
85
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
86
- ("learning", "learning:42", "Test Learning")
87
- ).lastrowid
88
-
89
- for i in range(8):
90
- file_id = db.execute(
91
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
92
- ("file", f"file:test{i}.py", f"test{i}.py")
93
- ).lastrowid
94
- db.execute(
95
- "INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
96
- (node_id, file_id, "touched", 1.0)
97
- )
98
- db.commit()
99
-
100
- results = [
101
- {"source_type": "learning", "source_id": "L42", "score": 0.6},
102
- {"source_type": "sensory", "source_id": "buffer#1", "score": 0.6},
103
- ]
104
- boosted = cognitive._kg_boost_results(results)
105
-
106
- # Learning with 8 edges should get boost
107
- assert boosted[0].get("kg_boost") is not None
108
- assert boosted[0]["score"] > 0.6
109
- expected_boost = min(0.08, 0.015 * math.log2(8 + 1))
110
- assert abs(boosted[0]["kg_boost"] - round(expected_boost, 4)) < 0.001
111
-
112
- # Sensory with no KG node should NOT get boost
113
- assert "kg_boost" not in boosted[1]
114
-
115
-
116
- def test_kg_boost_relevance_gate():
117
- """KG boost should not apply to low-relevance results (score < 0.45)."""
118
- import cognitive
119
-
120
- db = cognitive._get_db()
121
- db.executescript("""
122
- CREATE TABLE IF NOT EXISTS kg_nodes (
123
- id INTEGER PRIMARY KEY AUTOINCREMENT,
124
- node_type TEXT NOT NULL,
125
- node_ref TEXT NOT NULL UNIQUE,
126
- label TEXT NOT NULL DEFAULT '',
127
- properties TEXT DEFAULT '{}'
128
- );
129
- CREATE TABLE IF NOT EXISTS kg_edges (
130
- id INTEGER PRIMARY KEY AUTOINCREMENT,
131
- source_id INTEGER NOT NULL,
132
- target_id INTEGER NOT NULL,
133
- relation TEXT NOT NULL DEFAULT '',
134
- weight REAL DEFAULT 1.0,
135
- confidence REAL DEFAULT 1.0,
136
- valid_from TEXT,
137
- valid_until TEXT,
138
- source_memory_id TEXT DEFAULT '',
139
- properties TEXT DEFAULT '{}'
140
- );
141
- """)
142
-
143
- node_id = db.execute(
144
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
145
- ("learning", "learning:99", "Low Score Learning")
146
- ).lastrowid
147
- for i in range(20):
148
- fid = db.execute(
149
- "INSERT INTO kg_nodes (node_type, node_ref, label) VALUES (?, ?, ?)",
150
- ("file", f"file:low{i}.py", f"low{i}.py")
151
- ).lastrowid
152
- db.execute(
153
- "INSERT INTO kg_edges (source_id, target_id, relation, weight) VALUES (?, ?, ?, ?)",
154
- (node_id, fid, "touched", 1.0)
155
- )
156
- db.commit()
157
-
158
- results = [
159
- {"source_type": "learning", "source_id": "L99", "score": 0.3},
160
- ]
161
- boosted = cognitive._kg_boost_results(results)
162
- # Score below 0.45 gate → no boost
163
- assert boosted[0]["score"] == 0.3
164
- assert "kg_boost" not in boosted[0]
165
-
166
-
167
- def test_decay_formula():
168
- """Ebbinghaus decay should reduce strength over time (lambda operates on HOURS)."""
169
- import cognitive
170
- # STM decay: lambda=0.004126, after 7 days (168h) strength should be ~0.5
171
- initial = 1.0
172
- hours_7d = 7 * 24
173
- decayed = initial * math.exp(-cognitive.LAMBDA_STM * hours_7d)
174
- assert 0.45 < decayed < 0.55
175
-
176
- # LTM decay: lambda=0.000481, after 60 days (1440h) strength should be ~0.5
177
- hours_60d = 60 * 24
178
- decayed_ltm = initial * math.exp(-cognitive.LAMBDA_LTM * hours_60d)
179
- assert 0.45 < decayed_ltm < 0.55
180
-
181
-
182
- def test_apply_temporal_boost_historical():
183
- """Historical queries should get no temporal boost."""
184
- import cognitive
185
- results = [
186
- {"source_type": "learning", "source_id": "L1", "score": 0.7,
187
- "created_at": "2026-03-28T10:00:00"},
188
- ]
189
- boosted = cognitive._apply_temporal_boost(results, "what happened months ago")
190
- # Historical cue "months" should disable boost
191
- assert boosted[0]["score"] == 0.7
192
-
193
-
194
- def test_apply_temporal_boost_operational():
195
- """Operational queries should get a higher temporal boost."""
196
- import cognitive
197
- from datetime import datetime
198
- now_str = datetime.now().strftime("%Y-%m-%dT%H:%M:%S")
199
- results = [
200
- {"source_type": "learning", "source_id": "L1", "score": 0.7,
201
- "created_at": now_str},
202
- ]
203
- boosted = cognitive._apply_temporal_boost(results, "active backend issues today")
204
- # Very recent + operational → should get noticeable boost
205
- assert boosted[0]["score"] > 0.7
@@ -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)