secubox-openwrt/package/secubox/luci-app-gitea/htdocs/luci-static/resources/view/gitea/overview.js
CyberMind-FR d43a02a397 feat(gitea): Add self-hosted Git platform for OpenWrt
Add secubox-app-gitea and luci-app-gitea packages:

secubox-app-gitea:
- LXC container with Alpine 3.21 rootfs
- Gitea 1.22.6 binary (auto-detect amd64/arm64/armv7)
- HTTP (3000) and SSH (2222) ports
- SQLite database (embedded)
- giteactl: install/uninstall/update/backup/restore

luci-app-gitea:
- Cyberpunk themed dashboard
- Repository browser with clone URLs
- User management interface
- Server and security settings
- Backup/restore functionality
- 18 RPCD methods

Resource requirements: 256MB RAM minimum, ~100MB storage

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-24 10:03:31 +01:00

489 lines
14 KiB
JavaScript

'use strict';
'require view';
'require ui';
'require dom';
'require poll';
'require gitea.api as api';
return view.extend({
statusData: null,
statsData: null,
logsData: null,
load: function() {
return this.refreshData();
},
refreshData: function() {
var self = this;
return api.getDashboardData().then(function(data) {
self.statusData = data.status || {};
self.statsData = data.stats || {};
self.logsData = data.logs || [];
return data;
});
},
render: function() {
var self = this;
// Inject CSS
var cssLink = E('link', {
'rel': 'stylesheet',
'type': 'text/css',
'href': L.resource('gitea/dashboard.css')
});
var container = E('div', { 'class': 'gitea-dashboard' }, [
cssLink,
this.renderHeader(),
this.renderStatsGrid(),
this.renderMainGrid()
]);
// Poll for updates
poll.add(function() {
return self.refreshData().then(function() {
self.updateDynamicContent();
});
}, 10);
return container;
},
renderHeader: function() {
var status = this.statusData;
var statusClass = !status.installed ? 'not-installed' : (status.running ? 'running' : 'stopped');
var statusText = !status.installed ? _('Not Installed') : (status.running ? _('Running') : _('Stopped'));
return E('div', { 'class': 'gt-header' }, [
E('div', { 'class': 'gt-header-content' }, [
E('div', { 'class': 'gt-logo' }, '\uD83D\uDCE6'),
E('div', {}, [
E('h1', { 'class': 'gt-title' }, _('GITEA PLATFORM')),
E('p', { 'class': 'gt-subtitle' }, _('Self-Hosted Git Service for SecuBox'))
]),
E('div', { 'class': 'gt-status-badge ' + statusClass, 'id': 'gt-status-badge' }, [
E('span', {}, statusClass === 'running' ? '\u25CF' : '\u25CB'),
' ' + statusText
])
])
]);
},
renderStatsGrid: function() {
var status = this.statusData;
var stats = this.statsData;
var statItems = [
{
icon: '\uD83D\uDD0C',
label: _('Status'),
value: status.running ? _('Online') : _('Offline'),
id: 'stat-status',
cardClass: status.running ? 'success' : 'error'
},
{
icon: '\uD83D\uDCE6',
label: _('Repositories'),
value: stats.repo_count || status.repo_count || 0,
id: 'stat-repos'
},
{
icon: '\uD83D\uDC65',
label: _('Users'),
value: stats.user_count || 0,
id: 'stat-users'
},
{
icon: '\uD83D\uDCBE',
label: _('Disk Usage'),
value: stats.disk_usage || status.disk_usage || '0',
id: 'stat-disk'
},
{
icon: '\uD83C\uDF10',
label: _('HTTP Port'),
value: status.http_port || '3000',
id: 'stat-http'
},
{
icon: '\uD83D\uDD12',
label: _('SSH Port'),
value: status.ssh_port || '2222',
id: 'stat-ssh'
}
];
return E('div', { 'class': 'gt-stats-grid' },
statItems.map(function(stat) {
return E('div', { 'class': 'gt-stat-card ' + (stat.cardClass || '') }, [
E('div', { 'class': 'gt-stat-icon' }, stat.icon),
E('div', { 'class': 'gt-stat-content' }, [
E('div', { 'class': 'gt-stat-value', 'id': stat.id }, String(stat.value)),
E('div', { 'class': 'gt-stat-label' }, stat.label)
])
]);
})
);
},
renderMainGrid: function() {
return E('div', { 'class': 'gt-main-grid' }, [
this.renderControlCard(),
this.renderInfoCard(),
this.renderLogsCard()
]);
},
renderControlCard: function() {
var self = this;
var status = this.statusData;
var buttons = [];
if (!status.installed) {
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-primary',
'id': 'btn-install',
'click': function() { self.handleInstall(); }
}, [E('span', {}, '\uD83D\uDCE5'), ' ' + _('Install')])
);
} else {
if (status.running) {
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-danger',
'id': 'btn-stop',
'click': function() { self.handleStop(); }
}, [E('span', {}, '\u23F9'), ' ' + _('Stop')])
);
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-warning',
'id': 'btn-restart',
'click': function() { self.handleRestart(); }
}, [E('span', {}, '\uD83D\uDD04'), ' ' + _('Restart')])
);
} else {
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-success',
'id': 'btn-start',
'click': function() { self.handleStart(); }
}, [E('span', {}, '\u25B6'), ' ' + _('Start')])
);
}
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-primary',
'id': 'btn-update',
'click': function() { self.handleUpdate(); }
}, [E('span', {}, '\u2B06'), ' ' + _('Update')])
);
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-primary',
'id': 'btn-backup',
'click': function() { self.handleBackup(); }
}, [E('span', {}, '\uD83D\uDCBE'), ' ' + _('Backup')])
);
buttons.push(
E('button', {
'class': 'gt-btn gt-btn-danger',
'id': 'btn-uninstall',
'click': function() { self.handleUninstall(); }
}, [E('span', {}, '\uD83D\uDDD1'), ' ' + _('Uninstall')])
);
}
return E('div', { 'class': 'gt-card' }, [
E('div', { 'class': 'gt-card-header' }, [
E('div', { 'class': 'gt-card-title' }, [
E('span', {}, '\uD83C\uDFAE'),
' ' + _('Controls')
])
]),
E('div', { 'class': 'gt-card-body' }, [
E('div', { 'class': 'gt-btn-group', 'id': 'gt-controls' }, buttons),
E('div', { 'class': 'gt-progress', 'id': 'gt-progress-container', 'style': 'display:none' }, [
E('div', { 'class': 'gt-progress-bar', 'id': 'gt-progress-bar', 'style': 'width:0%' })
]),
E('div', { 'class': 'gt-progress-text', 'id': 'gt-progress-text', 'style': 'display:none' })
])
]);
},
renderInfoCard: function() {
var status = this.statusData;
var infoItems = [
{ label: _('Container'), value: status.container_name || 'gitea' },
{ label: _('Version'), value: status.version || '1.22.x' },
{ label: _('Data Path'), value: status.data_path || '/srv/gitea' },
{ label: _('Memory Limit'), value: status.memory_limit || '512M' },
{ label: _('Web Interface'), value: status.http_url, isLink: true },
{ label: _('SSH Clone'), value: status.ssh_url }
];
return E('div', { 'class': 'gt-card' }, [
E('div', { 'class': 'gt-card-header' }, [
E('div', { 'class': 'gt-card-title' }, [
E('span', {}, '\u2139\uFE0F'),
' ' + _('Information')
])
]),
E('div', { 'class': 'gt-card-body' }, [
E('ul', { 'class': 'gt-info-list', 'id': 'gt-info-list' },
infoItems.map(function(item) {
var valueEl;
if (item.isLink && item.value) {
valueEl = E('a', { 'href': item.value, 'target': '_blank' }, item.value);
} else {
valueEl = item.value || '-';
}
return E('li', {}, [
E('span', { 'class': 'gt-info-label' }, item.label),
E('span', { 'class': 'gt-info-value' }, valueEl)
]);
})
)
])
]);
},
renderLogsCard: function() {
var logs = this.logsData || [];
return E('div', { 'class': 'gt-card gt-card-full' }, [
E('div', { 'class': 'gt-card-header' }, [
E('div', { 'class': 'gt-card-title' }, [
E('span', {}, '\uD83D\uDCDC'),
' ' + _('Recent Logs')
])
]),
E('div', { 'class': 'gt-card-body' }, [
logs.length > 0 ?
E('div', { 'class': 'gt-logs', 'id': 'gt-logs' },
logs.slice(-20).map(function(line) {
return E('div', { 'class': 'gt-logs-line' }, line);
})
) :
E('div', { 'class': 'gt-empty' }, [
E('div', { 'class': 'gt-empty-icon' }, '\uD83D\uDCED'),
E('div', {}, _('No logs available'))
])
])
]);
},
updateDynamicContent: function() {
var status = this.statusData;
var stats = this.statsData;
// Update status badge
var badge = document.getElementById('gt-status-badge');
if (badge) {
var statusClass = !status.installed ? 'not-installed' : (status.running ? 'running' : 'stopped');
var statusText = !status.installed ? _('Not Installed') : (status.running ? _('Running') : _('Stopped'));
badge.className = 'gt-status-badge ' + statusClass;
badge.innerHTML = '';
badge.appendChild(E('span', {}, statusClass === 'running' ? '\u25CF' : '\u25CB'));
badge.appendChild(document.createTextNode(' ' + statusText));
}
// Update stats
var statStatus = document.getElementById('stat-status');
if (statStatus) {
statStatus.textContent = status.running ? _('Online') : _('Offline');
}
var statRepos = document.getElementById('stat-repos');
if (statRepos) {
statRepos.textContent = stats.repo_count || status.repo_count || 0;
}
var statUsers = document.getElementById('stat-users');
if (statUsers) {
statUsers.textContent = stats.user_count || 0;
}
// Update logs
var logsContainer = document.getElementById('gt-logs');
if (logsContainer && this.logsData) {
logsContainer.innerHTML = '';
this.logsData.slice(-20).forEach(function(line) {
logsContainer.appendChild(E('div', { 'class': 'gt-logs-line' }, line));
});
}
},
handleInstall: function() {
var self = this;
var btn = document.getElementById('btn-install');
if (btn) btn.disabled = true;
ui.showModal(_('Installing Gitea Platform'), [
E('p', {}, _('This will download Alpine Linux rootfs and the Gitea binary. This may take several minutes.')),
E('div', { 'class': 'gt-progress' }, [
E('div', { 'class': 'gt-progress-bar', 'id': 'modal-progress', 'style': 'width:0%' })
]),
E('p', { 'id': 'modal-status' }, _('Starting installation...'))
]);
api.install().then(function(result) {
if (result && result.started) {
self.pollInstallProgress();
} else {
ui.hideModal();
ui.addNotification(null, E('p', {}, result.message || _('Installation failed')), 'error');
}
}).catch(function(err) {
ui.hideModal();
ui.addNotification(null, E('p', {}, _('Installation failed: ') + err.message), 'error');
});
},
pollInstallProgress: function() {
var self = this;
var checkProgress = function() {
api.getInstallProgress().then(function(result) {
var progressBar = document.getElementById('modal-progress');
var statusText = document.getElementById('modal-status');
if (progressBar) {
progressBar.style.width = (result.progress || 0) + '%';
}
if (statusText) {
statusText.textContent = result.message || '';
}
if (result.status === 'completed') {
ui.hideModal();
ui.addNotification(null, E('p', {}, _('Gitea Platform installed successfully!')), 'success');
self.refreshData();
location.reload();
} else if (result.status === 'error') {
ui.hideModal();
ui.addNotification(null, E('p', {}, _('Installation failed: ') + result.message), 'error');
} else if (result.status === 'running') {
setTimeout(checkProgress, 3000);
} else {
setTimeout(checkProgress, 3000);
}
}).catch(function() {
setTimeout(checkProgress, 5000);
});
};
setTimeout(checkProgress, 2000);
},
handleStart: function() {
var self = this;
api.start().then(function(result) {
if (result && result.success) {
ui.addNotification(null, E('p', {}, _('Gitea Platform started')), 'success');
} else {
ui.addNotification(null, E('p', {}, result.message || _('Failed to start')), 'error');
}
self.refreshData();
});
},
handleStop: function() {
var self = this;
api.stop().then(function(result) {
if (result && result.success) {
ui.addNotification(null, E('p', {}, _('Gitea Platform stopped')), 'info');
} else {
ui.addNotification(null, E('p', {}, result.message || _('Failed to stop')), 'error');
}
self.refreshData();
});
},
handleRestart: function() {
var self = this;
api.restart().then(function(result) {
if (result && result.success) {
ui.addNotification(null, E('p', {}, _('Gitea Platform restarted')), 'success');
} else {
ui.addNotification(null, E('p', {}, result.message || _('Failed to restart')), 'error');
}
self.refreshData();
});
},
handleUpdate: function() {
var self = this;
ui.showModal(_('Updating Gitea'), [
E('p', {}, _('Downloading and installing the latest Gitea binary...')),
E('div', { 'class': 'spinning' })
]);
api.update().then(function(result) {
ui.hideModal();
if (result && result.started) {
ui.addNotification(null, E('p', {}, _('Update started. The server will restart automatically.')), 'info');
} else {
ui.addNotification(null, E('p', {}, result.message || _('Update failed')), 'error');
}
self.refreshData();
});
},
handleBackup: function() {
var self = this;
ui.showModal(_('Creating Backup'), [
E('p', {}, _('Backing up repositories and database...')),
E('div', { 'class': 'spinning' })
]);
api.createBackup().then(function(result) {
ui.hideModal();
if (result && result.success) {
ui.addNotification(null, E('p', {}, _('Backup created: ') + (result.file || '')), 'success');
} else {
ui.addNotification(null, E('p', {}, result.message || _('Backup failed')), 'error');
}
});
},
handleUninstall: function() {
var self = this;
ui.showModal(_('Confirm Uninstall'), [
E('p', {}, _('Are you sure you want to uninstall Gitea Platform? Your repositories will be preserved.')),
E('div', { 'class': 'right' }, [
E('button', {
'class': 'btn',
'click': ui.hideModal
}, _('Cancel')),
E('button', {
'class': 'btn cbi-button-negative',
'click': function() {
ui.hideModal();
api.uninstall().then(function(result) {
if (result && result.success) {
ui.addNotification(null, E('p', {}, _('Gitea Platform uninstalled')), 'info');
} else {
ui.addNotification(null, E('p', {}, result.message || _('Uninstall failed')), 'error');
}
self.refreshData();
location.reload();
});
}
}, _('Uninstall'))
])
]);
}
});