nexo-brain 2.0.0 → 2.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (370) hide show
  1. package/README.md +143 -44
  2. package/bin/nexo-brain.js +53 -26
  3. package/package.json +15 -3
  4. package/scripts/migrate-to-unified 2.sh +813 -0
  5. package/scripts/migrate-v1.5-to-v1.6 2.py +778 -0
  6. package/scripts/migrate-v1.7-to-v1.8 2.py +214 -0
  7. package/scripts/pre-commit-check 2.sh +55 -0
  8. package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
  9. package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
  10. package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
  11. package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
  12. package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
  13. package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
  14. package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
  15. package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
  16. package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
  17. package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
  18. package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
  19. package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
  20. package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
  21. package/src/auto_close_sessions 2.py +159 -0
  22. package/src/auto_update 2.py +634 -0
  23. package/src/claim_graph 2.py +323 -0
  24. package/src/cognitive/__init__ 2.py +62 -0
  25. package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
  26. package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
  27. package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
  28. package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
  29. package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
  30. package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
  31. package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
  32. package/src/cognitive/_core 2.py +567 -0
  33. package/src/cognitive/_decay 2.py +382 -0
  34. package/src/cognitive/_ingest 2.py +892 -0
  35. package/src/cognitive/_memory 2.py +912 -0
  36. package/src/cognitive/_search 2.py +949 -0
  37. package/src/cognitive/_trust 2.py +464 -0
  38. package/src/cognitive/_trust.py +10 -36
  39. package/src/crons/manifest 2.json +106 -0
  40. package/src/crons/manifest.json +106 -0
  41. package/src/crons/sync 2.py +217 -0
  42. package/src/crons/sync.py +217 -0
  43. package/src/dashboard/__init__ 2.py +0 -0
  44. package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
  45. package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
  46. package/src/dashboard/app 2.py +789 -0
  47. package/src/dashboard/app.py +16 -2
  48. package/src/dashboard/templates/dashboard.html +3 -2
  49. package/src/db/__init__ 2.py +89 -0
  50. package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
  51. package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
  52. package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
  53. package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
  54. package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
  55. package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
  56. package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
  57. package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
  58. package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
  59. package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
  60. package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
  61. package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
  62. package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
  63. package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
  64. package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
  65. package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
  66. package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
  67. package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
  68. package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
  69. package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
  70. package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
  71. package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
  72. package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
  73. package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
  74. package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
  75. package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
  76. package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
  77. package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
  78. package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
  79. package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
  80. package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
  81. package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
  82. package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
  83. package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
  84. package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
  85. package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
  86. package/src/db/_core 2.py +417 -0
  87. package/src/db/_credentials 2.py +124 -0
  88. package/src/db/_entities 2.py +178 -0
  89. package/src/db/_episodic 2.py +738 -0
  90. package/src/db/_episodic.py +1 -1
  91. package/src/db/_evolution 2.py +54 -0
  92. package/src/db/_fts 2.py +406 -0
  93. package/src/db/_learnings 2.py +168 -0
  94. package/src/db/_reminders 2.py +338 -0
  95. package/src/db/_reminders.py +9 -5
  96. package/src/db/_schema 2.py +364 -0
  97. package/src/db/_sessions 2.py +300 -0
  98. package/src/db/_tasks 2.py +91 -0
  99. package/src/evolution_cycle 2.py +266 -0
  100. package/src/hnsw_index 2.py +254 -0
  101. package/src/hooks/auto_capture 2.py +208 -0
  102. package/src/hooks/caffeinate-guard 2.sh +8 -0
  103. package/src/hooks/capture-session 2.sh +21 -0
  104. package/src/hooks/capture-session.sh +2 -0
  105. package/src/hooks/capture-tool-logs 2.sh +127 -0
  106. package/src/hooks/capture-tool-logs.sh +3 -2
  107. package/src/hooks/daily-briefing-check 2.sh +33 -0
  108. package/src/hooks/inbox-hook 2.sh +76 -0
  109. package/src/hooks/inbox-hook.sh +3 -2
  110. package/src/hooks/post-compact 2.sh +148 -0
  111. package/src/hooks/post-compact.sh +1 -1
  112. package/src/hooks/pre-compact 2.sh +151 -0
  113. package/src/hooks/pre-compact.sh +1 -1
  114. package/src/hooks/session-start 2.sh +268 -0
  115. package/src/hooks/session-start.sh +6 -3
  116. package/src/hooks/session-stop 2.sh +140 -0
  117. package/src/hooks/session-stop.sh +3 -2
  118. package/src/kg_populate 2.py +290 -0
  119. package/src/knowledge_graph 2.py +257 -0
  120. package/src/maintenance 2.py +59 -0
  121. package/src/migrate_embeddings 2.py +122 -0
  122. package/src/plugin_loader 2.py +202 -0
  123. package/src/plugins/__init__ 2.py +0 -0
  124. package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
  125. package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
  126. package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
  127. package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
  128. package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
  129. package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
  130. package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
  131. package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
  132. package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
  133. package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
  134. package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
  135. package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
  136. package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
  137. package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
  138. package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
  139. package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
  140. package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
  141. package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
  142. package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
  143. package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
  144. package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
  145. package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
  146. package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
  147. package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
  148. package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
  149. package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
  150. package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
  151. package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
  152. package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
  153. package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
  154. package/src/plugins/adaptive_mode 2.py +805 -0
  155. package/src/plugins/agents 2.py +52 -0
  156. package/src/plugins/artifact_registry 2.py +450 -0
  157. package/src/plugins/backup 2.py +104 -0
  158. package/src/plugins/cognitive_memory 2.py +564 -0
  159. package/src/plugins/core_rules 2.py +252 -0
  160. package/src/plugins/core_rules.py +34 -17
  161. package/src/plugins/cortex 2.py +299 -0
  162. package/src/plugins/entities 2.py +67 -0
  163. package/src/plugins/episodic_memory 2.py +533 -0
  164. package/src/plugins/evolution 2.py +115 -0
  165. package/src/plugins/guard 2.py +746 -0
  166. package/src/plugins/knowledge_graph_tools 2.py +105 -0
  167. package/src/plugins/preferences 2.py +47 -0
  168. package/src/plugins/update 2.py +256 -0
  169. package/src/plugins/update.py +18 -0
  170. package/src/requirements 2.txt +12 -0
  171. package/src/rules/__init__ 2.py +0 -0
  172. package/src/rules/core-rules 2.json +331 -0
  173. package/src/rules/migrate 2.py +207 -0
  174. package/src/scripts/check-context 2.py +264 -0
  175. package/src/scripts/check-context.py +4 -7
  176. package/src/scripts/deep-sleep/apply_findings.py +570 -167
  177. package/src/scripts/deep-sleep/collect.py +480 -0
  178. package/src/scripts/deep-sleep/extract-prompt.md +233 -0
  179. package/src/scripts/deep-sleep/extract.py +249 -0
  180. package/src/scripts/deep-sleep/synthesize-prompt.md +197 -0
  181. package/src/scripts/deep-sleep/synthesize.py +191 -0
  182. package/src/scripts/nexo-auto-update 2.py +6 -0
  183. package/src/scripts/nexo-backup 2.sh +25 -0
  184. package/src/scripts/nexo-brain-activation 2.sh +140 -0
  185. package/src/scripts/nexo-catchup 2.py +242 -0
  186. package/src/scripts/nexo-catchup.py +5 -8
  187. package/src/scripts/nexo-cognitive-decay 2.py +182 -0
  188. package/src/scripts/nexo-daily-self-audit 2.py +552 -0
  189. package/src/scripts/nexo-daily-self-audit.py +28 -19
  190. package/src/scripts/nexo-deep-sleep 2.sh +97 -0
  191. package/src/scripts/nexo-deep-sleep.sh +31 -16
  192. package/src/scripts/nexo-evolution-run 2.py +597 -0
  193. package/src/scripts/nexo-evolution-run.py +5 -20
  194. package/src/scripts/nexo-followup-hygiene 2.py +112 -0
  195. package/src/scripts/nexo-followup-hygiene.py +4 -2
  196. package/src/scripts/nexo-github-monitor 2.py +256 -0
  197. package/src/scripts/nexo-github-monitor.py +6 -9
  198. package/src/scripts/nexo-immune 2.py +927 -0
  199. package/src/scripts/nexo-immune.py +4 -17
  200. package/src/scripts/nexo-inbox-hook 2.sh +74 -0
  201. package/src/scripts/nexo-install 2.py +6 -0
  202. package/src/scripts/nexo-learning-housekeep 2.py +245 -0
  203. package/src/scripts/nexo-learning-validator 2.py +207 -0
  204. package/src/scripts/nexo-learning-validator.py +0 -29
  205. package/src/scripts/nexo-migrate 2.py +232 -0
  206. package/src/scripts/nexo-postmortem-consolidator 2.py +421 -0
  207. package/src/scripts/nexo-postmortem-consolidator.py +9 -20
  208. package/src/scripts/nexo-pre-commit 2.py +120 -0
  209. package/src/scripts/nexo-prevent-sleep 2.sh +29 -0
  210. package/src/scripts/nexo-proactive-dashboard 2.py +345 -0
  211. package/src/scripts/nexo-proactive-dashboard.py +1 -0
  212. package/src/scripts/nexo-reflection 2.py +253 -0
  213. package/src/scripts/nexo-runtime-preflight 2.py +274 -0
  214. package/src/scripts/nexo-send-email 2.py +25 -0
  215. package/src/scripts/nexo-send-reply 2.py +178 -0
  216. package/src/scripts/nexo-sleep 2.py +592 -0
  217. package/src/scripts/nexo-sleep.py +8 -18
  218. package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
  219. package/src/scripts/nexo-synthesis 2.py +253 -0
  220. package/src/scripts/nexo-synthesis.py +8 -19
  221. package/src/scripts/nexo-tcc-approve 2.sh +79 -0
  222. package/src/scripts/nexo-update 2.sh +161 -0
  223. package/src/scripts/nexo-watchdog 2.sh +878 -0
  224. package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
  225. package/src/server 2.py +733 -0
  226. package/src/server.py +6 -1
  227. package/src/storage_router 2.py +32 -0
  228. package/src/tools_coordination 2.py +102 -0
  229. package/src/tools_credentials 2.py +68 -0
  230. package/src/tools_learnings 2.py +220 -0
  231. package/src/tools_menu 2.py +227 -0
  232. package/src/tools_menu.py +1 -1
  233. package/src/tools_reminders 2.py +86 -0
  234. package/src/tools_reminders_crud 2.py +159 -0
  235. package/src/tools_reminders_crud.py +7 -0
  236. package/src/tools_sessions 2.py +476 -0
  237. package/src/tools_sessions.py +67 -0
  238. package/src/tools_task_history 2.py +57 -0
  239. package/templates/CLAUDE.md 2.template +63 -0
  240. package/templates/openclaw 2.json +13 -0
  241. package/tests/__init__ 2.py +0 -0
  242. package/tests/conftest 2.py +71 -0
  243. package/tests/test_cognitive 2.py +205 -0
  244. package/tests/test_knowledge_graph 2.py +140 -0
  245. package/tests/test_migrations 2.py +137 -0
  246. package/src/__pycache__/auto_close_sessions.cpython-310.pyc +0 -0
  247. package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
  248. package/src/__pycache__/auto_update.cpython-314.pyc +0 -0
  249. package/src/__pycache__/claim_graph.cpython-310.pyc +0 -0
  250. package/src/__pycache__/claim_graph.cpython-314.pyc +0 -0
  251. package/src/__pycache__/evolution_cycle.cpython-310.pyc +0 -0
  252. package/src/__pycache__/evolution_cycle.cpython-314.pyc +0 -0
  253. package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
  254. package/src/__pycache__/kg_populate.cpython-314.pyc +0 -0
  255. package/src/__pycache__/knowledge_graph.cpython-314.pyc +0 -0
  256. package/src/__pycache__/maintenance.cpython-310.pyc +0 -0
  257. package/src/__pycache__/maintenance.cpython-314.pyc +0 -0
  258. package/src/__pycache__/migrate_embeddings.cpython-310.pyc +0 -0
  259. package/src/__pycache__/migrate_embeddings.cpython-314.pyc +0 -0
  260. package/src/__pycache__/plugin_loader.cpython-314.pyc +0 -0
  261. package/src/__pycache__/server.cpython-310.pyc +0 -0
  262. package/src/__pycache__/server.cpython-314.pyc +0 -0
  263. package/src/__pycache__/storage_router.cpython-310.pyc +0 -0
  264. package/src/__pycache__/storage_router.cpython-314.pyc +0 -0
  265. package/src/__pycache__/tools_coordination.cpython-314.pyc +0 -0
  266. package/src/__pycache__/tools_credentials.cpython-314.pyc +0 -0
  267. package/src/__pycache__/tools_learnings.cpython-314.pyc +0 -0
  268. package/src/__pycache__/tools_menu.cpython-314.pyc +0 -0
  269. package/src/__pycache__/tools_reminders.cpython-314.pyc +0 -0
  270. package/src/__pycache__/tools_reminders_crud.cpython-314.pyc +0 -0
  271. package/src/__pycache__/tools_sessions.cpython-314.pyc +0 -0
  272. package/src/__pycache__/tools_task_history.cpython-314.pyc +0 -0
  273. package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
  274. package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
  275. package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
  276. package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
  277. package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
  278. package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
  279. package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
  280. package/src/dashboard/__pycache__/__init__.cpython-314.pyc +0 -0
  281. package/src/dashboard/__pycache__/app.cpython-314.pyc +0 -0
  282. package/src/hooks/__pycache__/auto_capture.cpython-310.pyc +0 -0
  283. package/src/hooks/__pycache__/auto_capture.cpython-314.pyc +0 -0
  284. package/src/plugins/__pycache__/__init__.cpython-314.pyc +0 -0
  285. package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
  286. package/src/plugins/__pycache__/agents.cpython-314.pyc +0 -0
  287. package/src/plugins/__pycache__/artifact_registry.cpython-314.pyc +0 -0
  288. package/src/plugins/__pycache__/backup.cpython-314.pyc +0 -0
  289. package/src/plugins/__pycache__/cognitive_memory.cpython-314.pyc +0 -0
  290. package/src/plugins/__pycache__/core_rules.cpython-314.pyc +0 -0
  291. package/src/plugins/__pycache__/cortex.cpython-314.pyc +0 -0
  292. package/src/plugins/__pycache__/entities.cpython-314.pyc +0 -0
  293. package/src/plugins/__pycache__/episodic_memory.cpython-314.pyc +0 -0
  294. package/src/plugins/__pycache__/evolution.cpython-314.pyc +0 -0
  295. package/src/plugins/__pycache__/guard.cpython-314.pyc +0 -0
  296. package/src/plugins/__pycache__/knowledge_graph_tools.cpython-314.pyc +0 -0
  297. package/src/plugins/__pycache__/preferences.cpython-314.pyc +0 -0
  298. package/src/rules/__pycache__/__init__.cpython-310.pyc +0 -0
  299. package/src/rules/__pycache__/__init__.cpython-314.pyc +0 -0
  300. package/src/rules/__pycache__/migrate.cpython-310.pyc +0 -0
  301. package/src/rules/__pycache__/migrate.cpython-314.pyc +0 -0
  302. package/src/scripts/__pycache__/check-context.cpython-310.pyc +0 -0
  303. package/src/scripts/__pycache__/check-context.cpython-314.pyc +0 -0
  304. package/src/scripts/__pycache__/nexo-auto-update.cpython-310.pyc +0 -0
  305. package/src/scripts/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
  306. package/src/scripts/__pycache__/nexo-catchup.cpython-310.pyc +0 -0
  307. package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
  308. package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-310.pyc +0 -0
  309. package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
  310. package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-310.pyc +0 -0
  311. package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
  312. package/src/scripts/__pycache__/nexo-evolution-run.cpython-310.pyc +0 -0
  313. package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
  314. package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-310.pyc +0 -0
  315. package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
  316. package/src/scripts/__pycache__/nexo-github-monitor.cpython-310.pyc +0 -0
  317. package/src/scripts/__pycache__/nexo-github-monitor.cpython-314.pyc +0 -0
  318. package/src/scripts/__pycache__/nexo-immune.cpython-310.pyc +0 -0
  319. package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
  320. package/src/scripts/__pycache__/nexo-install.cpython-310.pyc +0 -0
  321. package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
  322. package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-310.pyc +0 -0
  323. package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
  324. package/src/scripts/__pycache__/nexo-learning-validator.cpython-310.pyc +0 -0
  325. package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
  326. package/src/scripts/__pycache__/nexo-migrate.cpython-310.pyc +0 -0
  327. package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
  328. package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-310.pyc +0 -0
  329. package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
  330. package/src/scripts/__pycache__/nexo-pre-commit.cpython-310.pyc +0 -0
  331. package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
  332. package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-310.pyc +0 -0
  333. package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
  334. package/src/scripts/__pycache__/nexo-reflection.cpython-310.pyc +0 -0
  335. package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
  336. package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-310.pyc +0 -0
  337. package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
  338. package/src/scripts/__pycache__/nexo-send-email.cpython-310.pyc +0 -0
  339. package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
  340. package/src/scripts/__pycache__/nexo-send-reply.cpython-310.pyc +0 -0
  341. package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
  342. package/src/scripts/__pycache__/nexo-sleep.cpython-310.pyc +0 -0
  343. package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
  344. package/src/scripts/__pycache__/nexo-synthesis.cpython-310.pyc +0 -0
  345. package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
  346. package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-310.pyc +0 -0
  347. package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
  348. package/src/scripts/deep-sleep/__pycache__/analyze_session.cpython-310.pyc +0 -0
  349. package/src/scripts/deep-sleep/__pycache__/analyze_session.cpython-314.pyc +0 -0
  350. package/src/scripts/deep-sleep/__pycache__/apply_findings.cpython-310.pyc +0 -0
  351. package/src/scripts/deep-sleep/__pycache__/apply_findings.cpython-314.pyc +0 -0
  352. package/src/scripts/deep-sleep/__pycache__/collect_transcripts.cpython-310.pyc +0 -0
  353. package/src/scripts/deep-sleep/__pycache__/collect_transcripts.cpython-314.pyc +0 -0
  354. package/src/scripts/deep-sleep/analyze_session.py +0 -217
  355. package/src/scripts/deep-sleep/collect_transcripts.py +0 -145
  356. package/src/scripts/deep-sleep/prompt.md +0 -109
  357. package/tests/__pycache__/__init__.cpython-310.pyc +0 -0
  358. package/tests/__pycache__/__init__.cpython-314.pyc +0 -0
  359. package/tests/__pycache__/conftest.cpython-310-pytest-9.0.2.pyc +0 -0
  360. package/tests/__pycache__/conftest.cpython-310.pyc +0 -0
  361. package/tests/__pycache__/conftest.cpython-314-pytest-9.0.2.pyc +0 -0
  362. package/tests/__pycache__/test_cognitive.cpython-310-pytest-9.0.2.pyc +0 -0
  363. package/tests/__pycache__/test_cognitive.cpython-310.pyc +0 -0
  364. package/tests/__pycache__/test_cognitive.cpython-314-pytest-9.0.2.pyc +0 -0
  365. package/tests/__pycache__/test_knowledge_graph.cpython-310-pytest-9.0.2.pyc +0 -0
  366. package/tests/__pycache__/test_knowledge_graph.cpython-310.pyc +0 -0
  367. package/tests/__pycache__/test_knowledge_graph.cpython-314-pytest-9.0.2.pyc +0 -0
  368. package/tests/__pycache__/test_migrations.cpython-310-pytest-9.0.2.pyc +0 -0
  369. package/tests/__pycache__/test_migrations.cpython-310.pyc +0 -0
  370. package/tests/__pycache__/test_migrations.cpython-314-pytest-9.0.2.pyc +0 -0
