nexo-brain 5.3.26 → 5.3.28

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 (212) hide show
  1. package/.claude-plugin/plugin.json +1 -1
  2. package/package.json +1 -1
  3. package/src/hook_guardrails.py +44 -0
  4. package/src/server.py +3 -0
  5. package/src/tools_sessions.py +6 -1
  6. package/src/dashboard/static/favicon 2.svg +0 -32
  7. package/src/dashboard/static/nexo-logo 2.png +0 -0
  8. package/src/dashboard/static/nexo-logo 2.svg +0 -40
  9. package/src/dashboard/static/style 2.css +0 -2458
  10. package/src/dashboard/templates/adaptive 2.html +0 -118
  11. package/src/dashboard/templates/artifacts 2.html +0 -133
  12. package/src/dashboard/templates/backups 2.html +0 -136
  13. package/src/dashboard/templates/base 2.html +0 -417
  14. package/src/dashboard/templates/calendar 2.html +0 -591
  15. package/src/dashboard/templates/chat 2.html +0 -356
  16. package/src/dashboard/templates/claims 2.html +0 -259
  17. package/src/dashboard/templates/cortex 2.html +0 -321
  18. package/src/dashboard/templates/credentials 2.html +0 -128
  19. package/src/dashboard/templates/crons 2.html +0 -370
  20. package/src/dashboard/templates/dashboard 2.html +0 -494
  21. package/src/dashboard/templates/dreams 2.html +0 -252
  22. package/src/dashboard/templates/email 2.html +0 -160
  23. package/src/dashboard/templates/evolution 2.html +0 -189
  24. package/src/dashboard/templates/feed 2.html +0 -249
  25. package/src/dashboard/templates/followup_health 2.html +0 -170
  26. package/src/dashboard/templates/graph 2.html +0 -201
  27. package/src/dashboard/templates/guard 2.html +0 -259
  28. package/src/dashboard/templates/inbox 2.html +0 -251
  29. package/src/dashboard/templates/memory 2.html +0 -420
  30. package/src/dashboard/templates/operations 2.html +0 -608
  31. package/src/dashboard/templates/plugins 2.html +0 -185
  32. package/src/dashboard/templates/protocol 2.html +0 -199
  33. package/src/dashboard/templates/rules 2.html +0 -246
  34. package/src/dashboard/templates/sentiment 2.html +0 -247
  35. package/src/dashboard/templates/sessions 2.html +0 -218
  36. package/src/dashboard/templates/skills 2.html +0 -329
  37. package/src/dashboard/templates/somatic 2.html +0 -73
  38. package/src/dashboard/templates/triggers 2.html +0 -133
  39. package/src/dashboard/templates/trust 2.html +0 -360
  40. package/src/db/__init__ 2.py +0 -259
  41. package/src/db/_core 2.py +0 -437
  42. package/src/db/_credentials 2.py +0 -124
  43. package/src/db/_episodic 2.py +0 -762
  44. package/src/db/_evolution 2.py +0 -54
  45. package/src/db/_fts 2.py +0 -406
  46. package/src/db/_goal_profiles 2.py +0 -376
  47. package/src/db/_hot_context 2.py +0 -660
  48. package/src/db/_outcomes 2.py +0 -800
  49. package/src/db/_personal_scripts 2.py +0 -582
  50. package/src/db/_sessions 2.py +0 -330
  51. package/src/db/_tasks 2.py +0 -91
  52. package/src/db/_watchers 2.py +0 -173
  53. package/src/doctor/formatters 2.py +0 -52
  54. package/src/doctor/models 2.py +0 -69
  55. package/src/doctor/planes 2.py +0 -87
  56. package/src/doctor/providers/__init__ 2.py +0 -1
  57. package/src/doctor/providers/deep 2.py +0 -367
  58. package/src/evolution_cycle 2.py +0 -519
  59. package/src/hooks/auto_capture 2.py +0 -208
  60. package/src/hooks/caffeinate-guard 2.sh +0 -8
  61. package/src/hooks/capture-session 2.sh +0 -21
  62. package/src/hooks/capture-tool-logs 2.sh +0 -158
  63. package/src/hooks/daily-briefing-check 2.sh +0 -33
  64. package/src/hooks/heartbeat-enforcement 2.py +0 -90
  65. package/src/hooks/heartbeat-posttool 2.sh +0 -18
  66. package/src/hooks/inbox-hook 2.sh +0 -76
  67. package/src/hooks/post-compact 2.sh +0 -152
  68. package/src/hooks/pre-compact 2.sh +0 -169
  69. package/src/hooks/protocol-guardrail 2.sh +0 -10
  70. package/src/hooks/protocol-pretool-guardrail 2.sh +0 -9
  71. package/src/hooks/session-stop 2.sh +0 -52
  72. package/src/kg_populate 2.py +0 -292
  73. package/src/maintenance 2.py +0 -53
  74. package/src/memory_backends 2.py +0 -71
  75. package/src/migrate_embeddings 2.py +0 -124
  76. package/src/nexo_sdk 2.py +0 -103
  77. package/src/observability 2.py +0 -199
  78. package/src/plugin_loader 2.py +0 -217
  79. package/src/plugins/__init__ 2.py +0 -0
  80. package/src/plugins/artifact_registry 2.py +0 -450
  81. package/src/plugins/backup 2.py +0 -127
  82. package/src/plugins/claims_tools 2.py +0 -119
  83. package/src/plugins/cognitive_memory 2.py +0 -609
  84. package/src/plugins/core_rules 2.py +0 -252
  85. package/src/plugins/cortex 2.py +0 -1155
  86. package/src/plugins/entities 2.py +0 -67
  87. package/src/plugins/episodic_memory 2.py +0 -560
  88. package/src/plugins/evolution 2.py +0 -167
  89. package/src/plugins/goal_engine 2.py +0 -142
  90. package/src/plugins/guard 2.py +0 -862
  91. package/src/plugins/impact 2.py +0 -29
  92. package/src/plugins/knowledge_graph_tools 2.py +0 -137
  93. package/src/plugins/media_memory_tools 2.py +0 -98
  94. package/src/plugins/memory_export 2.py +0 -196
  95. package/src/plugins/outcomes 2.py +0 -130
  96. package/src/plugins/personal_scripts 2.py +0 -117
  97. package/src/plugins/preferences 2.py +0 -47
  98. package/src/plugins/protocol 2.py +0 -1449
  99. package/src/plugins/simple_api 2.py +0 -106
  100. package/src/plugins/skills 2.py +0 -341
  101. package/src/plugins/state_watchers 2.py +0 -79
  102. package/src/plugins/update 2.py +0 -986
  103. package/src/plugins/user_state_tools 2.py +0 -43
  104. package/src/plugins/workflow 2.py +0 -588
  105. package/src/protocol_settings 2.py +0 -59
  106. package/src/public_contribution 2.py +0 -466
  107. package/src/public_evolution_queue 2.py +0 -241
  108. package/src/requirements 2.txt +0 -14
  109. package/src/retroactive_learnings 2.py +0 -373
  110. package/src/rules/__init__ 2.py +0 -0
  111. package/src/rules/core-rules 2.json +0 -331
  112. package/src/rules/migrate 2.py +0 -207
  113. package/src/runtime_power 2.py +0 -874
  114. package/src/script_registry 2.py +0 -1559
  115. package/src/scripts/check-context 2.py +0 -272
  116. package/src/scripts/deep-sleep/apply_findings 2.py +0 -2327
  117. package/src/scripts/deep-sleep/collect 2.py +0 -928
  118. package/src/scripts/deep-sleep/extract 2.py +0 -330
  119. package/src/scripts/deep-sleep/extract-prompt 2.md +0 -285
  120. package/src/scripts/deep-sleep/synthesize 2.py +0 -312
  121. package/src/scripts/deep-sleep/synthesize-prompt 2.md +0 -336
  122. package/src/scripts/nexo-agent-run 2.py +0 -75
  123. package/src/scripts/nexo-auto-update 2.py +0 -6
  124. package/src/scripts/nexo-backup 2.sh +0 -25
  125. package/src/scripts/nexo-brain-activation 2.sh +0 -140
  126. package/src/scripts/nexo-catchup 2.py +0 -300
  127. package/src/scripts/nexo-cognitive-decay 2.py +0 -257
  128. package/src/scripts/nexo-cortex-cycle 2.py +0 -293
  129. package/src/scripts/nexo-cron-wrapper 2.sh +0 -53
  130. package/src/scripts/nexo-daily-self-audit 2.py +0 -2161
  131. package/src/scripts/nexo-dashboard 2.sh +0 -29
  132. package/src/scripts/nexo-deep-sleep 2.sh +0 -86
  133. package/src/scripts/nexo-evolution-run 2.py +0 -1664
  134. package/src/scripts/nexo-followup-hygiene 2.py +0 -139
  135. package/src/scripts/nexo-hook-record 2.py +0 -42
  136. package/src/scripts/nexo-immune 2.py +0 -936
  137. package/src/scripts/nexo-impact-scorer 2.py +0 -117
  138. package/src/scripts/nexo-inbox-hook 2.sh +0 -74
  139. package/src/scripts/nexo-install 2.py +0 -6
  140. package/src/scripts/nexo-learning-housekeep 2.py +0 -401
  141. package/src/scripts/nexo-learning-validator 2.py +0 -266
  142. package/src/scripts/nexo-migrate 2.py +0 -260
  143. package/src/scripts/nexo-outcome-checker 2.py +0 -127
  144. package/src/scripts/nexo-postmortem-consolidator 2.py +0 -456
  145. package/src/scripts/nexo-pre-commit 2.py +0 -120
  146. package/src/scripts/nexo-prevent-sleep 2.sh +0 -35
  147. package/src/scripts/nexo-proactive-dashboard 2.py +0 -354
  148. package/src/scripts/nexo-reflection 2.py +0 -256
  149. package/src/scripts/nexo-runtime-preflight 2.py +0 -274
  150. package/src/scripts/nexo-sleep 2.py +0 -631
  151. package/src/scripts/nexo-snapshot-restore 2.sh +0 -35
  152. package/src/scripts/nexo-sync-clients 2.py +0 -16
  153. package/src/scripts/nexo-synthesis 2.py +0 -475
  154. package/src/scripts/nexo-tcc-approve 2.sh +0 -79
  155. package/src/scripts/nexo-update 2.sh +0 -306
  156. package/src/scripts/nexo-watchdog 2.sh +0 -1207
  157. package/src/scripts/nexo-watchdog-smoke 2.py +0 -119
  158. package/src/scripts/rehydrate_learnings_from_archive 2.py +0 -245
  159. package/src/server 2.py +0 -1296
  160. package/src/skills/run-nexo-audit-phase/guide 2.md +0 -43
  161. package/src/skills/run-nexo-audit-phase/skill 2.json +0 -59
  162. package/src/skills/run-nexo-core-fix-cycle/guide 2.md +0 -17
  163. package/src/skills/run-nexo-core-fix-cycle/script 2.py +0 -276
  164. package/src/skills/run-nexo-core-fix-cycle/skill 2.json +0 -58
  165. package/src/skills/run-release-final-audit/guide 2.md +0 -16
  166. package/src/skills/run-release-final-audit/script 2.py +0 -259
  167. package/src/skills/run-release-final-audit/skill 2.json +0 -77
  168. package/src/skills/run-runtime-doctor/guide 2.md +0 -12
  169. package/src/skills/run-runtime-doctor/script 2.py +0 -21
  170. package/src/skills/run-runtime-doctor/skill 2.json +0 -25
  171. package/src/skills_runtime 2.py +0 -932
  172. package/src/state_watchers_runtime 2.py +0 -475
  173. package/src/storage_router 2.py +0 -32
  174. package/src/system_catalog 2.py +0 -786
  175. package/src/tools_coordination 2.py +0 -103
  176. package/src/tools_credentials 2.py +0 -68
  177. package/src/tools_drive 2.py +0 -487
  178. package/src/tools_hot_context 2.py +0 -163
  179. package/src/tools_learnings 2.py +0 -612
  180. package/src/tools_menu 2.py +0 -229
  181. package/src/tools_reminders 2.py +0 -88
  182. package/src/tools_reminders_crud 2.py +0 -363
  183. package/src/tools_sessions 2.py +0 -1054
  184. package/src/tools_system_catalog 2.py +0 -19
  185. package/src/tools_task_history 2.py +0 -57
  186. package/src/tools_transcripts 2.py +0 -98
  187. package/src/transcript_utils 2.py +0 -412
  188. package/src/user_context 2.py +0 -46
  189. package/src/user_data_portability 2.py +0 -328
  190. package/src/user_state_model 2.py +0 -170
  191. package/templates/CLAUDE.md 2.template +0 -108
  192. package/templates/CODEX.AGENTS.md 2.template +0 -66
  193. package/templates/launchagents/README 2.md +0 -132
  194. package/templates/launchagents/com.nexo.auto-close-sessions 2.plist +0 -39
  195. package/templates/launchagents/com.nexo.catchup 2.plist +0 -39
  196. package/templates/launchagents/com.nexo.cognitive-decay 2.plist +0 -40
  197. package/templates/launchagents/com.nexo.dashboard 2.plist +0 -43
  198. package/templates/launchagents/com.nexo.deep-sleep 2.plist +0 -43
  199. package/templates/launchagents/com.nexo.evolution 2.plist +0 -44
  200. package/templates/launchagents/com.nexo.followup-hygiene 2.plist +0 -45
  201. package/templates/launchagents/com.nexo.immune 2.plist +0 -41
  202. package/templates/launchagents/com.nexo.postmortem 2.plist +0 -45
  203. package/templates/launchagents/com.nexo.self-audit 2.plist +0 -47
  204. package/templates/launchagents/com.nexo.synthesis 2.plist +0 -45
  205. package/templates/launchagents/com.nexo.watchdog 2.plist +0 -37
  206. package/templates/nexo_helper 2.py +0 -301
  207. package/templates/openclaw 2.json +0 -13
  208. package/templates/plugin-template 2.py +0 -40
  209. package/templates/script-template 2.py +0 -59
  210. package/templates/script-template 2.sh +0 -13
  211. package/templates/skill-script-template 2.py +0 -48
  212. package/templates/skill-template 2.md +0 -33
