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.
Files changed (235) hide show
  1. package/README.md +3 -3
  2. package/bin/nexo-brain.js +53 -26
  3. package/package.json +1 -1
  4. package/scripts/migrate-to-unified 2.sh +813 -0
  5. package/scripts/migrate-v1.5-to-v1.6 2.py +778 -0
  6. package/scripts/migrate-v1.7-to-v1.8 2.py +214 -0
  7. package/scripts/pre-commit-check 2.sh +55 -0
  8. package/src/__pycache__/auto_update.cpython-310.pyc +0 -0
  9. package/src/__pycache__/hnsw_index.cpython-310.pyc +0 -0
  10. package/src/__pycache__/kg_populate.cpython-310.pyc +0 -0
  11. package/src/__pycache__/knowledge_graph.cpython-310.pyc +0 -0
  12. package/src/__pycache__/plugin_loader.cpython-310.pyc +0 -0
  13. package/src/__pycache__/tools_coordination.cpython-310.pyc +0 -0
  14. package/src/__pycache__/tools_credentials.cpython-310.pyc +0 -0
  15. package/src/__pycache__/tools_learnings.cpython-310.pyc +0 -0
  16. package/src/__pycache__/tools_menu.cpython-310.pyc +0 -0
  17. package/src/__pycache__/tools_reminders.cpython-310.pyc +0 -0
  18. package/src/__pycache__/tools_reminders_crud.cpython-310.pyc +0 -0
  19. package/src/__pycache__/tools_sessions.cpython-310.pyc +0 -0
  20. package/src/__pycache__/tools_task_history.cpython-310.pyc +0 -0
  21. package/src/auto_close_sessions 2.py +159 -0
  22. package/src/auto_update 2.py +634 -0
  23. package/src/claim_graph 2.py +323 -0
  24. package/src/cognitive/__init__ 2.py +62 -0
  25. package/src/cognitive/__pycache__/__init__.cpython-310.pyc +0 -0
  26. package/src/cognitive/__pycache__/_core.cpython-310.pyc +0 -0
  27. package/src/cognitive/__pycache__/_decay.cpython-310.pyc +0 -0
  28. package/src/cognitive/__pycache__/_ingest.cpython-310.pyc +0 -0
  29. package/src/cognitive/__pycache__/_memory.cpython-310.pyc +0 -0
  30. package/src/cognitive/__pycache__/_search.cpython-310.pyc +0 -0
  31. package/src/cognitive/__pycache__/_trust.cpython-310.pyc +0 -0
  32. package/src/cognitive/_core 2.py +567 -0
  33. package/src/cognitive/_decay 2.py +382 -0
  34. package/src/cognitive/_ingest 2.py +892 -0
  35. package/src/cognitive/_memory 2.py +912 -0
  36. package/src/cognitive/_search 2.py +949 -0
  37. package/src/cognitive/_trust 2.py +464 -0
  38. package/src/cognitive/_trust.py +10 -36
  39. package/src/crons/manifest 2.json +106 -0
  40. package/src/crons/sync 2.py +217 -0
  41. package/src/dashboard/__init__ 2.py +0 -0
  42. package/src/dashboard/__pycache__/__init__.cpython-310.pyc +0 -0
  43. package/src/dashboard/__pycache__/app.cpython-310.pyc +0 -0
  44. package/src/dashboard/app 2.py +789 -0
  45. package/src/db/__init__ 2.py +89 -0
  46. package/src/db/__pycache__/__init__.cpython-310.pyc +0 -0
  47. package/src/db/__pycache__/__init__.cpython-312.pyc +0 -0
  48. package/src/db/__pycache__/__init__.cpython-314.pyc +0 -0
  49. package/src/db/__pycache__/_core.cpython-310.pyc +0 -0
  50. package/src/db/__pycache__/_core.cpython-312.pyc +0 -0
  51. package/src/db/__pycache__/_core.cpython-314.pyc +0 -0
  52. package/src/db/__pycache__/_credentials.cpython-310.pyc +0 -0
  53. package/src/db/__pycache__/_credentials.cpython-312.pyc +0 -0
  54. package/src/db/__pycache__/_credentials.cpython-314.pyc +0 -0
  55. package/src/db/__pycache__/_entities.cpython-310.pyc +0 -0
  56. package/src/db/__pycache__/_entities.cpython-312.pyc +0 -0
  57. package/src/db/__pycache__/_entities.cpython-314.pyc +0 -0
  58. package/src/db/__pycache__/_episodic.cpython-310.pyc +0 -0
  59. package/src/db/__pycache__/_episodic.cpython-312.pyc +0 -0
  60. package/src/db/__pycache__/_episodic.cpython-314.pyc +0 -0
  61. package/src/db/__pycache__/_evolution.cpython-310.pyc +0 -0
  62. package/src/db/__pycache__/_evolution.cpython-312.pyc +0 -0
  63. package/src/db/__pycache__/_evolution.cpython-314.pyc +0 -0
  64. package/src/db/__pycache__/_fts.cpython-310.pyc +0 -0
  65. package/src/db/__pycache__/_fts.cpython-312.pyc +0 -0
  66. package/src/db/__pycache__/_fts.cpython-314.pyc +0 -0
  67. package/src/db/__pycache__/_learnings.cpython-310.pyc +0 -0
  68. package/src/db/__pycache__/_learnings.cpython-312.pyc +0 -0
  69. package/src/db/__pycache__/_learnings.cpython-314.pyc +0 -0
  70. package/src/db/__pycache__/_reminders.cpython-310.pyc +0 -0
  71. package/src/db/__pycache__/_reminders.cpython-312.pyc +0 -0
  72. package/src/db/__pycache__/_reminders.cpython-314.pyc +0 -0
  73. package/src/db/__pycache__/_schema.cpython-310.pyc +0 -0
  74. package/src/db/__pycache__/_schema.cpython-312.pyc +0 -0
  75. package/src/db/__pycache__/_schema.cpython-314.pyc +0 -0
  76. package/src/db/__pycache__/_sessions.cpython-310.pyc +0 -0
  77. package/src/db/__pycache__/_sessions.cpython-312.pyc +0 -0
  78. package/src/db/__pycache__/_sessions.cpython-314.pyc +0 -0
  79. package/src/db/__pycache__/_tasks.cpython-310.pyc +0 -0
  80. package/src/db/__pycache__/_tasks.cpython-312.pyc +0 -0
  81. package/src/db/__pycache__/_tasks.cpython-314.pyc +0 -0
  82. package/src/db/_core 2.py +417 -0
  83. package/src/db/_credentials 2.py +124 -0
  84. package/src/db/_entities 2.py +178 -0
  85. package/src/db/_episodic 2.py +738 -0
  86. package/src/db/_evolution 2.py +54 -0
  87. package/src/db/_fts 2.py +406 -0
  88. package/src/db/_learnings 2.py +168 -0
  89. package/src/db/_reminders 2.py +338 -0
  90. package/src/db/_schema 2.py +364 -0
  91. package/src/db/_sessions 2.py +300 -0
  92. package/src/db/_tasks 2.py +91 -0
  93. package/src/evolution_cycle 2.py +266 -0
  94. package/src/hnsw_index 2.py +254 -0
  95. package/src/hooks/auto_capture 2.py +208 -0
  96. package/src/hooks/caffeinate-guard 2.sh +8 -0
  97. package/src/hooks/capture-session 2.sh +21 -0
  98. package/src/hooks/capture-session.sh +2 -0
  99. package/src/hooks/capture-tool-logs 2.sh +127 -0
  100. package/src/hooks/capture-tool-logs.sh +3 -2
  101. package/src/hooks/daily-briefing-check 2.sh +33 -0
  102. package/src/hooks/inbox-hook 2.sh +76 -0
  103. package/src/hooks/inbox-hook.sh +3 -2
  104. package/src/hooks/post-compact 2.sh +148 -0
  105. package/src/hooks/post-compact.sh +1 -1
  106. package/src/hooks/pre-compact 2.sh +151 -0
  107. package/src/hooks/pre-compact.sh +1 -1
  108. package/src/hooks/session-start 2.sh +268 -0
  109. package/src/hooks/session-start.sh +6 -3
  110. package/src/hooks/session-stop 2.sh +140 -0
  111. package/src/hooks/session-stop.sh +1 -1
  112. package/src/kg_populate 2.py +290 -0
  113. package/src/knowledge_graph 2.py +257 -0
  114. package/src/maintenance 2.py +59 -0
  115. package/src/migrate_embeddings 2.py +122 -0
  116. package/src/plugin_loader 2.py +202 -0
  117. package/src/plugins/__init__ 2.py +0 -0
  118. package/src/plugins/__pycache__/__init__ 2.cpython-310.pyc +0 -0
  119. package/src/plugins/__pycache__/__init__.cpython-310.pyc +0 -0
  120. package/src/plugins/__pycache__/adaptive_mode 2.cpython-310.pyc +0 -0
  121. package/src/plugins/__pycache__/adaptive_mode.cpython-310.pyc +0 -0
  122. package/src/plugins/__pycache__/agents 2.cpython-310.pyc +0 -0
  123. package/src/plugins/__pycache__/agents.cpython-310.pyc +0 -0
  124. package/src/plugins/__pycache__/artifact_registry 2.cpython-310.pyc +0 -0
  125. package/src/plugins/__pycache__/artifact_registry.cpython-310.pyc +0 -0
  126. package/src/plugins/__pycache__/backup 2.cpython-310.pyc +0 -0
  127. package/src/plugins/__pycache__/backup.cpython-310.pyc +0 -0
  128. package/src/plugins/__pycache__/cognitive_memory 2.cpython-310.pyc +0 -0
  129. package/src/plugins/__pycache__/cognitive_memory.cpython-310.pyc +0 -0
  130. package/src/plugins/__pycache__/core_rules 2.cpython-310.pyc +0 -0
  131. package/src/plugins/__pycache__/core_rules.cpython-310.pyc +0 -0
  132. package/src/plugins/__pycache__/cortex 2.cpython-310.pyc +0 -0
  133. package/src/plugins/__pycache__/cortex.cpython-310.pyc +0 -0
  134. package/src/plugins/__pycache__/entities 2.cpython-310.pyc +0 -0
  135. package/src/plugins/__pycache__/entities.cpython-310.pyc +0 -0
  136. package/src/plugins/__pycache__/episodic_memory 2.cpython-310.pyc +0 -0
  137. package/src/plugins/__pycache__/episodic_memory.cpython-310.pyc +0 -0
  138. package/src/plugins/__pycache__/evolution 2.cpython-310.pyc +0 -0
  139. package/src/plugins/__pycache__/evolution.cpython-310.pyc +0 -0
  140. package/src/plugins/__pycache__/guard 2.cpython-310.pyc +0 -0
  141. package/src/plugins/__pycache__/guard.cpython-310.pyc +0 -0
  142. package/src/plugins/__pycache__/knowledge_graph_tools 2.cpython-310.pyc +0 -0
  143. package/src/plugins/__pycache__/knowledge_graph_tools.cpython-310.pyc +0 -0
  144. package/src/plugins/__pycache__/preferences 2.cpython-310.pyc +0 -0
  145. package/src/plugins/__pycache__/preferences.cpython-310.pyc +0 -0
  146. package/src/plugins/__pycache__/update 2.cpython-310.pyc +0 -0
  147. package/src/plugins/__pycache__/update.cpython-310.pyc +0 -0
  148. package/src/plugins/adaptive_mode 2.py +805 -0
  149. package/src/plugins/agents 2.py +52 -0
  150. package/src/plugins/artifact_registry 2.py +450 -0
  151. package/src/plugins/backup 2.py +104 -0
  152. package/src/plugins/cognitive_memory 2.py +564 -0
  153. package/src/plugins/core_rules 2.py +252 -0
  154. package/src/plugins/cortex 2.py +299 -0
  155. package/src/plugins/entities 2.py +67 -0
  156. package/src/plugins/episodic_memory 2.py +533 -0
  157. package/src/plugins/evolution 2.py +115 -0
  158. package/src/plugins/guard 2.py +746 -0
  159. package/src/plugins/knowledge_graph_tools 2.py +105 -0
  160. package/src/plugins/preferences 2.py +47 -0
  161. package/src/plugins/update 2.py +256 -0
  162. package/src/requirements 2.txt +12 -0
  163. package/src/rules/__init__ 2.py +0 -0
  164. package/src/rules/core-rules 2.json +331 -0
  165. package/src/rules/migrate 2.py +207 -0
  166. package/src/scripts/check-context 2.py +264 -0
  167. package/src/scripts/deep-sleep/apply_findings.py +58 -0
  168. package/src/scripts/deep-sleep/synthesize-prompt.md +30 -1
  169. package/src/scripts/nexo-auto-update 2.py +6 -0
  170. package/src/scripts/nexo-backup 2.sh +25 -0
  171. package/src/scripts/nexo-brain-activation 2.sh +140 -0
  172. package/src/scripts/nexo-catchup 2.py +242 -0
  173. package/src/scripts/nexo-cognitive-decay 2.py +182 -0
  174. package/src/scripts/nexo-daily-self-audit 2.py +552 -0
  175. package/src/scripts/nexo-deep-sleep 2.sh +97 -0
  176. package/src/scripts/nexo-evolution-run 2.py +597 -0
  177. package/src/scripts/nexo-followup-hygiene 2.py +112 -0
  178. package/src/scripts/nexo-github-monitor 2.py +256 -0
  179. package/src/scripts/nexo-immune 2.py +927 -0
  180. package/src/scripts/nexo-inbox-hook 2.sh +74 -0
  181. package/src/scripts/nexo-install 2.py +6 -0
  182. package/src/scripts/nexo-learning-housekeep 2.py +245 -0
  183. package/src/scripts/nexo-learning-validator 2.py +207 -0
  184. package/src/scripts/nexo-migrate 2.py +232 -0
  185. package/src/scripts/nexo-postmortem-consolidator 2.py +421 -0
  186. package/src/scripts/nexo-pre-commit 2.py +120 -0
  187. package/src/scripts/nexo-prevent-sleep 2.sh +29 -0
  188. package/src/scripts/nexo-proactive-dashboard 2.py +345 -0
  189. package/src/scripts/nexo-reflection 2.py +253 -0
  190. package/src/scripts/nexo-runtime-preflight 2.py +274 -0
  191. package/src/scripts/nexo-send-email 2.py +25 -0
  192. package/src/scripts/nexo-send-reply 2.py +178 -0
  193. package/src/scripts/nexo-sleep 2.py +592 -0
  194. package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
  195. package/src/scripts/nexo-synthesis 2.py +253 -0
  196. package/src/scripts/nexo-tcc-approve 2.sh +79 -0
  197. package/src/scripts/nexo-update 2.sh +161 -0
  198. package/src/scripts/nexo-watchdog 2.sh +878 -0
  199. package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
  200. package/src/server 2.py +733 -0
  201. package/src/server.py +6 -1
  202. package/src/storage_router 2.py +32 -0
  203. package/src/tools_coordination 2.py +102 -0
  204. package/src/tools_credentials 2.py +68 -0
  205. package/src/tools_learnings 2.py +220 -0
  206. package/src/tools_menu 2.py +227 -0
  207. package/src/tools_reminders 2.py +86 -0
  208. package/src/tools_reminders_crud 2.py +159 -0
  209. package/src/tools_reminders_crud.py +7 -0
  210. package/src/tools_sessions 2.py +476 -0
  211. package/src/tools_task_history 2.py +57 -0
  212. package/templates/CLAUDE.md 2.template +63 -0
  213. package/templates/openclaw 2.json +13 -0
  214. package/tests/__init__ 2.py +0 -0
  215. package/tests/conftest 2.py +71 -0
  216. package/tests/test_cognitive 2.py +205 -0
  217. package/tests/test_knowledge_graph 2.py +140 -0
  218. package/tests/test_migrations 2.py +137 -0
  219. package/src/__pycache__/hnsw_index.cpython-314.pyc +0 -0
  220. package/src/cognitive/__pycache__/__init__.cpython-312.pyc +0 -0
  221. package/src/cognitive/__pycache__/__init__.cpython-314.pyc +0 -0
  222. package/src/cognitive/__pycache__/_core.cpython-312.pyc +0 -0
  223. package/src/cognitive/__pycache__/_core.cpython-314.pyc +0 -0
  224. package/src/cognitive/__pycache__/_decay.cpython-312.pyc +0 -0
  225. package/src/cognitive/__pycache__/_decay.cpython-314.pyc +0 -0
  226. package/src/cognitive/__pycache__/_ingest.cpython-312.pyc +0 -0
  227. package/src/cognitive/__pycache__/_ingest.cpython-314.pyc +0 -0
  228. package/src/cognitive/__pycache__/_memory.cpython-312.pyc +0 -0
  229. package/src/cognitive/__pycache__/_memory.cpython-314.pyc +0 -0
  230. package/src/cognitive/__pycache__/_search.cpython-312.pyc +0 -0
  231. package/src/cognitive/__pycache__/_search.cpython-314.pyc +0 -0
  232. package/src/cognitive/__pycache__/_trust.cpython-312.pyc +0 -0
  233. package/src/cognitive/__pycache__/_trust.cpython-314.pyc +0 -0
  234. package/src/plugins/__pycache__/adaptive_mode.cpython-314.pyc +0 -0
  235. package/src/scripts/deep-sleep/__pycache__/extract.cpython-314.pyc +0 -0
