nexo-brain 2.1.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.
- package/README.md +3 -3
- package/bin/nexo-brain.js +53 -26
- package/package.json +1 -1
- package/scripts/migrate-to-unified 2.sh +813 -0
- package/scripts/migrate-v1.5-to-v1.6 2.py +778 -0
- package/scripts/migrate-v1.7-to-v1.8 2.py +214 -0
- package/scripts/pre-commit-check 2.sh +55 -0
- package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
- package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
- package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
- package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
- package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
- package/src/auto_close_sessions 2.py +159 -0
- package/src/auto_update 2.py +634 -0
- package/src/claim_graph 2.py +323 -0
- package/src/cognitive/__init__ 2.py +62 -0
- package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
- package/src/cognitive/_core 2.py +567 -0
- package/src/cognitive/_decay 2.py +382 -0
- package/src/cognitive/_ingest 2.py +892 -0
- package/src/cognitive/_memory 2.py +912 -0
- package/src/cognitive/_search 2.py +949 -0
- package/src/cognitive/_trust 2.py +464 -0
- package/src/cognitive/_trust.py +10 -36
- package/src/crons/manifest 2.json +106 -0
- package/src/crons/sync 2.py +217 -0
- package/src/dashboard/__init__ 2.py +0 -0
- package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
- package/src/dashboard/app 2.py +789 -0
- package/src/db/__init__ 2.py +89 -0
- package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
- package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
- package/src/db/_core 2.py +417 -0
- package/src/db/_credentials 2.py +124 -0
- package/src/db/_entities 2.py +178 -0
- package/src/db/_episodic 2.py +738 -0
- package/src/db/_evolution 2.py +54 -0
- package/src/db/_fts 2.py +406 -0
- package/src/db/_learnings 2.py +168 -0
- package/src/db/_reminders 2.py +338 -0
- package/src/db/_schema 2.py +364 -0
- package/src/db/_sessions 2.py +300 -0
- package/src/db/_tasks 2.py +91 -0
- package/src/evolution_cycle 2.py +266 -0
- package/src/hnsw_index 2.py +254 -0
- package/src/hooks/auto_capture 2.py +208 -0
- package/src/hooks/caffeinate-guard 2.sh +8 -0
- package/src/hooks/capture-session 2.sh +21 -0
- package/src/hooks/capture-session.sh +2 -0
- package/src/hooks/capture-tool-logs 2.sh +127 -0
- package/src/hooks/capture-tool-logs.sh +3 -2
- package/src/hooks/daily-briefing-check 2.sh +33 -0
- package/src/hooks/inbox-hook 2.sh +76 -0
- package/src/hooks/inbox-hook.sh +3 -2
- package/src/hooks/post-compact 2.sh +148 -0
- package/src/hooks/post-compact.sh +1 -1
- package/src/hooks/pre-compact 2.sh +151 -0
- package/src/hooks/pre-compact.sh +1 -1
- package/src/hooks/session-start 2.sh +268 -0
- package/src/hooks/session-start.sh +6 -3
- package/src/hooks/session-stop 2.sh +140 -0
- package/src/hooks/session-stop.sh +1 -1
- package/src/kg_populate 2.py +290 -0
- package/src/knowledge_graph 2.py +257 -0
- package/src/maintenance 2.py +59 -0
- package/src/migrate_embeddings 2.py +122 -0
- package/src/plugin_loader 2.py +202 -0
- package/src/plugins/__init__ 2.py +0 -0
- package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
- package/src/plugins/adaptive_mode 2.py +805 -0
- package/src/plugins/agents 2.py +52 -0
- package/src/plugins/artifact_registry 2.py +450 -0
- package/src/plugins/backup 2.py +104 -0
- package/src/plugins/cognitive_memory 2.py +564 -0
- package/src/plugins/core_rules 2.py +252 -0
- package/src/plugins/cortex 2.py +299 -0
- package/src/plugins/entities 2.py +67 -0
- package/src/plugins/episodic_memory 2.py +533 -0
- package/src/plugins/evolution 2.py +115 -0
- package/src/plugins/guard 2.py +746 -0
- package/src/plugins/knowledge_graph_tools 2.py +105 -0
- package/src/plugins/preferences 2.py +47 -0
- package/src/plugins/update 2.py +256 -0
- package/src/requirements 2.txt +12 -0
- package/src/rules/__init__ 2.py +0 -0
- package/src/rules/core-rules 2.json +331 -0
- package/src/rules/migrate 2.py +207 -0
- package/src/scripts/check-context 2.py +264 -0
- package/src/scripts/deep-sleep/apply_findings.py +58 -0
- package/src/scripts/deep-sleep/synthesize-prompt.md +30 -1
- package/src/scripts/nexo-auto-update 2.py +6 -0
- package/src/scripts/nexo-backup 2.sh +25 -0
- package/src/scripts/nexo-brain-activation 2.sh +140 -0
- package/src/scripts/nexo-catchup 2.py +242 -0
- package/src/scripts/nexo-cognitive-decay 2.py +182 -0
- package/src/scripts/nexo-daily-self-audit 2.py +552 -0
- package/src/scripts/nexo-deep-sleep 2.sh +97 -0
- package/src/scripts/nexo-evolution-run 2.py +597 -0
- package/src/scripts/nexo-followup-hygiene 2.py +112 -0
- package/src/scripts/nexo-github-monitor 2.py +256 -0
- package/src/scripts/nexo-immune 2.py +927 -0
- package/src/scripts/nexo-inbox-hook 2.sh +74 -0
- package/src/scripts/nexo-install 2.py +6 -0
- package/src/scripts/nexo-learning-housekeep 2.py +245 -0
- package/src/scripts/nexo-learning-validator 2.py +207 -0
- package/src/scripts/nexo-migrate 2.py +232 -0
- package/src/scripts/nexo-postmortem-consolidator 2.py +421 -0
- package/src/scripts/nexo-pre-commit 2.py +120 -0
- package/src/scripts/nexo-prevent-sleep 2.sh +29 -0
- package/src/scripts/nexo-proactive-dashboard 2.py +345 -0
- package/src/scripts/nexo-reflection 2.py +253 -0
- package/src/scripts/nexo-runtime-preflight 2.py +274 -0
- package/src/scripts/nexo-send-email 2.py +25 -0
- package/src/scripts/nexo-send-reply 2.py +178 -0
- package/src/scripts/nexo-sleep 2.py +592 -0
- package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
- package/src/scripts/nexo-synthesis 2.py +253 -0
- package/src/scripts/nexo-tcc-approve 2.sh +79 -0
- package/src/scripts/nexo-update 2.sh +161 -0
- package/src/scripts/nexo-watchdog 2.sh +878 -0
- package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
- package/src/server 2.py +733 -0
- package/src/server.py +6 -1
- package/src/storage_router 2.py +32 -0
- package/src/tools_coordination 2.py +102 -0
- package/src/tools_credentials 2.py +68 -0
- package/src/tools_learnings 2.py +220 -0
- package/src/tools_menu 2.py +227 -0
- package/src/tools_reminders 2.py +86 -0
- package/src/tools_reminders_crud 2.py +159 -0
- package/src/tools_reminders_crud.py +7 -0
- package/src/tools_sessions 2.py +476 -0
- package/src/tools_task_history 2.py +57 -0
- package/templates/CLAUDE.md 2.template +63 -0
- package/templates/openclaw 2.json +13 -0
- package/tests/__init__ 2.py +0 -0
- package/tests/conftest 2.py +71 -0
- package/tests/test_cognitive 2.py +205 -0
- package/tests/test_knowledge_graph 2.py +140 -0
- package/tests/test_migrations 2.py +137 -0
- package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
- package/src/scripts/deep-sleep/__pycache__/extract.cpython-314.pyc +0 -0
|
@@ -0,0 +1,112 @@
|
|
|
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()
|
|
@@ -0,0 +1,256 @@
|
|
|
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()
|