nexo-brain 2.4.0 → 2.5.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +80 -4
- package/bin/nexo-brain.js +238 -12
- package/bin/nexo.js +55 -0
- package/community/skills/.gitkeep +1 -0
- package/package.json +11 -3
- package/src/auto_update.py +193 -9
- package/src/cli.py +719 -0
- package/src/cognitive/_ingest.py +1 -1
- package/src/cognitive/_memory.py +4 -4
- package/src/crons/manifest.json +8 -0
- package/src/dashboard/app.py +700 -35
- package/src/dashboard/templates/adaptive.html +112 -218
- package/src/dashboard/templates/artifacts.html +133 -0
- package/src/dashboard/templates/backups.html +136 -0
- package/src/dashboard/templates/base.html +413 -0
- package/src/dashboard/templates/calendar.html +523 -654
- package/src/dashboard/templates/chat.html +356 -0
- package/src/dashboard/templates/claims.html +259 -0
- package/src/dashboard/templates/cortex.html +262 -0
- package/src/dashboard/templates/credentials.html +128 -0
- package/src/dashboard/templates/crons.html +370 -0
- package/src/dashboard/templates/dashboard.html +383 -578
- package/src/dashboard/templates/dreams.html +252 -0
- package/src/dashboard/templates/email.html +160 -0
- package/src/dashboard/templates/evolution.html +189 -0
- package/src/dashboard/templates/feed.html +249 -0
- package/src/dashboard/templates/followup_health.html +170 -0
- package/src/dashboard/templates/graph.html +191 -269
- package/src/dashboard/templates/guard.html +259 -0
- package/src/dashboard/templates/inbox.html +220 -346
- package/src/dashboard/templates/memory.html +317 -197
- package/src/dashboard/templates/operations.html +521 -698
- package/src/dashboard/templates/plugins.html +185 -0
- package/src/dashboard/templates/rules.html +246 -0
- package/src/dashboard/templates/sentiment.html +247 -0
- package/src/dashboard/templates/sessions.html +215 -182
- package/src/dashboard/templates/skills.html +329 -0
- package/src/dashboard/templates/somatic.html +68 -172
- package/src/dashboard/templates/triggers.html +133 -0
- package/src/dashboard/templates/trust.html +360 -0
- package/src/db/__init__.py +5 -0
- package/src/db/_schema.py +16 -1
- package/src/db/_sessions.py +22 -0
- package/src/db/_skills.py +980 -274
- package/src/doctor/__init__.py +1 -0
- package/src/doctor/formatters.py +52 -0
- package/src/doctor/models.py +44 -0
- package/src/doctor/orchestrator.py +42 -0
- package/src/doctor/providers/__init__.py +1 -0
- package/src/doctor/providers/boot.py +206 -0
- package/src/doctor/providers/deep.py +292 -0
- package/src/doctor/providers/runtime.py +686 -0
- package/src/evolution_cycle.py +86 -6
- package/src/hooks/post-compact.sh +5 -1
- package/src/hooks/pre-compact.sh +1 -1
- package/src/plugins/doctor.py +36 -0
- package/src/plugins/evolution.py +11 -3
- package/src/plugins/skills.py +135 -175
- package/src/requirements.txt +1 -0
- package/src/script_registry.py +322 -0
- package/src/scripts/deep-sleep/apply_findings.py +63 -48
- package/src/scripts/deep-sleep/extract-prompt.md +14 -0
- package/src/scripts/deep-sleep/synthesize-prompt.md +36 -0
- package/src/scripts/deep-sleep/synthesize.py +37 -1
- package/src/scripts/nexo-dashboard.sh +29 -0
- package/src/scripts/nexo-day-orchestrator.sh +139 -0
- package/src/scripts/nexo-evolution-run.py +141 -54
- package/src/scripts/nexo-learning-housekeep.py +1 -1
- package/src/scripts/nexo-watchdog.sh +1 -1
- package/src/server.py +9 -5
- package/src/skills/run-runtime-doctor/guide.md +12 -0
- package/src/skills/run-runtime-doctor/script.py +21 -0
- package/src/skills/run-runtime-doctor/skill.json +25 -0
- package/src/skills_runtime.py +347 -0
- package/src/tools_menu.py +3 -2
- package/src/tools_sessions.py +126 -0
- package/src/user_context.py +46 -0
- package/templates/nexo_helper.py +45 -0
- package/templates/script-template.py +44 -0
- package/templates/skill-script-template.py +39 -0
- package/templates/skill-template.md +33 -0
|
@@ -0,0 +1,329 @@
|
|
|
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">⚠ ${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,177 +1,73 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
<
|
|
9
|
-
<
|
|
10
|
-
<
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M8.25 6.75h12M8.25 12h12m-12 5.25h12M3.75 6.75h.007v.008H3.75V6.75zm.375 0a.375.375 0 11-.75 0 .375.375 0 01.75 0zM3.75 12h.007v.008H3.75V12zm.375 0a.375.375 0 11-.75 0 .375.375 0 01.75 0zm-.375 5.25h.007v.008H3.75v-.008zm.375 0a.375.375 0 11-.75 0 .375.375 0 01.75 0z" />
|
|
38
|
-
</svg>
|
|
39
|
-
</a>
|
|
40
|
-
<!-- Calendar -->
|
|
41
|
-
<a href="/calendar" title="Calendar" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-500 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
42
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
43
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M6.75 3v2.25M17.25 3v2.25M3 18.75V7.5a2.25 2.25 0 012.25-2.25h13.5A2.25 2.25 0 0121 7.5v11.25m-18 0A2.25 2.25 0 005.25 21h13.5A2.25 2.25 0 0021 18.75m-18 0v-7.5A2.25 2.25 0 015.25 9h13.5A2.25 2.25 0 0121 11.25v7.5" />
|
|
44
|
-
</svg>
|
|
45
|
-
</a>
|
|
46
|
-
<!-- Inbox -->
|
|
47
|
-
<a href="/inbox" title="Inbox" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-500 hover:text-slate-200 hover:bg-slate-800 transition-colors relative">
|
|
48
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
49
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M21.75 9v.906a2.25 2.25 0 01-1.183 1.981l-6.478 3.488M2.25 9v.906a2.25 2.25 0 001.183 1.981l6.478 3.488m8.839 2.51l-4.66-2.51m0 0l-1.023-.55a2.25 2.25 0 00-2.134 0l-1.022.55m0 0l-4.661 2.51m16.5 1.615a2.25 2.25 0 01-2.25 2.25h-15a2.25 2.25 0 01-2.25-2.25V8.844a2.25 2.25 0 011.183-1.981l7.5-4.039a2.25 2.25 0 012.134 0l7.5 4.039a2.25 2.25 0 011.183 1.98V19.5z" />
|
|
50
|
-
</svg>
|
|
51
|
-
<span id="inbox-badge" class="absolute top-1 right-1 w-2 h-2 bg-violet-500 rounded-full hidden"></span>
|
|
52
|
-
</a>
|
|
53
|
-
|
|
54
|
-
<div class="w-6 border-t border-slate-800 my-1"></div>
|
|
55
|
-
|
|
56
|
-
<!-- Memory -->
|
|
57
|
-
<a href="/memory" title="Memory" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-500 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
58
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
59
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M3.75 9.776c.112-.017.227-.026.344-.026h15.812c.117 0 .232.009.344.026m-16.5 0a2.25 2.25 0 00-1.883 2.542l.857 6a2.25 2.25 0 002.227 1.932H19.05a2.25 2.25 0 002.227-1.932l.857-6a2.25 2.25 0 00-1.883-2.542m-16.5 0V6A2.25 2.25 0 016 3.75h3.879a1.5 1.5 0 011.06.44l2.122 2.12a1.5 1.5 0 001.06.44H18A2.25 2.25 0 0120.25 9v.776" />
|
|
60
|
-
</svg>
|
|
61
|
-
</a>
|
|
62
|
-
<!-- Graph -->
|
|
63
|
-
<a href="/graph" title="Graph" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-500 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
64
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
65
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M7.5 14.25v2.25m3-4.5v4.5m3-6.75v6.75m3-9v9M6 20.25h12A2.25 2.25 0 0020.25 18V6A2.25 2.25 0 0018 3.75H6A2.25 2.25 0 003.75 6v12A2.25 2.25 0 006 20.25z" />
|
|
66
|
-
</svg>
|
|
67
|
-
</a>
|
|
68
|
-
<!-- Sessions -->
|
|
69
|
-
<a href="/sessions" title="Sessions" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-500 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
70
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
71
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M12 6v6h4.5m4.5 0a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
72
|
-
</svg>
|
|
73
|
-
</a>
|
|
74
|
-
<!-- Somatic (ACTIVE) -->
|
|
75
|
-
<a href="/somatic" title="Somatic" class="w-9 h-9 flex items-center justify-center rounded-lg bg-slate-800 text-slate-200 transition-colors">
|
|
76
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
77
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M21 8.25c0-2.485-2.099-4.5-4.688-4.5-1.935 0-3.597 1.126-4.312 2.733-.715-1.607-2.377-2.733-4.313-2.733C5.1 3.75 3 5.765 3 8.25c0 7.22 9 12 9 12s9-4.78 9-12z" />
|
|
78
|
-
</svg>
|
|
79
|
-
</a>
|
|
80
|
-
<!-- Adaptive -->
|
|
81
|
-
<a href="/adaptive" title="Adaptive" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-500 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
82
|
-
<svg class="w-5 h-5" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24">
|
|
83
|
-
<path stroke-linecap="round" stroke-linejoin="round" d="M10.5 6h9.75M10.5 6a1.5 1.5 0 11-3 0m3 0a1.5 1.5 0 10-3 0M3.75 6H7.5m3 12h9.75m-9.75 0a1.5 1.5 0 01-3 0m3 0a1.5 1.5 0 00-3 0m-3.75 0H7.5m9-6h3.75m-3.75 0a1.5 1.5 0 01-3 0m3 0a1.5 1.5 0 00-3 0m-9.75 0h9.75" />
|
|
84
|
-
</svg>
|
|
85
|
-
</a>
|
|
86
|
-
</nav>
|
|
87
|
-
|
|
88
|
-
<!-- Trust score -->
|
|
89
|
-
<div class="mt-auto flex flex-col items-center gap-0.5">
|
|
90
|
-
<span class="text-xs uppercase tracking-wider text-slate-600">Trust</span>
|
|
91
|
-
<span id="sidebar-trust" class="text-xs font-mono font-medium text-slate-400">--</span>
|
|
92
|
-
</div>
|
|
93
|
-
</aside>
|
|
94
|
-
|
|
95
|
-
<!-- Main -->
|
|
96
|
-
<main class="ml-14 min-h-screen bg-gray-950 text-slate-200">
|
|
97
|
-
<header class="h-12 border-b border-slate-800/50 flex items-center px-6">
|
|
98
|
-
<h1 class="text-sm font-display font-semibold">Somatic Markers</h1>
|
|
99
|
-
</header>
|
|
100
|
-
<div class="p-6 max-w-3xl">
|
|
101
|
-
<p class="text-sm text-slate-400 mb-6">Risk associations from past errors. Higher scores indicate areas needing extra caution.</p>
|
|
102
|
-
<div id="risks" class="space-y-2"></div>
|
|
103
|
-
<div id="empty" class="p-12 text-center text-slate-500 text-sm hidden">No somatic markers recorded</div>
|
|
104
|
-
</div>
|
|
105
|
-
</main>
|
|
106
|
-
|
|
107
|
-
<script>
|
|
108
|
-
function barColor(score) {
|
|
109
|
-
if (score > 5) return 'bg-red-500';
|
|
110
|
-
if (score > 2) return 'bg-amber-500';
|
|
111
|
-
return 'bg-emerald-500';
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
function scoreColor(score) {
|
|
115
|
-
if (score > 5) return 'text-red-400';
|
|
116
|
-
if (score > 2) return 'text-amber-400';
|
|
117
|
-
return 'text-emerald-400';
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
function loadData() {
|
|
121
|
-
fetch('/api/somatic')
|
|
122
|
-
.then(r => r.json())
|
|
123
|
-
.then(data => {
|
|
124
|
-
const risks = (data.risks || []).slice().sort((a, b) => {
|
|
125
|
-
const sa = Math.abs(a.cumulative_score || a.score || 0);
|
|
126
|
-
const sb = Math.abs(b.cumulative_score || b.score || 0);
|
|
127
|
-
return sb - sa;
|
|
128
|
-
});
|
|
129
|
-
|
|
130
|
-
const container = document.getElementById('risks');
|
|
131
|
-
const empty = document.getElementById('empty');
|
|
132
|
-
|
|
133
|
-
if (!risks.length) {
|
|
134
|
-
empty.classList.remove('hidden');
|
|
135
|
-
return;
|
|
136
|
-
}
|
|
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
|
+
});
|
|
137
37
|
|
|
138
|
-
|
|
38
|
+
const container = document.getElementById('risks');
|
|
39
|
+
const empty = document.getElementById('empty');
|
|
139
40
|
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
const target = (r.target || r.name || '').replace(/</g, '<');
|
|
145
|
-
const bc = barColor(score);
|
|
146
|
-
const sc = scoreColor(score);
|
|
41
|
+
if (!risks.length) {
|
|
42
|
+
empty.classList.remove('hidden');
|
|
43
|
+
return;
|
|
44
|
+
}
|
|
147
45
|
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
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, '<');
|
|
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>
|
|
155
63
|
</div>`;
|
|
156
|
-
|
|
157
|
-
})
|
|
158
|
-
.catch(err => {
|
|
159
|
-
document.getElementById('risks').innerHTML = `<p class="text-xs text-red-400">Failed to load: ${err.message}</p>`;
|
|
160
|
-
});
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
// Trust score
|
|
164
|
-
fetch('/api/stats')
|
|
165
|
-
.then(r => r.json())
|
|
166
|
-
.then(data => {
|
|
167
|
-
if (data.trust_score != null) {
|
|
168
|
-
document.getElementById('sidebar-trust').textContent = Number(data.trust_score).toFixed(0);
|
|
169
|
-
}
|
|
64
|
+
}).join('');
|
|
170
65
|
})
|
|
171
|
-
.catch(
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
</
|
|
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 %}
|