@@ -0,0 +1,264 @@
1
+ #!/usr/bin/env python3
2
+ """Context checker for NEXO operations - prevents duplicate actions.
3
+
4
+ Mechanical checks (email sent, file exists, action done) run in Python.
5
+ When the 'smart' command is used, passes context to Claude CLI for
6
+ intelligent duplicate/conflict detection that goes beyond file checks.
7
+ """
8
+
9
+ import os
10
+ import sys
11
+ import json
12
+ import hashlib
13
+ import subprocess
14
+ from datetime import datetime
15
+ from pathlib import Path
16
+
17
+ NEXO_HOME = Path(os.environ.get("NEXO_HOME", str(Path.home() / ".nexo")))
18
+
19
+ CLAUDE_CLI = Path.home() / ".local" / "bin" / "claude"
20
+
21
+ class ContextChecker:
22
+ def __init__(self):
23
+ self.state_dir = NEXO_HOME / 'state'
24
+ self.state_dir.mkdir(exist_ok=True)
25
+
26
+ def check_email_sent(self, to_addr, subject, since_hours=72):
27
+ """Check if email was already sent to address with subject."""
28
+ sent_path = Path.home() / 'mail' / '.nexo-sent' / '.Sent' # Configure for your mail setup
29
+ if not sent_path.exists():
30
+ return False
31
+
32
+ subject_lower = subject.lower()
33
+ to_lower = to_addr.lower()
34
+ cutoff = datetime.now().timestamp() - (since_hours * 3600)
35
+ cur_dir = sent_path / 'cur'
36
+ if not cur_dir.exists():
37
+ return False
38
+
39
+ for msg_file in cur_dir.iterdir():
40
+ try:
41
+ if msg_file.stat().st_mtime < cutoff:
42
+ continue
43
+ content = msg_file.read_text(errors='ignore')
44
+ except (OSError, UnicodeDecodeError):
45
+ continue
46
+
47
+ content_lower = content.lower()
48
+ if f"to:{to_lower}" in content_lower or f"to: {to_lower}" in content_lower:
49
+ if subject_lower in content_lower:
50
+ return True
51
+ return False
52
+
53
+ def check_file_exists(self, pattern, search_dirs=None):
54
+ """Check if file matching pattern exists in common locations."""
55
+ if search_dirs is None:
56
+ search_dirs = [
57
+ '/var/www/vhosts',
58
+ str(NEXO_HOME),
59
+ '/opt'
60
+ ]
61
+
62
+ for base_dir in search_dirs:
63
+ if not os.path.exists(base_dir):
64
+ continue
65
+ matches = []
66
+ try:
67
+ for root, _, files in os.walk(base_dir):
68
+ for filename in files:
69
+ if pattern in filename:
70
+ matches.append(str(Path(root) / filename))
71
+ if len(matches) >= 5:
72
+ return matches
73
+ except OSError:
74
+ continue
75
+ return []
76
+
77
+ def check_action_done(self, action_type, identifier, ttl_days=7):
78
+ """Check if action was already performed recently."""
79
+ action_file = self.state_dir / 'actions.json'
80
+
81
+ # Load existing actions
82
+ actions = {}
83
+ if action_file.exists():
84
+ with open(action_file) as f:
85
+ actions = json.load(f)
86
+
87
+ # Create action key
88
+ key = hashlib.md5(f"{action_type}:{identifier}".encode()).hexdigest()
89
+
90
+ # Check if exists and not expired
91
+ if key in actions:
92
+ action_time = datetime.fromisoformat(actions[key]['timestamp'])
93
+ age_days = (datetime.now() - action_time).days
94
+ if age_days < ttl_days:
95
+ return True, actions[key]
96
+
97
+ return False, None
98
+
99
+ def mark_action_done(self, action_type, identifier, metadata=None):
100
+ """Mark action as completed."""
101
+ action_file = self.state_dir / 'actions.json'
102
+
103
+ # Load existing actions
104
+ actions = {}
105
+ if action_file.exists():
106
+ with open(action_file) as f:
107
+ actions = json.load(f)
108
+
109
+ # Add new action
110
+ key = hashlib.md5(f"{action_type}:{identifier}".encode()).hexdigest()
111
+ actions[key] = {
112
+ 'type': action_type,
113
+ 'identifier': identifier,
114
+ 'timestamp': datetime.now().isoformat(),
115
+ 'metadata': metadata or {}
116
+ }
117
+
118
+ # Save
119
+ with open(action_file, 'w') as f:
120
+ json.dump(actions, f, indent=2)
121
+
122
+ return key
123
+
124
+ def smart_check(action_description: str, context: str = "") -> dict:
125
+ """Use Claude CLI to intelligently check if an action would be redundant.
126
+
127
+ Goes beyond simple file/hash checks — understands intent and context
128
+ to detect semantic duplicates (e.g., "send welcome email" vs
129
+ "email onboarding message" to same person).
130
+ """
131
+ checker = ContextChecker()
132
+
133
+ # Gather mechanical context first
134
+ state_file = checker.state_dir / 'actions.json'
135
+ recent_actions = {}
136
+ if state_file.exists():
137
+ try:
138
+ all_actions = json.loads(state_file.read_text())
139
+ cutoff = datetime.now().timestamp() - (7 * 86400)
140
+ for k, v in all_actions.items():
141
+ try:
142
+ ts = datetime.fromisoformat(v['timestamp']).timestamp()
143
+ if ts > cutoff:
144
+ recent_actions[k] = v
145
+ except (ValueError, KeyError):
146
+ pass
147
+ except Exception:
148
+ pass
149
+
150
+ if not CLAUDE_CLI.exists():
151
+ return {"redundant": False, "reason": "CLI unavailable, cannot smart-check"}
152
+
153
+ prompt = f"""You are a context deduplication engine for NEXO operations.
154
+
155
+ PROPOSED ACTION:
156
+ {action_description}
157
+
158
+ ADDITIONAL CONTEXT:
159
+ {context or "None"}
160
+
161
+ RECENT ACTIONS (last 7 days):
162
+ {json.dumps(list(recent_actions.values()), indent=1, default=str)}
163
+
164
+ Respond with ONLY valid JSON (no markdown):
165
+ {{
166
+ "redundant": true/false,
167
+ "confidence": 0.0-1.0,
168
+ "reason": "<one line explanation>",
169
+ "matching_action": "<identifier of matching action if redundant, else null>"
170
+ }}
171
+
172
+ Rules:
173
+ - Same recipient + same intent within 72h = redundant
174
+ - Same file modification with same content = redundant
175
+ - Similar but different scope (e.g., different recipients) = NOT redundant
176
+ - When in doubt, say not redundant (false negatives are cheaper than false positives)"""
177
+ )
178
+ if auth_check.returncode != 0:
179
+ # CLI not authenticated, skip gracefully
180
+ return {"redundant": False, "reason": "CLI not authenticated — skipped analysis", "suggestion": "N/A"}
181
+
182
+ env = os.environ.copy()
183
+ env["NEXO_HEADLESS"] = "1" # Skip stop hook post-mortem
184
+ env.pop("CLAUDECODE", None)
185
+ env.pop("CLAUDE_CODE", None)
186
+
187
+ try:
188
+ result = subprocess.run(
189
+ [str(CLAUDE_CLI), "-p", prompt, "--model", "opus", "--output-format", "text",
190
+ "--allowedTools", "Read,Write,Edit,Glob,Grep,Bash,mcp__nexo__*"],
191
+ capture_output=True, text=True, timeout=21600, env=env
192
+ )
193
+ if result.returncode == 0:
194
+ text = result.stdout.strip()
195
+ if "```json" in text:
196
+ text = text.split("```json")[1].split("```")[0]
197
+ elif "```" in text:
198
+ text = text.split("```")[1].split("```")[0]
199
+ return json.loads(text.strip())
200
+ except Exception:
201
+ pass
202
+
203
+ return {"redundant": False, "reason": "CLI check failed, defaulting to not redundant"}
204
+
205
+
206
+ def main():
207
+ """CLI interface for context checking."""
208
+ if len(sys.argv) < 3:
209
+ print("Usage: check-context.py <command> <args>")
210
+ print("Commands:")
211
+ print(" email <to> <subject> - Check if email was sent")
212
+ print(" file <pattern> - Check if file exists")
213
+ print(" action <type> <id> - Check if action was done")
214
+ print(" smart <description> [ctx] - Intelligent duplicate check via CLI")
215
+ sys.exit(1)
216
+
217
+ checker = ContextChecker()
218
+ command = sys.argv[1]
219
+
220
+ if command == 'email':
221
+ if len(sys.argv) < 4:
222
+ print("Usage: check-context.py email <to> <subject>")
223
+ sys.exit(1)
224
+ exists = checker.check_email_sent(sys.argv[2], sys.argv[3])
225
+ print("EXISTS" if exists else "NOT_FOUND")
226
+ sys.exit(0 if not exists else 1)
227
+
228
+ elif command == 'file':
229
+ files = checker.check_file_exists(sys.argv[2])
230
+ if files:
231
+ print("\n".join(files))
232
+ sys.exit(1)
233
+ else:
234
+ print("NOT_FOUND")
235
+ sys.exit(0)
236
+
237
+ elif command == 'action':
238
+ if len(sys.argv) < 4:
239
+ print("Usage: check-context.py action <type> <id>")
240
+ sys.exit(1)
241
+ done, data = checker.check_action_done(sys.argv[2], sys.argv[3])
242
+ if done:
243
+ print(f"DONE: {data}")
244
+ sys.exit(1)
245
+ else:
246
+ print("NOT_DONE")
247
+ sys.exit(0)
248
+
249
+ elif command == 'smart':
250
+ if len(sys.argv) < 3:
251
+ print("Usage: check-context.py smart <description> [context]")
252
+ sys.exit(1)
253
+ description = sys.argv[2]
254
+ context = sys.argv[3] if len(sys.argv) > 3 else ""
255
+ result = smart_check(description, context)
256
+ print(json.dumps(result, indent=2))
257
+ sys.exit(1 if result.get("redundant") else 0)
258
+
259
+ else:
260
+ print(f"Unknown command: {command}")
261
+ sys.exit(1)
262
+
263
+ if __name__ == '__main__':
264
+ main()
@@ -154,6 +154,55 @@ def update_calibration_mood(synthesis: dict) -> dict:
154
154
  return {"success": False, "error": str(e)}
155
155
 
156
156
 
157
+ def calibrate_trust_score(synthesis: dict, target_date: str) -> dict:
158
+ """Set the daily trust score from Deep Sleep analysis.
159
+
160
+ This is the authoritative score for the day — replaces incremental
161
+ adjustments with a holistic evaluation of the entire day.
162
+ """
163
+ trust_cal = synthesis.get("trust_calibration")
164
+ if not trust_cal or "score" not in trust_cal:
165
+ return {"success": False, "error": "no trust_calibration in synthesis"}
166
+
167
+ score = max(0, min(100, trust_cal["score"]))
168
+ reasoning = trust_cal.get("reasoning", "Deep Sleep calibration")
169
+ trend = trust_cal.get("trend", "stable")
170
+ highlights = trust_cal.get("highlights", [])
171
+ lowlights = trust_cal.get("lowlights", [])
172
+
173
+ context = (
174
+ f"Deep Sleep {target_date} | trend: {trend} | "
175
+ f"highlights: {', '.join(highlights[:3])} | "
176
+ f"lowlights: {', '.join(lowlights[:3])}"
177
+ )
178
+
179
+ try:
180
+ # Get current score for delta calculation
181
+ db = sqlite3.connect(str(COGNITIVE_DB))
182
+ row = db.execute(
183
+ "SELECT score FROM trust_score ORDER BY id DESC LIMIT 1"
184
+ ).fetchone()
185
+ old_score = row[0] if row else 50.0
186
+ delta = score - old_score
187
+
188
+ db.execute(
189
+ "INSERT INTO trust_score (score, event, delta, context) VALUES (?, ?, ?, ?)",
190
+ (score, f"deep_sleep_calibration: {reasoning[:200]}", delta, context[:500])
191
+ )
192
+ db.commit()
193
+ db.close()
194
+
195
+ return {
196
+ "success": True,
197
+ "old_score": old_score,
198
+ "new_score": score,
199
+ "delta": delta,
200
+ "trend": trend,
201
+ }
202
+ except Exception as e:
203
+ return {"success": False, "error": str(e)}
204
+
205
+
157
206
  def create_abandoned_followups(synthesis: dict) -> list[dict]:
