redisradar / index.html
uciwild's picture
app for monitoring redis database
2a89258 verified
<!DOCTYPE html>
<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>