504 lines
17 KiB
Plaintext
504 lines
17 KiB
Plaintext
<!DOCTYPE html>
|
||
<html lang="en">
|
||
<head>
|
||
<title data-text="title">Asteroid Radio - Admin Dashboard</title>
|
||
<meta charset="utf-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||
<link rel="stylesheet" type="text/css" href="/asteroid/static/asteroid.css">
|
||
</head>
|
||
<body>
|
||
<div class="container">
|
||
<h1>🎛️ ADMIN DASHBOARD</h1>
|
||
<div class="nav">
|
||
<a href="/asteroid/">← Back to Main</a>
|
||
<a href="/asteroid/player/">Web Player</a>
|
||
</div>
|
||
|
||
<!-- System Status -->
|
||
<div class="admin-section">
|
||
<h2>System Status</h2>
|
||
<div class="admin-grid">
|
||
<div class="status-card">
|
||
<h3>Server Status</h3>
|
||
<p class="status-good" data-text="server-status">🟢 Running</p>
|
||
</div>
|
||
<div class="status-card">
|
||
<h3>Database Status</h3>
|
||
<p class="status-good" data-text="database-status">🟢 Connected</p>
|
||
</div>
|
||
<div class="status-card">
|
||
<h3>Liquidsoap Status</h3>
|
||
<p class="status-error" data-text="liquidsoap-status">🔴 Not Running</p>
|
||
</div>
|
||
<div class="status-card">
|
||
<h3>Icecast Status</h3>
|
||
<p class="status-error" data-text="icecast-status">🔴 Not Running</p>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Music Library Management -->
|
||
<div class="admin-section">
|
||
<h2>Music Library Management</h2>
|
||
|
||
<!-- File Upload -->
|
||
<div class="upload-section">
|
||
<h3>Add Music Files</h3>
|
||
<div class="upload-info">
|
||
<p><strong>To add your own MP3 files:</strong></p>
|
||
<ol>
|
||
<li>Copy your MP3/FLAC/OGG/WAV files to: <code>/home/glenn/Projects/Code/asteroid/music/incoming/</code></li>
|
||
<li>Click "Copy Files to Library" below</li>
|
||
<li>Files will be moved to the library and added to the database</li>
|
||
</ol>
|
||
<p><em>Supported formats: MP3, FLAC, OGG, WAV</em></p>
|
||
</div>
|
||
<div class="upload-controls">
|
||
<button id="copy-files" class="btn btn-success">📁 Copy Files to Library</button>
|
||
<button id="open-incoming" class="btn btn-info">📂 Open Incoming Folder</button>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="admin-controls">
|
||
<button id="scan-library" class="btn btn-primary">🔍 Scan Library</button>
|
||
<button id="refresh-tracks" class="btn btn-secondary">🔄 Refresh Track List</button>
|
||
</div>
|
||
|
||
<div class="track-stats">
|
||
<p>Total Tracks: <span id="track-count" data-text="track-count">0</span></p>
|
||
<p>Library Path: <span data-text="library-path">/music/library/</span></p>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Track Management -->
|
||
<div class="admin-section">
|
||
<h2>Track Management</h2>
|
||
<div class="track-controls">
|
||
<input type="text" id="track-search" placeholder="Search tracks..." class="search-input">
|
||
<select id="sort-tracks" class="sort-select">
|
||
<option value="title">Sort by Title</option>
|
||
<option value="artist">Sort by Artist</option>
|
||
<option value="album">Sort by Album</option>
|
||
<option value="added-date">Sort by Date Added</option>
|
||
</select>
|
||
</div>
|
||
|
||
<div id="tracks-container" class="tracks-list">
|
||
<div class="loading">Loading tracks...</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Player Control -->
|
||
<div class="admin-section">
|
||
<h2>Player Control</h2>
|
||
<div class="card">
|
||
<h3>🎵 Player Control</h3>
|
||
<div class="player-controls">
|
||
<button class="btn btn-primary" onclick="playTrack()">▶️ Play</button>
|
||
<button class="btn btn-secondary" onclick="pausePlayer()">⏸️ Pause</button>
|
||
<button class="btn btn-secondary" onclick="stopPlayer()">⏹️ Stop</button>
|
||
<button class="btn btn-secondary" onclick="resumePlayer()">▶️ Resume</button>
|
||
</div>
|
||
<div id="player-status" class="status-info">
|
||
Status: <span id="status-text">Unknown</span><br>
|
||
Current Track: <span id="current-track">None</span>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="card">
|
||
<h3>👥 User Management</h3>
|
||
<div class="user-stats" id="user-stats">
|
||
<div class="stat-card">
|
||
<span class="stat-number" id="total-users">0</span>
|
||
<span class="stat-label">Total Users</span>
|
||
</div>
|
||
<div class="stat-card">
|
||
<span class="stat-number" id="active-users">0</span>
|
||
<span class="stat-label">Active Users</span>
|
||
</div>
|
||
<div class="stat-card">
|
||
<span class="stat-number" id="admin-users">0</span>
|
||
<span class="stat-label">Admins</span>
|
||
</div>
|
||
<div class="stat-card">
|
||
<span class="stat-number" id="dj-users">0</span>
|
||
<span class="stat-label">DJs</span>
|
||
</div>
|
||
</div>
|
||
<div class="controls">
|
||
<button class="btn btn-primary" onclick="loadUsers()">👥 Manage Users</button>
|
||
<button class="btn btn-secondary" onclick="showCreateUser()">➕ Create User</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<script>
|
||
// Admin Dashboard JavaScript
|
||
let tracks = [];
|
||
let currentTrackId = null;
|
||
|
||
// Load tracks on page load
|
||
document.addEventListener('DOMContentLoaded', function() {
|
||
loadTracks();
|
||
updatePlayerStatus();
|
||
|
||
// Setup event listeners
|
||
document.getElementById('scan-library').addEventListener('click', scanLibrary);
|
||
document.getElementById('refresh-tracks').addEventListener('click', loadTracks);
|
||
document.getElementById('track-search').addEventListener('input', filterTracks);
|
||
document.getElementById('sort-tracks').addEventListener('change', sortTracks);
|
||
document.getElementById('copy-files').addEventListener('click', copyFiles);
|
||
document.getElementById('open-incoming').addEventListener('click', openIncomingFolder);
|
||
|
||
// Player controls
|
||
document.getElementById('player-play').addEventListener('click', () => playTrack(currentTrackId));
|
||
document.getElementById('player-pause').addEventListener('click', pausePlayer);
|
||
document.getElementById('player-stop').addEventListener('click', stopPlayer);
|
||
document.getElementById('player-resume').addEventListener('click', resumePlayer);
|
||
});
|
||
|
||
// Load tracks from API
|
||
async function loadTracks() {
|
||
try {
|
||
const response = await fetch('/admin/tracks');
|
||
const data = await response.json();
|
||
|
||
if (data.status === 'success') {
|
||
tracks = data.tracks || [];
|
||
document.getElementById('track-count').textContent = tracks.length;
|
||
displayTracks(tracks);
|
||
}
|
||
} catch (error) {
|
||
console.error('Error loading tracks:', error);
|
||
document.getElementById('tracks-container').innerHTML = '<div class="error">Error loading tracks</div>';
|
||
}
|
||
}
|
||
|
||
// Display tracks in the UI
|
||
function displayTracks(trackList) {
|
||
const container = document.getElementById('tracks-container');
|
||
|
||
if (trackList.length === 0) {
|
||
container.innerHTML = '<div class="no-tracks">No tracks found. Click "Scan Library" to add tracks.</div>';
|
||
return;
|
||
}
|
||
|
||
const tracksHtml = trackList.map(track => `
|
||
<div class="track-item" data-track-id="${track.id}">
|
||
<div class="track-info">
|
||
<div class="track-title">${track.title || 'Unknown Title'}</div>
|
||
<div class="track-artist">${track.artist || 'Unknown Artist'}</div>
|
||
<div class="track-album">${track.album || 'Unknown Album'}</div>
|
||
</div>
|
||
<div class="track-actions">
|
||
<button onclick="playTrack(${track.id})" class="btn btn-sm btn-success">▶️ Play</button>
|
||
<button onclick="streamTrack(${track.id})" class="btn btn-sm btn-info">🎵 Stream</button>
|
||
<button onclick="deleteTrack(${track.id})" class="btn btn-sm btn-danger">🗑️ Delete</button>
|
||
</div>
|
||
</div>
|
||
`).join('');
|
||
|
||
container.innerHTML = tracksHtml;
|
||
}
|
||
|
||
// Scan music library
|
||
async function scanLibrary() {
|
||
const statusEl = document.getElementById('scan-status');
|
||
const scanBtn = document.getElementById('scan-library');
|
||
|
||
statusEl.textContent = 'Scanning...';
|
||
scanBtn.disabled = true;
|
||
|
||
try {
|
||
const response = await fetch('/admin/scan-library', { method: 'POST' });
|
||
const data = await response.json();
|
||
|
||
if (data.status === 'success') {
|
||
statusEl.textContent = `✅ Added ${data['tracks-added']} tracks`;
|
||
loadTracks(); // Refresh track list
|
||
} else {
|
||
statusEl.textContent = '❌ Scan failed';
|
||
}
|
||
} catch (error) {
|
||
statusEl.textContent = '❌ Scan error';
|
||
console.error('Scan error:', error);
|
||
} finally {
|
||
scanBtn.disabled = false;
|
||
setTimeout(() => statusEl.textContent = '', 3000);
|
||
}
|
||
}
|
||
|
||
// Filter tracks based on search
|
||
function filterTracks() {
|
||
const query = document.getElementById('track-search').value.toLowerCase();
|
||
const filtered = tracks.filter(track =>
|
||
(track.title || '').toLowerCase().includes(query) ||
|
||
(track.artist || '').toLowerCase().includes(query) ||
|
||
(track.album || '').toLowerCase().includes(query)
|
||
);
|
||
displayTracks(filtered);
|
||
}
|
||
|
||
// Sort tracks
|
||
function sortTracks() {
|
||
const sortBy = document.getElementById('sort-tracks').value;
|
||
const sorted = [...tracks].sort((a, b) => {
|
||
const aVal = a[sortBy] || '';
|
||
const bVal = b[sortBy] || '';
|
||
return aVal.localeCompare(bVal);
|
||
});
|
||
displayTracks(sorted);
|
||
}
|
||
|
||
// Player functions
|
||
async function playTrack(trackId) {
|
||
if (!trackId) {
|
||
alert('Please select a track to play');
|
||
return;
|
||
}
|
||
|
||
try {
|
||
const response = await fetch(`/api/play?track-id=${trackId}`, { method: 'POST' });
|
||
const data = await response.json();
|
||
|
||
if (data.status === 'success') {
|
||
currentTrackId = trackId;
|
||
updatePlayerStatus();
|
||
} else {
|
||
alert('Error playing track: ' + data.message);
|
||
}
|
||
} catch (error) {
|
||
console.error('Play error:', error);
|
||
alert('Error playing track');
|
||
}
|
||
}
|
||
|
||
async function pausePlayer() {
|
||
try {
|
||
await fetch('/api/pause', { method: 'POST' });
|
||
updatePlayerStatus();
|
||
} catch (error) {
|
||
console.error('Pause error:', error);
|
||
}
|
||
}
|
||
|
||
async function stopPlayer() {
|
||
try {
|
||
await fetch('/api/stop', { method: 'POST' });
|
||
currentTrackId = null;
|
||
updatePlayerStatus();
|
||
} catch (error) {
|
||
console.error('Stop error:', error);
|
||
}
|
||
}
|
||
|
||
async function resumePlayer() {
|
||
try {
|
||
await fetch('/api/resume', { method: 'POST' });
|
||
updatePlayerStatus();
|
||
} catch (error) {
|
||
console.error('Resume error:', error);
|
||
}
|
||
}
|
||
|
||
async function updatePlayerStatus() {
|
||
try {
|
||
const response = await fetch('/api/player-status');
|
||
const data = await response.json();
|
||
|
||
if (data.status === 'success') {
|
||
const player = data.player;
|
||
document.getElementById('player-state').textContent = player.state;
|
||
document.getElementById('current-track').textContent = player['current-track'] || 'None';
|
||
document.getElementById('current-position').textContent = player.position;
|
||
}
|
||
} catch (error) {
|
||
console.error('Error updating player status:', error);
|
||
}
|
||
}
|
||
|
||
function streamTrack(trackId) {
|
||
window.open(`/asteroid/tracks/${trackId}/stream`, '_blank');
|
||
}
|
||
|
||
function deleteTrack(trackId) {
|
||
if (confirm('Are you sure you want to delete this track?')) {
|
||
// TODO: Implement track deletion API
|
||
alert('Track deletion not yet implemented');
|
||
}
|
||
}
|
||
|
||
// Copy files from incoming to library
|
||
async function copyFiles() {
|
||
try {
|
||
const response = await fetch('/admin/copy-files');
|
||
const data = await response.json();
|
||
|
||
if (data.status === 'success') {
|
||
alert(`${data.message}`);
|
||
await loadTracks(); // Refresh track list
|
||
} else {
|
||
alert(`Error: ${data.message}`);
|
||
}
|
||
} catch (error) {
|
||
console.error('Error copying files:', error);
|
||
alert('Failed to copy files');
|
||
}
|
||
}
|
||
|
||
// Open incoming folder (for convenience)
|
||
function openIncomingFolder() {
|
||
alert('Copy your MP3 files to: /home/glenn/Projects/Code/asteroid/music/incoming/\n\nThen click "Copy Files to Library" to add them to your music collection.');
|
||
}
|
||
|
||
// User Management Functions
|
||
async function loadUserStats() {
|
||
try {
|
||
const response = await fetch('/asteroid/api/users/stats');
|
||
const result = await response.json();
|
||
|
||
if (result.status === 'success') {
|
||
const stats = result.stats;
|
||
document.getElementById('total-users').textContent = stats.total;
|
||
document.getElementById('active-users').textContent = stats.active;
|
||
document.getElementById('admin-users').textContent = stats.admins;
|
||
document.getElementById('dj-users').textContent = stats.djs;
|
||
}
|
||
} catch (error) {
|
||
console.error('Error loading user stats:', error);
|
||
}
|
||
}
|
||
|
||
async function loadUsers() {
|
||
try {
|
||
const response = await fetch('/asteroid/api/users');
|
||
const result = await response.json();
|
||
|
||
if (result.status === 'success') {
|
||
showUsersTable(result.users);
|
||
}
|
||
} catch (error) {
|
||
console.error('Error loading users:', error);
|
||
alert('Error loading users. Please try again.');
|
||
}
|
||
}
|
||
|
||
function showUsersTable(users) {
|
||
const container = document.createElement('div');
|
||
container.className = 'user-management';
|
||
container.innerHTML = `
|
||
<h3>User Management</h3>
|
||
<table class="users-table">
|
||
<thead>
|
||
<tr>
|
||
<th>Username</th>
|
||
<th>Email</th>
|
||
<th>Role</th>
|
||
<th>Status</th>
|
||
<th>Last Login</th>
|
||
<th>Actions</th>
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
${users.map(user => `
|
||
<tr>
|
||
<td>${user.username}</td>
|
||
<td>${user.email}</td>
|
||
<td>
|
||
<select onchange="updateUserRole('${user.id}', this.value)">
|
||
<option value="listener" ${user.role === 'listener' ? 'selected' : ''}>Listener</option>
|
||
<option value="dj" ${user.role === 'dj' ? 'selected' : ''}>DJ</option>
|
||
<option value="admin" ${user.role === 'admin' ? 'selected' : ''}>Admin</option>
|
||
</select>
|
||
</td>
|
||
<td>${user.active ? '✅ Active' : '❌ Inactive'}</td>
|
||
<td>${user['last-login'] ? new Date(user['last-login'] * 1000).toLocaleString() : 'Never'}</td>
|
||
<td class="user-actions">
|
||
${user.active ?
|
||
`<button class="btn btn-danger" onclick="deactivateUser('${user.id}')">Deactivate</button>` :
|
||
`<button class="btn btn-success" onclick="activateUser('${user.id}')">Activate</button>`
|
||
}
|
||
</td>
|
||
</tr>
|
||
`).join('')}
|
||
</tbody>
|
||
</table>
|
||
<button class="btn btn-secondary" onclick="hideUsersTable()">Close</button>
|
||
`;
|
||
|
||
document.body.appendChild(container);
|
||
}
|
||
|
||
function hideUsersTable() {
|
||
const userManagement = document.querySelector('.user-management');
|
||
if (userManagement) {
|
||
userManagement.remove();
|
||
}
|
||
}
|
||
|
||
async function updateUserRole(userId, newRole) {
|
||
try {
|
||
const formData = new FormData();
|
||
formData.append('role', newRole);
|
||
|
||
const response = await fetch(`/asteroid/api/users/${userId}/role`, {
|
||
method: 'POST',
|
||
body: formData
|
||
});
|
||
|
||
const result = await response.json();
|
||
|
||
if (result.status === 'success') {
|
||
loadUserStats();
|
||
alert('User role updated successfully');
|
||
} else {
|
||
alert('Error updating user role: ' + result.message);
|
||
}
|
||
} catch (error) {
|
||
console.error('Error updating user role:', error);
|
||
alert('Error updating user role. Please try again.');
|
||
}
|
||
}
|
||
|
||
async function deactivateUser(userId) {
|
||
if (!confirm('Are you sure you want to deactivate this user?')) {
|
||
return;
|
||
}
|
||
|
||
try {
|
||
const response = await fetch(`/asteroid/api/users/${userId}/deactivate`, {
|
||
method: 'POST'
|
||
});
|
||
|
||
const result = await response.json();
|
||
|
||
if (result.status === 'success') {
|
||
loadUsers();
|
||
loadUserStats();
|
||
alert('User deactivated successfully');
|
||
} else {
|
||
alert('Error deactivating user: ' + result.message);
|
||
}
|
||
} catch (error) {
|
||
console.error('Error deactivating user:', error);
|
||
alert('Error deactivating user. Please try again.');
|
||
}
|
||
}
|
||
|
||
function showCreateUser() {
|
||
window.location.href = '/asteroid/register';
|
||
}
|
||
|
||
// Load user stats on page load
|
||
loadUserStats();
|
||
|
||
// Update player status every 5 seconds
|
||
setInterval(updatePlayerStatus, 5000);
|
||
|
||
// Update user stats every 30 seconds
|
||
setInterval(loadUserStats, 30000);
|
||
</script>
|
||
</body>
|
||
</html>
|