nexo-brain 5.3.13 → 5.3.15

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 (230) hide show
  1. package/.claude-plugin/plugin.json +1 -1
  2. package/bin/nexo-brain.js +52 -1
  3. package/package.json +1 -1
  4. package/src/crons/sync.py +18 -4
  5. package/src/dashboard/static/favicon 2.svg +32 -0
  6. package/src/dashboard/static/nexo-logo 2.png +0 -0
  7. package/src/dashboard/static/nexo-logo 2.svg +40 -0
  8. package/src/dashboard/static/style 2.css +2458 -0
  9. package/src/dashboard/templates/adaptive 2.html +118 -0
  10. package/src/dashboard/templates/artifacts 2.html +133 -0
  11. package/src/dashboard/templates/backups 2.html +136 -0
  12. package/src/dashboard/templates/base 2.html +417 -0
  13. package/src/dashboard/templates/calendar 2.html +591 -0
  14. package/src/dashboard/templates/chat 2.html +356 -0
  15. package/src/dashboard/templates/claims 2.html +259 -0
  16. package/src/dashboard/templates/cortex 2.html +321 -0
  17. package/src/dashboard/templates/credentials 2.html +128 -0
  18. package/src/dashboard/templates/crons 2.html +370 -0
  19. package/src/dashboard/templates/dashboard 2.html +494 -0
  20. package/src/dashboard/templates/dreams 2.html +252 -0
  21. package/src/dashboard/templates/email 2.html +160 -0
  22. package/src/dashboard/templates/evolution 2.html +189 -0
  23. package/src/dashboard/templates/feed 2.html +249 -0
  24. package/src/dashboard/templates/followup_health 2.html +170 -0
  25. package/src/dashboard/templates/graph 2.html +201 -0
  26. package/src/dashboard/templates/guard 2.html +259 -0
  27. package/src/dashboard/templates/inbox 2.html +251 -0
  28. package/src/dashboard/templates/memory 2.html +420 -0
  29. package/src/dashboard/templates/operations 2.html +608 -0
  30. package/src/dashboard/templates/plugins 2.html +185 -0
  31. package/src/dashboard/templates/protocol 2.html +199 -0
  32. package/src/dashboard/templates/rules 2.html +246 -0
  33. package/src/dashboard/templates/sentiment 2.html +247 -0
  34. package/src/dashboard/templates/sessions 2.html +218 -0
  35. package/src/dashboard/templates/skills 2.html +329 -0
  36. package/src/dashboard/templates/somatic 2.html +73 -0
  37. package/src/dashboard/templates/triggers 2.html +133 -0
  38. package/src/dashboard/templates/trust 2.html +360 -0
  39. package/src/db/__init__ 2.py +259 -0
  40. package/src/db/_core 2.py +437 -0
  41. package/src/db/_credentials 2.py +124 -0
  42. package/src/db/_entities.py +1 -1
  43. package/src/db/_episodic 2.py +762 -0
  44. package/src/db/_evolution 2.py +54 -0
  45. package/src/db/_fts 2.py +406 -0
  46. package/src/db/_goal_profiles 2.py +376 -0
  47. package/src/db/_hot_context 2.py +660 -0
  48. package/src/db/_outcomes 2.py +800 -0
  49. package/src/db/_personal_scripts 2.py +582 -0
  50. package/src/db/_sessions 2.py +330 -0
  51. package/src/db/_tasks 2.py +91 -0
  52. package/src/db/_watchers 2.py +173 -0
  53. package/src/doctor/formatters 2.py +52 -0
  54. package/src/doctor/models 2.py +69 -0
  55. package/src/doctor/planes 2.py +87 -0
  56. package/src/doctor/providers/__init__ 2.py +1 -0
  57. package/src/doctor/providers/deep 2.py +367 -0
  58. package/src/evolution_cycle 2.py +519 -0
  59. package/src/hooks/auto_capture 2.py +208 -0
  60. package/src/hooks/caffeinate-guard 2.sh +8 -0
  61. package/src/hooks/capture-session 2.sh +21 -0
  62. package/src/hooks/capture-tool-logs 2.sh +158 -0
  63. package/src/hooks/daily-briefing-check 2.sh +33 -0
  64. package/src/hooks/heartbeat-enforcement 2.py +90 -0
  65. package/src/hooks/heartbeat-posttool 2.sh +18 -0
  66. package/src/hooks/inbox-hook 2.sh +76 -0
  67. package/src/hooks/post-compact 2.sh +152 -0
  68. package/src/hooks/pre-compact 2.sh +169 -0
  69. package/src/hooks/protocol-guardrail 2.sh +10 -0
  70. package/src/hooks/protocol-pretool-guardrail 2.sh +9 -0
  71. package/src/hooks/session-stop 2.sh +52 -0
  72. package/src/kg_populate 2.py +292 -0
  73. package/src/maintenance 2.py +53 -0
  74. package/src/memory_backends 2.py +71 -0
  75. package/src/migrate_embeddings 2.py +124 -0
  76. package/src/nexo_sdk 2.py +103 -0
  77. package/src/observability 2.py +199 -0
  78. package/src/plugin_loader 2.py +217 -0
  79. package/src/plugins/__init__ 2.py +0 -0
  80. package/src/plugins/agents.py +10 -3
  81. package/src/plugins/artifact_registry 2.py +450 -0
  82. package/src/plugins/backup 2.py +127 -0
  83. package/src/plugins/claims_tools 2.py +119 -0
  84. package/src/plugins/cognitive_memory 2.py +609 -0
  85. package/src/plugins/core_rules 2.py +252 -0
  86. package/src/plugins/cortex 2.py +1155 -0
  87. package/src/plugins/entities 2.py +67 -0
  88. package/src/plugins/episodic_memory 2.py +560 -0
  89. package/src/plugins/evolution 2.py +167 -0
  90. package/src/plugins/goal_engine 2.py +142 -0
  91. package/src/plugins/guard 2.py +862 -0
  92. package/src/plugins/impact 2.py +29 -0
  93. package/src/plugins/knowledge_graph_tools 2.py +137 -0
  94. package/src/plugins/media_memory_tools 2.py +98 -0
  95. package/src/plugins/memory_export 2.py +196 -0
  96. package/src/plugins/outcomes 2.py +130 -0
  97. package/src/plugins/personal_scripts 2.py +117 -0
  98. package/src/plugins/preferences 2.py +47 -0
  99. package/src/plugins/protocol 2.py +1449 -0
  100. package/src/plugins/schedule.py +2 -1
  101. package/src/plugins/simple_api 2.py +106 -0
  102. package/src/plugins/skills 2.py +341 -0
  103. package/src/plugins/state_watchers 2.py +79 -0
  104. package/src/plugins/update 2.py +986 -0
  105. package/src/plugins/user_state_tools 2.py +43 -0
  106. package/src/plugins/workflow 2.py +588 -0
  107. package/src/protocol_settings 2.py +59 -0
  108. package/src/public_contribution 2.py +466 -0
  109. package/src/public_evolution_queue 2.py +241 -0
  110. package/src/requirements 2.txt +14 -0
  111. package/src/requirements.txt +1 -1
  112. package/src/retroactive_learnings 2.py +373 -0
  113. package/src/rules/__init__ 2.py +0 -0
  114. package/src/rules/core-rules 2.json +331 -0
  115. package/src/rules/migrate 2.py +207 -0
  116. package/src/runtime_power 2.py +874 -0
  117. package/src/runtime_power.py +18 -1
  118. package/src/script_registry 2.py +1559 -0
  119. package/src/scripts/check-context 2.py +272 -0
  120. package/src/scripts/deep-sleep/apply_findings 2.py +2327 -0
  121. package/src/scripts/deep-sleep/collect 2.py +928 -0
  122. package/src/scripts/deep-sleep/extract 2.py +330 -0
  123. package/src/scripts/deep-sleep/extract-prompt 2.md +285 -0
  124. package/src/scripts/deep-sleep/synthesize 2.py +312 -0
  125. package/src/scripts/deep-sleep/synthesize-prompt 2.md +336 -0
  126. package/src/scripts/nexo-agent-run 2.py +75 -0
  127. package/src/scripts/nexo-auto-update 2.py +6 -0
  128. package/src/scripts/nexo-backup 2.sh +25 -0
  129. package/src/scripts/nexo-brain-activation 2.sh +140 -0
  130. package/src/scripts/nexo-catchup 2.py +300 -0
  131. package/src/scripts/nexo-cognitive-decay 2.py +257 -0
  132. package/src/scripts/nexo-cortex-cycle 2.py +293 -0
  133. package/src/scripts/nexo-cron-wrapper 2.sh +53 -0
  134. package/src/scripts/nexo-cron-wrapper.sh +7 -0
  135. package/src/scripts/nexo-daily-self-audit 2.py +2161 -0
  136. package/src/scripts/nexo-dashboard 2.sh +29 -0
  137. package/src/scripts/nexo-deep-sleep 2.sh +86 -0
  138. package/src/scripts/nexo-evolution-run 2.py +1664 -0
  139. package/src/scripts/nexo-followup-hygiene 2.py +139 -0
  140. package/src/scripts/nexo-hook-record 2.py +42 -0
  141. package/src/scripts/nexo-immune 2.py +936 -0
  142. package/src/scripts/nexo-impact-scorer 2.py +117 -0
  143. package/src/scripts/nexo-inbox-hook 2.sh +74 -0
  144. package/src/scripts/nexo-install 2.py +6 -0
  145. package/src/scripts/nexo-learning-housekeep 2.py +401 -0
  146. package/src/scripts/nexo-learning-validator 2.py +266 -0
  147. package/src/scripts/nexo-migrate 2.py +260 -0
  148. package/src/scripts/nexo-outcome-checker 2.py +127 -0
  149. package/src/scripts/nexo-postmortem-consolidator 2.py +456 -0
  150. package/src/scripts/nexo-pre-commit 2.py +120 -0
  151. package/src/scripts/nexo-prevent-sleep 2.sh +35 -0
  152. package/src/scripts/nexo-proactive-dashboard 2.py +354 -0
  153. package/src/scripts/nexo-reflection 2.py +256 -0
  154. package/src/scripts/nexo-runtime-preflight 2.py +274 -0
  155. package/src/scripts/nexo-sleep 2.py +631 -0
  156. package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
  157. package/src/scripts/nexo-sync-clients 2.py +16 -0
  158. package/src/scripts/nexo-synthesis 2.py +475 -0
  159. package/src/scripts/nexo-tcc-approve 2.sh +79 -0
  160. package/src/scripts/nexo-update 2.sh +306 -0
  161. package/src/scripts/nexo-watchdog 2.sh +1207 -0
  162. package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
  163. package/src/scripts/rehydrate_learnings_from_archive 2.py +245 -0
  164. package/src/server 2.py +1296 -0
  165. package/src/skills/run-nexo-audit-phase/guide 2.md +43 -0
  166. package/src/skills/run-nexo-audit-phase/skill 2.json +59 -0
  167. package/src/skills/run-nexo-core-fix-cycle/guide 2.md +17 -0
  168. package/src/skills/run-nexo-core-fix-cycle/script 2.py +276 -0
  169. package/src/skills/run-nexo-core-fix-cycle/skill 2.json +58 -0
  170. package/src/skills/run-release-final-audit/guide 2.md +16 -0
  171. package/src/skills/run-release-final-audit/script 2.py +259 -0
  172. package/src/skills/run-release-final-audit/skill 2.json +77 -0
  173. package/src/skills/run-runtime-doctor/guide 2.md +12 -0
  174. package/src/skills/run-runtime-doctor/script 2.py +21 -0
  175. package/src/skills/run-runtime-doctor/skill 2.json +25 -0
  176. package/src/skills_runtime 2.py +932 -0
  177. package/src/state_watchers_runtime 2.py +475 -0
  178. package/src/storage_router 2.py +32 -0
  179. package/src/system_catalog 2.py +786 -0
  180. package/src/tools_coordination 2.py +103 -0
  181. package/src/tools_credentials 2.py +68 -0
  182. package/src/tools_drive 2.py +487 -0
  183. package/src/tools_hot_context 2.py +163 -0
  184. package/src/tools_learnings 2.py +612 -0
  185. package/src/tools_menu 2.py +229 -0
  186. package/src/tools_reminders 2.py +88 -0
  187. package/src/tools_reminders_crud 2.py +363 -0
  188. package/src/tools_sessions 2.py +1054 -0
  189. package/src/tools_system_catalog 2.py +19 -0
  190. package/src/tools_task_history 2.py +57 -0
  191. package/src/tools_transcripts 2.py +98 -0
  192. package/src/transcript_utils 2.py +412 -0
  193. package/src/user_context 2.py +46 -0
  194. package/src/user_data_portability 2.py +328 -0
  195. package/src/user_state_model 2.py +170 -0
  196. package/templates/CLAUDE.md 2.template +108 -0
  197. package/templates/CODEX.AGENTS.md 2.template +66 -0
  198. package/templates/launchagents/README 2.md +132 -0
  199. package/templates/launchagents/com.nexo.auto-close-sessions 2.plist +39 -0
  200. package/templates/launchagents/com.nexo.auto-close-sessions.plist +1 -1
  201. package/templates/launchagents/com.nexo.catchup 2.plist +39 -0
  202. package/templates/launchagents/com.nexo.catchup.plist +1 -1
  203. package/templates/launchagents/com.nexo.cognitive-decay 2.plist +40 -0
  204. package/templates/launchagents/com.nexo.dashboard 2.plist +43 -0
  205. package/templates/launchagents/com.nexo.dashboard.plist +1 -1
  206. package/templates/launchagents/com.nexo.deep-sleep 2.plist +43 -0
  207. package/templates/launchagents/com.nexo.deep-sleep.plist +1 -1
  208. package/templates/launchagents/com.nexo.evolution 2.plist +44 -0
  209. package/templates/launchagents/com.nexo.evolution.plist +1 -1
  210. package/templates/launchagents/com.nexo.followup-hygiene 2.plist +45 -0
  211. package/templates/launchagents/com.nexo.followup-hygiene.plist +1 -1
  212. package/templates/launchagents/com.nexo.immune 2.plist +41 -0
  213. package/templates/launchagents/com.nexo.immune.plist +1 -1
  214. package/templates/launchagents/com.nexo.postmortem 2.plist +45 -0
  215. package/templates/launchagents/com.nexo.postmortem.plist +1 -1
  216. package/templates/launchagents/com.nexo.self-audit 2.plist +47 -0
  217. package/templates/launchagents/com.nexo.self-audit.plist +1 -1
  218. package/templates/launchagents/com.nexo.synthesis 2.plist +45 -0
  219. package/templates/launchagents/com.nexo.synthesis.plist +1 -1
  220. package/templates/launchagents/com.nexo.watchdog 2.plist +37 -0
  221. package/templates/launchagents/com.nexo.watchdog.plist +1 -1
  222. package/templates/nexo_helper 2.py +301 -0
  223. package/templates/openclaw 2.json +13 -0
  224. package/templates/plugin-template 2.py +40 -0
  225. package/templates/script-template 2.py +59 -0
  226. package/templates/script-template 2.sh +13 -0
  227. package/templates/script-template.py +5 -4
  228. package/templates/skill-script-template 2.py +48 -0
  229. package/templates/skill-script-template.py +2 -1
  230. package/templates/skill-template 2.md +33 -0
