secubox-openwrt/package/secubox/luci-app-ollama/htdocs/luci-static/resources/view/ollama/dashboard.js
CyberMind-FR b0cf6e2240 fix(ollama): Show start prompt when Ollama is not running
Instead of showing clickable model suggestions when Ollama is stopped,
display a helpful message prompting the user to start Ollama first.

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-02-02 07:15:38 +01:00

406 lines
17 KiB
JavaScript

'use strict';
'require view';
'require dom';
'require poll';
'require rpc';
var api = {
status: rpc.declare({ object: 'luci.ollama', method: 'status' }),
models: rpc.declare({ object: 'luci.ollama', method: 'models' }),
health: rpc.declare({ object: 'luci.ollama', method: 'health' }),
start: rpc.declare({ object: 'luci.ollama', method: 'start' }),
stop: rpc.declare({ object: 'luci.ollama', method: 'stop' }),
restart: rpc.declare({ object: 'luci.ollama', method: 'restart' }),
pull: rpc.declare({ object: 'luci.ollama', method: 'model_pull', params: ['name'] }),
remove: rpc.declare({ object: 'luci.ollama', method: 'model_remove', params: ['name'] }),
chat: rpc.declare({ object: 'luci.ollama', method: 'chat', params: ['model', 'message'] })
};
function fmtBytes(b) {
if (!b) return '-';
var u = ['B', 'KB', 'MB', 'GB'];
var i = 0;
while (b >= 1024 && i < u.length - 1) { b /= 1024; i++; }
return b.toFixed(1) + ' ' + u[i];
}
function fmtUptime(s) {
if (!s) return '-';
var h = Math.floor(s / 3600);
var m = Math.floor((s % 3600) / 60);
return h > 0 ? h + 'h ' + m + 'm' : m + 'm';
}
return view.extend({
css: `
:root { --ol-bg: #0f172a; --ol-card: #1e293b; --ol-border: #334155; --ol-text: #f1f5f9; --ol-muted: #94a3b8; --ol-accent: #f97316; --ol-success: #22c55e; --ol-danger: #ef4444; }
.ol-wrap { font-family: system-ui, sans-serif; background: var(--ol-bg); color: var(--ol-text); min-height: 100vh; padding: 1rem; }
.ol-header { display: flex; align-items: center; justify-content: space-between; margin-bottom: 1.5rem; padding-bottom: 1rem; border-bottom: 1px solid var(--ol-border); }
.ol-title { font-size: 1.5rem; font-weight: 700; display: flex; align-items: center; gap: 0.5rem; }
.ol-title span { font-size: 1.75rem; }
.ol-badge { padding: 0.25rem 0.75rem; border-radius: 1rem; font-size: 0.75rem; font-weight: 600; }
.ol-badge.on { background: rgba(34,197,94,0.2); color: var(--ol-success); }
.ol-badge.off { background: rgba(239,68,68,0.2); color: var(--ol-danger); }
.ol-stats { display: grid; grid-template-columns: repeat(auto-fit, minmax(140px, 1fr)); gap: 1rem; margin-bottom: 1.5rem; }
.ol-stat { background: var(--ol-card); border: 1px solid var(--ol-border); border-radius: 0.5rem; padding: 1rem; text-align: center; }
.ol-stat-val { font-size: 1.5rem; font-weight: 700; color: var(--ol-accent); }
.ol-stat-lbl { font-size: 0.7rem; color: var(--ol-muted); text-transform: uppercase; margin-top: 0.25rem; }
.ol-grid { display: grid; grid-template-columns: repeat(auto-fit, minmax(350px, 1fr)); gap: 1rem; }
.ol-card { background: var(--ol-card); border: 1px solid var(--ol-border); border-radius: 0.5rem; overflow: hidden; }
.ol-card-head { padding: 0.75rem 1rem; background: rgba(0,0,0,0.2); border-bottom: 1px solid var(--ol-border); font-weight: 600; display: flex; justify-content: space-between; align-items: center; }
.ol-card-body { padding: 1rem; }
.ol-btn { padding: 0.5rem 1rem; border: none; border-radius: 0.375rem; font-size: 0.8rem; font-weight: 500; cursor: pointer; transition: opacity 0.2s; }
.ol-btn:hover { opacity: 0.8; }
.ol-btn:disabled { opacity: 0.4; cursor: not-allowed; }
.ol-btn-primary { background: var(--ol-accent); color: #fff; }
.ol-btn-success { background: var(--ol-success); color: #fff; }
.ol-btn-danger { background: var(--ol-danger); color: #fff; }
.ol-btn-sm { padding: 0.25rem 0.5rem; font-size: 0.7rem; }
.ol-btns { display: flex; gap: 0.5rem; flex-wrap: wrap; }
.ol-input { width: 100%; padding: 0.5rem; border: 1px solid var(--ol-border); border-radius: 0.375rem; background: var(--ol-bg); color: var(--ol-text); font-size: 0.875rem; }
.ol-input:focus { outline: none; border-color: var(--ol-accent); }
.ol-row { display: flex; justify-content: space-between; padding: 0.5rem 0; border-bottom: 1px solid var(--ol-border); font-size: 0.875rem; }
.ol-row:last-child { border-bottom: none; }
.ol-row-lbl { color: var(--ol-muted); }
.ol-model { display: flex; align-items: center; justify-content: space-between; padding: 0.75rem; background: var(--ol-bg); border-radius: 0.375rem; margin-bottom: 0.5rem; }
.ol-model:last-child { margin-bottom: 0; }
.ol-model-name { font-weight: 500; }
.ol-model-size { font-size: 0.75rem; color: var(--ol-muted); }
.ol-empty { text-align: center; padding: 2rem; color: var(--ol-muted); }
.ol-suggest { margin-top: 1rem; }
.ol-suggest-title { font-size: 0.85rem; margin-bottom: 0.75rem; color: var(--ol-text); }
.ol-suggest-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(200px, 1fr)); gap: 0.5rem; }
.ol-suggest-item { background: var(--ol-bg); border: 1px solid var(--ol-border); border-radius: 0.375rem; padding: 0.75rem; cursor: pointer; transition: all 0.2s; }
.ol-suggest-item:hover { border-color: var(--ol-accent); }
.ol-suggest-name { font-weight: 600; font-size: 0.9rem; }
.ol-suggest-desc { font-size: 0.75rem; color: var(--ol-muted); margin-top: 0.25rem; }
.ol-suggest-size { font-size: 0.7rem; color: var(--ol-accent); margin-top: 0.25rem; }
.ol-chat-box { height: 200px; overflow-y: auto; background: var(--ol-bg); border: 1px solid var(--ol-border); border-radius: 0.375rem; padding: 0.75rem; margin-bottom: 0.75rem; font-size: 0.875rem; }
.ol-chat-msg { margin-bottom: 0.75rem; }
.ol-chat-msg.user { color: var(--ol-accent); }
.ol-chat-msg.ai { color: var(--ol-text); white-space: pre-wrap; }
.ol-chat-input { display: flex; gap: 0.5rem; }
.ol-chat-input input { flex: 1; }
.ol-select { padding: 0.5rem; border: 1px solid var(--ol-border); border-radius: 0.375rem; background: var(--ol-bg); color: var(--ol-text); font-size: 0.875rem; margin-bottom: 0.75rem; }
.ol-toast { position: fixed; bottom: 1rem; right: 1rem; padding: 0.75rem 1rem; border-radius: 0.375rem; font-size: 0.875rem; z-index: 9999; }
.ol-toast.success { background: var(--ol-success); color: #fff; }
.ol-toast.error { background: var(--ol-danger); color: #fff; }
`,
load: function() {
return Promise.all([
api.status().catch(function() { return {}; }),
api.models().catch(function() { return { models: [] }; })
]);
},
render: function(data) {
var self = this;
var status = data[0] || {};
var models = (data[1] && data[1].models) || [];
this.isRunning = status.running;
var view = E('div', { 'class': 'ol-wrap' }, [
E('style', {}, this.css),
// Header
E('div', { 'class': 'ol-header' }, [
E('div', { 'class': 'ol-title' }, [
E('span', {}, '\uD83E\uDD99'),
'Ollama'
]),
E('div', { 'class': 'ol-badge ' + (status.running ? 'on' : 'off'), 'id': 'ol-status' },
status.running ? 'Running' : 'Stopped')
]),
// Stats
E('div', { 'class': 'ol-stats', 'id': 'ol-stats' }, this.renderStats(status, models)),
// Cards Grid
E('div', { 'class': 'ol-grid' }, [
// Service Control
E('div', { 'class': 'ol-card' }, [
E('div', { 'class': 'ol-card-head' }, 'Service Control'),
E('div', { 'class': 'ol-card-body' }, [
E('div', {}, [
E('div', { 'class': 'ol-row' }, [
E('span', { 'class': 'ol-row-lbl' }, 'Status'),
E('span', { 'id': 'svc-status' }, status.running ? 'Running' : 'Stopped')
]),
E('div', { 'class': 'ol-row' }, [
E('span', { 'class': 'ol-row-lbl' }, 'Runtime'),
E('span', {}, status.runtime || 'none')
]),
E('div', { 'class': 'ol-row' }, [
E('span', { 'class': 'ol-row-lbl' }, 'Memory'),
E('span', {}, status.memory_limit || '2g')
]),
E('div', { 'class': 'ol-row' }, [
E('span', { 'class': 'ol-row-lbl' }, 'Data Path'),
E('span', {}, status.data_path || '/srv/ollama')
])
]),
E('div', { 'class': 'ol-btns', 'style': 'margin-top: 1rem;' }, [
E('button', { 'class': 'ol-btn ol-btn-success', 'click': function() { self.svcAction('start'); } }, 'Start'),
E('button', { 'class': 'ol-btn ol-btn-danger', 'click': function() { self.svcAction('stop'); } }, 'Stop'),
E('button', { 'class': 'ol-btn ol-btn-primary', 'click': function() { self.svcAction('restart'); } }, 'Restart')
])
])
]),
// Models
E('div', { 'class': 'ol-card' }, [
E('div', { 'class': 'ol-card-head' }, [
'Models',
E('span', { 'id': 'model-count' }, models.length + ' installed')
]),
E('div', { 'class': 'ol-card-body' }, [
E('div', { 'id': 'ol-models' }, this.renderModels(models)),
E('div', { 'style': 'margin-top: 1rem; display: flex; gap: 0.5rem;' }, [
E('input', { 'type': 'text', 'class': 'ol-input', 'id': 'pull-model', 'placeholder': 'Model name (e.g. tinyllama, llama2)' }),
E('button', { 'class': 'ol-btn ol-btn-primary', 'click': function() { self.pullModel(); } }, 'Pull')
])
])
]),
// Chat
E('div', { 'class': 'ol-card' }, [
E('div', { 'class': 'ol-card-head' }, 'Chat'),
E('div', { 'class': 'ol-card-body' }, [
E('select', { 'class': 'ol-select', 'id': 'chat-model', 'style': 'width: 100%;' },
models.length === 0
? [E('option', {}, '-- No models --')]
: models.map(function(m) { return E('option', { 'value': m.name }, m.name); })
),
E('div', { 'class': 'ol-chat-box', 'id': 'chat-box' }),
E('div', { 'class': 'ol-chat-input' }, [
E('input', { 'type': 'text', 'class': 'ol-input', 'id': 'chat-input', 'placeholder': 'Type a message...',
'keypress': function(e) { if (e.key === 'Enter') self.sendChat(); }
}),
E('button', { 'class': 'ol-btn ol-btn-primary', 'id': 'chat-send', 'click': function() { self.sendChat(); } }, 'Send')
])
])
])
])
]);
poll.add(L.bind(this.refresh, this), 15);
return view;
},
renderStats: function(status, models) {
return [
E('div', { 'class': 'ol-stat' }, [
E('div', { 'class': 'ol-stat-val' }, models.length.toString()),
E('div', { 'class': 'ol-stat-lbl' }, 'Models')
]),
E('div', { 'class': 'ol-stat' }, [
E('div', { 'class': 'ol-stat-val' }, status.running ? fmtUptime(status.uptime) : '-'),
E('div', { 'class': 'ol-stat-lbl' }, 'Uptime')
]),
E('div', { 'class': 'ol-stat' }, [
E('div', { 'class': 'ol-stat-val' }, (status.api_port || 11434).toString()),
E('div', { 'class': 'ol-stat-lbl' }, 'API Port')
]),
E('div', { 'class': 'ol-stat' }, [
E('div', { 'class': 'ol-stat-val' }, status.runtime || '-'),
E('div', { 'class': 'ol-stat-lbl' }, 'Runtime')
])
];
},
suggestedModels: [
{ name: 'tinyllama', desc: 'Tiny but capable, fast inference', size: '637 MB' },
{ name: 'llama3.2:1b', desc: 'Meta Llama 3.2 1B - lightweight', size: '1.3 GB' },
{ name: 'llama3.2:3b', desc: 'Meta Llama 3.2 3B - balanced', size: '2.0 GB' },
{ name: 'phi3:mini', desc: 'Microsoft Phi-3 Mini - efficient', size: '2.2 GB' },
{ name: 'gemma2:2b', desc: 'Google Gemma 2 2B - compact', size: '1.6 GB' },
{ name: 'qwen2.5:1.5b', desc: 'Alibaba Qwen 2.5 - multilingual', size: '986 MB' },
{ name: 'mistral', desc: 'Mistral 7B - high quality', size: '4.1 GB' },
{ name: 'codellama:7b', desc: 'Meta CodeLlama - coding tasks', size: '3.8 GB' }
],
renderModels: function(models) {
var self = this;
if (!models || models.length === 0) {
// If Ollama isn't running, show start prompt instead of suggestions
if (!this.isRunning) {
return E('div', {}, [
E('div', { 'class': 'ol-empty' }, [
E('div', { 'style': 'font-size: 2rem; margin-bottom: 0.5rem;' }, '\u26A0\uFE0F'),
E('div', {}, 'Ollama is not running'),
E('div', { 'style': 'margin-top: 0.5rem; font-size: 0.85rem;' }, 'Click "Start" above to launch Ollama')
])
]);
}
return E('div', {}, [
E('div', { 'class': 'ol-empty' }, 'No models installed'),
E('div', { 'class': 'ol-suggest' }, [
E('div', { 'class': 'ol-suggest-title' }, '\uD83D\uDCE5 Click to download a model:'),
E('div', { 'class': 'ol-suggest-grid' }, this.suggestedModels.map(function(m) {
return E('div', {
'class': 'ol-suggest-item',
'click': function() { self.pullModel(m.name); }
}, [
E('div', { 'class': 'ol-suggest-name' }, m.name),
E('div', { 'class': 'ol-suggest-desc' }, m.desc),
E('div', { 'class': 'ol-suggest-size' }, m.size)
]);
}))
])
]);
}
return E('div', {}, models.map(function(m) {
return E('div', { 'class': 'ol-model' }, [
E('div', {}, [
E('div', { 'class': 'ol-model-name' }, m.name),
E('div', { 'class': 'ol-model-size' }, fmtBytes(m.size))
]),
E('button', { 'class': 'ol-btn ol-btn-danger ol-btn-sm', 'click': function() { self.removeModel(m.name); } }, 'Remove')
]);
}));
},
refresh: function() {
var self = this;
return Promise.all([
api.status().catch(function() { return {}; }),
api.models().catch(function() { return { models: [] }; })
]).then(function(data) {
var status = data[0] || {};
var models = (data[1] && data[1].models) || [];
self.isRunning = status.running;
var badge = document.getElementById('ol-status');
if (badge) {
badge.className = 'ol-badge ' + (status.running ? 'on' : 'off');
badge.textContent = status.running ? 'Running' : 'Stopped';
}
var statsEl = document.getElementById('ol-stats');
if (statsEl) dom.content(statsEl, self.renderStats(status, models));
var modelsEl = document.getElementById('ol-models');
if (modelsEl) dom.content(modelsEl, self.renderModels(models));
var countEl = document.getElementById('model-count');
if (countEl) countEl.textContent = models.length + ' installed';
var svcEl = document.getElementById('svc-status');
if (svcEl) svcEl.textContent = status.running ? 'Running' : 'Stopped';
// Update chat model select
var sel = document.getElementById('chat-model');
if (sel && models.length > 0) {
var current = sel.value;
sel.innerHTML = '';
models.forEach(function(m) {
var opt = document.createElement('option');
opt.value = m.name;
opt.textContent = m.name;
if (m.name === current) opt.selected = true;
sel.appendChild(opt);
});
}
});
},
svcAction: function(action) {
var self = this;
var fn = action === 'start' ? api.start : action === 'stop' ? api.stop : api.restart;
fn().then(function(r) {
self.toast(r && r.success ? action + ' OK' : 'Failed: ' + ((r && r.error) || 'Unknown'), r && r.success);
if (r && r.success) setTimeout(function() { self.refresh(); }, 2000);
}).catch(function(e) { self.toast('Error: ' + e.message, false); });
},
pullModel: function(modelName) {
var self = this;
var input = document.getElementById('pull-model');
var name = modelName || (input ? input.value.trim() : '');
if (!name) { self.toast('Enter model name', false); return; }
self.toast('Pulling ' + name + '... (this may take a while)', true);
api.pull(name).then(function(r) {
self.toast(r && r.success ? 'Pulled ' + name : 'Failed: ' + ((r && r.error) || 'Unknown'), r && r.success);
if (r && r.success) {
if (input) input.value = '';
self.refresh();
}
}).catch(function(e) { self.toast('Error: ' + e.message, false); });
},
removeModel: function(name) {
var self = this;
if (!confirm('Remove model "' + name + '"?')) return;
api.remove(name).then(function(r) {
self.toast(r && r.success ? 'Removed ' + name : 'Failed: ' + ((r && r.error) || 'Unknown'), r && r.success);
if (r && r.success) self.refresh();
}).catch(function(e) { self.toast('Error: ' + e.message, false); });
},
sendChat: function() {
var self = this;
var modelEl = document.getElementById('chat-model');
var inputEl = document.getElementById('chat-input');
var boxEl = document.getElementById('chat-box');
var btnEl = document.getElementById('chat-send');
var model = modelEl ? modelEl.value : '';
var msg = inputEl ? inputEl.value.trim() : '';
if (!model || model === '-- No models --') { self.toast('Select a model', false); return; }
if (!msg) return;
// Add user message
var userDiv = document.createElement('div');
userDiv.className = 'ol-chat-msg user';
userDiv.textContent = '> ' + msg;
boxEl.appendChild(userDiv);
inputEl.value = '';
inputEl.disabled = true;
btnEl.disabled = true;
boxEl.scrollTop = boxEl.scrollHeight;
api.chat(model, msg).then(function(r) {
var aiDiv = document.createElement('div');
aiDiv.className = 'ol-chat-msg ai';
if (r && r.response) {
aiDiv.textContent = r.response;
} else if (r && r.error) {
aiDiv.textContent = 'Error: ' + r.error;
aiDiv.style.color = '#ef4444';
} else {
aiDiv.textContent = 'No response';
}
boxEl.appendChild(aiDiv);
boxEl.scrollTop = boxEl.scrollHeight;
}).catch(function(e) {
var errDiv = document.createElement('div');
errDiv.className = 'ol-chat-msg ai';
errDiv.textContent = 'Error: ' + e.message;
errDiv.style.color = '#ef4444';
boxEl.appendChild(errDiv);
}).finally(function() {
inputEl.disabled = false;
btnEl.disabled = false;
inputEl.focus();
});
},
toast: function(msg, success) {
var t = document.querySelector('.ol-toast');
if (t) t.remove();
t = document.createElement('div');
t.className = 'ol-toast ' + (success ? 'success' : 'error');
t.textContent = msg;
document.body.appendChild(t);
setTimeout(function() { t.remove(); }, 4000);
},
handleSaveApply: null,
handleSave: null,
handleReset: null
});