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,242 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
NEXO Catch-Up — Runs at Mac boot to execute any missed scheduled tasks.
|
|
4
|
+
|
|
5
|
+
When the Mac was asleep/off during scheduled times, launchd does NOT retry
|
|
6
|
+
missed StartCalendarInterval jobs. This script detects what was missed and
|
|
7
|
+
runs them in the correct order.
|
|
8
|
+
|
|
9
|
+
Scheduled tasks (ordered by intended run time):
|
|
10
|
+
03:00 — cognitive-decay (Ebbinghaus decay + STM→LTM promotion)
|
|
11
|
+
03:00 — evolution (weekly, Sundays only)
|
|
12
|
+
04:00 — sleep (session cleanup)
|
|
13
|
+
07:00 — self-audit (health checks + weekly cognitive GC on Sundays)
|
|
14
|
+
23:30 — postmortem (consolidation + sensory register)
|
|
15
|
+
|
|
16
|
+
Logic: For each task, check if its last successful run was before the
|
|
17
|
+
most recent scheduled time. If so, run it now.
|
|
18
|
+
"""
|
|
19
|
+
|
|
20
|
+
import json
|
|
21
|
+
import os
|
|
22
|
+
import subprocess
|
|
23
|
+
import sys
|
|
24
|
+
from datetime import datetime, timedelta
|
|
25
|
+
from pathlib import Path
|
|
26
|
+
|
|
27
|
+
CLAUDE_CLI = Path.home() / ".local" / "bin" / "claude"
|
|
28
|
+
|
|
29
|
+
HOME = Path.home()
|
|
30
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
|
|
31
|
+
LOG_DIR = NEXO_HOME / "logs"
|
|
32
|
+
LOG_DIR.mkdir(parents=True, exist_ok=True)
|
|
33
|
+
LOG_FILE = LOG_DIR / "catchup.log"
|
|
34
|
+
STATE_FILE = NEXO_HOME / "operations" / ".catchup-state.json"
|
|
35
|
+
|
|
36
|
+
SCRIPTS = NEXO_HOME / "scripts"
|
|
37
|
+
|
|
38
|
+
# Resolve Python: prefer NEXO's venv, then the same Python running this script
|
|
39
|
+
def _resolve_python() -> str:
|
|
40
|
+
"""Find the best Python to use for subprocess calls."""
|
|
41
|
+
# Check for NEXO_CODE env var pointing to the repo's src/
|
|
42
|
+
nexo_code = os.environ.get("NEXO_CODE", "")
|
|
43
|
+
if nexo_code:
|
|
44
|
+
venv_python = Path(nexo_code).parent / ".venv" / "bin" / "python"
|
|
45
|
+
if venv_python.exists():
|
|
46
|
+
return str(venv_python)
|
|
47
|
+
# Check for venv relative to NEXO_HOME
|
|
48
|
+
venv_python = NEXO_HOME / ".venv" / "bin" / "python"
|
|
49
|
+
if venv_python.exists():
|
|
50
|
+
return str(venv_python)
|
|
51
|
+
# Fall back to the same Python running this script
|
|
52
|
+
return sys.executable
|
|
53
|
+
|
|
54
|
+
NEXO_PYTHON = _resolve_python()
|
|
55
|
+
|
|
56
|
+
|
|
57
|
+
def log(msg: str):
|
|
58
|
+
ts = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
|
59
|
+
line = f"[{ts}] {msg}"
|
|
60
|
+
print(line, flush=True)
|
|
61
|
+
with open(LOG_FILE, "a") as f:
|
|
62
|
+
f.write(line + "\n")
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
def load_state() -> dict:
|
|
66
|
+
if STATE_FILE.exists():
|
|
67
|
+
try:
|
|
68
|
+
return json.loads(STATE_FILE.read_text())
|
|
69
|
+
except Exception:
|
|
70
|
+
pass
|
|
71
|
+
return {}
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
def save_state(state: dict):
|
|
75
|
+
STATE_FILE.parent.mkdir(parents=True, exist_ok=True)
|
|
76
|
+
STATE_FILE.write_text(json.dumps(state, indent=2))
|
|
77
|
+
|
|
78
|
+
|
|
79
|
+
def last_scheduled_time(hour: int, minute: int, weekday: int = None) -> datetime:
|
|
80
|
+
"""Calculate the most recent time this task should have run."""
|
|
81
|
+
now = datetime.now()
|
|
82
|
+
today_at = now.replace(hour=hour, minute=minute, second=0, microsecond=0)
|
|
83
|
+
|
|
84
|
+
if weekday is not None:
|
|
85
|
+
# Weekly task — find the most recent matching weekday
|
|
86
|
+
days_since = (now.weekday() - weekday) % 7
|
|
87
|
+
target = now - timedelta(days=days_since)
|
|
88
|
+
target = target.replace(hour=hour, minute=minute, second=0, microsecond=0)
|
|
89
|
+
if target > now:
|
|
90
|
+
target -= timedelta(weeks=1)
|
|
91
|
+
return target
|
|
92
|
+
|
|
93
|
+
# Daily task
|
|
94
|
+
if today_at <= now:
|
|
95
|
+
return today_at
|
|
96
|
+
else:
|
|
97
|
+
return today_at - timedelta(days=1)
|
|
98
|
+
|
|
99
|
+
|
|
100
|
+
def should_run(task_name: str, hour: int, minute: int, state: dict, weekday: int = None) -> bool:
|
|
101
|
+
"""Check if task needs catch-up: last run was before last scheduled time."""
|
|
102
|
+
last_run_str = state.get(task_name)
|
|
103
|
+
last_scheduled = last_scheduled_time(hour, minute, weekday)
|
|
104
|
+
|
|
105
|
+
if not last_run_str:
|
|
106
|
+
# Never ran — should run
|
|
107
|
+
return True
|
|
108
|
+
|
|
109
|
+
try:
|
|
110
|
+
last_run = datetime.fromisoformat(last_run_str)
|
|
111
|
+
except ValueError:
|
|
112
|
+
return True
|
|
113
|
+
|
|
114
|
+
return last_run < last_scheduled
|
|
115
|
+
|
|
116
|
+
|
|
117
|
+
def run_task(name: str, python: str, script: str, state: dict) -> bool:
|
|
118
|
+
"""Execute a task and update state."""
|
|
119
|
+
script_path = str(SCRIPTS / script)
|
|
120
|
+
if not Path(script_path).exists():
|
|
121
|
+
log(f" SKIP {name}: script not found ({script_path})")
|
|
122
|
+
return False
|
|
123
|
+
|
|
124
|
+
log(f" RUNNING {name}: {script}")
|
|
125
|
+
try:
|
|
126
|
+
result = subprocess.run(
|
|
127
|
+
[python, script_path],
|
|
128
|
+
capture_output=True, text=True, timeout=21600,
|
|
129
|
+
env={**os.environ, "HOME": str(HOME), "NEXO_CATCHUP": "1"}
|
|
130
|
+
)
|
|
131
|
+
if result.returncode == 0:
|
|
132
|
+
log(f" OK {name} (exit 0)")
|
|
133
|
+
else:
|
|
134
|
+
log(f" WARN {name} (exit {result.returncode})")
|
|
135
|
+
if result.stderr:
|
|
136
|
+
log(f" stderr: {result.stderr[:300]}")
|
|
137
|
+
state[name] = datetime.now().isoformat()
|
|
138
|
+
save_state(state)
|
|
139
|
+
return True
|
|
140
|
+
except subprocess.TimeoutExpired:
|
|
141
|
+
log(f" TIMEOUT {name} (300s)")
|
|
142
|
+
return False
|
|
143
|
+
except Exception as e:
|
|
144
|
+
log(f" ERROR {name}: {e}")
|
|
145
|
+
return False
|
|
146
|
+
|
|
147
|
+
|
|
148
|
+
def main():
|
|
149
|
+
log("=== NEXO Catch-Up starting (boot/wake) ===")
|
|
150
|
+
state = load_state()
|
|
151
|
+
|
|
152
|
+
# Define tasks in execution order (matching their intended schedule order)
|
|
153
|
+
# Note: auto-update is handled by the MCP server on startup, not by catchup.
|
|
154
|
+
tasks = [
|
|
155
|
+
# (name, hour, minute, python, script, weekday)
|
|
156
|
+
("cognitive-decay", 3, 0, NEXO_PYTHON, "nexo-cognitive-decay.py", None),
|
|
157
|
+
("evolution", 3, 0, NEXO_PYTHON, "nexo-evolution-run.py", 6), # Sunday = 6
|
|
158
|
+
("sleep", 4, 0, NEXO_PYTHON, "nexo-sleep.py", None),
|
|
159
|
+
("self-audit", 7, 0, NEXO_PYTHON, "nexo-daily-self-audit.py", None),
|
|
160
|
+
("github-monitor", 8, 0, NEXO_PYTHON, "nexo-github-monitor.py", None),
|
|
161
|
+
("postmortem", 23, 30, NEXO_PYTHON, "nexo-postmortem-consolidator.py", None),
|
|
162
|
+
]
|
|
163
|
+
|
|
164
|
+
ran = 0
|
|
165
|
+
skipped = 0
|
|
166
|
+
for name, hour, minute, python, script, weekday in tasks:
|
|
167
|
+
if should_run(name, hour, minute, state, weekday):
|
|
168
|
+
log(f" {name} — missed scheduled run, catching up...")
|
|
169
|
+
if run_task(name, python, script, state):
|
|
170
|
+
ran += 1
|
|
171
|
+
else:
|
|
172
|
+
skipped += 1
|
|
173
|
+
|
|
174
|
+
if ran == 0:
|
|
175
|
+
log("All tasks up to date, nothing to catch up.")
|
|
176
|
+
elif ran >= 3:
|
|
177
|
+
# Many tasks caught up — ask CLI to assess system state
|
|
178
|
+
_cli_post_catchup_assessment(ran, skipped, state)
|
|
179
|
+
else:
|
|
180
|
+
log(f"Caught up {ran} tasks, {skipped} already current.")
|
|
181
|
+
|
|
182
|
+
log("=== Catch-Up complete ===")
|
|
183
|
+
|
|
184
|
+
|
|
185
|
+
def _cli_post_catchup_assessment(ran: int, skipped: int, state: dict):
|
|
186
|
+
"""When 3+ tasks were missed, use CLI to assess if there are concerns."""
|
|
187
|
+
if not CLAUDE_CLI.exists():
|
|
188
|
+
log(f"Caught up {ran} tasks, {skipped} already current. (CLI unavailable for assessment)")
|
|
189
|
+
return
|
|
190
|
+
)
|
|
191
|
+
if auth_check.returncode != 0:
|
|
192
|
+
# CLI not authenticated, skip gracefully
|
|
193
|
+
print(f"[{datetime.now().strftime('%H:%M:%S')}] Claude CLI not authenticated. Skipping CLI analysis.")
|
|
194
|
+
return
|
|
195
|
+
|
|
196
|
+
assessment_file = LOG_DIR / "catchup-assessment.md"
|
|
197
|
+
state_summary = json.dumps(state, indent=2, default=str)
|
|
198
|
+
|
|
199
|
+
prompt = f"""You are the NEXO Catch-Up system. The Mac was off/asleep and {ran} scheduled tasks just ran as catch-up ({skipped} were already current).
|
|
200
|
+
|
|
201
|
+
Task run state (timestamps of last successful runs):
|
|
202
|
+
{state_summary}
|
|
203
|
+
|
|
204
|
+
Assess:
|
|
205
|
+
1. How long was the system likely offline? (compare timestamps to now)
|
|
206
|
+
2. Are there any tasks that depend on each other where order matters?
|
|
207
|
+
3. Any tasks that may have produced stale results because they ran late?
|
|
208
|
+
4. Should any task be re-run at its normal time today?
|
|
209
|
+
|
|
210
|
+
Write a brief assessment (max 20 lines) to: {assessment_file}
|
|
211
|
+
|
|
212
|
+
Format:
|
|
213
|
+
## Catch-Up Assessment — {datetime.now().strftime('%Y-%m-%d %H:%M')}
|
|
214
|
+
- Offline duration: ~Xh
|
|
215
|
+
- Tasks caught up: {ran}
|
|
216
|
+
- Concerns: ...
|
|
217
|
+
- Recommendation: ..."""
|
|
218
|
+
|
|
219
|
+
log(f"Caught up {ran} tasks — running CLI assessment...")
|
|
220
|
+
env = os.environ.copy()
|
|
221
|
+
env["NEXO_HEADLESS"] = "1" # Skip stop hook post-mortem
|
|
222
|
+
env.pop("CLAUDECODE", None)
|
|
223
|
+
env.pop("CLAUDE_CODE", None)
|
|
224
|
+
|
|
225
|
+
try:
|
|
226
|
+
result = subprocess.run(
|
|
227
|
+
[str(CLAUDE_CLI), "-p", prompt, "--model", "opus", "--output-format", "text",
|
|
228
|
+
"--allowedTools", "Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*"],
|
|
229
|
+
capture_output=True, text=True, timeout=21600, env=env
|
|
230
|
+
)
|
|
231
|
+
if result.returncode == 0:
|
|
232
|
+
log(f"Assessment written to {assessment_file}")
|
|
233
|
+
else:
|
|
234
|
+
log(f"CLI assessment exited {result.returncode}")
|
|
235
|
+
except subprocess.TimeoutExpired:
|
|
236
|
+
log("CLI assessment timed out (90s)")
|
|
237
|
+
except Exception as e:
|
|
238
|
+
log(f"CLI assessment error: {e}")
|
|
239
|
+
|
|
240
|
+
|
|
241
|
+
if __name__ == "__main__":
|
|
242
|
+
main()
|
|
@@ -0,0 +1,182 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""NEXO Cognitive Decay — Daily Ebbinghaus sweep + STM→LTM promotion."""
|
|
3
|
+
|
|
4
|
+
import json
|
|
5
|
+
import os
|
|
6
|
+
import sys
|
|
7
|
+
from pathlib import Path
|
|
8
|
+
|
|
9
|
+
NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
|
|
10
|
+
# Auto-detect: if running from repo (src/scripts/), use src/ as NEXO_CODE
|
|
11
|
+
_script_dir = Path(__file__).resolve().parent
|
|
12
|
+
_repo_src = _script_dir.parent # src/scripts/ -> src/
|
|
13
|
+
NEXO_CODE = Path(os.environ.get("NEXO_CODE", str(_repo_src) if (_repo_src / "server.py").exists() else str(NEXO_HOME)))
|
|
14
|
+
from datetime import datetime, timedelta
|
|
15
|
+
|
|
16
|
+
sys.path.insert(0, str(NEXO_CODE))
|
|
17
|
+
import cognitive
|
|
18
|
+
|
|
19
|
+
STATE_FILE = NEXO_HOME / "operations" / ".catchup-state.json"
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
def update_catchup_state():
|
|
23
|
+
"""Register successful run so catch-up script knows we ran."""
|
|
24
|
+
try:
|
|
25
|
+
state = json.loads(STATE_FILE.read_text()) if STATE_FILE.exists() else {}
|
|
26
|
+
except Exception:
|
|
27
|
+
state = {}
|
|
28
|
+
state["cognitive-decay"] = datetime.now().isoformat()
|
|
29
|
+
STATE_FILE.parent.mkdir(parents=True, exist_ok=True)
|
|
30
|
+
STATE_FILE.write_text(json.dumps(state, indent=2))
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
def main():
|
|
34
|
+
ts = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
|
35
|
+
print(f"[{ts}] Cognitive decay starting...")
|
|
36
|
+
|
|
37
|
+
# 0. Process quarantine FIRST — promote/reject/expire pending items
|
|
38
|
+
# BUG FIX 26-Mar-2026: quarantine was NEVER processed automatically.
|
|
39
|
+
# 78 items were stuck as pending indefinitely.
|
|
40
|
+
try:
|
|
41
|
+
q_result = cognitive.process_quarantine()
|
|
42
|
+
print(f"[{ts}] Quarantine: {q_result['promoted']} promoted, {q_result['rejected']} rejected, "
|
|
43
|
+
f"{q_result['expired']} expired, {q_result['still_pending']} still pending.")
|
|
44
|
+
except Exception as e:
|
|
45
|
+
print(f"[{ts}] Quarantine processing error: {e}")
|
|
46
|
+
|
|
47
|
+
# 0b. Purge test/dev memories from STM
|
|
48
|
+
try:
|
|
49
|
+
test_purged = cognitive.gc_test_memories()
|
|
50
|
+
if test_purged > 0:
|
|
51
|
+
print(f"[{ts}] Purged {test_purged} test/dev memories from STM.")
|
|
52
|
+
except Exception as e:
|
|
53
|
+
print(f"[{ts}] Test memory purge error: {e}")
|
|
54
|
+
|
|
55
|
+
# 1. Apply decay
|
|
56
|
+
cognitive.apply_decay()
|
|
57
|
+
print(f"[{ts}] Decay applied.")
|
|
58
|
+
|
|
59
|
+
# 2. Promote eligible STM → LTM
|
|
60
|
+
promoted = cognitive.promote_stm_to_ltm()
|
|
61
|
+
print(f"[{ts}] Promoted {promoted} STM memories to LTM.")
|
|
62
|
+
|
|
63
|
+
# 3. Garbage collect expired STM + sensory
|
|
64
|
+
gc_count = cognitive.gc_stm()
|
|
65
|
+
try:
|
|
66
|
+
gc_sensory = cognitive.gc_sensory(max_age_hours=48)
|
|
67
|
+
print(f"[{ts}] GC: removed {gc_count} expired STM, {gc_sensory} expired sensory.")
|
|
68
|
+
except Exception as e:
|
|
69
|
+
print(f"[{ts}] GC: removed {gc_count} expired STM. Sensory GC error: {e}")
|
|
70
|
+
|
|
71
|
+
# 4. Semantic consolidation — merge near-duplicate LTM (cosine > 0.9)
|
|
72
|
+
# With discriminative fusion: siblings (different environments) are linked, not merged
|
|
73
|
+
try:
|
|
74
|
+
result = cognitive.consolidate_semantic(threshold=0.9, dry_run=False)
|
|
75
|
+
merged = result.get("merged", [])
|
|
76
|
+
siblings = result.get("siblings", [])
|
|
77
|
+
if merged:
|
|
78
|
+
print(f"[{ts}] Consolidated {len(merged)} duplicate LTM pairs:")
|
|
79
|
+
for m in merged[:10]:
|
|
80
|
+
print(f"[{ts}] [{m['score']}] kept #{m['keep_id']} ({m['keep_access']} accesses), merged #{m['drop_id']}")
|
|
81
|
+
if siblings:
|
|
82
|
+
print(f"[{ts}] Linked {len(siblings)} sibling pairs (similar-but-incompatible):")
|
|
83
|
+
for s in siblings[:10]:
|
|
84
|
+
print(f"[{ts}] [{s['score']}] #{s['memory_a_id']} <> #{s['memory_b_id']} differ in: {', '.join(s['discriminators'])}")
|
|
85
|
+
if not merged and not siblings:
|
|
86
|
+
print(f"[{ts}] No semantic duplicates or siblings found (threshold=0.9)")
|
|
87
|
+
except Exception as e:
|
|
88
|
+
print(f"[{ts}] Consolidation error: {e}")
|
|
89
|
+
|
|
90
|
+
# 5. Correction fatigue — mark memories corrected 3+ times as unreliable
|
|
91
|
+
try:
|
|
92
|
+
fatigued = cognitive.check_correction_fatigue()
|
|
93
|
+
if fatigued:
|
|
94
|
+
print(f"[{ts}] CORRECTION FATIGUE: {len(fatigued)} memories corrected 3+ times in 7d:")
|
|
95
|
+
for f in fatigued:
|
|
96
|
+
print(f"[{ts}] LTM #{f['memory_id']} ({f['corrections_7d']}x): {f['content'][:80]}...")
|
|
97
|
+
else:
|
|
98
|
+
print(f"[{ts}] No correction fatigue detected.")
|
|
99
|
+
except Exception as e:
|
|
100
|
+
print(f"[{ts}] Correction fatigue check error: {e}")
|
|
101
|
+
|
|
102
|
+
# 6. Memory Dreaming — discover hidden connections between recent memories
|
|
103
|
+
try:
|
|
104
|
+
dream_result = cognitive.dream_cycle(max_insights=15)
|
|
105
|
+
scanned = dream_result["memories_scanned"]
|
|
106
|
+
created = dream_result["insights_created"]
|
|
107
|
+
candidates = dream_result["candidates_found"]
|
|
108
|
+
print(f"[{ts}] Dream cycle: scanned {scanned} recent memories, {candidates} candidates, {created} insights created.")
|
|
109
|
+
for insight in dream_result["insights"][:10]:
|
|
110
|
+
print(f"[{ts}] [{insight['similarity']}] {insight['title_a'][:40]} <-> {insight['title_b'][:40]}")
|
|
111
|
+
except Exception as e:
|
|
112
|
+
print(f"[{ts}] Dream cycle error: {e}")
|
|
113
|
+
|
|
114
|
+
# 7. Auto-merge duplicates (runs AFTER dream_cycle, higher threshold than consolidation)
|
|
115
|
+
try:
|
|
116
|
+
merge_result = cognitive.auto_merge_duplicates(threshold=0.92)
|
|
117
|
+
if merge_result["merged"] > 0:
|
|
118
|
+
print(f"[{ts}] Auto-merge: scanned {merge_result['scanned']}, merged {merge_result['merged']} duplicates, {merge_result['kept']} kept.")
|
|
119
|
+
for m in merge_result["merge_log"][:10]:
|
|
120
|
+
print(f"[{ts}] [{m['similarity']}] kept #{m['kept_id']}, dropped #{m['dropped_id']}")
|
|
121
|
+
else:
|
|
122
|
+
print(f"[{ts}] Auto-merge: scanned {merge_result['scanned']}, no duplicates above 0.92 threshold.")
|
|
123
|
+
except Exception as e:
|
|
124
|
+
print(f"[{ts}] Auto-merge error: {e}")
|
|
125
|
+
|
|
126
|
+
# 9. Adaptive weight learning — Ridge regression from feedback-annotated entries
|
|
127
|
+
try:
|
|
128
|
+
sys.path.insert(0, str(NEXO_CODE / "plugins"))
|
|
129
|
+
from adaptive_mode import learn_weights, prune_adaptive_log, check_weight_rollback
|
|
130
|
+
|
|
131
|
+
rollback = check_weight_rollback()
|
|
132
|
+
if rollback["status"] == "rolled_back":
|
|
133
|
+
print(f"[{ts}] WEIGHT ROLLBACK: {rollback['reason']}")
|
|
134
|
+
elif rollback["status"] == "ok":
|
|
135
|
+
print(f"[{ts}] Weight health: pre={rollback['pre_rate']}/day, post={rollback['post_rate']}/day")
|
|
136
|
+
elif rollback["status"] != "no_learned_weights":
|
|
137
|
+
print(f"[{ts}] Weight rollback: {rollback['status']}")
|
|
138
|
+
|
|
139
|
+
result = learn_weights()
|
|
140
|
+
if result["status"] in ("shadow", "active"):
|
|
141
|
+
mode_label = "SHADOW" if result["status"] == "shadow" else "ACTIVE"
|
|
142
|
+
print(f"[{ts}] Learned weights ({mode_label}) from {result['samples']} samples. Max drift: {result['max_drift']:.4f}")
|
|
143
|
+
for signal, weight in result["weights"].items():
|
|
144
|
+
drift = result["drift"][signal]
|
|
145
|
+
arrow = "+" if drift > 0 else "" if drift < 0 else "="
|
|
146
|
+
print(f"[{ts}] {signal}: {weight:.4f} ({arrow}{drift:.4f} from static)")
|
|
147
|
+
elif result["status"] == "insufficient_data":
|
|
148
|
+
print(f"[{ts}] Weight learning: {result['samples']}/{result['min_required']} samples (waiting)")
|
|
149
|
+
else:
|
|
150
|
+
print(f"[{ts}] Weight learning: {result['status']}")
|
|
151
|
+
|
|
152
|
+
pruned = prune_adaptive_log(max_age_days=90)
|
|
153
|
+
if pruned > 0:
|
|
154
|
+
print(f"[{ts}] Pruned {pruned} adaptive_log entries >90 days")
|
|
155
|
+
except Exception as e:
|
|
156
|
+
print(f"[{ts}] Adaptive weight learning error: {e}")
|
|
157
|
+
|
|
158
|
+
# 10. Project somatic events from nexo.db -> cognitive.db
|
|
159
|
+
try:
|
|
160
|
+
projected = cognitive.somatic_project_events()
|
|
161
|
+
if projected > 0:
|
|
162
|
+
print(f"[{ts}] Somatic projection: {projected} events projected to cognitive.db")
|
|
163
|
+
except Exception as e:
|
|
164
|
+
print(f"[{ts}] Somatic projection error: {e}")
|
|
165
|
+
|
|
166
|
+
# 11. Somatic marker nightly decay
|
|
167
|
+
try:
|
|
168
|
+
decayed = cognitive.somatic_nightly_decay(gamma=0.95)
|
|
169
|
+
print(f"[{ts}] Somatic decay: {decayed} markers processed (x0.95)")
|
|
170
|
+
except Exception as e:
|
|
171
|
+
print(f"[{ts}] Somatic decay error: {e}")
|
|
172
|
+
|
|
173
|
+
# 8. Stats
|
|
174
|
+
stats = cognitive.get_stats()
|
|
175
|
+
print(f"[{ts}] STM: {stats['stm_active']} active (+{stats.get('stm_promoted', 0)} promoted, {stats.get('stm_total', 0)} total) | LTM: {stats['ltm_active']} active, {stats['ltm_dormant']} dormant")
|
|
176
|
+
print(f"[{ts}] Done.")
|
|
177
|
+
|
|
178
|
+
update_catchup_state()
|
|
179
|
+
|
|
180
|
+
|
|
181
|
+
if __name__ == "__main__":
|
|
182
|
+
main()
|