@@ -0,0 +1,217 @@
1
+ from __future__ import annotations
2
+ """Dynamic plugin loader for NEXO MCP server."""
3
+
4
+ import importlib
5
+ import importlib.util
6
+ import os
7
+ import signal
8
+ import sys
9
+ import time
10
+
11
+ from db import get_db
12
+ from fastmcp.tools import Tool
13
+
14
+ SERVER_DIR = os.path.dirname(os.path.abspath(__file__))
15
+ PLUGINS_DIR = os.path.join(SERVER_DIR, "plugins")
16
+
17
+ # Personal plugins directory: NEXO_HOME/plugins/ (env var, defaults to ~/.nexo/)
18
+ NEXO_HOME = os.environ.get("NEXO_HOME", os.path.expanduser("~/.nexo"))
19
+ PERSONAL_PLUGINS_DIR = os.path.join(NEXO_HOME, "plugins")
20
+
21
+ PLUGIN_LOAD_TIMEOUT = 10 # seconds per plugin
22
+
23
+
24
+ class _PluginTimeout(Exception):
25
+ pass
26
+
27
+
28
+ def _timeout_handler(signum, frame):
29
+ raise _PluginTimeout("Plugin loading timed out")
30
+
31
+
32
+ def _ensure_src_in_path():
33
+ """Ensure server src/ is in sys.path so personal plugins can import db, cognitive, etc."""
34
+ if SERVER_DIR not in sys.path:
35
+ sys.path.insert(0, SERVER_DIR)
36
+
37
+
38
+ def load_all_plugins(mcp) -> int:
39
+ """Load all plugins from repo and personal directories at startup. Returns total tools loaded."""
40
+ _ensure_src_in_path()
41
+ total = 0
42
+
43
+ # Collect plugins: repo first, personal overrides
44
+ plugin_map = {} # filename -> (dir_path, source_label)
45
+
46
+ # 1. Repo plugins (base)
47
+ if os.path.isdir(PLUGINS_DIR):
48
+ for f in sorted(os.listdir(PLUGINS_DIR)):
49
+ if f.endswith(".py") and f != "__init__.py":
50
+ plugin_map[f] = (PLUGINS_DIR, "repo")
51
+
52
+ # 2. Personal plugins (override if same filename)
53
+ if os.path.isdir(PERSONAL_PLUGINS_DIR):
54
+ for f in sorted(os.listdir(PERSONAL_PLUGINS_DIR)):
55
+ if f.endswith(".py") and f != "__init__.py":
56
+ source = "personal (override)" if f in plugin_map else "personal"
57
+ plugin_map[f] = (PERSONAL_PLUGINS_DIR, source)
58
+
59
+ # Load all in sorted order
60
+ for f in sorted(plugin_map):
61
+ plugins_dir, source_label = plugin_map[f]
62
+ try:
63
+ old_handler = signal.signal(signal.SIGALRM, _timeout_handler)
64
+ signal.alarm(PLUGIN_LOAD_TIMEOUT)
65
+ try:
66
+ n = load_plugin(mcp, f, plugins_dir=plugins_dir)
67
+ total += n
68
+ print(f"[PLUGIN LOADED] {f} ({n} tools) from {source_label}: {plugins_dir}", file=sys.stderr)
69
+ finally:
70
+ signal.alarm(0)
71
+ signal.signal(signal.SIGALRM, old_handler)
72
+ except _PluginTimeout:
73
+ print(f"[PLUGIN TIMEOUT] {f}: skipped after {PLUGIN_LOAD_TIMEOUT}s", file=sys.stderr)
74
+ except Exception as e:
75
+ print(f"[PLUGIN ERROR] {f}: {e}", file=sys.stderr)
76
+ return total
77
+
78
+
79
+ def load_plugin(mcp, filename: str, plugins_dir: str | None = None) -> int:
80
+ """Load or reload a single plugin. Returns number of tools registered.
81
+
82
+ Args:
83
+ plugins_dir: Directory to load from. If None, searches repo PLUGINS_DIR first,
84
+ then PERSONAL_PLUGINS_DIR. Personal plugins are loaded via
85
+ importlib.util.spec_from_file_location.
86
+ """
87
+ if not filename.endswith(".py"):
88
+ filename += ".py"
89
+
90
+ # Reject path separators and traversal sequences before joining
91
+ if "/" in filename or "\\" in filename or ".." in filename:
92
+ raise ValueError(f"Invalid plugin filename (path separators or '..' not allowed): {filename}")
93
+
94
+ if plugins_dir is not None:
95
+ filepath = os.path.join(plugins_dir, filename)
96
+ if not os.path.isfile(filepath):
97
+ raise FileNotFoundError(f"Plugin not found: {filepath}")
98
+ else:
99
+ # Search repo first, then personal
100
+ repo_path = os.path.join(PLUGINS_DIR, filename)
101
+ personal_path = os.path.join(PERSONAL_PLUGINS_DIR, filename)
102
+ if os.path.isfile(repo_path):
103
+ plugins_dir = PLUGINS_DIR
104
+ filepath = repo_path
105
+ elif os.path.isfile(personal_path):
106
+ plugins_dir = PERSONAL_PLUGINS_DIR
107
+ filepath = personal_path
108
+ else:
109
+ raise FileNotFoundError(
110
+ f"Plugin not found in repo ({PLUGINS_DIR}) or personal ({PERSONAL_PLUGINS_DIR}): {filename}"
111
+ )
112
+
113
+ # Security: reject path traversal — resolved path must stay inside allowed directories
114
+ real_path = os.path.realpath(filepath)
115
+ real_plugins = os.path.realpath(PLUGINS_DIR)
116
+ real_personal = os.path.realpath(PERSONAL_PLUGINS_DIR)
117
+ if not (real_path.startswith(real_plugins + os.sep) or real_path.startswith(real_personal + os.sep)):
118
+ raise ValueError(
119
+ f"Path traversal blocked: {filename!r} resolves to {real_path}, "
120
+ f"which is outside {real_plugins} and {real_personal}"
121
+ )
122
+
123
+ module_name = f"plugins.{filename[:-3]}"
124
+
125
+ # For personal plugins (outside repo), use spec_from_file_location
126
+ if plugins_dir != PLUGINS_DIR:
127
+ _ensure_src_in_path()
128
+ spec = importlib.util.spec_from_file_location(module_name, filepath)
129
+ if spec is None or spec.loader is None:
130
+ raise ImportError(f"Cannot create module spec for {filepath}")
131
+ mod = importlib.util.module_from_spec(spec)
132
+ sys.modules[module_name] = mod
133
+ spec.loader.exec_module(mod)
134
+ elif module_name in sys.modules:
135
+ mod = importlib.reload(sys.modules[module_name])
136
+ else:
137
+ mod = importlib.import_module(module_name)
138
+
139
+ tools_list = getattr(mod, "TOOLS", [])
140
+ tool_names = []
141
+
142
+ for func, name, description in tools_list:
143
+ try:
144
+ mcp.local_provider.remove_tool(name)
145
+ except Exception:
146
+ pass
147
+ t = Tool.from_function(func, name=name, description=description)
148
+ mcp.add_tool(t)
149
+ tool_names.append(name)
150
+
151
+ source_label = "personal" if plugins_dir != PLUGINS_DIR else "repo"
152
+ _update_registry(filename, len(tool_names), ",".join(tool_names), source_label)
153
+
154
+ return len(tool_names)
155
+
156
+
157
+ def remove_plugin(mcp, filename: str) -> list[str]:
158
+ """Unregister a plugin's tools from MCP and clean the registry.
159
+
160
+ Does NOT delete plugin files — only unregisters tools to avoid
161
+ accidental deletion of code from repo or personal directories.
162
+ """
163
+ if not filename.endswith(".py"):
164
+ filename += ".py"
165
+
166
+ conn = get_db()
167
+ row = conn.execute("SELECT tool_names FROM plugins WHERE filename = ?", (filename,)).fetchone()
168
+
169
+ removed = []
170
+ if row and row["tool_names"]:
171
+ for name in row["tool_names"].split(","):
172
+ name = name.strip()
173
+ if name:
174
+ try:
175
+ mcp.local_provider.remove_tool(name)
176
+ removed.append(name)
177
+ except Exception:
178
+ pass
179
+
180
+ module_name = f"plugins.{filename[:-3]}"
181
+ sys.modules.pop(module_name, None)
182
+
183
+ conn = get_db()
184
+ conn.execute("DELETE FROM plugins WHERE filename = ?", (filename,))
185
+ conn.commit()
186
+
187
+ return removed
188
+
189
+
190
+ def list_plugins() -> list[dict]:
191
+ """List all registered plugins with source info (repo/personal)."""
192
+ conn = get_db()
193
+ rows = conn.execute(
194
+ "SELECT filename, tools_count, tool_names, loaded_at, created_by FROM plugins ORDER BY filename"
195
+ ).fetchall()
196
+ result = []
197
+ for r in rows:
198
+ d = dict(r)
199
+ d["source"] = d.get("created_by", "repo")
200
+ result.append(d)
201
+ return result
202
+
203
+
204
+ def _update_registry(filename: str, tools_count: int, tool_names: str, created_by: str):
205
+ """Insert or update plugin registry entry. Non-fatal on lock — tools still work."""
206
+ now = time.time()
207
+ try:
208
+ conn = get_db()
209
+ conn.execute(
210
+ "INSERT INTO plugins (filename, tools_count, tool_names, loaded_at, created_by) "
211
+ "VALUES (?, ?, ?, ?, ?) "
212
+ "ON CONFLICT(filename) DO UPDATE SET tools_count=?, tool_names=?, loaded_at=?, created_by=?",
213
+ (filename, tools_count, tool_names, now, created_by, tools_count, tool_names, now, created_by),
214
+ )
215
+ conn.commit()
216
+ except Exception as e:
217
+ print(f"[PLUGIN REGISTRY] Skipped update for {filename}: {e}")
File without changes
@@ -1,6 +1,12 @@
1
1
  """Agents plugin — registry of known agent types with their configs."""