158
207
  """Create followups for truly abandoned projects."""
159
208
  results = []
@@ -527,6 +576,15 @@ def main():
527
576
  else:
528
577
  print(f" Mood skip: {mood_result.get('error', '?')}")
529
578
 
579
+ # Calibrate trust score (authoritative daily score from Deep Sleep)
580
+ print("[apply] Calibrating trust score...")
581
+ trust_result = calibrate_trust_score(synthesis, target_date)
582
+ if trust_result.get("success"):
583
+ stats["applied"] += 1
584
+ print(f" Trust: {trust_result['old_score']:.0f} → {trust_result['new_score']:.0f} (Δ{trust_result['delta']:+.0f}, {trust_result['trend']})")
585
+ else:
586
+ print(f" Trust skip: {trust_result.get('error', '?')}")
587
+
530
588
  # Create followups for abandoned projects
531
589
  abandoned_results = create_abandoned_followups(synthesis)
532
590
  for r in abandoned_results:
@@ -52,7 +52,28 @@ Consolidate `abandoned_projects` from all sessions:
52
52
  - Cross-reference across sessions — was the abandoned work picked up later in another session?
53
53
  - Only flag projects that are truly abandoned (no followup AND not resumed)
54
54
 
55
- ### 7. Consolidated Actions
55
+ ### 7. Trust Calibration (CRITICAL)
56
+ Score the agent's performance for the day on a scale of 0-100. This score becomes the agent's trust score and directly affects its autonomy level the next day. Be fair but honest.
57
+
58
+ Scoring guide:
59
+ - **90-100**: Flawless day. Zero corrections needed. Proactive. Anticipated user needs. Deployed code without issues.
60
+ - **70-89**: Good day. Minor corrections, quickly resolved. Mostly proactive. User satisfied.
61
+ - **50-69**: Average day. Some corrections, some reactive behavior. Mixed results.
62
+ - **30-49**: Below average. Multiple corrections. Repeated mistakes. User had to push.
63
+ - **0-29**: Bad day. Many corrections, repeated errors, user frustrated. Broke things.
64
+
65
+ Consider ALL of these:
66
+ - Number and severity of corrections (most important signal)
67
+ - Tasks completed successfully vs failed
68
+ - Did the agent act autonomously or wait to be told?
69
+ - Did the agent catch its own mistakes or did the user?
70
+ - Did the agent repeat known errors (worst offense)?
71
+ - User emotional signals throughout the day
72
+ - Code deployed: did it work first try?
73
+
74
+ The score should feel fair. A day with 2 minor corrections and 10 tasks completed is still a good day (75+). A day with 1 catastrophic error might be a 40 even if everything else was fine.
75
+
76
+ ### 8. Consolidated Actions
56
77
  Merge and deduplicate all findings into a final action list. Each action should have:
