nexo-brain 2.4.0 → 2.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +65 -2
- package/bin/nexo-brain.js +208 -11
- package/bin/nexo.js +55 -0
- package/community/skills/.gitkeep +1 -0
- package/package.json +5 -2
- package/src/auto_update.py +158 -8
- package/src/cli.py +605 -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/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 +2 -1
- 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 +2 -1
- 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
|
@@ -1,185 +1,218 @@
|
|
|
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
|
-
<!-- Inbox -->
|
|
37
|
-
<a href="/inbox" title="Inbox" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-400 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
38
|
-
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><polyline points="22 12 16 12 14 15 10 15 8 12 2 12"/><path d="M5.45 5.11L2 12v6a2 2 0 0 0 2 2h16a2 2 0 0 0 2-2v-6l-3.45-6.89A2 2 0 0 0 16.76 4H7.24a2 2 0 0 0-1.79 1.11z"/></svg>
|
|
39
|
-
</a>
|
|
40
|
-
|
|
41
|
-
<div class="w-6 h-px bg-slate-800 my-1"></div>
|
|
42
|
-
|
|
43
|
-
<!-- Memory -->
|
|
44
|
-
<a href="/memory" title="Memory" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-400 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
45
|
-
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><path d="M12 2a7 7 0 0 1 7 7c0 2.38-1.19 4.47-3 5.74V17a1 1 0 0 1-1 1H9a1 1 0 0 1-1-1v-2.26C6.19 13.47 5 11.38 5 9a7 7 0 0 1 7-7z"/><path d="M9 21h6"/><path d="M10 21v1a1 1 0 0 0 1 1h2a1 1 0 0 0 1-1v-1"/></svg>
|
|
46
|
-
</a>
|
|
47
|
-
<!-- Graph -->
|
|
48
|
-
<a href="/graph" title="Knowledge Graph" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-400 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
49
|
-
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><path d="M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71"/><path d="M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71"/></svg>
|
|
50
|
-
</a>
|
|
51
|
-
<!-- Sessions — ACTIVE -->
|
|
52
|
-
<a href="/sessions" title="Sessions" class="w-9 h-9 flex items-center justify-center rounded-lg bg-violet-600/20 text-violet-400 transition-colors">
|
|
53
|
-
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
|
|
54
|
-
</a>
|
|
55
|
-
<!-- Somatic -->
|
|
56
|
-
<a href="/somatic" title="Somatic" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-400 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
57
|
-
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><path d="M20.84 4.61a5.5 5.5 0 0 0-7.78 0L12 5.67l-1.06-1.06a5.5 5.5 0 0 0-7.78 7.78l1.06 1.06L12 21.23l7.78-7.78 1.06-1.06a5.5 5.5 0 0 0 0-7.78z"/></svg>
|
|
58
|
-
</a>
|
|
59
|
-
<!-- Adaptive -->
|
|
60
|
-
<a href="/adaptive" title="Adaptive" class="w-9 h-9 flex items-center justify-center rounded-lg text-slate-400 hover:text-slate-200 hover:bg-slate-800 transition-colors">
|
|
61
|
-
<svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.75" stroke-linecap="round" stroke-linejoin="round"><circle cx="12" cy="12" r="3"/><path d="M12 1v4M12 19v4M4.22 4.22l2.83 2.83M16.95 16.95l2.83 2.83M1 12h4M19 12h4M4.22 19.78l2.83-2.83M16.95 7.05l2.83-2.83"/></svg>
|
|
62
|
-
</a>
|
|
63
|
-
</nav>
|
|
64
|
-
|
|
65
|
-
<!-- Trust score -->
|
|
66
|
-
<div class="flex flex-col items-center gap-0.5 mt-2">
|
|
67
|
-
<span class="text-xs font-display font-semibold text-slate-500 uppercase tracking-widest">Trust</span>
|
|
68
|
-
<span id="trust-score" class="text-sm font-mono font-bold text-violet-400">--</span>
|
|
1
|
+
{% extends "base.html" %}
|
|
2
|
+
|
|
3
|
+
{% block title %}Session Replay{% endblock %}
|
|
4
|
+
{% block page_title %}Session Replay{% endblock %}
|
|
5
|
+
|
|
6
|
+
{% block content %}
|
|
7
|
+
<div class="space-y-5">
|
|
8
|
+
<!-- Active sessions -->
|
|
9
|
+
<div>
|
|
10
|
+
<div class="text-xs uppercase tracking-wider text-slate-400 font-medium mb-3">Active Sessions</div>
|
|
11
|
+
<div class="grid grid-cols-3 gap-4" id="active-sessions">
|
|
12
|
+
<div class="text-xs text-slate-600 py-4 text-center col-span-3">loading...</div>
|
|
13
|
+
</div>
|
|
14
|
+
</div>
|
|
15
|
+
|
|
16
|
+
<!-- Main content: Diaries + Checkpoints -->
|
|
17
|
+
<div class="grid grid-cols-3 gap-5">
|
|
18
|
+
<!-- Diary timeline -->
|
|
19
|
+
<div class="col-span-2 bg-slate-900/50 border border-slate-800/50 rounded-xl p-5 card">
|
|
20
|
+
<div class="flex items-center justify-between mb-4">
|
|
21
|
+
<div class="text-xs uppercase tracking-wider text-slate-400 font-medium">Session Diaries</div>
|
|
22
|
+
<button onclick="loadMore()" id="load-more-btn" class="text-xs px-2.5 py-1 rounded-md bg-slate-800/50 text-slate-400 hover:bg-slate-700 transition-colors hidden">Load more</button>
|
|
23
|
+
</div>
|
|
24
|
+
<div class="space-y-3" id="diaries">
|
|
25
|
+
<div class="text-xs text-slate-600 py-4 text-center">loading...</div>
|
|
26
|
+
</div>
|
|
27
|
+
</div>
|
|
28
|
+
|
|
29
|
+
<!-- Sidebar: Checkpoints + Mental states -->
|
|
30
|
+
<div class="space-y-5">
|
|
31
|
+
<!-- Mental State -->
|
|
32
|
+
<div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-5 card">
|
|
33
|
+
<div class="text-xs uppercase tracking-wider text-slate-400 font-medium mb-3">Mental States</div>
|
|
34
|
+
<div class="space-y-2" id="mental-states">
|
|
35
|
+
<div class="text-xs text-slate-600">loading...</div>
|
|
69
36
|
</div>
|
|
70
|
-
|
|
37
|
+
</div>
|
|
71
38
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
<
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
<div class="p-6 max-w-3xl">
|
|
79
|
-
<div id="sessions" class="space-y-3"></div>
|
|
80
|
-
<button id="load-more" onclick="loadMore()" class="mt-4 w-full py-2 text-xs text-slate-400 bg-slate-900/50 border border-slate-800/50 rounded-lg hover:bg-slate-800 transition-colors" style="display:none">Load more</button>
|
|
39
|
+
<!-- Checkpoints -->
|
|
40
|
+
<div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-5 card">
|
|
41
|
+
<div class="text-xs uppercase tracking-wider text-slate-400 font-medium mb-3">Checkpoints</div>
|
|
42
|
+
<div class="space-y-2" id="checkpoints">
|
|
43
|
+
<div class="text-xs text-slate-600">loading...</div>
|
|
81
44
|
</div>
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
</
|
|
185
|
-
|
|
45
|
+
</div>
|
|
46
|
+
</div>
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
|
|
50
|
+
<style>
|
|
51
|
+
@keyframes livePulse { 0%, 100% { box-shadow: 0 0 0 0 rgba(16,185,129,0.4); } 50% { box-shadow: 0 0 0 6px rgba(16,185,129,0); } }
|
|
52
|
+
.live-pulse { animation: livePulse 2s ease-in-out infinite; }
|
|
53
|
+
</style>
|
|
54
|
+
{% endblock %}
|
|
55
|
+
|
|
56
|
+
{% block scripts %}
|
|
57
|
+
<script>
|
|
58
|
+
let diaryOffset = 0;
|
|
59
|
+
const PAGE_SIZE = 15;
|
|
60
|
+
let exhausted = false;
|
|
61
|
+
|
|
62
|
+
const STATE_BADGES = {
|
|
63
|
+
focused: 'bg-violet-500/15 text-violet-400 border-violet-500/30',
|
|
64
|
+
energized: 'bg-emerald-500/15 text-emerald-400 border-emerald-500/30',
|
|
65
|
+
tired: 'bg-amber-500/15 text-amber-400 border-amber-500/30',
|
|
66
|
+
frustrated: 'bg-red-500/15 text-red-400 border-red-500/30',
|
|
67
|
+
calm: 'bg-blue-500/15 text-blue-400 border-blue-500/30',
|
|
68
|
+
flow: 'bg-pink-500/15 text-pink-400 border-pink-500/30',
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
function renderActiveSession(s) {
|
|
72
|
+
const sid = s.session_id || s.id || '';
|
|
73
|
+
const shortId = String(sid).substring(0, 12);
|
|
74
|
+
const lastHB = s.last_heartbeat || s.created_at;
|
|
75
|
+
const cutoff = Date.now() - 15 * 60 * 1000;
|
|
76
|
+
const isLive = new Date(lastHB).getTime() > cutoff;
|
|
77
|
+
|
|
78
|
+
return `<div class="bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 card ${isLive ? 'live-pulse' : ''}">
|
|
79
|
+
<div class="flex items-center gap-2 mb-2">
|
|
80
|
+
<span class="relative flex h-2 w-2">
|
|
81
|
+
${isLive ? `<span class="animate-ping absolute inline-flex h-full w-full rounded-full bg-emerald-400 opacity-75"></span>` : ''}
|
|
82
|
+
<span class="relative inline-flex rounded-full h-2 w-2 ${isLive ? 'bg-emerald-500' : 'bg-slate-600'}"></span>
|
|
83
|
+
</span>
|
|
84
|
+
<span class="text-xs font-mono text-violet-400">${escapeHtml(shortId)}</span>
|
|
85
|
+
<span class="ml-auto text-[10px] text-slate-500">${isLive ? 'LIVE' : 'idle'}</span>
|
|
86
|
+
</div>
|
|
87
|
+
<div class="text-[10px] text-slate-500">Started ${relativeTime(s.created_at)}</div>
|
|
88
|
+
${lastHB ? `<div class="text-[10px] text-slate-600">Last heartbeat: ${relativeTime(lastHB)}</div>` : ''}
|
|
89
|
+
</div>`;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function renderDiary(d) {
|
|
93
|
+
const sid = d.session_id || d.id || '';
|
|
94
|
+
const state = d.mental_state || '';
|
|
95
|
+
const stateLower = state.toLowerCase();
|
|
96
|
+
const badgeClass = STATE_BADGES[stateLower] || 'bg-slate-500/15 text-slate-400 border-slate-500/30';
|
|
97
|
+
|
|
98
|
+
return `<div class="bg-slate-800/30 border border-slate-800/30 rounded-lg p-4 hover:border-slate-700/50 transition-colors">
|
|
99
|
+
<div class="flex items-center gap-2 mb-2 flex-wrap">
|
|
100
|
+
<span class="text-xs font-mono text-violet-400">${escapeHtml(String(sid).substring(0, 12))}</span>
|
|
101
|
+
<span class="text-[10px] text-slate-500">${escapeHtml(d.created_at || '')}</span>
|
|
102
|
+
${d.domain ? `<span class="text-[10px] px-1.5 py-0.5 rounded bg-slate-800 text-slate-400">${escapeHtml(d.domain)}</span>` : ''}
|
|
103
|
+
${state ? `<span class="text-[10px] px-1.5 py-0.5 rounded border ${badgeClass}">${escapeHtml(state)}</span>` : ''}
|
|
104
|
+
</div>
|
|
105
|
+
${d.summary ? `<p class="text-sm text-slate-200 leading-relaxed mb-2 whitespace-pre-wrap">${escapeHtml(d.summary)}</p>` : ''}
|
|
106
|
+
${d.self_critique ? `<div class="border-l-2 border-amber-500/30 pl-3 mb-2"><p class="text-xs text-amber-300/80 leading-relaxed whitespace-pre-wrap">${escapeHtml(d.self_critique)}</p></div>` : ''}
|
|
107
|
+
</div>`;
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
async function loadMore() {
|
|
111
|
+
if (exhausted) return;
|
|
112
|
+
const btn = document.getElementById('load-more-btn');
|
|
113
|
+
btn.textContent = 'Loading...'; btn.disabled = true;
|
|
114
|
+
|
|
115
|
+
try {
|
|
116
|
+
const data = await fetchJSON(`/api/sessions?limit=${PAGE_SIZE}&offset=${diaryOffset}`);
|
|
117
|
+
if (!data) throw new Error('No data');
|
|
118
|
+
const diaries = data.diaries || data.sessions || [];
|
|
119
|
+
|
|
120
|
+
if (diaries.length > 0) {
|
|
121
|
+
document.getElementById('diaries').insertAdjacentHTML('beforeend', diaries.map(renderDiary).join(''));
|
|
122
|
+
diaryOffset += diaries.length;
|
|
123
|
+
}
|
|
124
|
+
if (diaries.length < PAGE_SIZE) {
|
|
125
|
+
exhausted = true; btn.classList.add('hidden');
|
|
126
|
+
} else {
|
|
127
|
+
btn.textContent = 'Load more'; btn.disabled = false;
|
|
128
|
+
}
|
|
129
|
+
} catch (err) {
|
|
130
|
+
btn.textContent = 'Error -- retry'; btn.disabled = false;
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
async function loadSessions() {
|
|
135
|
+
const data = await fetchJSON('/api/sessions?limit=' + PAGE_SIZE);
|
|
136
|
+
if (!data) return;
|
|
137
|
+
|
|
138
|
+
// Active sessions
|
|
139
|
+
const sessions = data.sessions || [];
|
|
140
|
+
const activeContainer = document.getElementById('active-sessions');
|
|
141
|
+
const cutoff = Date.now() - 15 * 60 * 1000;
|
|
142
|
+
const activeSessions = sessions.filter(s => new Date(s.last_heartbeat || s.created_at || 0).getTime() > cutoff);
|
|
143
|
+
|
|
144
|
+
if (activeSessions.length > 0) {
|
|
145
|
+
activeContainer.innerHTML = activeSessions.map(renderActiveSession).join('');
|
|
146
|
+
} else {
|
|
147
|
+
activeContainer.innerHTML = '<div class="col-span-3 bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-xs text-slate-500 text-center">No active sessions</div>';
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// Diaries
|
|
151
|
+
const diaries = data.diaries || sessions;
|
|
152
|
+
const diaryContainer = document.getElementById('diaries');
|
|
153
|
+
if (diaries.length > 0) {
|
|
154
|
+
diaryContainer.innerHTML = diaries.map(renderDiary).join('');
|
|
155
|
+
diaryOffset = diaries.length;
|
|
156
|
+
if (diaries.length >= PAGE_SIZE) {
|
|
157
|
+
document.getElementById('load-more-btn').classList.remove('hidden');
|
|
158
|
+
}
|
|
159
|
+
} else {
|
|
160
|
+
diaryContainer.innerHTML = '<div class="text-xs text-slate-600 py-4 text-center">No session diaries found</div>';
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
// Mental states
|
|
164
|
+
const mentalContainer = document.getElementById('mental-states');
|
|
165
|
+
const states = {};
|
|
166
|
+
diaries.forEach(d => {
|
|
167
|
+
if (d.mental_state) {
|
|
168
|
+
const s = d.mental_state.toLowerCase();
|
|
169
|
+
states[s] = (states[s] || 0) + 1;
|
|
170
|
+
}
|
|
171
|
+
});
|
|
172
|
+
const stateEntries = Object.entries(states).sort((a, b) => b[1] - a[1]);
|
|
173
|
+
if (stateEntries.length) {
|
|
174
|
+
mentalContainer.innerHTML = stateEntries.map(([state, count]) => {
|
|
175
|
+
const badgeClass = STATE_BADGES[state] || 'bg-slate-500/15 text-slate-400 border-slate-500/30';
|
|
176
|
+
return `<div class="flex items-center justify-between">
|
|
177
|
+
<span class="text-xs px-2 py-0.5 rounded border ${badgeClass}">${escapeHtml(state)}</span>
|
|
178
|
+
<span class="text-xs font-mono text-slate-500">${count}</span>
|
|
179
|
+
</div>`;
|
|
180
|
+
}).join('');
|
|
181
|
+
} else {
|
|
182
|
+
mentalContainer.innerHTML = '<div class="text-xs text-slate-600">No mental state data</div>';
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// Checkpoints
|
|
186
|
+
const checkpoints = data.checkpoints || [];
|
|
187
|
+
const cpContainer = document.getElementById('checkpoints');
|
|
188
|
+
if (checkpoints.length) {
|
|
189
|
+
cpContainer.innerHTML = checkpoints.slice(0, 10).map(cp => {
|
|
190
|
+
return `<div class="flex items-center gap-2 py-1.5 border-b border-slate-800/30 last:border-0">
|
|
191
|
+
<svg class="w-3.5 h-3.5 text-slate-500 flex-shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24"><path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 13l4 4L19 7"/></svg>
|
|
192
|
+
<span class="text-xs text-slate-400 truncate flex-1">${escapeHtml(cp.label || cp.session_id || 'checkpoint')}</span>
|
|
193
|
+
<span class="text-[10px] text-slate-600 font-mono">${relativeTime(cp.created_at)}</span>
|
|
194
|
+
</div>`;
|
|
195
|
+
}).join('');
|
|
196
|
+
} else {
|
|
197
|
+
cpContainer.innerHTML = '<div class="text-xs text-slate-600">No checkpoints</div>';
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
loadSessions();
|
|
202
|
+
setInterval(() => {
|
|
203
|
+
// Refresh active sessions only
|
|
204
|
+
fetchJSON('/api/sessions?limit=5').then(data => {
|
|
205
|
+
if (!data) return;
|
|
206
|
+
const sessions = data.sessions || [];
|
|
207
|
+
const activeContainer = document.getElementById('active-sessions');
|
|
208
|
+
const cutoff = Date.now() - 15 * 60 * 1000;
|
|
209
|
+
const activeSessions = sessions.filter(s => new Date(s.last_heartbeat || s.created_at || 0).getTime() > cutoff);
|
|
210
|
+
if (activeSessions.length > 0) {
|
|
211
|
+
activeContainer.innerHTML = activeSessions.map(renderActiveSession).join('');
|
|
212
|
+
} else {
|
|
213
|
+
activeContainer.innerHTML = '<div class="col-span-3 bg-slate-900/50 border border-slate-800/50 rounded-xl p-4 text-xs text-slate-500 text-center">No active sessions</div>';
|
|
214
|
+
}
|
|
215
|
+
});
|
|
216
|
+
}, 60000);
|
|
217
|
+
</script>
|
|
218
|
+
{% endblock %}
|