@@ -1,329 +0,0 @@
1
- {% extends "base.html" %}
2
-
3
- {% block title %}Skills Lab{% endblock %}
4
- {% block page_title %}Skills Lab{% endblock %}
5
-
6
- {% block head %}
7
- <style>
8
- .kanban-col {
9
- min-height: 300px;
10
- transition: background 0.2s;
11
- }
12
- .kanban-col:hover {
13
- background: rgba(30, 41, 59, 0.3);
14
- }
15
- .skill-card {
16
- transition: all 0.25s ease;
17
- cursor: pointer;
18
- animation: slideIn 0.3s ease-out forwards;
19
- }
20
- .skill-card:hover {
21
- border-color: rgba(124, 58, 237, 0.4);
22
- box-shadow: 0 4px 20px rgba(124, 58, 237, 0.1);
23
- transform: translateY(-1px);
24
- }
25
- .skill-card .detail-panel {
26
- max-height: 0;
27
- overflow: hidden;
28
- transition: max-height 0.4s ease, opacity 0.3s ease;
29
- opacity: 0;
30
- }
31
- .skill-card.expanded .detail-panel {
32
- max-height: 600px;
33
- opacity: 1;
34
- }
35
- .tag-pill {
36
- transition: transform 0.15s;
37
- }
38
- .tag-pill:hover {
39
- transform: scale(1.05);
40
- }
41
- .trust-bar-fill {
42
- transition: width 0.8s cubic-bezier(0.22, 1, 0.36, 1);
43
- }
44
- .level-badge-trace { background: rgba(148, 163, 184, 0.1); color: #94a3b8; border-color: rgba(148, 163, 184, 0.2); }
45
- .level-badge-draft { background: rgba(251, 191, 36, 0.1); color: #fbbf24; border-color: rgba(251, 191, 36, 0.2); }
46
- .level-badge-candidate { background: rgba(96, 165, 250, 0.1); color: #60a5fa; border-color: rgba(96, 165, 250, 0.2); }
47
- .level-badge-published { background: rgba(52, 211, 153, 0.1); color: #34d399; border-color: rgba(52, 211, 153, 0.2); }
48
- .level-badge-archived { background: rgba(107, 114, 128, 0.1); color: #6b7280; border-color: rgba(107, 114, 128, 0.2); }
49
- .drag-hint {
50
- cursor: grab;
51
- opacity: 0.3;
52
- transition: opacity 0.2s;
53
- }
54
- .skill-card:hover .drag-hint {
55
- opacity: 0.6;
56
- }
57
- @keyframes slideIn {
58
- from { opacity: 0; transform: translateY(8px); }
59
- to { opacity: 1; transform: translateY(0); }
60
- }
61
- </style>
62
- {% endblock %}
63
-
64
- {% block header_actions %}
65
- <div class="flex items-center gap-2">
66
- <input type="text" id="skill-search" placeholder="Filter skills..."
67
- class="bg-slate-800 border border-slate-700 rounded-lg px-3 py-1.5 text-xs text-slate-200 placeholder-slate-500 focus:outline-none focus:ring-1 focus:ring-violet-500 w-52"
68
- oninput="filterSkills()">
69
- </div>
70
- {% endblock %}
71
-
72
- {% block content %}
73
- <!-- Stats Header -->
74
- <div class="grid grid-cols-2 sm:grid-cols-4 lg:grid-cols-6 gap-3 mb-6" id="stats-row">
75
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-center">
76
- <div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Total Skills</div>
77
- <div class="text-2xl font-display font-bold text-slate-200" id="stat-total">--</div>
78
- </div>
79
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-center">
80
- <div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Trace</div>
81
- <div class="text-2xl font-display font-bold text-slate-400" id="stat-trace">--</div>
82
- </div>
83
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-center">
84
- <div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Draft</div>
85
- <div class="text-2xl font-display font-bold text-amber-400" id="stat-draft">--</div>
86
- </div>
87
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-center">
88
- <div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Candidate</div>
89
- <div class="text-2xl font-display font-bold text-blue-400" id="stat-candidate">--</div>
90
- </div>
91
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-center">
92
- <div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Published</div>
93
- <div class="text-2xl font-display font-bold text-emerald-400" id="stat-published">--</div>
94
- </div>
95
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-center">
96
- <div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Archived</div>
97
- <div class="text-2xl font-display font-bold text-gray-500" id="stat-archived">--</div>
98
- </div>
99
- </div>
100
-
101
- <!-- Top Skills by Usage -->
102
- <div class="mb-6">
103
- <h2 class="text-xs uppercase tracking-wider text-slate-500 font-semibold mb-3">Top Skills by Usage</h2>
104
- <div class="flex gap-3 overflow-x-auto pb-2" id="top-skills">
105
- <div class="text-xs text-slate-600 py-3">Loading...</div>
106
- </div>
107
- </div>
108
-
109
- <!-- Kanban Board -->
110
- <div class="mb-6">
111
- <h2 class="text-xs uppercase tracking-wider text-slate-500 font-semibold mb-3">Skill Lifecycle</h2>
112
- <div class="grid grid-cols-1 md:grid-cols-3 lg:grid-cols-5 gap-3" id="kanban-board">
113
- <!-- Columns generated by JS -->
114
- </div>
115
- </div>
116
-
117
- <!-- Recent Usage Timeline -->
118
- <div>
119
- <h2 class="text-xs uppercase tracking-wider text-slate-500 font-semibold mb-3">Recent Usage</h2>
120
- <div class="space-y-2" id="usage-timeline">
121
- <div class="text-xs text-slate-600 text-center py-6">Loading usage data...</div>
122
- </div>
123
- </div>
124
- {% endblock %}
125
-
126
- {% block scripts %}
127
- <script>
128
- const REFRESH_MS = 60000;
129
- const LEVELS = ['trace', 'draft', 'candidate', 'published', 'archived'];
130
- const LEVEL_COLORS = {
131
- trace: { bg: 'bg-slate-500/10', border: 'border-slate-500/20', text: 'text-slate-400', dot: 'bg-slate-400' },
132
- draft: { bg: 'bg-amber-500/10', border: 'border-amber-500/20', text: 'text-amber-400', dot: 'bg-amber-400' },
133
- candidate: { bg: 'bg-blue-500/10', border: 'border-blue-500/20', text: 'text-blue-400', dot: 'bg-blue-400' },
134
- published: { bg: 'bg-emerald-500/10', border: 'border-emerald-500/20', text: 'text-emerald-400', dot: 'bg-emerald-400' },
135
- archived: { bg: 'bg-gray-500/10', border: 'border-gray-500/20', text: 'text-gray-500', dot: 'bg-gray-500' },
136
- };
137
-
138
- let allSkills = [];
139
-
140
- const TAG_COLORS = [
141
- 'bg-violet-500/15 text-violet-300', 'bg-cyan-500/15 text-cyan-300',
142
- 'bg-pink-500/15 text-pink-300', 'bg-emerald-500/15 text-emerald-300',
143
- 'bg-amber-500/15 text-amber-300', 'bg-blue-500/15 text-blue-300',
144
- 'bg-rose-500/15 text-rose-300', 'bg-indigo-500/15 text-indigo-300',
145
- ];
146
-
147
- function tagColor(tag) {
148
- let hash = 0;
149
- for (let i = 0; i < tag.length; i++) hash = tag.charCodeAt(i) + ((hash << 5) - hash);
150
- return TAG_COLORS[Math.abs(hash) % TAG_COLORS.length];
151
- }
152
-
153
- function trustColor(score) {
154
- if (score >= 0.8) return 'bg-emerald-500';
155
- if (score >= 0.5) return 'bg-violet-500';
156
- if (score >= 0.3) return 'bg-amber-500';
157
- return 'bg-red-500';
158
- }
159
-
160
- function renderSkillCard(skill) {
161
- const level = (skill.level || 'trace').toLowerCase();
162
- const colors = LEVEL_COLORS[level] || LEVEL_COLORS.trace;
163
- const trust = skill.trust_score != null ? skill.trust_score : 0;
164
- const tags = skill.tags || [];
165
- const useCount = skill.use_count || 0;
166
- const steps = skill.steps || [];
167
- const gotchas = skill.gotchas || [];
168
- const desc = skill.description || '';
169
-
170
- const tagsHtml = tags.slice(0, 5).map(t =>
171
- `<span class="tag-pill text-[10px] px-1.5 py-0.5 rounded-full ${tagColor(t)}">${escapeHtml(t)}</span>`
172
- ).join('');
173
-
174
- const stepsHtml = steps.length > 0
175
- ? `<div class="mt-3"><div class="text-[10px] uppercase tracking-wider text-slate-500 font-semibold mb-1">Steps</div><ol class="list-decimal list-inside space-y-0.5">${steps.map(s => `<li class="text-xs text-slate-400">${escapeHtml(typeof s === 'string' ? s : s.description || JSON.stringify(s))}</li>`).join('')}</ol></div>`
176
- : '';
177
-
178
- const gotchasHtml = gotchas.length > 0
179
- ? `<div class="mt-3"><div class="text-[10px] uppercase tracking-wider text-amber-500/60 font-semibold mb-1">Gotchas</div><ul class="space-y-0.5">${gotchas.map(g => `<li class="text-xs text-amber-300/70">&#x26A0; ${escapeHtml(typeof g === 'string' ? g : JSON.stringify(g))}</li>`).join('')}</ul></div>`
180
- : '';
181
-
182
- return `<div class="skill-card bg-slate-900/50 border border-slate-800/50 rounded-xl p-3 mb-2" data-name="${escapeHtml(skill.name || '')}" data-level="${level}" onclick="this.classList.toggle('expanded')">
183
- <div class="flex items-start justify-between mb-1.5">
184
- <div class="flex items-center gap-1.5 min-w-0">
185
- <svg class="w-3 h-3 drag-hint text-slate-600 flex-shrink-0" fill="currentColor" viewBox="0 0 20 20"><circle cx="6" cy="5" r="1.5"/><circle cx="14" cy="5" r="1.5"/><circle cx="6" cy="10" r="1.5"/><circle cx="14" cy="10" r="1.5"/><circle cx="6" cy="15" r="1.5"/><circle cx="14" cy="15" r="1.5"/></svg>
186
- <span class="text-sm font-medium text-slate-200 truncate">${escapeHtml(skill.name || 'Unnamed')}</span>
187
- </div>
188
- <span class="text-[10px] font-mono text-slate-600 flex-shrink-0 ml-2">#${skill.id || '?'}</span>
189
- </div>
190
-
191
- <!-- Trust score bar -->
192
- <div class="flex items-center gap-2 mb-2">
193
- <div class="flex-1 h-1.5 bg-slate-800 rounded-full overflow-hidden">
194
- <div class="trust-bar-fill h-full rounded-full ${trustColor(trust)}" style="width:${(trust * 100).toFixed(0)}%"></div>
195
- </div>
196
- <span class="text-[10px] font-mono text-slate-500">${trust.toFixed(2)}</span>
197
- </div>
198
-
199
- <div class="flex items-center gap-2 flex-wrap">
200
- <span class="text-[10px] font-mono text-slate-500">${useCount} uses</span>
201
- ${tagsHtml}
202
- </div>
203
-
204
- <!-- Expandable detail -->
205
- <div class="detail-panel">
206
- <div class="mt-3 pt-3 border-t border-slate-800/50">
207
- ${desc ? `<p class="text-xs text-slate-400 leading-relaxed mb-2">${escapeHtml(desc)}</p>` : ''}
208
- ${stepsHtml}
209
- ${gotchasHtml}
210
- </div>
211
- </div>
212
- </div>`;
213
- }
214
-
215
- function renderKanban(skills) {
216
- const board = document.getElementById('kanban-board');
217
- const grouped = {};
218
- LEVELS.forEach(l => grouped[l] = []);
219
- skills.forEach(s => {
220
- const level = (s.level || 'trace').toLowerCase();
221
- if (grouped[level]) grouped[level].push(s);
222
- else grouped.trace.push(s);
223
- });
224
-
225
- board.innerHTML = LEVELS.map(level => {
226
- const colors = LEVEL_COLORS[level];
227
- const items = grouped[level];
228
- return `<div class="kanban-col ${colors.bg} border ${colors.border} rounded-xl p-3">
229
- <div class="flex items-center gap-2 mb-3 pb-2 border-b ${colors.border}">
230
- <span class="w-2 h-2 rounded-full ${colors.dot}"></span>
231
- <span class="text-xs uppercase tracking-wider ${colors.text} font-semibold">${level}</span>
232
- <span class="text-[10px] font-mono text-slate-600 ml-auto">${items.length}</span>
233
- </div>
234
- <div class="space-y-0">
235
- ${items.length === 0
236
- ? `<div class="text-[10px] text-slate-600 text-center py-4">No skills</div>`
237
- : items.map(renderSkillCard).join('')}
238
- </div>
239
- </div>`;
240
- }).join('');
241
- }
242
-
243
- function renderTopSkills(skills) {
244
- const container = document.getElementById('top-skills');
245
- const sorted = [...skills].sort((a, b) => (b.use_count || 0) - (a.use_count || 0)).slice(0, 8);
246
- if (sorted.length === 0) {
247
- container.innerHTML = '<div class="text-xs text-slate-600 py-3">No skill usage data</div>';
248
- return;
249
- }
250
- const maxUse = sorted[0].use_count || 1;
251
- container.innerHTML = sorted.map(s => {
252
- const pct = ((s.use_count || 0) / maxUse * 100).toFixed(0);
253
- const level = (s.level || 'trace').toLowerCase();
254
- const colors = LEVEL_COLORS[level] || LEVEL_COLORS.trace;
255
- return `<div class="flex-shrink-0 bg-slate-900/50 border border-slate-800/50 rounded-xl p-3 w-44">
256
- <div class="text-sm font-medium text-slate-200 truncate mb-1">${escapeHtml(s.name || 'Unnamed')}</div>
257
- <div class="flex items-center gap-2 mb-1">
258
- <div class="flex-1 h-1 bg-slate-800 rounded-full overflow-hidden">
259
- <div class="h-full rounded-full bg-violet-500" style="width:${pct}%"></div>
260
- </div>
261
- </div>
262
- <div class="flex items-center justify-between">
263
- <span class="text-[10px] font-mono text-slate-500">${s.use_count || 0} uses</span>
264
- <span class="text-[10px] px-1 py-0.5 rounded level-badge-${level}">${level}</span>
265
- </div>
266
- </div>`;
267
- }).join('');
268
- }
269
-
270
- function renderUsageTimeline(usage) {
271
- const container = document.getElementById('usage-timeline');
272
- if (!usage || usage.length === 0) {
273
- container.innerHTML = '<div class="text-xs text-slate-600 text-center py-6">No recent usage</div>';
274
- return;
275
- }
276
- container.innerHTML = usage.slice(0, 20).map(u => {
277
- const skillName = u.skill_name || u.name || `Skill #${u.skill_id || '?'}`;
278
- const when = u.created_at ? relativeTime(u.created_at) : '';
279
- const result = u.result || u.outcome || '';
280
- return `<div class="flex items-center gap-3 bg-slate-900/30 border border-slate-800/30 rounded-lg px-3 py-2">
281
- <div class="w-1.5 h-1.5 rounded-full bg-violet-500 flex-shrink-0"></div>
282
- <span class="text-xs text-slate-300 font-medium">${escapeHtml(skillName)}</span>
283
- ${result ? `<span class="text-[10px] px-1.5 py-0.5 rounded ${result === 'success' ? 'bg-emerald-500/10 text-emerald-400' : 'bg-red-500/10 text-red-400'}">${escapeHtml(result)}</span>` : ''}
284
- <span class="text-[10px] text-slate-600 font-mono ml-auto">${escapeHtml(when)}</span>
285
- </div>`;
286
- }).join('');
287
- }
288
-
289
- function filterSkills() {
290
- const q = document.getElementById('skill-search').value.trim().toLowerCase();
291
- const filtered = q ? allSkills.filter(s =>
292
- (s.name || '').toLowerCase().includes(q) ||
293
- (s.description || '').toLowerCase().includes(q) ||
294
- (s.tags || []).some(t => t.toLowerCase().includes(q))
295
- ) : allSkills;
296
- renderKanban(filtered);
297
- }
298
-
299
- async function loadSkills() {
300
- const data = await fetchJSON('/api/skills');
301
- if (!data) return;
302
-
303
- allSkills = data.skills || [];
304
- const usage = data.usage || [];
305
- const levels = data.levels || {};
306
-
307
- // Stats
308
- document.getElementById('stat-total').textContent = formatNumber(allSkills.length);
309
- const countByLevel = {};
310
- LEVELS.forEach(l => countByLevel[l] = 0);
311
- allSkills.forEach(s => {
312
- const l = (s.level || 'trace').toLowerCase();
313
- if (countByLevel[l] !== undefined) countByLevel[l]++;
314
- });
315
- LEVELS.forEach(l => {
316
- const el = document.getElementById(`stat-${l}`);
317
- if (el) el.textContent = formatNumber(levels[l] != null ? levels[l] : countByLevel[l]);
318
- });
319
-
320
- renderKanban(allSkills);
321
- renderTopSkills(allSkills);
322
- renderUsageTimeline(usage);
323
- }
324
-
325
- // Init
326
- loadSkills();
327
- setInterval(loadSkills, REFRESH_MS);
328
- </script>
329
- {% endblock %}
@@ -1,73 +0,0 @@
1
- {% extends "base.html" %}
2
-
3
- {% block title %}Somatic Markers{% endblock %}
4
- {% block page_title %}Somatic Markers{% endblock %}
5
-
6
- {% block content %}
7
- <div class="max-w-3xl">
8
- <p class="text-sm text-slate-400 mb-6">Risk associations from past errors. Higher scores indicate areas needing extra caution.</p>
9
- <div id="risks" class="space-y-2"></div>
10
- <div id="empty" class="p-12 text-center text-slate-500 text-sm hidden">No somatic markers recorded</div>
11
- </div>
12
- {% endblock %}
13
-
14
- {% block scripts %}
15
- <script>
16
- function barColor(score) {
17
- if (score > 5) return 'bg-red-500';
18
- if (score > 2) return 'bg-amber-500';
19
- return 'bg-emerald-500';
20
- }
21
-
22
- function scoreColor(score) {
23
- if (score > 5) return 'text-red-400';
24
- if (score > 2) return 'text-amber-400';
25
- return 'text-emerald-400';
26
- }
27
-
28
- function loadData() {
29
- fetch('/api/somatic')
30
- .then(r => r.json())
31
- .then(data => {
32
- const risks = (data.risks || []).slice().sort((a, b) => {
33
- const sa = Math.abs(a.cumulative_score || a.score || 0);
34
- const sb = Math.abs(b.cumulative_score || b.score || 0);
35
- return sb - sa;
36
- });
37
-
38
- const container = document.getElementById('risks');
39
- const empty = document.getElementById('empty');
40
-
41
- if (!risks.length) {
42
- empty.classList.remove('hidden');
43
- return;
44
- }
45
-
46
- const max = Math.max(...risks.map(r => Math.abs(r.cumulative_score || r.score || 0)), 1);
47
-
48
- container.innerHTML = risks.map(r => {
49
- const score = Math.abs(r.cumulative_score || r.score || 0);
50
- const pct = Math.min((score / max) * 100, 100);
51
- const target_type = r.target_type || r.type || '';
52
- const target = (r.target || r.name || '').replace(/</g, '&lt;');
53
- const bc = barColor(score);
54
- const sc = scoreColor(score);
55
-
56
- return `<div class="flex items-center gap-3 py-1">
57
- <span class="text-xs w-20 text-right text-slate-400 flex-shrink-0 font-medium">${target_type}</span>
58
- <span class="text-sm w-56 truncate text-slate-200 flex-shrink-0">${target}</span>
59
- <div class="flex-1 h-2.5 bg-slate-800 rounded-full overflow-hidden">
60
- <div class="h-full rounded-full ${bc}" style="width:${pct.toFixed(1)}%"></div>
61
- </div>
62
- <span class="text-sm font-mono ${sc} w-12 text-right font-medium">${score.toFixed(1)}</span>
63
- </div>`;
64
- }).join('');
65
- })
66
- .catch(err => {
67
- document.getElementById('risks').innerHTML = `<p class="text-xs text-red-400">Failed to load: ${err.message}</p>`;
68
- });
69
- }
70
-
71
- loadData();
72
- </script>
73
- {% endblock %}
@@ -1,133 +0,0 @@
1
- {% extends "base.html" %}
2
-
3
- {% block title %}Prospective Triggers{% endblock %}
4
- {% block page_title %}Prospective Triggers{% endblock %}
5
-
6
- {% block content %}
7
- <div class="space-y-5">
8
- <!-- Stats -->
9
- <div class="grid grid-cols-3 gap-4">
10
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 card">
11
- <div class="text-xs uppercase tracking-wider text-slate-400 font-medium mb-2">Total Triggers</div>
12
- <div class="text-xl font-mono font-semibold text-slate-200" id="total-triggers">--</div>
13
- </div>
14
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 card">
15
- <div class="flex items-center gap-2 mb-2">
16
- <span class="w-2 h-2 rounded-full bg-emerald-500 glow-dot" style="color: #10B981;"></span>
17
- <div class="text-xs uppercase tracking-wider text-emerald-400/70 font-medium">Armed</div>
18
- </div>
19
- <div class="text-xl font-mono font-semibold text-emerald-400" id="armed-count">--</div>
20
- </div>
21
- <div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 card">
22
- <div class="text-xs uppercase tracking-wider text-slate-400 font-medium mb-2">Fired</div>
23
- <div class="text-xl font-mono font-semibold text-slate-400" id="fired-count">--</div>
24
- </div>
25
- </div>
26
-
27
- <!-- Triggers grid -->
28
- <div>
29
- <div class="flex items-center gap-3 mb-3">
30
- <div class="text-xs uppercase tracking-wider text-slate-400 font-medium">Armed Triggers</div>
31
- <span class="w-2 h-2 rounded-full bg-emerald-500 glow-dot" style="color: #10B981;"></span>
32
- </div>
33
- <div class="grid grid-cols-2 gap-4" id="armed-triggers">
34
- <div class="text-xs text-slate-600 py-4 text-center col-span-2">loading...</div>
35
- </div>
36
- </div>
37
-
38
- <div>
39
- <div class="text-xs uppercase tracking-wider text-slate-400 font-medium mb-3">Fired / Inactive</div>
40
- <div class="grid grid-cols-2 gap-4" id="fired-triggers">
41
- <div class="text-xs text-slate-600 py-4 text-center col-span-2">loading...</div>
42
- </div>
43
- </div>
44
- </div>
45
-
46
- <style>
47
- @keyframes triggerPulse {
48
- 0%, 100% { border-color: rgba(16,185,129,0.2); }
49
- 50% { border-color: rgba(16,185,129,0.5); }
50
- }
51
- .trigger-armed { animation: triggerPulse 3s ease-in-out infinite; }
52
- </style>
53
- {% endblock %}
54
-
55
- {% block scripts %}
56
- <script>
57
- function renderTrigger(t, isArmed) {
58
- const pattern = t.trigger_pattern || t.pattern || '--';
59
- const action = t.action || '--';
60
- const context = t.context || '';
61
- const status = t.status || 'armed';
62
- const firedAt = t.fired_at;
63
-
64
- if (isArmed) {
65
- return `<div class="bg-slate-900/50 border border-emerald-500/20 rounded-xl p-5 card trigger-armed">
66
- <div class="flex items-center gap-2 mb-3">
67
- <span class="relative flex h-2 w-2">
68
- <span class="animate-ping absolute inline-flex h-full w-full rounded-full bg-emerald-400 opacity-75"></span>
69
- <span class="relative inline-flex rounded-full h-2 w-2 bg-emerald-500"></span>
70
- </span>
71
- <span class="text-xs font-medium text-emerald-400 uppercase tracking-wider">Armed</span>
72
- </div>
73
- <div class="mb-3">
74
- <div class="text-[10px] text-slate-500 uppercase mb-1">Pattern</div>
75
- <div class="text-sm text-slate-200 font-mono bg-slate-800/50 rounded-lg px-3 py-2">${escapeHtml(pattern)}</div>
76
- </div>
77
- <div class="mb-3">
78
- <div class="text-[10px] text-slate-500 uppercase mb-1">Action</div>
79
- <div class="text-xs text-slate-300">${escapeHtml(action)}</div>
80
- </div>
81
- ${context ? `<div class="text-[10px] text-slate-500 mt-2">${escapeHtml(context)}</div>` : ''}
82
- </div>`;
83
- } else {
84
- return `<div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-5 card opacity-60 hover:opacity-80 transition-opacity">
85
- <div class="flex items-center gap-2 mb-3">
86
- <span class="w-2 h-2 rounded-full bg-slate-600"></span>
87
- <span class="text-xs font-medium text-slate-500 uppercase tracking-wider">${escapeHtml(status)}</span>
88
- ${firedAt ? `<span class="ml-auto text-[10px] text-slate-600">${relativeTime(firedAt)}</span>` : ''}
89
- </div>
90
- <div class="mb-3">
91
- <div class="text-[10px] text-slate-600 uppercase mb-1">Pattern</div>
92
- <div class="text-xs text-slate-400 font-mono bg-slate-800/30 rounded-lg px-3 py-2">${escapeHtml(pattern)}</div>
93
- </div>
94
- <div>
95
- <div class="text-[10px] text-slate-600 uppercase mb-1">Action</div>
96
- <div class="text-xs text-slate-500">${escapeHtml(action)}</div>
97
- </div>
98
- ${context ? `<div class="text-[10px] text-slate-600 mt-2">${escapeHtml(context)}</div>` : ''}
99
- </div>`;
100
- }
101
- }
102
-
103
- async function loadTriggers() {
104
- const data = await fetchJSON('/api/triggers');
105
- if (!data) return;
106
-
107
- const triggers = data.triggers || [];
108
- const armed = triggers.filter(t => t.status === 'armed' || t.status === 'active');
109
- const fired = triggers.filter(t => t.status !== 'armed' && t.status !== 'active');
110
-
111
- document.getElementById('total-triggers').textContent = triggers.length;
112
- document.getElementById('armed-count').textContent = data.armed ?? armed.length;
113
- document.getElementById('fired-count').textContent = data.fired ?? fired.length;
114
-
115
- const armedContainer = document.getElementById('armed-triggers');
116
- if (armed.length) {
117
- armedContainer.innerHTML = armed.map(t => renderTrigger(t, true)).join('');
118
- } else {
119
- armedContainer.innerHTML = '<div class="col-span-2 bg-slate-900/50 border border-slate-800/50 rounded-xl p-5 text-xs text-slate-500 text-center">No armed triggers</div>';
120
- }
121
-
122
- const firedContainer = document.getElementById('fired-triggers');
123
- if (fired.length) {
124
- firedContainer.innerHTML = fired.map(t => renderTrigger(t, false)).join('');
125
- } else {
126
- firedContainer.innerHTML = '<div class="col-span-2 bg-slate-900/50 border border-slate-800/50 rounded-xl p-5 text-xs text-slate-500 text-center">No fired triggers</div>';
127
- }
128
- }
129
-
130
- loadTriggers();
131
- setInterval(loadTriggers, 60000);
132
- </script>
133
- {% endblock %}