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,112 +0,0 @@
1
- #!/usr/bin/env python3
2
- import os
3
- """
4
- NEXO Followup Hygiene — Weekly cleanup of followup/reminder statuses.
5
-
6
- Runs Sundays via LaunchAgent (or manually). Tasks:
7
- 1. Normalize dirty statuses (COMPLETED YYYY-MM-DD -> COMPLETED)
8
- 2. Flag PENDING followups >14 days without updates as STALE
9
- 3. Generate summary of orphaned/forgotten followups for synthesis
10
-
11
- No CLI needed — this is pure mechanical cleanup.
12
- """
13
-
14
- import json
15
- import sqlite3
16
- import sys
17
- from datetime import datetime, date, timedelta
18
- from pathlib import Path
19
-
20
- NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
21
-
22
- NEXO_DB = NEXO_HOME / "data" / "nexo.db"
23
- COORD_DIR = NEXO_HOME / "coordination"
24
- LOG_FILE = NEXO_HOME / "logs" / "followup-hygiene.log"
25
-
26
- TODAY = date.today().isoformat()
27
-
28
-
29
- def log(msg):
30
- ts = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
31
- line = f"[{ts}] {msg}"
32
- print(line, flush=True)
33
- LOG_FILE.parent.mkdir(parents=True, exist_ok=True)
34
- with open(LOG_FILE, "a") as f:
35
- f.write(line + "\n")
36
-
37
-
38
- def main():
39
- log("=== Followup Hygiene starting ===")
40
-
41
- if not NEXO_DB.exists():
42
- log("nexo.db not found")
43
- return
44
-
45
- conn = sqlite3.connect(str(NEXO_DB))
46
- conn.row_factory = sqlite3.Row
47
-
48
- # 1. Normalize dirty statuses
49
- dirty_f = conn.execute("SELECT COUNT(*) FROM followups WHERE status LIKE 'COMPLETED %'").fetchone()[0]
50
- dirty_r = conn.execute("SELECT COUNT(*) FROM reminders WHERE status LIKE 'COMPLETED %'").fetchone()[0]
51
-
52
- if dirty_f > 0:
53
- conn.execute("UPDATE followups SET status='COMPLETED' WHERE status LIKE 'COMPLETED %'")
54
- log(f"Normalized {dirty_f} dirty followup statuses")
55
-
56
- if dirty_r > 0:
57
- conn.execute("UPDATE reminders SET status='COMPLETED' WHERE status LIKE 'COMPLETED %'")
58
- log(f"Normalized {dirty_r} dirty reminder statuses")
59
-
60
- # 2. Flag stale followups (PENDING >14 days, no updates)
61
- cutoff = (date.today() - timedelta(days=14)).isoformat()
62
- stale = conn.execute(
63
- "SELECT id, description, date, updated_at FROM followups "
64
- "WHERE status NOT LIKE 'COMPLETED%' "
65
- "AND status NOT IN ('DELETED','archived','blocked','waiting') "
66
- "AND date != '' AND date < ? "
67
- "ORDER BY date",
68
- (cutoff,)
69
- ).fetchall()
70
-
71
- if stale:
72
- log(f"Found {len(stale)} stale followups (>14 days overdue):")
73
- for s in stale[:10]:
74
- log(f" {s['id']}: {s['description'][:60]} (due: {s['date']})")
75
-
76
- # 3. Orphaned followups (no date, no recent update)
77
- orphans = conn.execute(
78
- "SELECT id, description FROM followups "
79
- "WHERE status NOT LIKE 'COMPLETED%' "
80
- "AND status NOT IN ('DELETED','archived','blocked','waiting') "
81
- "AND (date IS NULL OR date = '') "
82
- "ORDER BY id"
83
- ).fetchall()
84
-
85
- if orphans:
86
- log(f"Found {len(orphans)} orphaned followups (no date):")
87
- for o in orphans[:10]:
88
- log(f" {o['id']}: {o['description'][:60]}")
89
-
90
- conn.commit()
91
- conn.close()
92
-
93
- # 4. Write summary for synthesis
94
- summary = {
95
- "date": TODAY,
96
- "dirty_normalized": dirty_f + dirty_r,
97
- "stale_count": len(stale) if stale else 0,
98
- "orphan_count": len(orphans) if orphans else 0,
99
- "stale_ids": [s["id"] for s in stale[:20]] if stale else [],
100
- "orphan_ids": [o["id"] for o in orphans[:20]] if orphans else [],
101
- }
102
-
103
- summary_file = COORD_DIR / "followup-hygiene-summary.json"
104
- summary_file.parent.mkdir(parents=True, exist_ok=True)
105
- summary_file.write_text(json.dumps(summary, indent=2))
106
-
107
- log(f"Summary: {dirty_f + dirty_r} normalized, {len(stale) if stale else 0} stale, {len(orphans) if orphans else 0} orphans")
108
- log("=== Followup Hygiene complete ===")
109
-
110
-
111
- if __name__ == "__main__":
112
- main()
@@ -1,256 +0,0 @@
1
- #!/usr/bin/env python3
2
- """
3
- NEXO GitHub Monitor — Wrapper + CLI pattern.
4
- Python: gh CLI API calls, data collection.
5
- CLI: Generates rich analysis and suggested responses for issues/PRs.
6
-
7
- Runs at 08:00 via LaunchAgent.
8
- Results saved to ~/.nexo/github-status.json.
9
- """
10
-
11
- import json
12
- import os
13
- import subprocess
14
- import sys
15
- from datetime import datetime, timedelta
16
- from pathlib import Path
17
-
18
- NEXO_HOME = Path(os.environ.get("NEXO_HOME", Path.home() / ".nexo"))
19
- STATUS_FILE = NEXO_HOME / "github-status.json"
20
- LOG_FILE = NEXO_HOME / "logs" / "github-monitor.log"
21
- REPO = "wazionapps/nexo"
22
- CLAUDE_CLI = Path.home() / ".local" / "bin" / "claude"
23
-
24
-
25
- def log(msg: str):
26
- ts = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
27
- line = f"[{ts}] {msg}"
28
- print(line, flush=True)
29
- LOG_FILE.parent.mkdir(parents=True, exist_ok=True)
30
- with open(LOG_FILE, "a") as f:
31
- f.write(line + "\n")
32
-
33
-
34
- def gh_api(endpoint: str) -> dict | list | None:
35
- """Call GitHub API via gh."""
36
- try:
37
- result = subprocess.run(
38
- ["gh", "api", endpoint],
39
- capture_output=True, text=True, timeout=21600
40
- )
41
- if result.returncode == 0:
42
- return json.loads(result.stdout)
43
- except Exception:
44
- pass
45
- return None
46
-
47
-
48
- def collect_data():
49
- """Collect all GitHub data — mechanical work."""
50
- data = {
51
- "timestamp": datetime.now().isoformat(),
52
- "repo": REPO,
53
- "issues": [],
54
- "prs": [],
55
- "latest_release": None,
56
- "unreleased_commits": 0,
57
- }
58
-
59
- # Issues
60
- log("Fetching issues...")
61
- issues = gh_api(f"repos/{REPO}/issues?state=open&per_page=50")
62
- if issues:
63
- for issue in issues:
64
- if "pull_request" in issue:
65
- continue
66
- item = {
67
- "number": issue["number"],
68
- "title": issue["title"][:80],
69
- "body": (issue.get("body") or "")[:500],
70
- "created": issue["created_at"][:10],
71
- "comments": issue["comments"],
72
- "labels": [l["name"] for l in issue.get("labels", [])],
73
- "author": issue.get("user", {}).get("login", ""),
74
- }
75
- # Get comment bodies for context
76
- if issue["comments"] > 0:
77
- comments = gh_api(f"repos/{REPO}/issues/{issue['number']}/comments?per_page=5")
78
- if comments:
79
- item["comment_bodies"] = [
80
- {"author": c.get("user", {}).get("login", ""), "body": c.get("body", "")[:300]}
81
- for c in comments[:5]
82
- ]
83
- data["issues"].append(item)
84
-
85
- # PRs
86
- log("Fetching PRs...")
87
- prs = gh_api(f"repos/{REPO}/pulls?state=open&per_page=50")
88
- if prs:
89
- for pr in prs:
90
- reviews = gh_api(f"repos/{REPO}/pulls/{pr['number']}/reviews") or []
91
- item = {
92
- "number": pr["number"],
93
- "title": pr["title"][:80],
94
- "body": (pr.get("body") or "")[:500],
95
- "author": pr["user"]["login"],
96
- "created": pr["created_at"][:10],
97
- "reviews": len(reviews),
98
- "changed_files": pr.get("changed_files", 0),
99
- }
100
- data["prs"].append(item)
101
-
102
- # Releases
103
- log("Fetching releases...")
104
- releases = gh_api(f"repos/{REPO}/releases?per_page=1")
105
- if releases and len(releases) > 0:
106
- data["latest_release"] = releases[0].get("tag_name", "none")
107
- tag = releases[0].get("tag_name", "")
108
- if tag:
109
- try:
110
- result = subprocess.run(
111
- ["gh", "api", f"repos/{REPO}/compare/{tag}...main"],
112
- capture_output=True, text=True, timeout=21600
113
- )
114
- if result.returncode == 0:
115
- compare = json.loads(result.stdout)
116
- data["unreleased_commits"] = compare.get("ahead_by", 0)
117
- except Exception:
118
- pass
119
-
120
- return data
121
-
122
-
123
- def analyze_via_cli(data):
124
- """Pass collected data to CLI for analysis and suggested responses."""
125
- data_json = json.dumps(data, ensure_ascii=False)
126
-
127
- prompt = f"""Analyze this GitHub repository status for NEXO Brain (wazionapps/nexo).
128
-
129
- DATA:
130
- {data_json}
131
-
132
- Generate a status report with:
133
- 1. SUMMARY: counts of open issues, PRs, unresponded items
134
- 2. For each UNRESPONDED ISSUE (comments=0): suggest a response in English (technical, helpful, friendly)
135
- 3. For each PR: brief assessment (looks good / needs changes / needs review)
136
- 4. RELEASE STATUS: if >10 unreleased commits, recommend a release
137
- 5. ALERTS: anything needing immediate attention (stale issues >7d, etc.)
138
-
139
- Return as JSON:
140
- {{
141
- "summary": {{
142
- "open_issues": N,
143
- "unresponded_issues": N,
144
- "stale_issues": N,
145
- "open_prs": N,
146
- "unreviewed_prs": N,
147
- "unreleased_commits": N
148
- }},
149
- "issue_responses": [
150
- {{"number": N, "suggested_response": "text"}},
151
- ...
152
- ],
153
- "pr_assessments": [
154
- {{"number": N, "assessment": "text"}},
155
- ...
156
- ],
157
- "alerts": ["alert1", ...],
158
- "release_recommendation": "text or null"
159
- }}"""
160
- )
161
- if auth_check.returncode != 0:
162
- # CLI not authenticated, skip gracefully
163
- return ""
164
-
165
- env = os.environ.copy()
166
- env["NEXO_HEADLESS"] = "1" # Skip stop hook post-mortem
167
- env.pop("CLAUDECODE", None)
168
- env.pop("CLAUDE_CODE", None)
169
-
170
- result = subprocess.run(
171
- [str(CLAUDE_CLI), "-p", prompt,
172
- "--model", "opus", "--output-format", "text",
173
- "--allowedTools", "Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*"],
174
- capture_output=True, text=True, timeout=21600, env=env
175
- )
176
-
177
- if result.returncode != 0:
178
- log(f"CLI analysis failed: {result.stderr[:200]}")
179
- return None
180
-
181
- output = result.stdout.strip()
182
- start = output.find("{")
183
- end = output.rfind("}") + 1
184
- if start >= 0 and end > start:
185
- return json.loads(output[start:end])
186
- return None
187
-
188
-
189
- def main():
190
- log("=== NEXO GitHub Monitor ===")
191
-
192
- # Step 1: Collect data (mechanical)
193
- data = collect_data()
194
-
195
- # Step 2: Analyze via CLI (intelligent)
196
- log("Analyzing via CLI...")
197
- analysis = analyze_via_cli(data)
198
-
199
- # Build status file
200
- status = {
201
- "timestamp": data["timestamp"],
202
- "repo": REPO,
203
- "issues": {
204
- "open": len(data["issues"]),
205
- "unresponded": sum(1 for i in data["issues"] if i["comments"] == 0),
206
- "stale": sum(1 for i in data["issues"]
207
- if i["comments"] == 0 and i["created"] < (datetime.now() - timedelta(days=7)).strftime('%Y-%m-%d')),
208
- "items": [{"number": i["number"], "title": i["title"], "created": i["created"],
209
- "comments": i["comments"], "labels": i["labels"]} for i in data["issues"]],
210
- },
211
- "prs": {
212
- "open": len(data["prs"]),
213
- "unreviewed": sum(1 for p in data["prs"] if p["reviews"] == 0),
214
- "items": [{"number": p["number"], "title": p["title"], "author": p["author"],
215
- "created": p["created"], "reviews": p["reviews"]} for p in data["prs"]],
216
- },
217
- "releases": {
218
- "latest": data["latest_release"] or "none",
219
- "unreleased_commits": data["unreleased_commits"],
220
- },
221
- "alerts": [],
222
- }
223
-
224
- # Merge CLI analysis
225
- if analysis:
226
- status["alerts"] = analysis.get("alerts", [])
227
- status["issue_responses"] = analysis.get("issue_responses", [])
228
- status["pr_assessments"] = analysis.get("pr_assessments", [])
229
- status["release_recommendation"] = analysis.get("release_recommendation")
230
- else:
231
- # Fallback alerts without CLI
232
- if status["issues"]["unresponded"] > 0:
233
- status["alerts"].append(f"{status['issues']['unresponded']} issues without response")
234
- if status["issues"]["stale"] > 0:
235
- status["alerts"].append(f"{status['issues']['stale']} stale issues (>7d)")
236
- if status["prs"]["unreviewed"] > 0:
237
- status["alerts"].append(f"{status['prs']['unreviewed']} PRs awaiting review")
238
- if data["unreleased_commits"] > 10:
239
- status["alerts"].append(f"{data['unreleased_commits']} unreleased commits")
240
-
241
- # Log summary
242
- log(f"Issues: {status['issues']['open']} open ({status['issues']['unresponded']} unresponded)")
243
- log(f"PRs: {status['prs']['open']} open ({status['prs']['unreviewed']} unreviewed)")
244
- log(f"Latest release: {status['releases']['latest']}")
245
- if status["alerts"]:
246
- log(f"ALERTS: {'; '.join(status['alerts'])}")
247
-
248
- # Save
249
- STATUS_FILE.parent.mkdir(parents=True, exist_ok=True)
250
- STATUS_FILE.write_text(json.dumps(status, indent=2))
251
- log(f"Status saved to {STATUS_FILE}")
252
- log("=== Done ===")
253
-
254
-
255
- if __name__ == "__main__":
256
- main()