2
2
  from db import create_agent, get_agent, list_agents, update_agent, delete_agent
3
3
 
4
+ try:
5
+ from client_preferences import resolve_user_model
6
+ _DEFAULT_AGENT_MODEL = resolve_user_model()
7
+ except Exception:
8
+ _DEFAULT_AGENT_MODEL = ""
9
+
4
10
  def handle_agent_get(id: str) -> str:
5
11
  """Get an agent's full profile by ID."""
6
12
  a = get_agent(id)
@@ -11,11 +17,12 @@ def handle_agent_get(id: str) -> str:
11
17
  if a["rules"]: lines.append(f" Reglas: {a['rules']}")
12
18
  return "\n".join(lines)
13
19
 
14
- def handle_agent_create(id: str, name: str, specialization: str, model: str = "sonnet",
20
+ def handle_agent_create(id: str, name: str, specialization: str, model: str = "",
15
21
  tools: str = "", context_files: str = "", rules: str = "") -> str:
16
22
  """Register a new agent in the registry."""
17
- create_agent(id, name, specialization, model, tools, context_files, rules)
18
- return f"Agent '{id}' ({name}) registered. Model: {model}"
23
+ effective_model = model or _DEFAULT_AGENT_MODEL
24
+ create_agent(id, name, specialization, effective_model, tools, context_files, rules)
25
+ return f"Agent '{id}' ({name}) registered. Model: {effective_model}"
19
26
 
20
27
  def handle_agent_update(id: str, name: str = "", specialization: str = "", model: str = "",
21
28
  tools: str = "", context_files: str = "", rules: str = "") -> str: