Spaces:
Running
Running
| <html lang="en"> | |
| <head> | |
| <meta charset="UTF-8"> | |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
| <title>RedisRadar - Real-time Redis Monitoring</title> | |
| <script src="https://cdn.tailwindcss.com"></script> | |
| <script src="https://unpkg.com/feather-icons"></script> | |
| <script src="https://cdn.jsdelivr.net/npm/chart.js"></script> | |
| <script src="https://cdn.jsdelivr.net/npm/vanta@latest/dist/vanta.net.min.js"></script> | |
| <script> | |
| tailwind.config = { | |
| theme: { | |
| extend: { | |
| colors: { | |
| primary: '#d82c20', // Redis red | |
| secondary: '#4f5b66' // Dark slate | |
| } | |
| } | |
| } | |
| } | |
| </script> | |
| <style> | |
| .dashboard-card { | |
| transition: all 0.3s ease; | |
| } | |
| .dashboard-card:hover { | |
| transform: translateY(-5px); | |
| box-shadow: 0 10px 25px -5px rgba(0, 0, 0, 0.1); | |
| } | |
| #vanta-background { | |
| position: absolute; | |
| top: 0; | |
| left: 0; | |
| width: 100%; | |
| height: 100%; | |
| z-index: -1; | |
| opacity: 0.15; | |
| } | |
| </style> | |
| </head> | |
| <body class="bg-gray-50 min-h-screen"> | |
| <div id="vanta-background"></div> | |
| <!-- Navigation --> | |
| <nav class="bg-white shadow-sm"> | |
| <div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8"> | |
| <div class="flex justify-between h-16"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0 flex items-center"> | |
| <i data-feather="database" class="text-primary h-6 w-6"></i> | |
| <span class="ml-2 text-xl font-bold text-primary">RedisRadar</span> | |
| </div> | |
| <div class="hidden sm:ml-6 sm:flex sm:space-x-8"> | |
| <a href="#" class="border-primary text-gray-900 inline-flex items-center px-1 pt-1 border-b-2 text-sm font-medium"> | |
| Dashboard | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:border-gray-300 hover:text-gray-700 inline-flex items-center px-1 pt-1 border-b-2 text-sm font-medium"> | |
| Keys | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:border-gray-300 hover:text-gray-700 inline-flex items-center px-1 pt-1 border-b-2 text-sm font-medium"> | |
| Performance | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:border-gray-300 hover:text-gray-700 inline-flex items-center px-1 pt-1 border-b-2 text-sm font-medium"> | |
| Configuration | |
| </a> | |
| </div> | |
| </div> | |
| <div class="hidden sm:ml-6 sm:flex sm:items-center"> | |
| <button type="button" class="bg-white p-1 rounded-full text-gray-400 hover:text-gray-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary"> | |
| <i data-feather="bell" class="h-6 w-6"></i> | |
| </button> | |
| <div class="ml-3 relative"> | |
| <div> | |
| <button type="button" class="bg-white rounded-full flex text-sm focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary" id="user-menu-button" aria-expanded="false" aria-haspopup="true"> | |
| <img class="h-8 w-8 rounded-full" src="http://static.photos/technology/200x200/42" alt=""> | |
| </button> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="-mr-2 flex items-center sm:hidden"> | |
| <button type="button" class="inline-flex items-center justify-center p-2 rounded-md text-gray-400 hover:text-gray-500 hover:bg-gray-100 focus:outline-none focus:ring-2 focus:ring-inset focus:ring-primary" aria-controls="mobile-menu" aria-expanded="false"> | |
| <i data-feather="menu" class="block h-6 w-6"></i> | |
| <i data-feather="x" class="hidden h-6 w-6"></i> | |
| </button> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Mobile menu, show/hide based on menu state. --> | |
| <div class="sm:hidden hidden" id="mobile-menu"> | |
| <div class="pt-2 pb-3 space-y-1"> | |
| <a href="#" class="bg-primary-50 border-primary text-primary block pl-3 pr-4 py-2 border-l-4 text-base font-medium">Dashboard</a> | |
| <a href="#" class="border-transparent text-gray-500 hover:bg-gray-50 hover:border-gray-300 hover:text-gray-700 block pl-3 pr-4 py-2 border-l-4 text-base font-medium">Keys</a> | |
| <a href="#" class="border-transparent text-gray-500 hover:bg-gray-50 hover:border-gray-300 hover:text-gray-700 block pl-3 pr-4 py-2 border-l-4 text-base font-medium">Performance</a> | |
| <a href="#" class="border-transparent text-gray-500 hover:bg-gray-50 hover:border-gray-300 hover:text-gray-700 block pl-3 pr-4 py-2 border-l-4 text-base font-medium">Configuration</a> | |
| </div> | |
| <div class="pt-4 pb-3 border-t border-gray-200"> | |
| <div class="flex items-center px-4"> | |
| <div class="flex-shrink-0"> | |
| <img class="h-10 w-10 rounded-full" src="http://static.photos/technology/200x200/42" alt=""> | |
| </div> | |
| <div class="ml-3"> | |
| <div class="text-base font-medium text-gray-800">Admin User</div> | |
| <div class="text-sm font-medium text-gray-500">admin@example.com</div> | |
| </div> | |
| <button type="button" class="ml-auto bg-white flex-shrink-0 p-1 rounded-full text-gray-400 hover:text-gray-500 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-primary"> | |
| <i data-feather="bell" class="h-6 w-6"></i> | |
| </button> | |
| </div> | |
| </div> | |
| </div> | |
| </nav> | |
| <!-- Main Content --> | |
| <div class="py-6"> | |
| <div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8"> | |
| <h1 class="text-2xl font-semibold text-gray-900">Redis Dashboard</h1> | |
| <div class="py-4"> | |
| <div class="border-b border-gray-200"> | |
| <nav class="-mb-px flex space-x-8"> | |
| <a href="#" class="border-primary text-primary whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"> | |
| Overview | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"> | |
| Memory | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"> | |
| Clients | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"> | |
| Persistence | |
| </a> | |
| <a href="#" class="border-transparent text-gray-500 hover:text-gray-700 hover:border-gray-300 whitespace-nowrap py-4 px-1 border-b-2 font-medium text-sm"> | |
| Stats | |
| </a> | |
| </nav> | |
| </div> | |
| </div> | |
| <!-- Stats Cards --> | |
| <div class="grid grid-cols-1 gap-5 sm:grid-cols-2 lg:grid-cols-4 mb-6"> | |
| <div class="bg-white overflow-hidden shadow rounded-lg dashboard-card"> | |
| <div class="px-4 py-5 sm:p-6"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0 bg-primary-100 p-3 rounded-md"> | |
| <i data-feather="activity" class="text-primary h-6 w-6"></i> | |
| </div> | |
| <div class="ml-5 w-0 flex-1"> | |
| <dl> | |
| <dt class="text-sm font-medium text-gray-500 truncate">Connected Clients</dt> | |
| <dd> | |
| <div class="text-lg font-medium text-gray-900">127</div> | |
| </dd> | |
| </dl> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-gray-50 px-4 py-4 sm:px-6"> | |
| <div class="text-sm"> | |
| <a href="#" class="font-medium text-primary hover:text-primary-700"> View all<span class="sr-only"> Connected Clients stats</span></a> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-white overflow-hidden shadow rounded-lg dashboard-card"> | |
| <div class="px-4 py-5 sm:p-6"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0 bg-blue-100 p-3 rounded-md"> | |
| <i data-feather="database" class="text-blue-600 h-6 w-6"></i> | |
| </div> | |
| <div class="ml-5 w-0 flex-1"> | |
| <dl> | |
| <dt class="text-sm font-medium text-gray-500 truncate">Total Keys</dt> | |
| <dd> | |
| <div class="text-lg font-medium text-gray-900">8,947</div> | |
| </dd> | |
| </dl> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-gray-50 px-4 py-4 sm:px-6"> | |
| <div class="text-sm"> | |
| <a href="#" class="font-medium text-blue-600 hover:text-blue-500"> View all<span class="sr-only"> Total Keys stats</span></a> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-white overflow-hidden shadow rounded-lg dashboard-card"> | |
| <div class="px-4 py-5 sm:p-6"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0 bg-green-100 p-3 rounded-md"> | |
| <i data-feather="cpu" class="text-green-600 h-6 w-6"></i> | |
| </div> | |
| <div class="ml-5 w-0 flex-1"> | |
| <dl> | |
| <dt class="text-sm font-medium text-gray-500 truncate">Memory Usage</dt> | |
| <dd> | |
| <div class="text-lg font-medium text-gray-900">1.2GB</div> | |
| </dd> | |
| </dl> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-gray-50 px-4 py-4 sm:px-6"> | |
| <div class="text-sm"> | |
| <a href="#" class="font-medium text-green-600 hover:text-green-500"> View details<span class="sr-only"> Memory Usage stats</span></a> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-white overflow-hidden shadow rounded-lg dashboard-card"> | |
| <div class="px-4 py-5 sm:p-6"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0 bg-purple-100 p-3 rounded-md"> | |
| <i data-feather="zap" class="text-purple-600 h-6 w-6"></i> | |
| </div> | |
| <div class="ml-5 w-0 flex-1"> | |
| <dl> | |
| <dt class="text-sm font-medium text-gray-500 truncate">Ops/sec</dt> | |
| <dd> | |
| <div class="text-lg font-medium text-gray-900">2,349</div> | |
| </dd> | |
| </dl> | |
| </div> | |
| </div> | |
| </div> | |
| <div class="bg-gray-50 px-4 py-4 sm:px-6"> | |
| <div class="text-sm"> | |
| <a href="#" class="font-medium text-purple-600 hover:text-purple-500"> View performance<span class="sr-only"> Ops/sec stats</span></a> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Charts Section --> | |
| <div class="grid grid-cols-1 lg:grid-cols-2 gap-6 mb-6"> | |
| <!-- Memory Usage Chart --> | |
| <div class="bg-white shadow rounded-lg p-6 dashboard-card"> | |
| <h2 class="text-lg leading-6 font-medium text-gray-900 mb-4">Memory Usage</h2> | |
| <div class="h-64"> | |
| <canvas id="memoryChart"></canvas> | |
| </div> | |
| </div> | |
| <!-- Commands Per Second Chart --> | |
| <div class="bg-white shadow rounded-lg p-6 dashboard-card"> | |
| <h2 class="text-lg leading-6 font-medium text-gray-900 mb-4">Commands Per Second</h2> | |
| <div class="h-64"> | |
| <canvas id="commandsChart"></canvas> | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Recent Activity --> | |
| <div class="bg-white shadow rounded-lg overflow-hidden dashboard-card"> | |
| <div class="px-4 py-5 sm:px-6 border-b border-gray-200"> | |
| <h3 class="text-lg leading-6 font-medium text-gray-900">Recent Activity</h3> | |
| </div> | |
| <div class="bg-white overflow-hidden"> | |
| <ul class="divide-y divide-gray-200"> | |
| <li class="px-6 py-4"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0"> | |
| <i data-feather="activity" class="h-5 w-5 text-green-500"></i> | |
| </div> | |
| <div class="ml-4"> | |
| <p class="text-sm font-medium text-gray-900">New client connected</p> | |
| <p class="text-sm text-gray-500">Client ID: 128 from 192.168.1.42</p> | |
| </div> | |
| <div class="ml-auto text-sm text-gray-500"> | |
| 2m ago | |
| </div> | |
| </div> | |
| </li> | |
| <li class="px-6 py-4"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0"> | |
| <i data-feather="database" class="h-5 w-5 text-blue-500"></i> | |
| </div> | |
| <div class="ml-4"> | |
| <p class="text-sm font-medium text-gray-900">Key expiration</p> | |
| <p class="text-sm text-gray-500">Key: user:session:42 expired</p> | |
| </div> | |
| <div class="ml-auto text-sm text-gray-500"> | |
| 5m ago | |
| </div> | |
| </div> | |
| </li> | |
| <li class="px-6 py-4"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0"> | |
| <i data-feather="alert-triangle" class="h-5 w-5 text-yellow-500"></i> | |
| </div> | |
| <div class="ml-4"> | |
| <p class="text-sm font-medium text-gray-900">Memory threshold reached</p> | |
| <p class="text-sm text-gray-500">85% of maxmemory used</p> | |
| </div> | |
| <div class="ml-auto text-sm text-gray-500"> | |
| 12m ago | |
| </div> | |
| </div> | |
| </li> | |
| <li class="px-6 py-4"> | |
| <div class="flex items-center"> | |
| <div class="flex-shrink-0"> | |
| <i data-feather="zap" class="h-5 w-5 text-purple-500"></i> | |
| </div> | |
| <div class="ml-4"> | |
| <p class="text-sm font-medium text-gray-900">Performance spike</p> | |
| <p class="text-sm text-gray-500">5,231 ops/sec detected</p> | |
| </div> | |
| <div class="ml-auto text-sm text-gray-500"> | |
| 23m ago | |
| </div> | |
| </div> | |
| </li> | |
| </ul> | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| <script> | |
| // Initialize Vanta.js background | |
| VANTA.NET({ | |
| el: "#vanta-background", | |
| mouseControls: true, | |
| touchControls: true, | |
| gyroControls: false, | |
| minHeight: 200.00, | |
| minWidth: 200.00, | |
| scale: 1.00, | |
| scaleMobile: 1.00, | |
| color: 0xd82c20, | |
| backgroundColor: 0xf8fafc, | |
| points: 10.00, | |
| maxDistance: 20.00, | |
| spacing: 15.00 | |
| }); | |
| // Initialize charts | |
| document.addEventListener('DOMContentLoaded', function() { | |
| feather.replace(); | |
| // Memory Usage Chart | |
| const memoryCtx = document.getElementById('memoryChart').getContext('2d'); | |
| const memoryChart = new Chart(memoryCtx, { | |
| type: 'line', | |
| data: { | |
| labels: ['00:00', '03:00', '06:00', '09:00', '12:00', '15:00', '18:00', '21:00'], | |
| datasets: [{ | |
| label: 'Used Memory', | |
| data: [645, 892, 1123, 1456, 1234, 1567, 1890, 2100], | |
| backgroundColor: 'rgba(216, 44, 32, 0.1)', | |
| borderColor: '#d82c20', | |
| borderWidth: 2, | |
| tension: 0.1, | |
| fill: true | |
| }, { | |
| label: 'Max Memory', | |
| data: [2500, 2500, 2500, 2500, 2500, 2500, 2500, 2500], | |
| backgroundColor: 'rgba(75, 85, 99, 0.05)', | |
| borderColor: 'rgba(75, 85, 99, 0.5)', | |
| borderWidth: 1, | |
| borderDash: [5, 5], | |
| fill: true | |
| }] | |
| }, | |
| options: { | |
| responsive: true, | |
| maintainAspectRatio: false, | |
| plugins: { | |
| legend: { | |
| position: 'top', | |
| }, | |
| tooltip: { | |
| mode: 'index', | |
| intersect: false, | |
| } | |
| }, | |
| scales: { | |
| y: { | |
| beginAtZero: true, | |
| title: { | |
| display: true, | |
| text: 'Memory (MB)' | |
| } | |
| } | |
| } | |
| } | |
| }); | |
| // Commands Per Second Chart | |
| const commandsCtx = document.getElementById('commandsChart').getContext('2d'); | |
| const commandsChart = new Chart(commandsCtx, { | |
| type: 'bar', | |
| data: { | |
| labels: ['GET', 'SET', 'DEL', 'INCR', 'DECR', 'LPUSH', 'RPUSH', 'LPOP', 'RPOP'], | |
| datasets: [{ | |
| label: 'Commands Per Second', | |
| data: [1200, 800, 200, 150, 100, 300, 250, 180, 160], | |
| backgroundColor: [ | |
| 'rgba(216, 44, 32, 0.7)', | |
| 'rgba(216, 44, 32, 0.6)', | |
| 'rgba(216, 44, 32, 0.5)', | |
| 'rgba(216, 44, 32, 0.4)', | |
| 'rgba(216, 44, 32, 0.3)', | |
| 'rgba(216, 44, 32, 0.4)', | |
| 'rgba(216, 44, 32, 0.5)', | |
| 'rgba(216, 44, 32, 0.6)', | |
| 'rgba(216, 44, 32, 0.7)' | |
| ], | |
| borderColor: [ | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)', | |
| 'rgba(216, 44, 32, 1)' | |
| ], | |
| borderWidth: 1 | |
| }] | |
| }, | |
| options: { | |
| responsive: true, | |
| maintainAspectRatio: false, | |
| plugins: { | |
| legend: { | |
| display: false | |
| }, | |
| tooltip: { | |
| mode: 'index', | |
| intersect: false, | |
| } | |
| }, | |
| scales: { | |
| y: { | |
| beginAtZero: true, | |
| title: { | |
| display: true, | |
| text: 'Commands/sec' | |
| } | |
| } | |
| } | |
| } | |
| }); | |
| // Mobile menu toggle | |
| const mobileMenuButton = document.querySelector('[aria-controls="mobile-menu"]'); | |
| const mobileMenu = document.getElementById('mobile-menu'); | |
| mobileMenuButton.addEventListener('click', function() { | |
| const expanded = this.getAttribute('aria-expanded') === 'true'; | |
| this.setAttribute('aria-expanded', !expanded); | |
| mobileMenu.classList.toggle('hidden'); | |
| // Toggle menu icons | |
| const menuIcon = this.querySelector('i:first-child'); | |
| const closeIcon = this.querySelector('i:last-child'); | |
| menuIcon.classList.toggle('hidden'); | |
| closeIcon.classList.toggle('hidden'); | |
| }); | |
| }); | |
| </script> | |
| </body> | |
| </html> | |