nexo-brain 5.3.19 → 5.3.21
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/.claude-plugin/plugin.json +1 -1
- package/bin/nexo-brain.js +52 -10
- package/package.json +1 -1
- package/src/auto_update.py +11 -8
- package/src/dashboard/static/favicon 2.svg +32 -0
- package/src/dashboard/static/nexo-logo 2.png +0 -0
- package/src/dashboard/static/nexo-logo 2.svg +40 -0
- package/src/dashboard/static/style 2.css +2458 -0
- package/src/dashboard/templates/adaptive 2.html +118 -0
- package/src/dashboard/templates/artifacts 2.html +133 -0
- package/src/dashboard/templates/backups 2.html +136 -0
- package/src/dashboard/templates/base 2.html +417 -0
- package/src/dashboard/templates/calendar 2.html +591 -0
- package/src/dashboard/templates/chat 2.html +356 -0
- package/src/dashboard/templates/claims 2.html +259 -0
- package/src/dashboard/templates/cortex 2.html +321 -0
- package/src/dashboard/templates/credentials 2.html +128 -0
- package/src/dashboard/templates/crons 2.html +370 -0
- package/src/dashboard/templates/dashboard 2.html +494 -0
- package/src/dashboard/templates/dreams 2.html +252 -0
- package/src/dashboard/templates/email 2.html +160 -0
- package/src/dashboard/templates/evolution 2.html +189 -0
- package/src/dashboard/templates/feed 2.html +249 -0
- package/src/dashboard/templates/followup_health 2.html +170 -0
- package/src/dashboard/templates/graph 2.html +201 -0
- package/src/dashboard/templates/guard 2.html +259 -0
- package/src/dashboard/templates/inbox 2.html +251 -0
- package/src/dashboard/templates/memory 2.html +420 -0
- package/src/dashboard/templates/operations 2.html +608 -0
- package/src/dashboard/templates/plugins 2.html +185 -0
- package/src/dashboard/templates/protocol 2.html +199 -0
- package/src/dashboard/templates/rules 2.html +246 -0
- package/src/dashboard/templates/sentiment 2.html +247 -0
- package/src/dashboard/templates/sessions 2.html +218 -0
- package/src/dashboard/templates/skills 2.html +329 -0
- package/src/dashboard/templates/somatic 2.html +73 -0
- package/src/dashboard/templates/triggers 2.html +133 -0
- package/src/dashboard/templates/trust 2.html +360 -0
- package/src/db/__init__ 2.py +259 -0
- package/src/db/_core 2.py +437 -0
- package/src/db/_credentials 2.py +124 -0
- package/src/db/_episodic 2.py +762 -0
- package/src/db/_evolution 2.py +54 -0
- package/src/db/_fts 2.py +406 -0
- package/src/db/_goal_profiles 2.py +376 -0
- package/src/db/_hot_context 2.py +660 -0
- package/src/db/_outcomes 2.py +800 -0
- package/src/db/_personal_scripts 2.py +582 -0
- package/src/db/_sessions 2.py +330 -0
- package/src/db/_tasks 2.py +91 -0
- package/src/db/_watchers 2.py +173 -0
- package/src/doctor/formatters 2.py +52 -0
- package/src/doctor/models 2.py +69 -0
- package/src/doctor/planes 2.py +87 -0
- package/src/doctor/providers/__init__ 2.py +1 -0
- package/src/doctor/providers/deep 2.py +367 -0
- package/src/evolution_cycle 2.py +519 -0
- package/src/hooks/auto_capture 2.py +208 -0
- package/src/hooks/caffeinate-guard 2.sh +8 -0
- package/src/hooks/capture-session 2.sh +21 -0
- package/src/hooks/capture-tool-logs 2.sh +158 -0
- package/src/hooks/daily-briefing-check 2.sh +33 -0
- package/src/hooks/heartbeat-enforcement 2.py +90 -0
- package/src/hooks/heartbeat-posttool 2.sh +18 -0
- package/src/hooks/inbox-hook 2.sh +76 -0
- package/src/hooks/post-compact 2.sh +152 -0
- package/src/hooks/pre-compact 2.sh +169 -0
- package/src/hooks/protocol-guardrail 2.sh +10 -0
- package/src/hooks/protocol-pretool-guardrail 2.sh +9 -0
- package/src/hooks/session-stop 2.sh +52 -0
- package/src/kg_populate 2.py +292 -0
- package/src/maintenance 2.py +53 -0
- package/src/memory_backends 2.py +71 -0
- package/src/migrate_embeddings 2.py +124 -0
- package/src/nexo_sdk 2.py +103 -0
- package/src/observability 2.py +199 -0
- package/src/plugin_loader 2.py +217 -0
- package/src/plugins/__init__ 2.py +0 -0
- package/src/plugins/artifact_registry 2.py +450 -0
- package/src/plugins/backup 2.py +127 -0
- package/src/plugins/claims_tools 2.py +119 -0
- package/src/plugins/cognitive_memory 2.py +609 -0
- package/src/plugins/core_rules 2.py +252 -0
- package/src/plugins/cortex 2.py +1155 -0
- package/src/plugins/entities 2.py +67 -0
- package/src/plugins/episodic_memory 2.py +560 -0
- package/src/plugins/evolution 2.py +167 -0
- package/src/plugins/goal_engine 2.py +142 -0
- package/src/plugins/guard 2.py +862 -0
- package/src/plugins/impact 2.py +29 -0
- package/src/plugins/knowledge_graph_tools 2.py +137 -0
- package/src/plugins/media_memory_tools 2.py +98 -0
- package/src/plugins/memory_export 2.py +196 -0
- package/src/plugins/outcomes 2.py +130 -0
- package/src/plugins/personal_scripts 2.py +117 -0
- package/src/plugins/preferences 2.py +47 -0
- package/src/plugins/protocol 2.py +1449 -0
- package/src/plugins/simple_api 2.py +106 -0
- package/src/plugins/skills 2.py +341 -0
- package/src/plugins/state_watchers 2.py +79 -0
- package/src/plugins/update 2.py +986 -0
- package/src/plugins/user_state_tools 2.py +43 -0
- package/src/plugins/workflow 2.py +588 -0
- package/src/protocol_settings 2.py +59 -0
- package/src/public_contribution 2.py +466 -0
- package/src/public_evolution_queue 2.py +241 -0
- package/src/requirements 2.txt +14 -0
- package/src/retroactive_learnings 2.py +373 -0
- package/src/rules/__init__ 2.py +0 -0
- package/src/rules/core-rules 2.json +331 -0
- package/src/rules/migrate 2.py +207 -0
- package/src/runtime_power 2.py +874 -0
- package/src/script_registry 2.py +1559 -0
- package/src/scripts/check-context 2.py +272 -0
- package/src/scripts/deep-sleep/apply_findings 2.py +2327 -0
- package/src/scripts/deep-sleep/collect 2.py +928 -0
- package/src/scripts/deep-sleep/extract 2.py +330 -0
- package/src/scripts/deep-sleep/extract-prompt 2.md +285 -0
- package/src/scripts/deep-sleep/synthesize 2.py +312 -0
- package/src/scripts/deep-sleep/synthesize-prompt 2.md +336 -0
- package/src/scripts/nexo-agent-run 2.py +75 -0
- package/src/scripts/nexo-auto-update 2.py +6 -0
- package/src/scripts/nexo-backup 2.sh +25 -0
- package/src/scripts/nexo-brain-activation 2.sh +140 -0
- package/src/scripts/nexo-catchup 2.py +300 -0
- package/src/scripts/nexo-cognitive-decay 2.py +257 -0
- package/src/scripts/nexo-cortex-cycle 2.py +293 -0
- package/src/scripts/nexo-cron-wrapper 2.sh +53 -0
- package/src/scripts/nexo-daily-self-audit 2.py +2161 -0
- package/src/scripts/nexo-dashboard 2.sh +29 -0
- package/src/scripts/nexo-deep-sleep 2.sh +86 -0
- package/src/scripts/nexo-evolution-run 2.py +1664 -0
- package/src/scripts/nexo-followup-hygiene 2.py +139 -0
- package/src/scripts/nexo-hook-record 2.py +42 -0
- package/src/scripts/nexo-immune 2.py +936 -0
- package/src/scripts/nexo-impact-scorer 2.py +117 -0
- package/src/scripts/nexo-inbox-hook 2.sh +74 -0
- package/src/scripts/nexo-install 2.py +6 -0
- package/src/scripts/nexo-learning-housekeep 2.py +401 -0
- package/src/scripts/nexo-learning-validator 2.py +266 -0
- package/src/scripts/nexo-migrate 2.py +260 -0
- package/src/scripts/nexo-outcome-checker 2.py +127 -0
- package/src/scripts/nexo-postmortem-consolidator 2.py +456 -0
- package/src/scripts/nexo-pre-commit 2.py +120 -0
- package/src/scripts/nexo-prevent-sleep 2.sh +35 -0
- package/src/scripts/nexo-proactive-dashboard 2.py +354 -0
- package/src/scripts/nexo-reflection 2.py +256 -0
- package/src/scripts/nexo-runtime-preflight 2.py +274 -0
- package/src/scripts/nexo-sleep 2.py +631 -0
- package/src/scripts/nexo-snapshot-restore 2.sh +35 -0
- package/src/scripts/nexo-sync-clients 2.py +16 -0
- package/src/scripts/nexo-synthesis 2.py +475 -0
- package/src/scripts/nexo-tcc-approve 2.sh +79 -0
- package/src/scripts/nexo-update 2.sh +306 -0
- package/src/scripts/nexo-watchdog 2.sh +1207 -0
- package/src/scripts/nexo-watchdog-smoke 2.py +119 -0
- package/src/scripts/rehydrate_learnings_from_archive 2.py +245 -0
- package/src/server 2.py +1296 -0
- package/src/skills/run-nexo-audit-phase/guide 2.md +43 -0
- package/src/skills/run-nexo-audit-phase/skill 2.json +59 -0
- package/src/skills/run-nexo-core-fix-cycle/guide 2.md +17 -0
- package/src/skills/run-nexo-core-fix-cycle/script 2.py +276 -0
- package/src/skills/run-nexo-core-fix-cycle/skill 2.json +58 -0
- package/src/skills/run-release-final-audit/guide 2.md +16 -0
- package/src/skills/run-release-final-audit/script 2.py +259 -0
- package/src/skills/run-release-final-audit/skill 2.json +77 -0
- package/src/skills/run-runtime-doctor/guide 2.md +12 -0
- package/src/skills/run-runtime-doctor/script 2.py +21 -0
- package/src/skills/run-runtime-doctor/skill 2.json +25 -0
- package/src/skills_runtime 2.py +932 -0
- package/src/state_watchers_runtime 2.py +475 -0
- package/src/storage_router 2.py +32 -0
- package/src/system_catalog 2.py +786 -0
- package/src/tools_coordination 2.py +103 -0
- package/src/tools_credentials 2.py +68 -0
- package/src/tools_drive 2.py +487 -0
- package/src/tools_hot_context 2.py +163 -0
- package/src/tools_learnings 2.py +612 -0
- package/src/tools_menu 2.py +229 -0
- package/src/tools_reminders 2.py +88 -0
- package/src/tools_reminders_crud 2.py +363 -0
- package/src/tools_sessions 2.py +1054 -0
- package/src/tools_system_catalog 2.py +19 -0
- package/src/tools_task_history 2.py +57 -0
- package/src/tools_transcripts 2.py +98 -0
- package/src/transcript_utils 2.py +412 -0
- package/src/user_context 2.py +46 -0
- package/src/user_data_portability 2.py +328 -0
- package/src/user_state_model 2.py +170 -0
- package/templates/CLAUDE.md 2.template +108 -0
- package/templates/CODEX.AGENTS.md 2.template +66 -0
- package/templates/launchagents/README 2.md +132 -0
- package/templates/launchagents/com.nexo.auto-close-sessions 2.plist +39 -0
- package/templates/launchagents/com.nexo.catchup 2.plist +39 -0
- package/templates/launchagents/com.nexo.cognitive-decay 2.plist +40 -0
- package/templates/launchagents/com.nexo.dashboard 2.plist +43 -0
- package/templates/launchagents/com.nexo.deep-sleep 2.plist +43 -0
- package/templates/launchagents/com.nexo.evolution 2.plist +44 -0
- package/templates/launchagents/com.nexo.followup-hygiene 2.plist +45 -0
- package/templates/launchagents/com.nexo.immune 2.plist +41 -0
- package/templates/launchagents/com.nexo.postmortem 2.plist +45 -0
- package/templates/launchagents/com.nexo.self-audit 2.plist +47 -0
- package/templates/launchagents/com.nexo.synthesis 2.plist +45 -0
- package/templates/launchagents/com.nexo.watchdog 2.plist +37 -0
- package/templates/nexo_helper 2.py +301 -0
- package/templates/openclaw 2.json +13 -0
- package/templates/plugin-template 2.py +40 -0
- package/templates/script-template 2.py +59 -0
- package/templates/script-template 2.sh +13 -0
- package/templates/skill-script-template 2.py +48 -0
- package/templates/skill-template 2.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 %}
|
|
@@ -0,0 +1,73 @@
|
|
|
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, '<');
|
|
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 %}
|
|
@@ -0,0 +1,133 @@
|
|
|
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 %}
|