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,214 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
NEXO Migration Script: v1.7.x -> v1.8.0 (Hybrid Architecture)
|
|
4
|
+
|
|
5
|
+
Migrates CLAUDE.md to the new hybrid architecture where:
|
|
6
|
+
- CLAUDE.md = bootstrap (identity, profile, format, autonomy, project atlas)
|
|
7
|
+
- MCP instructions field = tool-coupled behavioral rules
|
|
8
|
+
- nexo_context_packet = on-demand area-specific context
|
|
9
|
+
|
|
10
|
+
The MCP server now carries all tool-coupled rules in its `instructions` field,
|
|
11
|
+
so CLAUDE.md no longer needs to duplicate them. This reduces CLAUDE.md from
|
|
12
|
+
~130 lines to ~50 lines, saving ~3K context tokens per session.
|
|
13
|
+
|
|
14
|
+
Safe to run multiple times (idempotent). Creates a backup before modifying.
|
|
15
|
+
|
|
16
|
+
Usage:
|
|
17
|
+
python3 migrate-v1.7-to-v1.8.py [--dry-run] [--nexo-home /path/to/nexo]
|
|
18
|
+
"""
|
|
19
|
+
|
|
20
|
+
import argparse
|
|
21
|
+
import os
|
|
22
|
+
import re
|
|
23
|
+
import shutil
|
|
24
|
+
import sys
|
|
25
|
+
from datetime import datetime
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
# Sections that are now in the MCP instructions field and should be REMOVED from CLAUDE.md
|
|
29
|
+
MCP_OWNED_SECTIONS = [
|
|
30
|
+
"Heartbeat",
|
|
31
|
+
"Guard",
|
|
32
|
+
"Delegation",
|
|
33
|
+
"Reminders & Followups",
|
|
34
|
+
"Reminders y Followups",
|
|
35
|
+
"Memory",
|
|
36
|
+
"Memoria",
|
|
37
|
+
"Trust Score",
|
|
38
|
+
"Adaptive Mode",
|
|
39
|
+
"Dissonance",
|
|
40
|
+
"Disonancia",
|
|
41
|
+
"Observe the User",
|
|
42
|
+
"Observar a Francisco", # legacy personal CLAUDE.md files
|
|
43
|
+
"Observar al Usuario",
|
|
44
|
+
"Change Log",
|
|
45
|
+
"Session Diary",
|
|
46
|
+
"Cortex",
|
|
47
|
+
"Operational Codex",
|
|
48
|
+
]
|
|
49
|
+
|
|
50
|
+
# Sections that STAY in CLAUDE.md (bootstrap layer)
|
|
51
|
+
BOOTSTRAP_SECTIONS = [
|
|
52
|
+
"Startup",
|
|
53
|
+
"User Profile",
|
|
54
|
+
"Francisco", # legacy personal CLAUDE.md files
|
|
55
|
+
"Formato",
|
|
56
|
+
"Format",
|
|
57
|
+
"Autonomy",
|
|
58
|
+
"Autonomía",
|
|
59
|
+
"Project Atlas",
|
|
60
|
+
"Atlas de Proyectos",
|
|
61
|
+
"Hooks",
|
|
62
|
+
"Menu",
|
|
63
|
+
"Platforms",
|
|
64
|
+
"Plataformas",
|
|
65
|
+
"Repo",
|
|
66
|
+
]
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
def find_nexo_home(override=None):
|
|
70
|
+
if override:
|
|
71
|
+
return override
|
|
72
|
+
candidates = [
|
|
73
|
+
os.path.expanduser("~/nexo"),
|
|
74
|
+
os.path.expanduser("~/.nexo"),
|
|
75
|
+
os.path.expanduser("~/claude/nexo-mcp"),
|
|
76
|
+
]
|
|
77
|
+
for c in candidates:
|
|
78
|
+
if os.path.isdir(c):
|
|
79
|
+
return c
|
|
80
|
+
return None
|
|
81
|
+
|
|
82
|
+
|
|
83
|
+
def find_claude_md():
|
|
84
|
+
"""Find the CLAUDE.md file that contains NEXO instructions."""
|
|
85
|
+
candidates = [
|
|
86
|
+
os.path.expanduser("~/.claude/CLAUDE.md"),
|
|
87
|
+
os.path.expanduser("~/CLAUDE.md"),
|
|
88
|
+
]
|
|
89
|
+
for c in candidates:
|
|
90
|
+
if os.path.isfile(c):
|
|
91
|
+
return c
|
|
92
|
+
return None
|
|
93
|
+
|
|
94
|
+
|
|
95
|
+
def parse_sections(content):
|
|
96
|
+
"""Parse markdown into sections by ## headers."""
|
|
97
|
+
sections = []
|
|
98
|
+
current_header = None
|
|
99
|
+
current_lines = []
|
|
100
|
+
|
|
101
|
+
for line in content.split("\n"):
|
|
102
|
+
if line.startswith("## "):
|
|
103
|
+
if current_header is not None:
|
|
104
|
+
sections.append((current_header, "\n".join(current_lines)))
|
|
105
|
+
current_header = line[3:].strip()
|
|
106
|
+
current_lines = [line]
|
|
107
|
+
else:
|
|
108
|
+
current_lines.append(line)
|
|
109
|
+
|
|
110
|
+
if current_header is not None:
|
|
111
|
+
sections.append((current_header, "\n".join(current_lines)))
|
|
112
|
+
elif current_lines:
|
|
113
|
+
sections.append(("_preamble", "\n".join(current_lines)))
|
|
114
|
+
|
|
115
|
+
return sections
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
def is_mcp_owned(header):
|
|
119
|
+
"""Check if a section header matches an MCP-owned section."""
|
|
120
|
+
header_lower = header.lower()
|
|
121
|
+
for section in MCP_OWNED_SECTIONS:
|
|
122
|
+
if section.lower() in header_lower:
|
|
123
|
+
return True
|
|
124
|
+
return False
|
|
125
|
+
|
|
126
|
+
|
|
127
|
+
def migrate_claude_md(path, dry_run=False):
|
|
128
|
+
"""Slim down CLAUDE.md by removing MCP-owned sections."""
|
|
129
|
+
with open(path, "r") as f:
|
|
130
|
+
content = f.read()
|
|
131
|
+
|
|
132
|
+
original_lines = len(content.strip().split("\n"))
|
|
133
|
+
sections = parse_sections(content)
|
|
134
|
+
|
|
135
|
+
# Separate preamble, bootstrap, and MCP-owned sections
|
|
136
|
+
preamble = ""
|
|
137
|
+
kept = []
|
|
138
|
+
removed = []
|
|
139
|
+
|
|
140
|
+
for header, body in sections:
|
|
141
|
+
if header == "_preamble":
|
|
142
|
+
preamble = body
|
|
143
|
+
elif is_mcp_owned(header):
|
|
144
|
+
removed.append(header)
|
|
145
|
+
else:
|
|
146
|
+
kept.append((header, body))
|
|
147
|
+
|
|
148
|
+
if not removed:
|
|
149
|
+
print(" CLAUDE.md already migrated (no MCP-owned sections found).")
|
|
150
|
+
return False
|
|
151
|
+
|
|
152
|
+
# Add hybrid architecture note to preamble
|
|
153
|
+
if "MCP instructions" not in preamble and "instructions" not in preamble:
|
|
154
|
+
preamble = preamble.rstrip() + (
|
|
155
|
+
"\nTool-coupled behavioral rules (heartbeat, guard, trust, memory, diary) "
|
|
156
|
+
"now live in the MCP server instructions field and are injected automatically.\n"
|
|
157
|
+
)
|
|
158
|
+
|
|
159
|
+
# Reconstruct
|
|
160
|
+
new_content = preamble + "\n"
|
|
161
|
+
for header, body in kept:
|
|
162
|
+
new_content += "\n" + body + "\n"
|
|
163
|
+
|
|
164
|
+
new_lines = len(new_content.strip().split("\n"))
|
|
165
|
+
|
|
166
|
+
print(f" Sections removed (now in MCP): {', '.join(removed)}")
|
|
167
|
+
print(f" Lines: {original_lines} → {new_lines} (saved {original_lines - new_lines})")
|
|
168
|
+
|
|
169
|
+
if dry_run:
|
|
170
|
+
print(" [DRY RUN] No changes written.")
|
|
171
|
+
return True
|
|
172
|
+
|
|
173
|
+
# Backup
|
|
174
|
+
backup = path + f".backup-{datetime.now().strftime(\"%Y%m%d-%H%M%S\")}"
|
|
175
|
+
shutil.copy2(path, backup)
|
|
176
|
+
print(f" Backup: {backup}")
|
|
177
|
+
|
|
178
|
+
with open(path, "w") as f:
|
|
179
|
+
f.write(new_content)
|
|
180
|
+
print(" CLAUDE.md updated.")
|
|
181
|
+
return True
|
|
182
|
+
|
|
183
|
+
|
|
184
|
+
def main():
|
|
185
|
+
parser = argparse.ArgumentParser(description="Migrate NEXO to v1.8 hybrid architecture")
|
|
186
|
+
parser.add_argument("--dry-run", action="store_true", help="Show what would change without modifying files")
|
|
187
|
+
parser.add_argument("--nexo-home", help="Override NEXO home directory")
|
|
188
|
+
args = parser.parse_args()
|
|
189
|
+
|
|
190
|
+
print("NEXO v1.7 → v1.8 Migration (Hybrid Architecture)")
|
|
191
|
+
print("=" * 50)
|
|
192
|
+
print()
|
|
193
|
+
|
|
194
|
+
# Step 1: Find and migrate CLAUDE.md
|
|
195
|
+
claude_md = find_claude_md()
|
|
196
|
+
if claude_md:
|
|
197
|
+
print(f"Found CLAUDE.md: {claude_md}")
|
|
198
|
+
migrate_claude_md(claude_md, dry_run=args.dry_run)
|
|
199
|
+
else:
|
|
200
|
+
print("No CLAUDE.md found (skipping).")
|
|
201
|
+
|
|
202
|
+
print()
|
|
203
|
+
print("Migration complete.")
|
|
204
|
+
print()
|
|
205
|
+
print("What changed:")
|
|
206
|
+
print(" - CLAUDE.md now contains only bootstrap (identity, format, autonomy)")
|
|
207
|
+
print(" - Tool-coupled rules are in the MCP server instructions field")
|
|
208
|
+
print(" - Context-specific rules load on-demand via nexo_context_packet")
|
|
209
|
+
print()
|
|
210
|
+
print("The MCP server must be restarted for instructions to take effect.")
|
|
211
|
+
|
|
212
|
+
|
|
213
|
+
if __name__ == "__main__":
|
|
214
|
+
main()
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
# Pre-commit hook: prevent private data from being committed to the public repo.
|
|
3
|
+
# Installed by nexo-brain or manually: cp scripts/pre-commit-check.sh .git/hooks/pre-commit
|
|
4
|
+
|
|
5
|
+
RED='\033[0;31m'
|
|
6
|
+
NC='\033[0m'
|
|
7
|
+
|
|
8
|
+
# Add patterns specific to your private data here.
|
|
9
|
+
# These are checked against staged files to prevent accidental leaks.
|
|
10
|
+
# The pre-commit-check.sh script itself is excluded from scanning.
|
|
11
|
+
BLOCKED_PATTERNS=(
|
|
12
|
+
# Add your own patterns below, e.g.:
|
|
13
|
+
# "my-private-api-key"
|
|
14
|
+
# "my-private-domain.com"
|
|
15
|
+
# "my-server-ip"
|
|
16
|
+
)
|
|
17
|
+
|
|
18
|
+
STAGED_FILES=$(git diff --cached --name-only --diff-filter=ACMR)
|
|
19
|
+
|
|
20
|
+
if [ -z "$STAGED_FILES" ]; then
|
|
21
|
+
exit 0
|
|
22
|
+
fi
|
|
23
|
+
|
|
24
|
+
FOUND=0
|
|
25
|
+
for pattern in "${BLOCKED_PATTERNS[@]}"; do
|
|
26
|
+
MATCHES=$(echo "$STAGED_FILES" | xargs grep -l "$pattern" 2>/dev/null)
|
|
27
|
+
if [ -n "$MATCHES" ]; then
|
|
28
|
+
echo -e "${RED}BLOCKED: Found private data pattern '$pattern' in:${NC}"
|
|
29
|
+
echo "$MATCHES" | sed 's/^/ /'
|
|
30
|
+
FOUND=1
|
|
31
|
+
fi
|
|
32
|
+
done
|
|
33
|
+
|
|
34
|
+
# Also check for .db files, tokens, credentials
|
|
35
|
+
DB_FILES=$(echo "$STAGED_FILES" | grep -E '\.(db|db-wal|db-shm|key|pem)$')
|
|
36
|
+
if [ -n "$DB_FILES" ]; then
|
|
37
|
+
echo -e "${RED}BLOCKED: Database/key files staged:${NC}"
|
|
38
|
+
echo "$DB_FILES" | sed 's/^/ /'
|
|
39
|
+
FOUND=1
|
|
40
|
+
fi
|
|
41
|
+
|
|
42
|
+
TOKEN_FILES=$(echo "$STAGED_FILES" | grep -E '_token\.|credentials|\.env$')
|
|
43
|
+
if [ -n "$TOKEN_FILES" ]; then
|
|
44
|
+
echo -e "${RED}BLOCKED: Token/credential files staged:${NC}"
|
|
45
|
+
echo "$TOKEN_FILES" | sed 's/^/ /'
|
|
46
|
+
FOUND=1
|
|
47
|
+
fi
|
|
48
|
+
|
|
49
|
+
if [ $FOUND -eq 1 ]; then
|
|
50
|
+
echo ""
|
|
51
|
+
echo -e "${RED}Commit blocked. Remove private data before pushing to public repo.${NC}"
|
|
52
|
+
exit 1
|
|
53
|
+
fi
|
|
54
|
+
|
|
55
|
+
exit 0
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
@@ -0,0 +1,159 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Auto-close orphan sessions and promote diary drafts.
|
|
3
|
+
|
|
4
|
+
Runs every 5 minutes via LaunchAgent (com.nexo.auto-close-sessions).
|
|
5
|
+
Finds sessions that exceeded TTL without a diary and promotes their
|
|
6
|
+
draft to a real diary entry marked as source=auto-close.
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
import json
|
|
10
|
+
import os
|
|
11
|
+
import sys
|
|
12
|
+
import datetime
|
|
13
|
+
|
|
14
|
+
# Ensure we can import from the source directory
|
|
15
|
+
sys.path.insert(0, os.path.dirname(os.path.abspath(__file__)))
|
|
16
|
+
os.environ["NEXO_SKIP_FS_INDEX"] = "1" # Skip FTS rebuild on import
|
|
17
|
+
|
|
18
|
+
from db import (
|
|
19
|
+
init_db, get_db, get_diary_draft, delete_diary_draft,
|
|
20
|
+
get_orphan_sessions, write_session_diary, now_epoch,
|
|
21
|
+
SESSION_STALE_SECONDS,
|
|
22
|
+
)
|
|
23
|
+
|
|
24
|
+
NEXO_HOME = os.environ.get("NEXO_HOME", os.path.expanduser("~/.nexo"))
|
|
25
|
+
LOG_DIR = os.path.join(NEXO_HOME, "operations", "tool-logs")
|
|
26
|
+
AUTO_CLOSE_LOG = os.path.join(NEXO_HOME, "coordination", "auto-close.log")
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
def get_tool_log_summary(sid: str) -> str:
|
|
30
|
+
"""Extract tool names from today's tool log for this session."""
|
|
31
|
+
today = datetime.date.today().isoformat()
|
|
32
|
+
log_path = os.path.join(LOG_DIR, f"{today}.jsonl")
|
|
33
|
+
if not os.path.exists(log_path):
|
|
34
|
+
return ""
|
|
35
|
+
|
|
36
|
+
tools = []
|
|
37
|
+
try:
|
|
38
|
+
with open(log_path) as f:
|
|
39
|
+
for line in f:
|
|
40
|
+
try:
|
|
41
|
+
entry = json.loads(line)
|
|
42
|
+
if entry.get("session_id") == sid:
|
|
43
|
+
tool = entry.get("tool_name", "")
|
|
44
|
+
if tool and tool not in ("Read", "Grep", "Glob"):
|
|
45
|
+
tools.append(tool)
|
|
46
|
+
except json.JSONDecodeError:
|
|
47
|
+
continue
|
|
48
|
+
except Exception:
|
|
49
|
+
pass
|
|
50
|
+
|
|
51
|
+
if tools:
|
|
52
|
+
seen = set()
|
|
53
|
+
unique = []
|
|
54
|
+
for t in tools:
|
|
55
|
+
if t not in seen:
|
|
56
|
+
seen.add(t)
|
|
57
|
+
unique.append(t)
|
|
58
|
+
return f"Tools used: {', '.join(unique[-15:])}"
|
|
59
|
+
return ""
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
def promote_draft_to_diary(sid: str, draft: dict, task: str = ""):
|
|
63
|
+
"""Promote a diary draft to a real session diary entry."""
|
|
64
|
+
tasks = json.loads(draft.get("tasks_seen", "[]"))
|
|
65
|
+
change_ids = json.loads(draft.get("change_ids", "[]"))
|
|
66
|
+
decision_ids = json.loads(draft.get("decision_ids", "[]"))
|
|
67
|
+
context_hint = draft.get("last_context_hint", "")
|
|
68
|
+
hb_count = draft.get("heartbeat_count", 0)
|
|
69
|
+
|
|
70
|
+
summary_parts = []
|
|
71
|
+
if draft.get("summary_draft"):
|
|
72
|
+
summary_parts.append(draft["summary_draft"])
|
|
73
|
+
|
|
74
|
+
tool_summary = get_tool_log_summary(sid)
|
|
75
|
+
if tool_summary:
|
|
76
|
+
summary_parts.append(tool_summary)
|
|
77
|
+
|
|
78
|
+
summary = " | ".join(summary_parts) if summary_parts else f"Auto-closed session ({hb_count} heartbeats)"
|
|
79
|
+
|
|
80
|
+
# Build decisions from actual decision records
|
|
81
|
+
decisions_text = ""
|
|
82
|
+
if decision_ids:
|
|
83
|
+
conn = get_db()
|
|
84
|
+
placeholders = ",".join("?" * len(decision_ids))
|
|
85
|
+
rows = conn.execute(
|
|
86
|
+
f"SELECT id, decision, domain FROM decisions WHERE id IN ({placeholders})",
|
|
87
|
+
decision_ids
|
|
88
|
+
).fetchall()
|
|
89
|
+
if rows:
|
|
90
|
+
decisions_text = json.dumps([
|
|
91
|
+
{"id": r["id"], "decision": r["decision"][:100], "domain": r["domain"]}
|
|
92
|
+
for r in rows
|
|
93
|
+
])
|
|
94
|
+
|
|
95
|
+
# Build context_next
|
|
96
|
+
context_next = ""
|
|
97
|
+
if context_hint:
|
|
98
|
+
context_next = f"Last topic: {context_hint}"
|
|
99
|
+
if tasks:
|
|
100
|
+
context_next += f" | Tasks: {', '.join(tasks[-5:])}"
|
|
101
|
+
|
|
102
|
+
write_session_diary(
|
|
103
|
+
session_id=sid,
|
|
104
|
+
decisions=decisions_text or "No decisions logged",
|
|
105
|
+
summary=summary,
|
|
106
|
+
discarded="",
|
|
107
|
+
pending=f"Changes: {change_ids}" if change_ids else "",
|
|
108
|
+
context_next=context_next,
|
|
109
|
+
mental_state=f"[auto-close] Session ended without explicit diary. Draft promoted. {hb_count} heartbeats recorded.",
|
|
110
|
+
domain="",
|
|
111
|
+
user_signals="",
|
|
112
|
+
self_critique="[auto-close] No self-critique available — session terminated without cleanup.",
|
|
113
|
+
source="auto-close",
|
|
114
|
+
)
|
|
115
|
+
delete_diary_draft(sid)
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
def main():
|
|
119
|
+
init_db()
|
|
120
|
+
conn = get_db()
|
|
121
|
+
|
|
122
|
+
orphans = get_orphan_sessions(SESSION_STALE_SECONDS)
|
|
123
|
+
if not orphans:
|
|
124
|
+
print(f"[{datetime.datetime.now().isoformat(timespec='seconds')}] No stale sessions")
|
|
125
|
+
return
|
|
126
|
+
|
|
127
|
+
for session in orphans:
|
|
128
|
+
sid = session["sid"]
|
|
129
|
+
draft = get_diary_draft(sid)
|
|
130
|
+
|
|
131
|
+
if draft:
|
|
132
|
+
promote_draft_to_diary(sid, draft, task=session.get("task", ""))
|
|
133
|
+
else:
|
|
134
|
+
write_session_diary(
|
|
135
|
+
session_id=sid,
|
|
136
|
+
decisions="No decisions logged",
|
|
137
|
+
summary=f"Auto-closed session. Task: {session.get('task', 'unknown')}",
|
|
138
|
+
context_next="",
|
|
139
|
+
mental_state="[auto-close] No draft available. Minimal diary.",
|
|
140
|
+
self_critique="[auto-close] Session terminated without diary or draft.",
|
|
141
|
+
source="auto-close",
|
|
142
|
+
)
|
|
143
|
+
|
|
144
|
+
# Clean up the session
|
|
145
|
+
conn.execute("DELETE FROM tracked_files WHERE sid = ?", (sid,))
|
|
146
|
+
conn.execute("DELETE FROM sessions WHERE sid = ?", (sid,))
|
|
147
|
+
conn.execute("DELETE FROM session_diary_draft WHERE sid = ?", (sid,))
|
|
148
|
+
|
|
149
|
+
conn.commit()
|
|
150
|
+
|
|
151
|
+
# Log what we did
|
|
152
|
+
os.makedirs(os.path.dirname(AUTO_CLOSE_LOG), exist_ok=True)
|
|
153
|
+
with open(AUTO_CLOSE_LOG, "a") as f:
|
|
154
|
+
ts = datetime.datetime.now().isoformat(timespec="seconds")
|
|
155
|
+
f.write(f"{ts} — auto-closed {len(orphans)} session(s): {[s['sid'] for s in orphans]}\n")
|
|
156
|
+
|
|
157
|
+
|
|
158
|
+
if __name__ == "__main__":
|
|
159
|
+
main()
|