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