@@ -113,7 +113,7 @@ def consolidate_with_cli(data: dict) -> bool:
113
113
 
114
114
  diaries_with_critique = [
115
115
  d for d in data["diaries"]
116
- if d.get("self_critique") and not (d["self_critique"] or "").strip().lower().startswith("no self-critique")
116
+ if d.get("self_critique") and not str(d["self_critique"]).strip().lower().startswith("no self-critique")
117
117
  ]
118
118
 
119
119
  if not diaries_with_critique:
@@ -125,8 +125,10 @@ def consolidate_with_cli(data: dict) -> bool:
125
125
  if len(diaries_json) > 12000:
126
126
  diaries_json = diaries_json[:12000] + "\n... (truncated)"
127
127
 
128
- prompt = f"""You are NEXO's nightly consolidator. Your job is to review the self-critiques
129
- from today and decide which deserve to become permanent rules (feedback_postmortem_*.md).
128
+ prompt = f"""FIRST: Call nexo_startup(task='nightly postmortem consolidation') to register this session.
129
+
130
+ You are NEXO's nightly consolidator. Your job is to review the self-critiques
131
+ from today and decide which deserve to become permanent rules. Use nexo_learning_add for permanent rules and nexo_followup_create for action items.
130
132
 
131
133
  DATE: {data['date']}
