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,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()
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
# NEXO Cron Wrapper — Records execution in cron_runs table.
|
|
3
|
+
# Usage: nexo-cron-wrapper.sh <cron_id> <command...>
|
|
4
|
+
# Example: nexo-cron-wrapper.sh deep-sleep bash nexo-deep-sleep.sh
|
|
5
|
+
#
|
|
6
|
+
# Wraps any cron command to automatically record start/end/exit_code/summary.
|
|
7
|
+
# Used by sync.py when generating LaunchAgents from manifest.json.
|
|
8
|
+
|
|
9
|
+
set -uo pipefail
|
|
10
|
+
|
|
11
|
+
CRON_ID="${1:?Usage: nexo-cron-wrapper.sh <cron_id> <command...>}"
|
|
12
|
+
shift
|
|
13
|
+
|
|
14
|
+
NEXO_HOME="${NEXO_HOME:-$HOME/.nexo}"
|
|
15
|
+
DB="$NEXO_HOME/data/nexo.db"
|
|
16
|
+
|
|
17
|
+
# Record start
|
|
18
|
+
RUN_ID=$(sqlite3 "$DB" "INSERT INTO cron_runs (cron_id) VALUES ('$CRON_ID'); SELECT last_insert_rowid();" 2>/dev/null)
|
|
19
|
+
|
|
20
|
+
if [ -z "$RUN_ID" ]; then
|
|
21
|
+
# DB not ready — run without tracking
|
|
22
|
+
exec "$@"
|
|
23
|
+
fi
|
|
24
|
+
|
|
25
|
+
# Run the actual command, capture output
|
|
26
|
+
OUTPUT_FILE=$(mktemp)
|
|
27
|
+
"$@" > "$OUTPUT_FILE" 2>&1
|
|
28
|
+
EXIT_CODE=$?
|
|
29
|
+
|
|
30
|
+
# Extract summary (last meaningful line, max 500 chars)
|
|
31
|
+
SUMMARY=$(tail -5 "$OUTPUT_FILE" | grep -v "^$" | tail -1 | head -c 500 | sed "s/'/''/g")
|
|
32
|
+
|
|
33
|
+
# Extract error if failed
|
|
34
|
+
ERROR=""
|
|
35
|
+
if [ $EXIT_CODE -ne 0 ]; then
|
|
36
|
+
ERROR=$(grep -i "error\|exception\|fail\|traceback" "$OUTPUT_FILE" | tail -1 | head -c 500 | sed "s/'/''/g")
|
|
37
|
+
fi
|
|
38
|
+
|
|
39
|
+
# Record end
|
|
40
|
+
sqlite3 "$DB" "
|
|
41
|
+
UPDATE cron_runs SET
|
|
42
|
+
ended_at = datetime('now'),
|
|
43
|
+
exit_code = $EXIT_CODE,
|
|
44
|
+
summary = '$SUMMARY',
|
|
45
|
+
error = '$ERROR',
|
|
46
|
+
duration_secs = ROUND((julianday(datetime('now')) - julianday(started_at)) * 86400, 1)
|
|
47
|
+
WHERE id = $RUN_ID;
|
|
48
|
+
" 2>/dev/null
|
|
49
|
+
|
|
50
|
+
# Clean output
|
|
51
|
+
rm -f "$OUTPUT_FILE"
|
|
52
|
+
|
|
53
|
+
exit $EXIT_CODE
|