nexo-brain 2.1.0 → 2.3.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 +7 -7
- 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/migrate-v1.7-to-v1.8.py +2 -2
- package/scripts/nexo-preflight.sh +236 -0
- package/scripts/pre-commit-check 2.sh +55 -0
- package/src/__pycache__/auto_close_sessions.cpython-314.pyc +0 -0
- package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
- package/src/__pycache__/hnsw_index.cpython-314.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__/plugin_loader.cpython-314.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/auto_update.py +25 -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__/__init__.cpython-312.pyc +0 -0
- package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
- package/src/cognitive/__pycache__/_core.cpython-310.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-310.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-310.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-310.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-310.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-310.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/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/__pycache__/sync.cpython-314.pyc +0 -0
- package/src/crons/manifest 2.json +106 -0
- package/src/crons/manifest.json +6 -13
- package/src/crons/sync 2.py +217 -0
- package/src/crons/sync.py +151 -6
- 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/__init__.py +13 -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__/_cron_runs.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_cron_runs.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__/_skills.cpython-310.pyc +0 -0
- package/src/db/__pycache__/_skills.cpython-312.pyc +0 -0
- package/src/db/__pycache__/_skills.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/_cron_runs.py +74 -0
- package/src/db/_entities 2.py +178 -0
- package/src/db/_episodic 2.py +738 -0
- package/src/db/_episodic.py +40 -6
- 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/_schema.py +64 -0
- package/src/db/_sessions 2.py +300 -0
- package/src/db/_skills.py +514 -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 +14 -102
- 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__/__init__.cpython-314.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__/adaptive_mode.cpython-314.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__/schedule.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/schedule.cpython-314.pyc +0 -0
- package/src/plugins/__pycache__/skills.cpython-310.pyc +0 -0
- package/src/plugins/__pycache__/skills.cpython-314.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/episodic_memory.py +5 -3
- 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/schedule.py +212 -0
- package/src/plugins/skills.py +264 -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/__pycache__/nexo-auto-update.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-catchup.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-cognitive-decay.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-daily-self-audit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-evolution-run.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-followup-hygiene.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-immune.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-install.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-housekeep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-learning-validator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-migrate.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-postmortem-consolidator.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-pre-commit.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-proactive-dashboard.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-reflection.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-runtime-preflight.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-email.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-send-reply.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-sleep.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-synthesis.cpython-314.pyc +0 -0
- package/src/scripts/__pycache__/nexo-watchdog-smoke.cpython-314.pyc +0 -0
- package/src/scripts/check-context 2.py +264 -0
- package/src/scripts/deep-sleep/apply_findings.py +168 -8
- package/src/scripts/deep-sleep/collect.py +33 -11
- package/src/scripts/deep-sleep/extract-prompt.md +38 -0
- package/src/scripts/deep-sleep/extract.py +80 -8
- package/src/scripts/deep-sleep/synthesize-prompt.md +59 -2
- package/src/scripts/deep-sleep/synthesize.py +3 -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-catchup.py +65 -29
- package/src/scripts/nexo-cognitive-decay 2.py +182 -0
- package/src/scripts/nexo-cron-wrapper.sh +53 -0
- package/src/scripts/nexo-daily-self-audit 2.py +552 -0
- package/src/scripts/nexo-daily-self-audit.py +4 -2
- package/src/scripts/nexo-deep-sleep 2.sh +97 -0
- package/src/scripts/nexo-deep-sleep.sh +66 -77
- package/src/scripts/nexo-evolution-run 2.py +597 -0
- package/src/scripts/nexo-evolution-run.py +13 -0
- package/src/scripts/nexo-followup-hygiene 2.py +112 -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-housekeep.py +156 -1
- package/src/scripts/nexo-learning-validator 2.py +207 -0
- package/src/scripts/nexo-learning-validator.py +19 -0
- package/src/scripts/nexo-migrate 2.py +232 -0
- package/src/scripts/nexo-postmortem-consolidator 2.py +421 -0
- package/src/scripts/nexo-postmortem-consolidator.py +3 -2
- 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-sleep.py +16 -11
- package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
- package/src/scripts/nexo-synthesis 2.py +253 -0
- package/src/scripts/nexo-synthesis.py +46 -3
- 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/scripts/nexo-watchdog.sh +72 -19
- package/src/server 2.py +733 -0
- package/src/server.py +11 -2
- 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/scripts/deep-sleep/__pycache__/extract.cpython-314.pyc +0 -0
- /package/src/scripts/{nexo-github-monitor.py → nexo-github-monitor 2.py} +0 -0
|
@@ -0,0 +1,140 @@
|
|
|
1
|
+
#!/usr/bin/env bash
|
|
2
|
+
# nexo-brain-activation.sh — NF24: Spontaneous Activation
|
|
3
|
+
# Reads user_model.json, detects significant shifts, and generates insights
|
|
4
|
+
# for NEXO startup. If nothing relevant: exit 0 with no output.
|
|
5
|
+
|
|
6
|
+
set -euo pipefail
|
|
7
|
+
|
|
8
|
+
BRAIN_DIR="~/.nexo/brain"
|
|
9
|
+
MODEL_FILE="$BRAIN_DIR/user_model.json"
|
|
10
|
+
SUMMARIES_DIR="$BRAIN_DIR/daily_summaries"
|
|
11
|
+
|
|
12
|
+
# --- Guard: required file ---
|
|
13
|
+
if [[ ! -f "$MODEL_FILE" ]]; then
|
|
14
|
+
exit 0
|
|
15
|
+
fi
|
|
16
|
+
|
|
17
|
+
# --- Inline Python to parse JSON and analyze ---
|
|
18
|
+
python3 - <<'PYEOF'
|
|
19
|
+
import json
|
|
20
|
+
import sys
|
|
21
|
+
import os
|
|
22
|
+
import glob
|
|
23
|
+
from datetime import datetime, timedelta
|
|
24
|
+
|
|
25
|
+
BRAIN_DIR = os.path.expanduser("~/.nexo/brain")
|
|
26
|
+
MODEL_FILE = os.path.join(BRAIN_DIR, "user_model.json")
|
|
27
|
+
SUMMARIES_DIR = os.path.join(BRAIN_DIR, "daily_summaries")
|
|
28
|
+
|
|
29
|
+
# ── Load model ──────────────────────────────────────────────────────────
|
|
30
|
+
try:
|
|
31
|
+
with open(MODEL_FILE) as f:
|
|
32
|
+
model = json.load(f)
|
|
33
|
+
except Exception:
|
|
34
|
+
sys.exit(0)
|
|
35
|
+
|
|
36
|
+
insights = []
|
|
37
|
+
|
|
38
|
+
# ── 1. Analyze evolution_log — last 3-5 entries ─────────────────────
|
|
39
|
+
evolution = model.get("evolution_log", [])
|
|
40
|
+
recent = evolution[-5:] if len(evolution) >= 5 else evolution
|
|
41
|
+
|
|
42
|
+
# Detect entries from the last 3 days
|
|
43
|
+
today = datetime.now().date()
|
|
44
|
+
cutoff = today - timedelta(days=3)
|
|
45
|
+
|
|
46
|
+
recent_obs = []
|
|
47
|
+
for entry in recent:
|
|
48
|
+
try:
|
|
49
|
+
entry_date = datetime.strptime(entry["date"], "%Y-%m-%d").date()
|
|
50
|
+
if entry_date >= cutoff:
|
|
51
|
+
recent_obs.append(entry)
|
|
52
|
+
except Exception:
|
|
53
|
+
pass
|
|
54
|
+
|
|
55
|
+
if recent_obs:
|
|
56
|
+
for obs in recent_obs[-2:]: # max 2 most recent
|
|
57
|
+
insights.append(obs["observation"].strip())
|
|
58
|
+
|
|
59
|
+
# ── 2. Detect trait changes >0.1 between entries (if history exists) ──
|
|
60
|
+
# The current model only has a snapshot; if in the future there is history
|
|
61
|
+
# in evolution_log with trait deltas, this can be expanded here.
|
|
62
|
+
# Detect the most extreme trait as the dominant identity signal.
|
|
63
|
+
traits = model.get("traits", {})
|
|
64
|
+
if traits:
|
|
65
|
+
dominant = max(traits, key=lambda k: traits[k])
|
|
66
|
+
dominant_val = traits[dominant]
|
|
67
|
+
weakest = min(traits, key=lambda k: traits[k])
|
|
68
|
+
weakest_val = traits[weakest]
|
|
69
|
+
# Only report if extreme (>0.9 or <0.2) to avoid noise
|
|
70
|
+
if dominant_val >= 0.9:
|
|
71
|
+
insights.append(f"Dominant trait: {dominant}={dominant_val} (highest recorded)")
|
|
72
|
+
if weakest_val <= 0.2:
|
|
73
|
+
insights.append(f"Lowest trait: {weakest}={weakest_val} (low tolerance active)")
|
|
74
|
+
|
|
75
|
+
# ── 3. Recent contradictions (last 3 days) ─────────────────────────
|
|
76
|
+
contradictions = model.get("contradictions", [])
|
|
77
|
+
recent_contradictions = []
|
|
78
|
+
for c in contradictions:
|
|
79
|
+
try:
|
|
80
|
+
c_date = datetime.strptime(c["date"], "%Y-%m-%d").date()
|
|
81
|
+
if c_date >= cutoff:
|
|
82
|
+
recent_contradictions.append(c)
|
|
83
|
+
except Exception:
|
|
84
|
+
pass
|
|
85
|
+
|
|
86
|
+
for c in recent_contradictions:
|
|
87
|
+
insights.append(f"Contradiction detected: {c['description'].strip()}")
|
|
88
|
+
|
|
89
|
+
# ── 4. Current active focus ─────────────────────────────────────────────────
|
|
90
|
+
current_focus = model.get("current_focus", [])
|
|
91
|
+
# Only include if there are focus items (and it's not initial startup)
|
|
92
|
+
if len(current_focus) > 0 and len(insights) == 0:
|
|
93
|
+
# If no other insights, report focus as minimum context
|
|
94
|
+
focus_str = ", ".join(current_focus)
|
|
95
|
+
insights.append(f"Current focus: {focus_str}")
|
|
96
|
+
|
|
97
|
+
# ── 5. Goals: active/dormant changes ─────────────────────────────────────
|
|
98
|
+
goals_active = model.get("goals_active", [])
|
|
99
|
+
goals_dormant = model.get("goals_dormant", [])
|
|
100
|
+
# If there are dormant goals, mention them (may need reactivation)
|
|
101
|
+
if goals_dormant:
|
|
102
|
+
insights.append(f"Goal dormant: {goals_dormant[0]}")
|
|
103
|
+
|
|
104
|
+
# ── Read last daily summary ─────────────────────────────────────────────
|
|
105
|
+
summary_text = ""
|
|
106
|
+
try:
|
|
107
|
+
files = sorted(glob.glob(os.path.join(SUMMARIES_DIR, "*.md")))
|
|
108
|
+
if files:
|
|
109
|
+
with open(files[-1]) as f:
|
|
110
|
+
lines = [l.rstrip() for l in f.readlines() if l.strip()]
|
|
111
|
+
# Take up to 3 lines of content (exclude header)
|
|
112
|
+
content_lines = [l for l in lines if not l.startswith("# Resumen") and not l.startswith("# Summary")]
|
|
113
|
+
summary_text = " | ".join(content_lines[:3])
|
|
114
|
+
except Exception:
|
|
115
|
+
pass
|
|
116
|
+
|
|
117
|
+
# ── Output ────────────────────────────────────────────────────────────────
|
|
118
|
+
# If no real insights, exit without output
|
|
119
|
+
if not insights and not summary_text:
|
|
120
|
+
sys.exit(0)
|
|
121
|
+
|
|
122
|
+
# Deduplicate and limit
|
|
123
|
+
seen = set()
|
|
124
|
+
unique_insights = []
|
|
125
|
+
for ins in insights:
|
|
126
|
+
key = ins[:60]
|
|
127
|
+
if key not in seen:
|
|
128
|
+
seen.add(key)
|
|
129
|
+
unique_insights.append(ins)
|
|
130
|
+
|
|
131
|
+
if unique_insights:
|
|
132
|
+
print("BRAIN_INSIGHTS:")
|
|
133
|
+
for ins in unique_insights[:5]:
|
|
134
|
+
print(f"- {ins}")
|
|
135
|
+
|
|
136
|
+
if summary_text:
|
|
137
|
+
print("RECENT_SUMMARY:")
|
|
138
|
+
print(summary_text[:400])
|
|
139
|
+
|
|
140
|
+
PYEOF
|
|
@@ -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()
|
|
@@ -1,20 +1,14 @@
|
|
|
1
1
|
#!/usr/bin/env python3
|
|
2
2
|
"""
|
|
3
|
-
NEXO Catch-Up — Runs at
|
|
3
|
+
NEXO Catch-Up — Runs at boot/wake to recover any missed scheduled tasks.
|
|
4
4
|
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
5
|
+
Tasks are loaded dynamically from crons/manifest.json (single source of truth).
|
|
6
|
+
Only scheduled crons (with hour/minute) are recovered — interval-based crons
|
|
7
|
+
(immune, watchdog, auto-close) restart automatically via launchd/systemd.
|
|
8
8
|
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
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.
|
|
9
|
+
Logic: For each scheduled task, check if its last successful run was before
|
|
10
|
+
the most recent scheduled time. If so, run it now. Only marks success on exit 0.
|
|
11
|
+
Uses cron/launchd weekday convention (0=Sunday) converted to Python (0=Monday).
|
|
18
12
|
"""
|
|
19
13
|
|
|
20
14
|
import json
|
|
@@ -52,6 +46,49 @@ def _resolve_python() -> str:
|
|
|
52
46
|
return sys.executable
|
|
53
47
|
|
|
54
48
|
NEXO_PYTHON = _resolve_python()
|
|
49
|
+
NEXO_CODE = Path(os.environ.get("NEXO_CODE", str(Path(__file__).resolve().parent.parent)))
|
|
50
|
+
MANIFEST = NEXO_CODE / "crons" / "manifest.json"
|
|
51
|
+
|
|
52
|
+
|
|
53
|
+
def _load_tasks_from_manifest() -> list[tuple]:
|
|
54
|
+
"""Read scheduled tasks from manifest.json — single source of truth.
|
|
55
|
+
|
|
56
|
+
Only includes crons with a schedule (hour/minute). Excludes interval-based
|
|
57
|
+
crons (immune, watchdog, auto-close) and run_at_load (catchup itself).
|
|
58
|
+
Returns: list of (name, hour, minute, python_or_bash, script, weekday)
|
|
59
|
+
"""
|
|
60
|
+
if not MANIFEST.exists():
|
|
61
|
+
log(f"WARNING: manifest not found at {MANIFEST}, using empty task list")
|
|
62
|
+
return []
|
|
63
|
+
|
|
64
|
+
with open(MANIFEST) as f:
|
|
65
|
+
data = json.load(f)
|
|
66
|
+
|
|
67
|
+
tasks = []
|
|
68
|
+
for cron in data.get("crons", []):
|
|
69
|
+
schedule = cron.get("schedule")
|
|
70
|
+
if not schedule or "hour" not in schedule:
|
|
71
|
+
continue # Skip interval-based and run_at_load crons
|
|
72
|
+
if cron["id"] == "catchup":
|
|
73
|
+
continue # Don't catch up ourselves
|
|
74
|
+
|
|
75
|
+
script = cron["script"]
|
|
76
|
+
script_type = cron.get("type", "python")
|
|
77
|
+
interpreter = NEXO_PYTHON if script_type == "python" else "/bin/bash"
|
|
78
|
+
weekday = schedule.get("weekday")
|
|
79
|
+
|
|
80
|
+
tasks.append((
|
|
81
|
+
cron["id"],
|
|
82
|
+
schedule["hour"],
|
|
83
|
+
schedule["minute"],
|
|
84
|
+
interpreter,
|
|
85
|
+
Path(script).name,
|
|
86
|
+
weekday,
|
|
87
|
+
))
|
|
88
|
+
|
|
89
|
+
# Sort by hour, minute for correct execution order
|
|
90
|
+
tasks.sort(key=lambda t: (t[1], t[2]))
|
|
91
|
+
return tasks
|
|
55
92
|
|
|
56
93
|
|
|
57
94
|
def log(msg: str):
|
|
@@ -83,7 +120,11 @@ def last_scheduled_time(hour: int, minute: int, weekday: int = None) -> datetime
|
|
|
83
120
|
|
|
84
121
|
if weekday is not None:
|
|
85
122
|
# Weekly task — find the most recent matching weekday
|
|
86
|
-
|
|
123
|
+
# Manifest uses cron/launchd convention: 0=Sunday, 6=Saturday
|
|
124
|
+
# Python datetime.weekday() uses: 0=Monday, 6=Sunday
|
|
125
|
+
# Convert: manifest 0 (Sun) -> python 6, manifest 1 (Mon) -> python 0, etc.
|
|
126
|
+
py_weekday = (weekday - 1) % 7
|
|
127
|
+
days_since = (now.weekday() - py_weekday) % 7
|
|
87
128
|
target = now - timedelta(days=days_since)
|
|
88
129
|
target = target.replace(hour=hour, minute=minute, second=0, microsecond=0)
|
|
89
130
|
if target > now:
|
|
@@ -130,13 +171,14 @@ def run_task(name: str, python: str, script: str, state: dict) -> bool:
|
|
|
130
171
|
)
|
|
131
172
|
if result.returncode == 0:
|
|
132
173
|
log(f" OK {name} (exit 0)")
|
|
174
|
+
state[name] = datetime.now().isoformat()
|
|
175
|
+
save_state(state)
|
|
176
|
+
return True
|
|
133
177
|
else:
|
|
134
|
-
log(f"
|
|
178
|
+
log(f" FAIL {name} (exit {result.returncode})")
|
|
135
179
|
if result.stderr:
|
|
136
180
|
log(f" stderr: {result.stderr[:300]}")
|
|
137
|
-
|
|
138
|
-
save_state(state)
|
|
139
|
-
return True
|
|
181
|
+
return False
|
|
140
182
|
except subprocess.TimeoutExpired:
|
|
141
183
|
log(f" TIMEOUT {name} (300s)")
|
|
142
184
|
return False
|
|
@@ -149,17 +191,8 @@ def main():
|
|
|
149
191
|
log("=== NEXO Catch-Up starting (boot/wake) ===")
|
|
150
192
|
state = load_state()
|
|
151
193
|
|
|
152
|
-
#
|
|
153
|
-
|
|
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
|
-
]
|
|
194
|
+
# Read tasks from manifest — single source of truth
|
|
195
|
+
tasks = _load_tasks_from_manifest()
|
|
163
196
|
|
|
164
197
|
ran = 0
|
|
165
198
|
skipped = 0
|
|
@@ -187,6 +220,9 @@ def _cli_post_catchup_assessment(ran: int, skipped: int, state: dict):
|
|
|
187
220
|
if not CLAUDE_CLI.exists():
|
|
188
221
|
log(f"Caught up {ran} tasks, {skipped} already current. (CLI unavailable for assessment)")
|
|
189
222
|
return
|
|
223
|
+
auth_check = subprocess.run(
|
|
224
|
+
[str(CLAUDE_CLI), "-p", "reply OK", "--output-format", "text"],
|
|
225
|
+
capture_output=True, text=True, timeout=30
|
|
190
226
|
)
|
|
191
227
|
if auth_check.returncode != 0:
|
|
192
228
|
# CLI not authenticated, skip gracefully
|