132
134
  SESSIONS TODAY: {len(data['diaries'])} total, {len(diaries_with_critique)} with self-critique
@@ -185,30 +187,17 @@ INSTRUCTIONS:
185
187
  Execute without asking."""
186
188
 
187
189
  log(f"Stage 2: Invoking Claude CLI (opus) with {len(diaries_with_critique)} critiques...")
188
-
189
- # Verify Claude CLI is authenticated before calling
190
- try:
191
- auth_check = subprocess.run(
192
- [str(CLAUDE_CLI), "-p", "Reply with exactly: ok", "--bare", "--output-format", "text", "--model", "haiku"],
193
- capture_output=True, text=True, timeout=15
194
- )
195
- if auth_check.returncode != 0:
196
- log("Stage 2: Claude CLI not available or not authenticated. Skipping Stage 2.")
197
- return False
198
- except Exception:
199
- log("Stage 2: Claude CLI check failed. Skipping Stage 2.")
200
- return False
201
-
202
190
  env = os.environ.copy()
191
+ env["NEXO_HEADLESS"] = "1" # Skip stop hook post-mortem
203
192
  env.pop("CLAUDECODE", None)
204
193
  env.pop("CLAUDE_CODE", None)
205
194
 
206
195
  try:
207
196
  result = subprocess.run(
208
197
  [str(CLAUDE_CLI), "-p", prompt, "--model", "opus",
209
- "--output-format", "text", "--bare",
210
- "--allowedTools", "Read,Write,Edit,Glob,Grep"],
211
- capture_output=True, text=True, timeout=300, env=env
198
+ "--output-format", "text",
199
+ "--allowedTools", "Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*"],
200
+ capture_output=True, text=True, timeout=21600, env=env
212
201
  )
213
202
 
214
203
  if result.returncode != 0:
@@ -0,0 +1,120 @@
1
+ #!/usr/bin/env /Library/Frameworks/Python.framework/Versions/3.12/bin/python3
2
+ """
3
+ NEXO Pre-Commit Validator — Dynamic version
4
+ Queries nexo.db learnings to surface relevant warnings/blockers before commit.
5
+ Runs standalone (no MCP dependency, just sqlite3).
6
+ """
7
+ import os
8
+ import sqlite3
9
+ import subprocess
10
+ import sys
11
+ from pathlib import Path
12
+
13
+ NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
14
+ NEXO_DB = NEXO_HOME / "data" / "nexo.db"
15
+
16
+
17
+ def get_staged_files():
18
+ """Get list of staged files."""
19
+ result = subprocess.run(
20
+ ['git', 'diff', '--cached', '--name-only'],
21
+ capture_output=True, text=True
22
+ )
23
+ return [f for f in result.stdout.strip().split('\n') if f]
24
+
25
+
26
+ def check_file(filepath, conn):
27
+ """Dynamic checks from learnings DB."""
28
+ errors, warnings = [], []
29
+
30
+ filename = Path(filepath).name
31
+ parent_dir = Path(filepath).parent.name
32
+
33
+ # 1. Find learnings mentioning this file or directory
34
+ file_learnings = conn.execute(
35
+ "SELECT id, title, content FROM learnings WHERE INSTR(content, ?) > 0 OR INSTR(content, ?) > 0",
36
+ (filename, parent_dir)
37
+ ).fetchall()
38
+
39
+ # 2. Check repetition count for each matching learning
40
+ for row in file_learnings:
41
+ lid, title = row[0], row[1]
42
+ rep_count = conn.execute(
43
+ "SELECT COUNT(*) FROM error_repetitions WHERE original_learning_id = ?",
44
+ (lid,)
45
+ ).fetchone()[0]
46
+
47
+ if rep_count >= 5:
48
+ errors.append(f"BLOCKED #{lid} ({rep_count}x repeated): {title[:80]}")
49
+ elif rep_count >= 3:
50
+ warnings.append(f"WARNING #{lid} ({rep_count}x repeated): {title[:80]}")
51
+
52
+ # 3. Universal rules (SIEMPRE/NUNCA/ANTES) matching this file
53
+ universal = conn.execute(
54
+ "SELECT id, title, content FROM learnings WHERE "
55
+ "(content LIKE '%SIEMPRE%' OR content LIKE '%NUNCA%' OR content LIKE '%ANTES%') "
56
+ "AND (INSTR(content, ?) > 0 OR INSTR(content, ?) > 0)",
57
+ (filename, parent_dir)
58
+ ).fetchall()
59
+
60
+ for row in universal:
61
+ lid, title = row[0], row[1]
62
+ # Don't duplicate if already found above
63
+ if not any(f"#{lid}" in e for e in errors + warnings):
64
+ warnings.append(f"RULE #{lid}: {title[:80]}")
65
+
66
+ return errors, warnings
67
+
68
+
69
+ def main():
70
+ """Run pre-commit validation."""
71
+ staged = get_staged_files()
72
+ if not staged:
73
+ print("No staged files to check")
74
+ return 0
75
+
76
+ if not NEXO_DB.exists():
77
+ print("nexo.db not found — skipping dynamic checks")
78
+ return 0
79
+
80
+ conn = sqlite3.connect(str(NEXO_DB), timeout=5)
81
+
82
+ all_errors = {}
83
+ all_warnings = {}
84
+
85
+ for filepath in staged:
86
+ errors, warnings = check_file(filepath, conn)
87
+ if errors:
88
+ all_errors[filepath] = errors
89
+ if warnings:
90
+ all_warnings[filepath] = warnings
91
+
92
+ conn.close()
93
+
94
+ # Print warnings (non-blocking)
95
+ if all_warnings:
96
+ print("\nWARNINGS:")
97
+ for filepath, warns in all_warnings.items():
98
+ print(f"\n {filepath}:")
99
+ for w in warns:
100
+ print(f" - {w}")
101
+
102
+ # Print errors (blocking)
103
+ if all_errors:
104
+ print("\nBLOCKED — Fix these before committing:\n")
105
+ for filepath, errs in all_errors.items():
106
+ print(f" {filepath}:")
107
+ for e in errs:
108
+ print(f" - {e}")
109
+ print()
110
+ return 1
111
+
112
+ if all_warnings:
113
+ print("\nPre-commit passed with warnings\n")
114
+ else:
115
+ print("Pre-commit validation passed")
116
+ return 0
117
+
118
+
119
+ if __name__ == '__main__':
120
+ sys.exit(main())
@@ -0,0 +1,29 @@
1
+ #!/bin/bash
2
+ # NEXO Prevent Sleep — keeps the machine awake so nocturnal processes run.
3
+ #
4
+ # macOS: uses caffeinate -s -i (prevent system + idle sleep)
5
+ # Linux: uses systemd-inhibit or caffeine if available, otherwise no-op
6
+ #
7
+ # Run as LaunchAgent (KeepAlive) or systemd service.
8
+
9
+ case "$(uname -s)" in
10
+ Darwin)
11
+ exec caffeinate -s -i -w $$
12
+ ;;
13
+ Linux)
14
+ if command -v systemd-inhibit &>/dev/null; then
15
+ exec systemd-inhibit --what=idle:sleep --who=nexo --why="NEXO nocturnal processes" sleep infinity
16
+ elif command -v caffeine &>/dev/null; then
17
+ exec caffeine
18
+ else
19
+ echo "[NEXO] No sleep prevention tool found. Install systemd-inhibit or caffeine."
20
+ echo "[NEXO] Nocturnal processes may not run if the system sleeps."
21
+ # Keep running so launchd/systemd doesn't restart loop
22
+ exec sleep infinity
23
+ fi
24
+ ;;
25
+ *)
26
+ echo "[NEXO] Unsupported platform: $(uname -s)"
27
+ exit 1
28
+ ;;
29
+ esac
@@ -0,0 +1,345 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ NEXO Proactive Dashboard — Surfaces issues and opportunities without the user asking.
4
+
5
+ Scans: overdue followups, forgotten reminders, unresolved learnings,
6
+ inactive systems, user patterns, and more.
7
+
8
+ Usage:
9
+ python3 nexo-proactive-dashboard.py # Full scan, text output
10
+ python3 nexo-proactive-dashboard.py --json # JSON output for programmatic use
11
+ python3 nexo-proactive-dashboard.py --brief # One-liner alerts only
12
+ """
13
+
14
+ import json
15
+ import os
16
+ import sqlite3
17
+ import subprocess
18
+ import sys
19
+ from datetime import datetime, timedelta
20
+ from pathlib import Path
21
+
22
+ NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
23
+
24
+ NEXO_DB = NEXO_HOME / "data" / "nexo.db"
25
+
26
+
27
+ def get_db():
28
+ conn = sqlite3.connect(str(NEXO_DB), timeout=10)
29
+ conn.row_factory = sqlite3.Row
30
+ return conn
31
+
32
+
33
+ def check_overdue_followups() -> list[dict]:
34
+ """Find followups that are overdue and not completed."""
35
+ conn = get_db()
36
+ now_epoch = datetime.now().timestamp()
37
+ rows = conn.execute("""
38
+ SELECT id, description, date, created_at, reasoning
39
+ FROM followups
40
+ WHERE status NOT LIKE 'COMPLETED%'
41
+ AND status NOT IN ('DELETED','archived','blocked','waiting')
42
+ AND date IS NOT NULL AND date != ''
43
+ ORDER BY date ASC
44
+ """).fetchall()
45
+ conn.close()
46
+ alerts = []
47
+ for r in rows:
48
+ due_str = r["date"]
49
+ try:
50
+ due = datetime.fromisoformat(due_str) if due_str else None
51
+ if due and due < datetime.now():
52
+ days_overdue = (datetime.now() - due).days
53
+ alerts.append({
54
+ "type": "overdue_followup",
55
+ "severity": "high" if days_overdue > 3 else "medium",
56
+ "title": f"Followup overdue by {days_overdue}d: {r['description'][:80]}",
57
+ "id": r["id"],
58
+ "days_overdue": days_overdue,
59
+ })
60
+ except (ValueError, TypeError):
61
+ pass
62
+ return alerts
63
+
64
+
65
+ def check_overdue_reminders() -> list[dict]:
66
+ """Find reminders that are overdue."""
67
+ conn = get_db()
68
+ rows = conn.execute("""
69
+ SELECT id, description, date, status
70
+ FROM reminders
71
+ WHERE status NOT IN ('COMPLETED', 'CANCELLED')
72
+ AND date IS NOT NULL AND date != ''
73
+ ORDER BY date ASC
74
+ """).fetchall()
75
+ conn.close()
76
+ alerts = []
77
+ for r in rows:
78
+ due_str = r["date"]
79
+ try:
80
+ due = datetime.fromisoformat(due_str) if due_str else None
81
+ if due and due < datetime.now():
82
+ days_overdue = (datetime.now() - due).days
83
+ alerts.append({
84
+ "type": "overdue_reminder",
85
+ "severity": "high" if days_overdue > 7 else "medium",
86
+ "title": f"Reminder overdue by {days_overdue}d: {r['description'][:80]}",
87
+ "id": r["id"],
88
+ "days_overdue": days_overdue,
89
+ })
90
+ except (ValueError, TypeError):
91
+ pass
92
+ return alerts
93
+
94
+
95
+ def check_stale_ideas() -> list[dict]:
96
+ """Find reminders/ideas without due dates that have been sitting for too long."""
97
+ conn = get_db()
98
+ rows = conn.execute("""
99
+ SELECT id, description, created_at
100
+ FROM reminders
101
+ WHERE status NOT IN ('COMPLETED', 'CANCELLED')
102
+ AND (date IS NULL OR date = '')
103
+ ORDER BY created_at ASC
104
+ """).fetchall()
105
+ conn.close()
106
+ alerts = []
107
+ stale_count = 0
108
+ for r in rows:
109
+ try:
110
+ # created_at is epoch float
111
+ created = datetime.fromtimestamp(r["created_at"])
112
+ age_days = (datetime.now() - created).days
113
+ except (ValueError, TypeError, OSError):
114
+ age_days = 0
115
+ if age_days > 14:
116
+ stale_count += 1
117
+
118
+ if stale_count > 10:
119
+ alerts.append({
120
+ "type": "stale_ideas",
121
+ "severity": "low",
122
+ "title": f"{stale_count} ideas/reminders without date have been sitting for >14 days. Review or archive.",
123
+ "count": stale_count,
124
+ })
125
+ return alerts
126
+
127
+
128
+ def check_session_gaps() -> list[dict]:
129
+ """Detect if NEXO hasn't been active for unusual periods."""
130
+ conn = get_db()
131
+ row = conn.execute("""
132
+ SELECT MAX(created_at) as last_diary FROM session_diary
133
+ """).fetchone()
134
+ conn.close()
135
+ alerts = []
136
+ if row and row["last_diary"]:
137
+ try:
138
+ last = datetime.fromisoformat(row["last_diary"])
139
+ gap_hours = (datetime.now() - last).total_seconds() / 3600
140
+ if gap_hours > 48:
141
+ alerts.append({
142
+ "type": "session_gap",
143
+ "severity": "low",
144
+ "title": f"No sessions recorded in {gap_hours:.0f}h ({gap_hours/24:.1f} days)",
145
+ "gap_hours": gap_hours,
146
+ })
147
+ except (ValueError, TypeError):
148
+ pass
149
+ return alerts
150
+
151
+
152
+ def check_evolution_status() -> list[dict]:
153
+ """Check if evolution system is healthy."""
154
+ alerts = []
155
+ obj_file = NEXO_HOME / "brain" / "evolution-objective.json"
156
+ if obj_file.exists():
157
+ obj = json.loads(obj_file.read_text())
158
+ if not obj.get("evolution_enabled", True):
159
+ alerts.append({
160
+ "type": "evolution_disabled",
161
+ "severity": "high",
162
+ "title": f"Evolution DISABLED: {obj.get('disabled_reason', 'unknown')}",
163
+ })
164
+ if obj.get("consecutive_failures", 0) > 0:
165
+ alerts.append({
166
+ "type": "evolution_failures",
167
+ "severity": "medium",
168
+ "title": f"Evolution: {obj['consecutive_failures']} consecutive failures",
169
+ })
170
+
171
+ # Check dimension regression
172
+ for dim, data in obj.get("dimensions", {}).items():
173
+ current = data.get("current", 0)
174
+ if current < 30:
175
+ alerts.append({
176
+ "type": "dimension_low",
177
+ "severity": "medium",
178
+ "title": f"Dimension '{dim}' baja: {current}%",
179
+ "dimension": dim,
180
+ "score": current,
181
+ })
182
+ return alerts
183
+
184
+
185
+ def check_pending_proposals() -> list[dict]:
186
+ """Check for evolution proposals awaiting the user's review."""
187
+ conn = get_db()
188
+ rows = conn.execute("""
189
+ SELECT id, dimension, proposal, created_at
190
+ FROM evolution_log
191
+ WHERE status = 'proposed' AND classification = 'propose'
192
+ ORDER BY created_at DESC
193
+ """).fetchall()
194
+ conn.close()
195
+ if rows:
196
+ return [{
197
+ "type": "pending_proposals",
198
+ "severity": "low",
199
+ "title": f"{len(rows)} evolution proposals pending review",
200
+ "count": len(rows),
201
+ "proposals": [{"id": r["id"], "dim": r["dimension"], "text": r["proposal"][:80]} for r in rows],
202
+ }]
203
+ return []
204
+
205
+
206
+ def check_recurring_errors() -> list[dict]:
207
+ """Detect learnings that keep appearing (same issue reported multiple times)."""
208
+ conn = get_db()
209
+ rows = conn.execute("""
210
+ SELECT category, COUNT(*) as cnt
211
+ FROM learnings
212
+ WHERE created_at > datetime('now', '-7 days')
213
+ GROUP BY category
214
+ HAVING cnt >= 5
215
+ ORDER BY cnt DESC
216
+ """).fetchall()
217
+ conn.close()
218
+ alerts = []
219
+ for r in rows:
220
+ alerts.append({
221
+ "type": "recurring_errors",
222
+ "severity": "medium",
223
+ "title": f"Category '{r['category']}' has {r['cnt']} learnings this week — possible systemic issue",
224
+ "category": r["category"],
225
+ "count": r["cnt"],
226
+ })
227
+ return alerts
228
+
229
+
230
+ def check_cron_health() -> list[dict]:
231
+ """Check if critical cron jobs are running."""
232
+ alerts = []
233
+
234
+ # Check backup cron
235
+ backup_dir = NEXO_HOME / "backups"
236
+ if backup_dir.exists():
237
+ backups = sorted(backup_dir.glob("nexo-*.db"), key=lambda p: p.stat().st_mtime, reverse=True)
238
+ if backups:
239
+ last_backup_age = (datetime.now().timestamp() - backups[0].stat().st_mtime) / 3600
240
+ if last_backup_age > 4:
241
+ alerts.append({
242
+ "type": "backup_stale",
243
+ "severity": "high",
244
+ "title": f"Last nexo.db backup {last_backup_age:.1f}h (should be hourly)",
245
+ })
246
+
247
+ # Check immune system
248
+ immune_status = NEXO_HOME / "coordination" / "immune-status.json"
249
+ if immune_status.exists():
250
+ try:
251
+ status = json.loads(immune_status.read_text())
252
+ if status.get("status") == "degraded":
253
+ alerts.append({
254
+ "type": "immune_degraded",
255
+ "severity": "high",
256
+ "title": f"Immune system degraded: {status.get('reason', '?')}",
257
+ })
258
+ except (json.JSONDecodeError, KeyError):
259
+ pass
260
+
261
+ return alerts
262
+
263
+
264
+ def run_all_checks() -> list[dict]:
265
+ """Run all proactive checks and return sorted alerts."""
266
+ all_alerts = []
267
+ checks = [
268
+ check_overdue_followups,
269
+ check_overdue_reminders,
270
+ check_stale_ideas,
271
+ check_session_gaps,
272
+ check_evolution_status,
273
+ check_pending_proposals,
274
+ check_recurring_errors,
275
+ check_cron_health,
276
+ ]
277
+
278
+ for check in checks:
279
+ try:
280
+ all_alerts.extend(check())
281
+ except Exception as e:
282
+ all_alerts.append({
283
+ "type": "check_error",
284
+ "severity": "low",
285
+ "title": f"Check {check.__name__} failed: {e}",
286
+ })
287
+
288
+ # Sort by severity
289
+ severity_order = {"high": 0, "medium": 1, "low": 2}
290
+ all_alerts.sort(key=lambda a: severity_order.get(a.get("severity", "low"), 3))
291
+
292
+ return all_alerts
293
+
294
+
295
+ def format_text(alerts: list[dict]) -> str:
296
+ """Format alerts as readable text."""
297
+ if not alerts:
298
+ return "No proactive alerts. All clear."
299
+
300
+ severity_icons = {"high": "!!!", "medium": " ! ", "low": " . "}
301
+ lines = [f"NEXO Proactive Dashboard — {len(alerts)} alerts\n"]
302
+
303
+ current_severity = None
304
+ for a in alerts:
305
+ sev = a.get("severity", "low")
306
+ if sev != current_severity:
307
+ current_severity = sev
308
+ label = {"high": "URGENTE", "medium": "ATENCION", "low": "INFO"}.get(sev, sev)
309
+ lines.append(f"\n [{label}]")
310
+ icon = severity_icons.get(sev, " . ")
311
+ lines.append(f" {icon} {a['title']}")
312
+
313
+ return "\n".join(lines)
314
+
315
+
316
+ def format_brief(alerts: list[dict]) -> str:
317
+ """One-liner summary."""
318
+ high = sum(1 for a in alerts if a.get("severity") == "high")
319
+ med = sum(1 for a in alerts if a.get("severity") == "medium")
320
+ low = sum(1 for a in alerts if a.get("severity") == "low")
321
+ if not alerts:
322
+ return "Dashboard: clean"
323
+ return f"Dashboard: {high} urgent, {med} attention, {low} info"
324
+
325
+
326
+ def main():
327
+ output_json = "--json" in sys.argv
328
+ brief = "--brief" in sys.argv
329
+
330
+ alerts = run_all_checks()
331
+
332
+ if output_json:
333
+ print(json.dumps(alerts, indent=2, default=str))
334
+ elif brief:
335
+ print(format_brief(alerts))
336
+ else:
337
+ print(format_text(alerts))
338
+
339
+ # Exit code = number of high severity alerts
340
+ high_count = sum(1 for a in alerts if a.get("severity") == "high")
341
+ sys.exit(min(high_count, 125))
342
+
343
+
344
+ if __name__ == "__main__":
345
+ main()
@@ -38,6 +38,7 @@ def check_overdue_followups() -> list[dict]:
38
38
  SELECT id, description, date, created_at, reasoning
39
39
  FROM followups
40
40
  WHERE status NOT LIKE 'COMPLETED%'
41
+ AND status NOT IN ('DELETED','archived','blocked','waiting')
41
42
  AND date IS NOT NULL AND date != ''
42
43
  ORDER BY date ASC
43
44
  """).fetchall()