57
78
  - `action_type`: `learning_add`, `followup_create`, `morning_briefing_item`
58
79
  - `action_class`: `auto_apply` (confidence >= 0.8, reversible) or `draft_for_morning` (confidence < 0.8 or high impact)
@@ -147,6 +168,14 @@ Return ONLY valid JSON. No markdown code fences. No explanation text.
147
168
  }
148
169
  ],
149
170
 
171
+ "trust_calibration": {
172
+ "score": 72,
173
+ "reasoning": "Why this score -- based on corrections, completions, autonomy, proactivity, and user satisfaction signals across ALL sessions",
174
+ "highlights": ["What went well"],
175
+ "lowlights": ["What went poorly"],
176
+ "trend": "improving|stable|declining"
177
+ },
178
+
150
179
  "summary": "2-3 sentence overall assessment of the day"
151
180
  }
152
181
  ```
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env python3
2
+ """DEPRECATED: Updates are handled automatically by NEXO on startup."""
3
+ import sys
4
+ print("This script is deprecated. NEXO auto-updates on startup.")
5
+ print("To update manually, use the nexo_update MCP tool.")
6
+ sys.exit(0)
@@ -0,0 +1,25 @@
1
+ #!/bin/bash
2
+ # NEXO DB hourly backup — crontab: 0 * * * * $NEXO_HOME/scripts/nexo-backup.sh
3
+ NEXO_HOME="${NEXO_HOME:-$HOME/.nexo}"
4
+ NEXO_DIR="$NEXO_HOME"
5
+ BACKUP_DIR="$NEXO_HOME/backups"
6
+ WEEKLY_DIR="$BACKUP_DIR/weekly"
7
+ DB="$NEXO_HOME/data/nexo.db"
8
+ RETENTION_HOURS=48
9
+
10
+ mkdir -p "$BACKUP_DIR" "$WEEKLY_DIR"
11
+
12
+ # Hourly backup
13
+ TIMESTAMP=$(date +%Y-%m-%d-%H%M)
14
+ sqlite3 "$DB" ".backup '$BACKUP_DIR/nexo-$TIMESTAMP.db'"
15
+
16
+ # Weekly backup — save one per week (Sundays)
17
+ WEEK=$(date +%Y-W%V)
18
+ WEEKLY_FILE="$WEEKLY_DIR/weekly-$WEEK.db"
19
+ if [ ! -f "$WEEKLY_FILE" ] && [ "$(date +%u)" = "7" ]; then
20
+ cp "$BACKUP_DIR/nexo-$TIMESTAMP.db" "$WEEKLY_FILE"
21
+ fi
22
+
23
+ # Cleanup: hourly >48h, weekly >90 days
24
+ find "$BACKUP_DIR" -maxdepth 1 -name "nexo-*.db" -mmin +$((RETENTION_HOURS * 60)) -delete
25
+ find "$WEEKLY_DIR" -name "weekly-*.db" -mtime +90 -delete
@@ -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