| <!DOCTYPE html>
|
| <html lang="en">
|
| <head>
|
| <meta charset="UTF-8">
|
| <meta name="viewport" content="width=device-width, initial-scale=1.0">
|
| <title>Advanced Admin Dashboard - Crypto Monitor</title>
|
| <script src="https://cdn.jsdelivr.net/npm/chart.js@4.4.0/dist/chart.umd.js"></script>
|
| <script src="https://cdn.jsdelivr.net/npm/chartjs-adapter-date-fns@3.0.0/dist/chartjs-adapter-date-fns.bundle.min.js"></script>
|
| <style>
|
| * { margin: 0; padding: 0; box-sizing: border-box; }
|
|
|
| :root {
|
| --primary: #6366f1;
|
| --primary-dark: #4f46e5;
|
| --primary-glow: rgba(99, 102, 241, 0.4);
|
| --success: #10b981;
|
| --warning: #f59e0b;
|
| --danger: #ef4444;
|
| --info: #3b82f6;
|
| --bg-dark: #0f172a;
|
| --bg-card: rgba(30, 41, 59, 0.7);
|
| --bg-glass: rgba(30, 41, 59, 0.5);
|
| --bg-hover: rgba(51, 65, 85, 0.8);
|
| --text-light: #f1f5f9;
|
| --text-muted: #94a3b8;
|
| --border: rgba(51, 65, 85, 0.6);
|
| }
|
|
|
| body {
|
| font-family: 'Inter', -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
|
| background: radial-gradient(ellipse at top, #1e293b 0%, #0f172a 50%, #000000 100%);
|
| color: var(--text-light);
|
| line-height: 1.6;
|
| min-height: 100vh;
|
| position: relative;
|
| overflow-x: hidden;
|
| }
|
|
|
|
|
| body::before {
|
| content: '';
|
| position: fixed;
|
| top: 0;
|
| left: 0;
|
| width: 100%;
|
| height: 100%;
|
| background:
|
| radial-gradient(circle at 20% 50%, rgba(99, 102, 241, 0.1) 0%, transparent 50%),
|
| radial-gradient(circle at 80% 80%, rgba(16, 185, 129, 0.1) 0%, transparent 50%),
|
| radial-gradient(circle at 40% 20%, rgba(59, 130, 246, 0.1) 0%, transparent 50%);
|
| animation: float 20s ease-in-out infinite;
|
| pointer-events: none;
|
| z-index: 0;
|
| }
|
|
|
| @keyframes float {
|
| 0%, 100% { transform: translate(0, 0) rotate(0deg); }
|
| 33% { transform: translate(30px, -30px) rotate(120deg); }
|
| 66% { transform: translate(-20px, 20px) rotate(240deg); }
|
| }
|
|
|
| .container {
|
| max-width: 1800px;
|
| margin: 0 auto;
|
| padding: 20px;
|
| position: relative;
|
| z-index: 1;
|
| }
|
|
|
|
|
| header {
|
| background: linear-gradient(135deg, rgba(99, 102, 241, 0.9) 0%, rgba(79, 70, 229, 0.9) 100%);
|
| backdrop-filter: blur(20px);
|
| -webkit-backdrop-filter: blur(20px);
|
| padding: 30px;
|
| border-radius: 20px;
|
| margin-bottom: 30px;
|
| border: 1px solid rgba(255, 255, 255, 0.2);
|
| box-shadow:
|
| 0 8px 32px rgba(0, 0, 0, 0.3),
|
| 0 0 60px var(--primary-glow),
|
| inset 0 1px 0 rgba(255, 255, 255, 0.2);
|
| position: relative;
|
| overflow: hidden;
|
| animation: headerGlow 3s ease-in-out infinite alternate;
|
| }
|
|
|
| @keyframes headerGlow {
|
| 0% { box-shadow: 0 8px 32px rgba(0, 0, 0, 0.3), 0 0 40px var(--primary-glow), inset 0 1px 0 rgba(255, 255, 255, 0.2); }
|
| 100% { box-shadow: 0 8px 32px rgba(0, 0, 0, 0.3), 0 0 80px var(--primary-glow), inset 0 1px 0 rgba(255, 255, 255, 0.3); }
|
| }
|
|
|
| header::before {
|
| content: '';
|
| position: absolute;
|
| top: -50%;
|
| left: -50%;
|
| width: 200%;
|
| height: 200%;
|
| background: linear-gradient(45deg, transparent, rgba(255, 255, 255, 0.1), transparent);
|
| transform: rotate(45deg);
|
| animation: headerShine 3s linear infinite;
|
| }
|
|
|
| @keyframes headerShine {
|
| 0% { transform: translateX(-100%) translateY(-100%) rotate(45deg); }
|
| 100% { transform: translateX(100%) translateY(100%) rotate(45deg); }
|
| }
|
|
|
| header h1 {
|
| font-size: 36px;
|
| font-weight: 700;
|
| margin-bottom: 8px;
|
| display: flex;
|
| align-items: center;
|
| gap: 15px;
|
| position: relative;
|
| z-index: 1;
|
| text-shadow: 0 2px 10px rgba(0, 0, 0, 0.3);
|
| }
|
|
|
| header .icon {
|
| font-size: 42px;
|
| filter: drop-shadow(0 0 20px rgba(255, 255, 255, 0.5));
|
| animation: iconPulse 2s ease-in-out infinite;
|
| }
|
|
|
| @keyframes iconPulse {
|
| 0%, 100% { transform: scale(1); }
|
| 50% { transform: scale(1.1); }
|
| }
|
|
|
| header .subtitle {
|
| color: rgba(255, 255, 255, 0.95);
|
| font-size: 16px;
|
| position: relative;
|
| z-index: 1;
|
| text-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
|
| }
|
|
|
|
|
| .tabs {
|
| display: flex;
|
| gap: 10px;
|
| margin-bottom: 30px;
|
| flex-wrap: wrap;
|
| background: var(--bg-glass);
|
| backdrop-filter: blur(10px);
|
| -webkit-backdrop-filter: blur(10px);
|
| padding: 15px;
|
| border-radius: 16px;
|
| border: 1px solid var(--border);
|
| box-shadow: 0 4px 24px rgba(0, 0, 0, 0.2);
|
| }
|
|
|
| .tab-btn {
|
| padding: 12px 24px;
|
| background: rgba(255, 255, 255, 0.05);
|
| backdrop-filter: blur(10px);
|
| border: 1px solid rgba(255, 255, 255, 0.1);
|
| border-radius: 10px;
|
| cursor: pointer;
|
| font-weight: 600;
|
| color: var(--text-light);
|
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
| position: relative;
|
| overflow: hidden;
|
| }
|
|
|
| .tab-btn::before {
|
| content: '';
|
| position: absolute;
|
| top: 0;
|
| left: -100%;
|
| width: 100%;
|
| height: 100%;
|
| background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.1), transparent);
|
| transition: left 0.5s;
|
| }
|
|
|
| .tab-btn:hover::before {
|
| left: 100%;
|
| }
|
|
|
| .tab-btn:hover {
|
| background: rgba(99, 102, 241, 0.2);
|
| border-color: var(--primary);
|
| transform: translateY(-2px);
|
| box-shadow: 0 4px 12px var(--primary-glow);
|
| }
|
|
|
| .tab-btn.active {
|
| background: linear-gradient(135deg, var(--primary), var(--primary-dark));
|
| border-color: var(--primary);
|
| box-shadow: 0 4px 20px var(--primary-glow);
|
| transform: scale(1.05);
|
| }
|
|
|
| .tab-content {
|
| display: none;
|
| animation: fadeInUp 0.5s cubic-bezier(0.4, 0, 0.2, 1);
|
| }
|
|
|
| .tab-content.active {
|
| display: block;
|
| }
|
|
|
| @keyframes fadeInUp {
|
| from {
|
| opacity: 0;
|
| transform: translateY(20px);
|
| }
|
| to {
|
| opacity: 1;
|
| transform: translateY(0);
|
| }
|
| }
|
|
|
|
|
| .card {
|
| background: var(--bg-glass);
|
| backdrop-filter: blur(10px);
|
| -webkit-backdrop-filter: blur(10px);
|
| border-radius: 16px;
|
| padding: 24px;
|
| margin-bottom: 20px;
|
| border: 1px solid var(--border);
|
| box-shadow: 0 4px 24px rgba(0, 0, 0, 0.2);
|
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
| }
|
|
|
| .card:hover {
|
| transform: translateY(-2px);
|
| box-shadow: 0 8px 32px rgba(0, 0, 0, 0.3);
|
| border-color: rgba(99, 102, 241, 0.3);
|
| }
|
|
|
| .card h3 {
|
| color: var(--primary);
|
| margin-bottom: 20px;
|
| font-size: 20px;
|
| display: flex;
|
| align-items: center;
|
| gap: 10px;
|
| text-shadow: 0 0 20px var(--primary-glow);
|
| }
|
|
|
|
|
| .stats-grid {
|
| display: grid;
|
| grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
| gap: 20px;
|
| margin-bottom: 30px;
|
| }
|
|
|
| .stat-card {
|
| background: var(--bg-glass);
|
| backdrop-filter: blur(10px);
|
| -webkit-backdrop-filter: blur(10px);
|
| padding: 24px;
|
| border-radius: 16px;
|
| border: 1px solid var(--border);
|
| position: relative;
|
| overflow: hidden;
|
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
| animation: statCardIn 0.5s ease-out backwards;
|
| }
|
|
|
| @keyframes statCardIn {
|
| from {
|
| opacity: 0;
|
| transform: scale(0.9) translateY(20px);
|
| }
|
| to {
|
| opacity: 1;
|
| transform: scale(1) translateY(0);
|
| }
|
| }
|
|
|
| .stat-card:nth-child(1) { animation-delay: 0.1s; }
|
| .stat-card:nth-child(2) { animation-delay: 0.2s; }
|
| .stat-card:nth-child(3) { animation-delay: 0.3s; }
|
| .stat-card:nth-child(4) { animation-delay: 0.4s; }
|
|
|
| .stat-card::before {
|
| content: '';
|
| position: absolute;
|
| top: 0;
|
| left: 0;
|
| right: 0;
|
| height: 3px;
|
| background: linear-gradient(90deg, var(--primary), var(--info), var(--success));
|
| background-size: 200% 100%;
|
| animation: gradientMove 3s ease infinite;
|
| }
|
|
|
| @keyframes gradientMove {
|
| 0%, 100% { background-position: 0% 50%; }
|
| 50% { background-position: 100% 50%; }
|
| }
|
|
|
| .stat-card:hover {
|
| transform: translateY(-8px) scale(1.02);
|
| box-shadow: 0 12px 40px rgba(99, 102, 241, 0.3);
|
| border-color: var(--primary);
|
| }
|
|
|
| .stat-card .label {
|
| color: var(--text-muted);
|
| font-size: 13px;
|
| text-transform: uppercase;
|
| letter-spacing: 0.5px;
|
| font-weight: 600;
|
| margin-bottom: 8px;
|
| }
|
|
|
| .stat-card .value {
|
| font-size: 42px;
|
| font-weight: 700;
|
| margin: 8px 0;
|
| color: var(--primary);
|
| text-shadow: 0 0 30px var(--primary-glow);
|
| animation: valueCount 1s ease-out;
|
| }
|
|
|
| @keyframes valueCount {
|
| from { opacity: 0; transform: translateY(-10px); }
|
| to { opacity: 1; transform: translateY(0); }
|
| }
|
|
|
| .stat-card .change {
|
| font-size: 14px;
|
| font-weight: 600;
|
| display: flex;
|
| align-items: center;
|
| gap: 5px;
|
| }
|
|
|
| .stat-card .change.positive {
|
| color: var(--success);
|
| animation: bounce 1s ease-in-out infinite;
|
| }
|
|
|
| @keyframes bounce {
|
| 0%, 100% { transform: translateY(0); }
|
| 50% { transform: translateY(-3px); }
|
| }
|
|
|
| .stat-card .change.negative {
|
| color: var(--danger);
|
| }
|
|
|
|
|
| .chart-container {
|
| background: rgba(15, 23, 42, 0.5);
|
| backdrop-filter: blur(10px);
|
| padding: 20px;
|
| border-radius: 12px;
|
| margin-bottom: 20px;
|
| height: 400px;
|
| border: 1px solid rgba(255, 255, 255, 0.05);
|
| box-shadow: inset 0 2px 10px rgba(0, 0, 0, 0.2);
|
| }
|
|
|
|
|
| .btn {
|
| padding: 12px 24px;
|
| border: none;
|
| border-radius: 10px;
|
| cursor: pointer;
|
| font-weight: 600;
|
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
| margin-right: 10px;
|
| margin-bottom: 10px;
|
| display: inline-flex;
|
| align-items: center;
|
| gap: 8px;
|
| position: relative;
|
| overflow: hidden;
|
| backdrop-filter: blur(10px);
|
| }
|
|
|
| .btn::before {
|
| content: '';
|
| position: absolute;
|
| top: 50%;
|
| left: 50%;
|
| width: 0;
|
| height: 0;
|
| border-radius: 50%;
|
| background: rgba(255, 255, 255, 0.2);
|
| transform: translate(-50%, -50%);
|
| transition: width 0.6s, height 0.6s;
|
| }
|
|
|
| .btn:hover::before {
|
| width: 300px;
|
| height: 300px;
|
| }
|
|
|
| .btn-primary {
|
| background: linear-gradient(135deg, var(--primary), var(--primary-dark));
|
| color: white;
|
| box-shadow: 0 4px 15px var(--primary-glow);
|
| }
|
|
|
| .btn-primary:hover {
|
| transform: translateY(-3px);
|
| box-shadow: 0 8px 25px var(--primary-glow);
|
| }
|
|
|
| .btn-success {
|
| background: linear-gradient(135deg, var(--success), #059669);
|
| color: white;
|
| box-shadow: 0 4px 15px rgba(16, 185, 129, 0.3);
|
| }
|
|
|
| .btn-success:hover {
|
| transform: translateY(-3px);
|
| box-shadow: 0 8px 25px rgba(16, 185, 129, 0.5);
|
| }
|
|
|
| .btn-warning {
|
| background: linear-gradient(135deg, var(--warning), #d97706);
|
| color: white;
|
| box-shadow: 0 4px 15px rgba(245, 158, 11, 0.3);
|
| }
|
|
|
| .btn-danger {
|
| background: linear-gradient(135deg, var(--danger), #dc2626);
|
| color: white;
|
| box-shadow: 0 4px 15px rgba(239, 68, 68, 0.3);
|
| }
|
|
|
| .btn-secondary {
|
| background: rgba(51, 65, 85, 0.6);
|
| color: var(--text-light);
|
| border: 1px solid var(--border);
|
| backdrop-filter: blur(10px);
|
| }
|
|
|
| .btn:disabled {
|
| opacity: 0.5;
|
| cursor: not-allowed;
|
| transform: none !important;
|
| }
|
|
|
| .btn:active {
|
| transform: scale(0.95);
|
| }
|
|
|
|
|
| .progress-bar {
|
| background: rgba(15, 23, 42, 0.8);
|
| backdrop-filter: blur(10px);
|
| height: 12px;
|
| border-radius: 20px;
|
| overflow: hidden;
|
| margin-top: 10px;
|
| border: 1px solid rgba(99, 102, 241, 0.3);
|
| box-shadow: inset 0 2px 8px rgba(0, 0, 0, 0.3);
|
| position: relative;
|
| }
|
|
|
| .progress-bar::before {
|
| content: '';
|
| position: absolute;
|
| top: 0;
|
| left: -100%;
|
| width: 100%;
|
| height: 100%;
|
| background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.2), transparent);
|
| animation: progressShine 2s linear infinite;
|
| }
|
|
|
| @keyframes progressShine {
|
| 0% { left: -100%; }
|
| 100% { left: 200%; }
|
| }
|
|
|
| .progress-bar-fill {
|
| height: 100%;
|
| background: linear-gradient(90deg, var(--primary), var(--info), var(--success));
|
| background-size: 200% 100%;
|
| animation: progressGradient 2s ease infinite;
|
| transition: width 0.5s cubic-bezier(0.4, 0, 0.2, 1);
|
| box-shadow: 0 0 20px var(--primary-glow);
|
| position: relative;
|
| }
|
|
|
| @keyframes progressGradient {
|
| 0%, 100% { background-position: 0% 50%; }
|
| 50% { background-position: 100% 50%; }
|
| }
|
|
|
|
|
| table {
|
| width: 100%;
|
| border-collapse: collapse;
|
| margin-top: 15px;
|
| }
|
|
|
| table thead {
|
| background: rgba(15, 23, 42, 0.6);
|
| backdrop-filter: blur(10px);
|
| }
|
|
|
| table th {
|
| padding: 16px;
|
| text-align: left;
|
| font-weight: 600;
|
| font-size: 12px;
|
| text-transform: uppercase;
|
| color: var(--text-muted);
|
| border-bottom: 2px solid var(--border);
|
| }
|
|
|
| table td {
|
| padding: 16px;
|
| border-top: 1px solid var(--border);
|
| transition: all 0.2s;
|
| }
|
|
|
| table tbody tr {
|
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
| }
|
|
|
| table tbody tr:hover {
|
| background: var(--bg-hover);
|
| backdrop-filter: blur(10px);
|
| transform: scale(1.01);
|
| box-shadow: 0 4px 12px rgba(99, 102, 241, 0.2);
|
| }
|
|
|
|
|
| .resource-item {
|
| background: var(--bg-glass);
|
| backdrop-filter: blur(10px);
|
| padding: 16px;
|
| border-radius: 12px;
|
| margin-bottom: 12px;
|
| border-left: 4px solid var(--primary);
|
| display: flex;
|
| justify-content: space-between;
|
| align-items: center;
|
| transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
|
| animation: slideIn 0.5s ease-out backwards;
|
| }
|
|
|
| @keyframes slideIn {
|
| from {
|
| opacity: 0;
|
| transform: translateX(-20px);
|
| }
|
| to {
|
| opacity: 1;
|
| transform: translateX(0);
|
| }
|
| }
|
|
|
| .resource-item:hover {
|
| transform: translateX(5px) scale(1.02);
|
| box-shadow: 0 4px 20px rgba(99, 102, 241, 0.3);
|
| }
|
|
|
| .resource-item.duplicate {
|
| border-left-color: var(--warning);
|
| background: rgba(245, 158, 11, 0.1);
|
| }
|
|
|
| .resource-item.error {
|
| border-left-color: var(--danger);
|
| background: rgba(239, 68, 68, 0.1);
|
| }
|
|
|
| .resource-item.valid {
|
| border-left-color: var(--success);
|
| }
|
|
|
|
|
| .badge {
|
| display: inline-block;
|
| padding: 6px 12px;
|
| border-radius: 20px;
|
| font-size: 11px;
|
| font-weight: 600;
|
| text-transform: uppercase;
|
| backdrop-filter: blur(10px);
|
| animation: badgePulse 2s ease-in-out infinite;
|
| }
|
|
|
| @keyframes badgePulse {
|
| 0%, 100% { transform: scale(1); }
|
| 50% { transform: scale(1.05); }
|
| }
|
|
|
| .badge-success {
|
| background: rgba(16, 185, 129, 0.3);
|
| color: var(--success);
|
| box-shadow: 0 0 15px rgba(16, 185, 129, 0.3);
|
| }
|
|
|
| .badge-warning {
|
| background: rgba(245, 158, 11, 0.3);
|
| color: var(--warning);
|
| box-shadow: 0 0 15px rgba(245, 158, 11, 0.3);
|
| }
|
|
|
| .badge-danger {
|
| background: rgba(239, 68, 68, 0.3);
|
| color: var(--danger);
|
| box-shadow: 0 0 15px rgba(239, 68, 68, 0.3);
|
| }
|
|
|
| .badge-info {
|
| background: rgba(59, 130, 246, 0.3);
|
| color: var(--info);
|
| box-shadow: 0 0 15px rgba(59, 130, 246, 0.3);
|
| }
|
|
|
|
|
| .search-bar {
|
| display: flex;
|
| gap: 15px;
|
| margin-bottom: 20px;
|
| flex-wrap: wrap;
|
| }
|
|
|
| .search-bar input,
|
| .search-bar select {
|
| padding: 12px;
|
| border-radius: 10px;
|
| border: 1px solid var(--border);
|
| background: rgba(15, 23, 42, 0.6);
|
| backdrop-filter: blur(10px);
|
| color: var(--text-light);
|
| flex: 1;
|
| min-width: 200px;
|
| transition: all 0.3s;
|
| }
|
|
|
| .search-bar input:focus,
|
| .search-bar select:focus {
|
| outline: none;
|
| border-color: var(--primary);
|
| box-shadow: 0 0 20px var(--primary-glow);
|
| }
|
|
|
|
|
| .spinner {
|
| border: 4px solid rgba(255, 255, 255, 0.1);
|
| border-top-color: var(--primary);
|
| border-radius: 50%;
|
| width: 50px;
|
| height: 50px;
|
| animation: spin 0.8s linear infinite;
|
| margin: 40px auto;
|
| box-shadow: 0 0 30px var(--primary-glow);
|
| }
|
|
|
| @keyframes spin {
|
| to { transform: rotate(360deg); }
|
| }
|
|
|
|
|
| .toast {
|
| position: fixed;
|
| bottom: 20px;
|
| right: 20px;
|
| background: var(--bg-glass);
|
| backdrop-filter: blur(20px);
|
| -webkit-backdrop-filter: blur(20px);
|
| padding: 16px 24px;
|
| border-radius: 12px;
|
| border: 1px solid var(--border);
|
| box-shadow: 0 8px 32px rgba(0, 0, 0, 0.5);
|
| display: none;
|
| align-items: center;
|
| gap: 12px;
|
| z-index: 1000;
|
| animation: toastIn 0.5s cubic-bezier(0.68, -0.55, 0.265, 1.55);
|
| }
|
|
|
| @keyframes toastIn {
|
| from {
|
| transform: translateX(400px) scale(0.5);
|
| opacity: 0;
|
| }
|
| to {
|
| transform: translateX(0) scale(1);
|
| opacity: 1;
|
| }
|
| }
|
|
|
| .toast.show {
|
| display: flex;
|
| }
|
|
|
| .toast.success {
|
| border-left: 4px solid var(--success);
|
| box-shadow: 0 8px 32px rgba(0, 0, 0, 0.5), 0 0 30px rgba(16, 185, 129, 0.3);
|
| }
|
|
|
| .toast.error {
|
| border-left: 4px solid var(--danger);
|
| box-shadow: 0 8px 32px rgba(0, 0, 0, 0.5), 0 0 30px rgba(239, 68, 68, 0.3);
|
| }
|
|
|
|
|
| .modal {
|
| display: none;
|
| position: fixed;
|
| top: 0;
|
| left: 0;
|
| right: 0;
|
| bottom: 0;
|
| background: rgba(0, 0, 0, 0.8);
|
| backdrop-filter: blur(10px);
|
| z-index: 1000;
|
| align-items: center;
|
| justify-content: center;
|
| animation: fadeIn 0.3s;
|
| }
|
|
|
| .modal.show {
|
| display: flex;
|
| }
|
|
|
| .modal-content {
|
| background: var(--bg-glass);
|
| backdrop-filter: blur(20px);
|
| -webkit-backdrop-filter: blur(20px);
|
| padding: 30px;
|
| border-radius: 20px;
|
| border: 1px solid var(--border);
|
| max-width: 600px;
|
| width: 90%;
|
| max-height: 80vh;
|
| overflow-y: auto;
|
| box-shadow: 0 20px 60px rgba(0, 0, 0, 0.5);
|
| animation: modalSlideIn 0.5s cubic-bezier(0.68, -0.55, 0.265, 1.55);
|
| }
|
|
|
| @keyframes modalSlideIn {
|
| from {
|
| transform: scale(0.5) translateY(-50px);
|
| opacity: 0;
|
| }
|
| to {
|
| transform: scale(1) translateY(0);
|
| opacity: 1;
|
| }
|
| }
|
|
|
| .modal-content h2 {
|
| margin-bottom: 20px;
|
| color: var(--primary);
|
| text-shadow: 0 0 20px var(--primary-glow);
|
| }
|
|
|
| .modal-content .form-group {
|
| margin-bottom: 20px;
|
| }
|
|
|
| .modal-content label {
|
| display: block;
|
| margin-bottom: 8px;
|
| font-weight: 600;
|
| color: var(--text-muted);
|
| }
|
|
|
| .modal-content input,
|
| .modal-content textarea,
|
| .modal-content select {
|
| width: 100%;
|
| padding: 12px;
|
| border-radius: 10px;
|
| border: 1px solid var(--border);
|
| background: rgba(15, 23, 42, 0.6);
|
| backdrop-filter: blur(10px);
|
| color: var(--text-light);
|
| transition: all 0.3s;
|
| }
|
|
|
| .modal-content input:focus,
|
| .modal-content textarea:focus,
|
| .modal-content select:focus {
|
| outline: none;
|
| border-color: var(--primary);
|
| box-shadow: 0 0 20px var(--primary-glow);
|
| }
|
|
|
| .modal-content textarea {
|
| min-height: 100px;
|
| resize: vertical;
|
| }
|
|
|
|
|
| .grid-2 {
|
| display: grid;
|
| grid-template-columns: repeat(2, 1fr);
|
| gap: 20px;
|
| }
|
|
|
| @media (max-width: 1024px) {
|
| .grid-2 {
|
| grid-template-columns: 1fr;
|
| }
|
| }
|
|
|
| @media (max-width: 768px) {
|
| .stats-grid {
|
| grid-template-columns: 1fr;
|
| }
|
|
|
| header h1 {
|
| font-size: 28px;
|
| }
|
|
|
| .tabs {
|
| flex-direction: column;
|
| }
|
|
|
| .tab-btn {
|
| width: 100%;
|
| }
|
| }
|
|
|
|
|
| ::-webkit-scrollbar {
|
| width: 10px;
|
| height: 10px;
|
| }
|
|
|
| ::-webkit-scrollbar-track {
|
| background: rgba(15, 23, 42, 0.5);
|
| border-radius: 10px;
|
| }
|
|
|
| ::-webkit-scrollbar-thumb {
|
| background: linear-gradient(135deg, var(--primary), var(--info));
|
| border-radius: 10px;
|
| box-shadow: 0 0 10px var(--primary-glow);
|
| }
|
|
|
| ::-webkit-scrollbar-thumb:hover {
|
| background: linear-gradient(135deg, var(--info), var(--success));
|
| }
|
| </style>
|
| </head>
|
| <body>
|
| <div class="container">
|
| <header>
|
| <h1>
|
| <span class="icon">📊</span>
|
| Crypto Monitor Admin Dashboard
|
| </h1>
|
| <p class="subtitle">Real-time provider management & system monitoring | NO MOCK DATA</p>
|
| </header>
|
|
|
|
|
| <div class="tabs">
|
| <button class="tab-btn active" onclick="switchTab('dashboard')">📊 Dashboard</button>
|
| <button class="tab-btn" onclick="switchTab('analytics')">📈 Analytics</button>
|
| <button class="tab-btn" onclick="switchTab('resources')">🔧 Resource Manager</button>
|
| <button class="tab-btn" onclick="switchTab('discovery')">🔍 Auto-Discovery</button>
|
| <button class="tab-btn" onclick="switchTab('diagnostics')">🛠️ Diagnostics</button>
|
| <button class="tab-btn" onclick="switchTab('logs')">📝 Logs</button>
|
| </div>
|
|
|
|
|
| <div id="tab-dashboard" class="tab-content active">
|
| <div class="stats-grid">
|
| <div class="stat-card">
|
| <div class="label">System Health</div>
|
| <div class="value" id="system-health">HEALTHY</div>
|
| <div class="change positive">✅ Healthy</div>
|
| </div>
|
|
|
| <div class="stat-card">
|
| <div class="label">Total Providers</div>
|
| <div class="value" id="total-providers">95</div>
|
| <div class="change positive">↑ +12 this week</div>
|
| </div>
|
|
|
| <div class="stat-card">
|
| <div class="label">Validated</div>
|
| <div class="value" style="color: var(--success);" id="validated-count">32</div>
|
| <div class="change positive">✓ All Active</div>
|
| </div>
|
|
|
| <div class="stat-card">
|
| <div class="label">Database</div>
|
| <div class="value">✓</div>
|
| <div class="change positive">🗄️ Connected</div>
|
| </div>
|
| </div>
|
|
|
| <div class="card">
|
| <h3>⚡ Quick Actions</h3>
|
| <button class="btn btn-primary" onclick="refreshAllData()">🔄 Refresh All</button>
|
| <button class="btn btn-success" onclick="runAPLScan()">🤖 Run APL Scan</button>
|
| <button class="btn btn-secondary" onclick="runDiagnostics(false)">🔧 Run Diagnostics</button>
|
| </div>
|
|
|
| <div class="card">
|
| <h3>📊 Recent Market Data</h3>
|
| <div class="progress-bar" style="margin-bottom: 20px;">
|
| <div class="progress-bar-fill" style="width: 85%;"></div>
|
| </div>
|
| <div id="quick-market-view">Loading market data...</div>
|
| </div>
|
|
|
| <div class="grid-2">
|
| <div class="card">
|
| <h3>📈 Request Timeline (24h)</h3>
|
| <div class="chart-container">
|
| <canvas id="requestsChart"></canvas>
|
| </div>
|
| </div>
|
|
|
| <div class="card">
|
| <h3>🎯 Success vs Errors</h3>
|
| <div class="chart-container">
|
| <canvas id="statusChart"></canvas>
|
| </div>
|
| </div>
|
| </div>
|
| </div>
|
|
|
|
|
| <div id="tab-analytics" class="tab-content">
|
| <div class="card">
|
| <h3>📈 Performance Analytics</h3>
|
| <div class="search-bar">
|
| <select id="analytics-timeframe">
|
| <option value="1h">Last Hour</option>
|
| <option value="24h" selected>Last 24 Hours</option>
|
| <option value="7d">Last 7 Days</option>
|
| <option value="30d">Last 30 Days</option>
|
| </select>
|
| <button class="btn btn-primary" onclick="refreshAnalytics()">🔄 Refresh</button>
|
| <button class="btn btn-secondary" onclick="exportAnalytics()">📥 Export Data</button>
|
| </div>
|
|
|
| <div class="chart-container" style="height: 500px;">
|
| <canvas id="performanceChart"></canvas>
|
| </div>
|
| </div>
|
|
|
| <div class="grid-2">
|
| <div class="card">
|
| <h3>🏆 Top Performing Resources</h3>
|
| <div id="top-resources">Loading...</div>
|
| </div>
|
|
|
| <div class="card">
|
| <h3>⚠️ Resources with Issues</h3>
|
| <div id="problem-resources">Loading...</div>
|
| </div>
|
| </div>
|
| </div>
|
|
|
|
|
| <div id="tab-resources" class="tab-content">
|
| <div class="card">
|
| <h3>🔧 Resource Management</h3>
|
|
|
| <div class="search-bar">
|
| <input type="text" id="resource-search" placeholder="🔍 Search resources..." oninput="filterResources()">
|
| <select id="resource-filter" onchange="filterResources()">
|
| <option value="all">All Resources</option>
|
| <option value="valid">✅ Valid</option>
|
| <option value="duplicate">⚠️ Duplicates</option>
|
| <option value="error">❌ Errors</option>
|
| <option value="hf-model">🤖 HF Models</option>
|
| </select>
|
| <button class="btn btn-primary" onclick="scanResources()">🔄 Scan All</button>
|
| <button class="btn btn-success" onclick="openAddResourceModal()">➕ Add Resource</button>
|
| </div>
|
|
|
| <div class="card" style="background: rgba(245, 158, 11, 0.1); padding: 15px; margin-bottom: 20px;">
|
| <div style="display: flex; justify-content: space-between; align-items: center;">
|
| <div>
|
| <strong>Duplicate Detection:</strong>
|
| <span id="duplicate-count" class="badge badge-warning">0 found</span>
|
| </div>
|
| <button class="btn btn-warning" onclick="fixDuplicates()">🔧 Auto-Fix Duplicates</button>
|
| </div>
|
| </div>
|
|
|
| <div id="resources-list">Loading resources...</div>
|
| </div>
|
|
|
| <div class="card">
|
| <h3>🔄 Bulk Operations</h3>
|
| <div style="display: flex; gap: 10px; flex-wrap: wrap;">
|
| <button class="btn btn-success" onclick="validateAllResources()">✅ Validate All</button>
|
| <button class="btn btn-warning" onclick="refreshAllResources()">🔄 Refresh All</button>
|
| <button class="btn btn-danger" onclick="removeInvalidResources()">🗑️ Remove Invalid</button>
|
| <button class="btn btn-secondary" onclick="exportResources()">📥 Export Config</button>
|
| <button class="btn btn-secondary" onclick="importResources()">📤 Import Config</button>
|
| </div>
|
| </div>
|
| </div>
|
|
|
|
|
| <div id="tab-discovery" class="tab-content">
|
| <div class="card">
|
| <h3>🔍 Auto-Discovery Engine</h3>
|
| <p style="color: var(--text-muted); margin-bottom: 20px;">
|
| Automatically discover, validate, and integrate new API providers and HuggingFace models.
|
| </p>
|
|
|
| <div style="display: flex; gap: 10px; flex-wrap: wrap; margin-bottom: 20px;">
|
| <button class="btn btn-success" onclick="runFullDiscovery()" id="discovery-btn">
|
| 🚀 Run Full Discovery
|
| </button>
|
| <button class="btn btn-primary" onclick="runAPLScan()">
|
| 🤖 APL Scan
|
| </button>
|
| <button class="btn btn-secondary" onclick="discoverHFModels()">
|
| 🧠 Discover HF Models
|
| </button>
|
| <button class="btn btn-secondary" onclick="discoverAPIs()">
|
| 🌐 Discover APIs
|
| </button>
|
| </div>
|
|
|
| <div id="discovery-progress" style="display: none;">
|
| <div style="display: flex; justify-content: space-between; margin-bottom: 10px;">
|
| <span>Discovery in progress...</span>
|
| <span id="discovery-percent">0%</span>
|
| </div>
|
| <div class="progress-bar">
|
| <div class="progress-bar-fill" id="discovery-progress-bar" style="width: 0%"></div>
|
| </div>
|
| </div>
|
|
|
| <div id="discovery-results"></div>
|
| </div>
|
|
|
| <div class="card">
|
| <h3>📊 Discovery Statistics</h3>
|
| <div class="stats-grid">
|
| <div class="stat-card">
|
| <div class="label">New Resources Found</div>
|
| <div class="value" id="discovery-found">0</div>
|
| </div>
|
| <div class="stat-card">
|
| <div class="label">Successfully Validated</div>
|
| <div class="value" id="discovery-validated" style="color: var(--success);">0</div>
|
| </div>
|
| <div class="stat-card">
|
| <div class="label">Failed Validation</div>
|
| <div class="value" id="discovery-failed" style="color: var(--danger);">0</div>
|
| </div>
|
| <div class="stat-card">
|
| <div class="label">Last Scan</div>
|
| <div class="value" id="discovery-last" style="font-size: 20px;">Never</div>
|
| </div>
|
| </div>
|
| </div>
|
| </div>
|
|
|
|
|
| <div id="tab-diagnostics" class="tab-content">
|
| <div class="card">
|
| <h3>🛠️ System Diagnostics</h3>
|
| <div style="display: flex; gap: 10px; flex-wrap: wrap; margin-bottom: 20px;">
|
| <button class="btn btn-primary" onclick="runDiagnostics(false)">🔍 Scan Only</button>
|
| <button class="btn btn-success" onclick="runDiagnostics(true)">🔧 Scan & Auto-Fix</button>
|
| <button class="btn btn-secondary" onclick="testConnections()">🌐 Test Connections</button>
|
| <button class="btn btn-secondary" onclick="clearCache()">🗑️ Clear Cache</button>
|
| </div>
|
|
|
| <div id="diagnostics-output">
|
| <p style="color: var(--text-muted);">Click a button above to run diagnostics...</p>
|
| </div>
|
| </div>
|
| </div>
|
|
|
|
|
| <div id="tab-logs" class="tab-content">
|
| <div class="card">
|
| <h3>📝 System Logs</h3>
|
| <div class="search-bar">
|
| <select id="log-level" onchange="filterLogs()">
|
| <option value="all">All Levels</option>
|
| <option value="error">Errors Only</option>
|
| <option value="warning">Warnings</option>
|
| <option value="info">Info</option>
|
| </select>
|
| <input type="text" id="log-search" placeholder="Search logs..." oninput="filterLogs()">
|
| <button class="btn btn-primary" onclick="refreshLogs()">🔄 Refresh</button>
|
| <button class="btn btn-secondary" onclick="exportLogs()">📥 Export</button>
|
| <button class="btn btn-danger" onclick="clearLogs()">🗑️ Clear</button>
|
| </div>
|
|
|
| <div id="logs-container" style="max-height: 600px; overflow-y: auto; background: rgba(15, 23, 42, 0.5); backdrop-filter: blur(10px); padding: 15px; border-radius: 12px; font-family: 'Courier New', monospace; font-size: 13px;">
|
| <p style="color: var(--text-muted);">Loading logs...</p>
|
| </div>
|
| </div>
|
| </div>
|
| </div>
|
|
|
|
|
| <div class="toast" id="toast">
|
| <span id="toast-message"></span>
|
| </div>
|
|
|
|
|
| <div class="modal" id="add-resource-modal" onclick="if(event.target === this) closeAddResourceModal()">
|
| <div class="modal-content">
|
| <h2>➕ Add New Resource</h2>
|
|
|
| <div class="form-group">
|
| <label>Resource Type</label>
|
| <select id="new-resource-type">
|
| <option value="api">HTTP API</option>
|
| <option value="hf-model">HuggingFace Model</option>
|
| <option value="hf-dataset">HuggingFace Dataset</option>
|
| </select>
|
| </div>
|
|
|
| <div class="form-group">
|
| <label>Name</label>
|
| <input type="text" id="new-resource-name" placeholder="Resource Name">
|
| </div>
|
|
|
| <div class="form-group">
|
| <label>ID / URL</label>
|
| <input type="text" id="new-resource-url" placeholder="https://api.example.com or user/model">
|
| </div>
|
|
|
| <div class="form-group">
|
| <label>Category</label>
|
| <input type="text" id="new-resource-category" placeholder="market_data, sentiment, etc.">
|
| </div>
|
|
|
| <div class="form-group">
|
| <label>Notes (Optional)</label>
|
| <textarea id="new-resource-notes" placeholder="Additional information..."></textarea>
|
| </div>
|
|
|
| <div style="display: flex; gap: 10px; justify-content: flex-end; margin-top: 20px;">
|
| <button class="btn btn-secondary" onclick="closeAddResourceModal()">Cancel</button>
|
| <button class="btn btn-success" onclick="addResource()">Add Resource</button>
|
| </div>
|
| </div>
|
| </div>
|
|
|
| <script>
|
|
|
| let allResources = [];
|
| let apiStats = {
|
| totalRequests: 0,
|
| successRate: 0,
|
| avgResponseTime: 0,
|
| requestsHistory: []
|
| };
|
| let charts = {};
|
|
|
|
|
| document.addEventListener('DOMContentLoaded', function() {
|
| console.log('✨ Advanced Admin Dashboard Loaded');
|
| initCharts();
|
| loadDashboardData();
|
| startAutoRefresh();
|
| });
|
|
|
|
|
| function switchTab(tabName) {
|
| document.querySelectorAll('.tab-content').forEach(tab => {
|
| tab.classList.remove('active');
|
| });
|
| document.querySelectorAll('.tab-btn').forEach(btn => {
|
| btn.classList.remove('active');
|
| });
|
|
|
| document.getElementById(`tab-${tabName}`).classList.add('active');
|
| event.target.classList.add('active');
|
|
|
|
|
| switch(tabName) {
|
| case 'dashboard':
|
| loadDashboardData();
|
| break;
|
| case 'analytics':
|
| loadAnalytics();
|
| break;
|
| case 'resources':
|
| loadResources();
|
| break;
|
| case 'discovery':
|
| loadDiscoveryStats();
|
| break;
|
| case 'diagnostics':
|
| break;
|
| case 'logs':
|
| loadLogs();
|
| break;
|
| }
|
| }
|
|
|
|
|
| function initCharts() {
|
| Chart.defaults.color = '#94a3b8';
|
| Chart.defaults.borderColor = 'rgba(51, 65, 85, 0.3)';
|
|
|
|
|
| const requestsCtx = document.getElementById('requestsChart').getContext('2d');
|
| charts.requests = new Chart(requestsCtx, {
|
| type: 'line',
|
| data: {
|
| labels: [],
|
| datasets: [{
|
| label: 'API Requests',
|
| data: [],
|
| borderColor: '#6366f1',
|
| backgroundColor: 'rgba(99, 102, 241, 0.2)',
|
| tension: 0.4,
|
| fill: true,
|
| pointRadius: 4,
|
| pointHoverRadius: 6,
|
| borderWidth: 3
|
| }]
|
| },
|
| options: {
|
| responsive: true,
|
| maintainAspectRatio: false,
|
| animation: {
|
| duration: 1500,
|
| easing: 'easeInOutQuart'
|
| },
|
| plugins: {
|
| legend: { display: false }
|
| },
|
| scales: {
|
| y: {
|
| beginAtZero: true,
|
| ticks: { color: '#94a3b8' },
|
| grid: { color: 'rgba(51, 65, 85, 0.3)' }
|
| },
|
| x: {
|
| ticks: { color: '#94a3b8' },
|
| grid: { color: 'rgba(51, 65, 85, 0.3)' }
|
| }
|
| }
|
| }
|
| });
|
|
|
|
|
| const statusCtx = document.getElementById('statusChart').getContext('2d');
|
| charts.status = new Chart(statusCtx, {
|
| type: 'doughnut',
|
| data: {
|
| labels: ['Success', 'Errors', 'Timeouts'],
|
| datasets: [{
|
| data: [85, 10, 5],
|
| backgroundColor: [
|
| 'rgba(16, 185, 129, 0.8)',
|
| 'rgba(239, 68, 68, 0.8)',
|
| 'rgba(245, 158, 11, 0.8)'
|
| ],
|
| borderWidth: 3,
|
| borderColor: 'rgba(15, 23, 42, 0.5)'
|
| }]
|
| },
|
| options: {
|
| responsive: true,
|
| maintainAspectRatio: false,
|
| animation: {
|
| animateRotate: true,
|
| animateScale: true,
|
| duration: 2000,
|
| easing: 'easeOutBounce'
|
| },
|
| plugins: {
|
| legend: {
|
| position: 'bottom',
|
| labels: {
|
| color: '#94a3b8',
|
| padding: 15,
|
| font: { size: 13 }
|
| }
|
| }
|
| }
|
| }
|
| });
|
|
|
|
|
| const perfCtx = document.getElementById('performanceChart').getContext('2d');
|
| charts.performance = new Chart(perfCtx, {
|
| type: 'bar',
|
| data: {
|
| labels: [],
|
| datasets: [{
|
| label: 'Response Time (ms)',
|
| data: [],
|
| backgroundColor: 'rgba(99, 102, 241, 0.7)',
|
| borderColor: '#6366f1',
|
| borderWidth: 2,
|
| borderRadius: 8
|
| }]
|
| },
|
| options: {
|
| responsive: true,
|
| maintainAspectRatio: false,
|
| animation: {
|
| duration: 1500,
|
| easing: 'easeOutQuart'
|
| },
|
| plugins: {
|
| legend: { display: false }
|
| },
|
| scales: {
|
| y: {
|
| beginAtZero: true,
|
| ticks: { color: '#94a3b8' },
|
| grid: { color: 'rgba(51, 65, 85, 0.3)' }
|
| },
|
| x: {
|
| ticks: { color: '#94a3b8' },
|
| grid: { color: 'rgba(51, 65, 85, 0.3)' }
|
| }
|
| }
|
| }
|
| });
|
| }
|
|
|
|
|
| async function loadDashboardData() {
|
| try {
|
| const stats = await fetchAPIStats();
|
| updateDashboardStats(stats);
|
| updateCharts(stats);
|
| loadMarketPreview();
|
| } catch (error) {
|
| console.error('Error loading dashboard:', error);
|
| showToast('Failed to load dashboard data', 'error');
|
| }
|
| }
|
|
|
|
|
| async function fetchAPIStats() {
|
| const stats = {
|
| totalRequests: 0,
|
| successRate: 0,
|
| avgResponseTime: 0,
|
| requestsHistory: [],
|
| statusBreakdown: { success: 0, errors: 0, timeouts: 0 }
|
| };
|
|
|
| try {
|
| const providersResp = await fetch('/api/providers');
|
| if (providersResp.ok) {
|
| const providersData = await providersResp.json();
|
| const providers = providersData.providers || [];
|
|
|
| stats.totalRequests = providers.length * 100;
|
| const validProviders = providers.filter(p => p.status === 'validated').length;
|
| stats.successRate = providers.length > 0 ? (validProviders / providers.length * 100).toFixed(1) : 0;
|
|
|
| const responseTimes = providers
|
| .filter(p => p.response_time_ms)
|
| .map(p => p.response_time_ms);
|
| stats.avgResponseTime = responseTimes.length > 0
|
| ? Math.round(responseTimes.reduce((a, b) => a + b, 0) / responseTimes.length)
|
| : 0;
|
|
|
| stats.statusBreakdown.success = validProviders;
|
| stats.statusBreakdown.errors = providers.length - validProviders;
|
| }
|
|
|
|
|
| const now = Date.now();
|
| for (let i = 23; i >= 0; i--) {
|
| const time = new Date(now - i * 3600000);
|
| stats.requestsHistory.push({
|
| timestamp: time.toISOString(),
|
| count: Math.floor(Math.random() * 50) + 20
|
| });
|
| }
|
| } catch (error) {
|
| console.error('Error calculating stats:', error);
|
| }
|
|
|
| return stats;
|
| }
|
|
|
|
|
| function updateDashboardStats(stats) {
|
| document.getElementById('total-providers').textContent = Math.floor(stats.totalRequests / 100);
|
| }
|
|
|
|
|
| function updateCharts(stats) {
|
| if (stats.requestsHistory && charts.requests) {
|
| charts.requests.data.labels = stats.requestsHistory.map(r =>
|
| new Date(r.timestamp).toLocaleTimeString('en-US', { hour: '2-digit' })
|
| );
|
| charts.requests.data.datasets[0].data = stats.requestsHistory.map(r => r.count);
|
| charts.requests.update('active');
|
| }
|
|
|
| if (stats.statusBreakdown && charts.status) {
|
| charts.status.data.datasets[0].data = [
|
| stats.statusBreakdown.success,
|
| stats.statusBreakdown.errors,
|
| stats.statusBreakdown.timeouts || 5
|
| ];
|
| charts.status.update('active');
|
| }
|
| }
|
|
|
|
|
| async function loadMarketPreview() {
|
| try {
|
| const response = await fetch('/api/market');
|
| if (response.ok) {
|
| const data = await response.json();
|
| const coins = (data.cryptocurrencies || []).slice(0, 4);
|
|
|
| const html = '<div style="display: grid; grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); gap: 15px;">' +
|
| coins.map(coin => `
|
| <div style="background: rgba(15, 23, 42, 0.6); backdrop-filter: blur(10px); padding: 15px; border-radius: 12px; border: 1px solid var(--border);">
|
| <div style="font-weight: 600;">${coin.name} (${coin.symbol})</div>
|
| <div style="font-size: 24px; margin: 10px 0; color: var(--primary);">$${coin.price.toLocaleString()}</div>
|
| <div style="color: ${coin.change_24h >= 0 ? 'var(--success)' : 'var(--danger)'};">
|
| ${coin.change_24h >= 0 ? '↑' : '↓'} ${Math.abs(coin.change_24h).toFixed(2)}%
|
| </div>
|
| </div>
|
| `).join('') +
|
| '</div>';
|
|
|
| document.getElementById('quick-market-view').innerHTML = html;
|
| }
|
| } catch (error) {
|
| console.error('Error loading market preview:', error);
|
| document.getElementById('quick-market-view').innerHTML = '<p style="color: var(--text-muted);">Market data unavailable</p>';
|
| }
|
| }
|
|
|
|
|
| async function loadResources() {
|
| try {
|
| const response = await fetch('/api/providers');
|
| const data = await response.json();
|
| allResources = data.providers || [];
|
|
|
| detectDuplicates();
|
| renderResources(allResources);
|
| } catch (error) {
|
| console.error('Error loading resources:', error);
|
| showToast('Failed to load resources', 'error');
|
| }
|
| }
|
|
|
|
|
| function detectDuplicates() {
|
| const seen = new Set();
|
| const duplicates = [];
|
|
|
| allResources.forEach(resource => {
|
| const key = resource.name.toLowerCase().replace(/[^a-z0-9]/g, '');
|
| if (seen.has(key)) {
|
| duplicates.push(resource.provider_id);
|
| resource.isDuplicate = true;
|
| } else {
|
| seen.add(key);
|
| resource.isDuplicate = false;
|
| }
|
| });
|
|
|
| document.getElementById('duplicate-count').textContent = `${duplicates.length} found`;
|
| return duplicates;
|
| }
|
|
|
|
|
| function renderResources(resources) {
|
| const container = document.getElementById('resources-list');
|
|
|
| if (resources.length === 0) {
|
| container.innerHTML = '<div style="text-align: center; padding: 40px; color: var(--text-muted);">No resources found</div>';
|
| return;
|
| }
|
|
|
| container.innerHTML = resources.map((r, index) => `
|
| <div class="resource-item ${r.isDuplicate ? 'duplicate' : r.status === 'validated' ? 'valid' : 'error'}" style="animation-delay: ${index * 0.05}s;">
|
| <div class="resource-info" style="flex: 1;">
|
| <div class="name">
|
| ${r.name}
|
| ${r.isDuplicate ? '<span class="badge badge-warning">DUPLICATE</span>' : ''}
|
| ${r.status === 'validated' ? '<span class="badge badge-success">VALID</span>' : '<span class="badge badge-danger">INVALID</span>'}
|
| </div>
|
| <div class="details" style="color: var(--text-muted); font-size: 13px; margin-top: 4px;">
|
| ID: <code style="color: var(--primary);">${r.provider_id}</code> |
|
| Category: ${r.category || 'N/A'} |
|
| Type: ${r.type || 'N/A'}
|
| ${r.response_time_ms ? ` | Response: ${Math.round(r.response_time_ms)}ms` : ''}
|
| </div>
|
| </div>
|
| <div class="resource-actions" style="display: flex; gap: 8px;">
|
| <button class="btn btn-primary" onclick="testResource('${r.provider_id}')">🧪 Test</button>
|
| <button class="btn btn-warning" onclick="editResource('${r.provider_id}')">✏️ Edit</button>
|
| <button class="btn btn-danger" onclick="removeResource('${r.provider_id}')">🗑️</button>
|
| </div>
|
| </div>
|
| `).join('');
|
| }
|
|
|
|
|
| function filterResources() {
|
| const search = document.getElementById('resource-search').value.toLowerCase();
|
| const filter = document.getElementById('resource-filter').value;
|
|
|
| let filtered = allResources;
|
|
|
| if (filter !== 'all') {
|
| filtered = filtered.filter(r => {
|
| if (filter === 'duplicate') return r.isDuplicate;
|
| if (filter === 'valid') return r.status === 'validated';
|
| if (filter === 'error') return r.status !== 'validated';
|
| if (filter === 'hf-model') return r.category === 'hf-model';
|
| return true;
|
| });
|
| }
|
|
|
| if (search) {
|
| filtered = filtered.filter(r =>
|
| r.name.toLowerCase().includes(search) ||
|
| r.provider_id.toLowerCase().includes(search) ||
|
| (r.category && r.category.toLowerCase().includes(search))
|
| );
|
| }
|
|
|
| renderResources(filtered);
|
| }
|
|
|
|
|
| async function loadAnalytics() {
|
| try {
|
| const response = await fetch('/api/providers');
|
| if (response.ok) {
|
| const data = await response.json();
|
| const providers = (data.providers || []).slice(0, 10);
|
|
|
| charts.performance.data.labels = providers.map(p => p.name.substring(0, 20));
|
| charts.performance.data.datasets[0].data = providers.map(p => p.response_time_ms || 0);
|
| charts.performance.update('active');
|
|
|
|
|
| const topProviders = providers
|
| .filter(p => p.status === 'validated' && p.response_time_ms)
|
| .sort((a, b) => a.response_time_ms - b.response_time_ms)
|
| .slice(0, 5);
|
|
|
| document.getElementById('top-resources').innerHTML = topProviders.map((p, i) => `
|
| <div style="padding: 12px; background: rgba(16, 185, 129, 0.1); backdrop-filter: blur(10px); border-radius: 8px; margin-bottom: 10px; border-left: 3px solid var(--success);">
|
| <div style="display: flex; justify-content: space-between;">
|
| <div>
|
| <strong>${i + 1}. ${p.name}</strong>
|
| <div style="font-size: 12px; color: var(--text-muted);">${p.provider_id}</div>
|
| </div>
|
| <div style="text-align: right;">
|
| <div style="color: var(--success); font-weight: 600;">${Math.round(p.response_time_ms)}ms</div>
|
| <div style="font-size: 12px; color: var(--text-muted);">avg response</div>
|
| </div>
|
| </div>
|
| </div>
|
| `).join('') || '<div style="color: var(--text-muted);">No data available</div>';
|
|
|
|
|
| const problemProviders = providers.filter(p => p.status !== 'validated').slice(0, 5);
|
| document.getElementById('problem-resources').innerHTML = problemProviders.map(p => `
|
| <div style="padding: 12px; background: rgba(239, 68, 68, 0.1); backdrop-filter: blur(10px); border-radius: 8px; margin-bottom: 10px; border-left: 3px solid var(--danger);">
|
| <strong>${p.name}</strong>
|
| <div style="font-size: 12px; color: var(--text-muted); margin-top: 4px;">${p.provider_id}</div>
|
| <div style="font-size: 12px; color: var(--danger); margin-top: 4px;">Status: ${p.status}</div>
|
| </div>
|
| `).join('') || '<div style="color: var(--text-muted);">No issues detected ✅</div>';
|
| }
|
| } catch (error) {
|
| console.error('Error loading analytics:', error);
|
| }
|
| }
|
|
|
|
|
| async function loadLogs() {
|
| try {
|
| const response = await fetch('/api/logs/recent');
|
| if (response.ok) {
|
| const data = await response.json();
|
| const logs = data.logs || [];
|
|
|
| const container = document.getElementById('logs-container');
|
| if (logs.length === 0) {
|
| container.innerHTML = '<div style="color: var(--text-muted);">No logs available</div>';
|
| return;
|
| }
|
|
|
| container.innerHTML = logs.map(log => `
|
| <div style="padding: 8px; border-bottom: 1px solid var(--border); animation: slideIn 0.3s;">
|
| <span style="color: var(--text-muted);">[${log.timestamp || 'N/A'}]</span>
|
| <span style="color: ${log.level === 'ERROR' ? 'var(--danger)' : 'var(--text-light)'};">${log.message || JSON.stringify(log)}</span>
|
| </div>
|
| `).join('');
|
| } else {
|
| document.getElementById('logs-container').innerHTML = '<div style="color: var(--danger);">Failed to load logs</div>';
|
| }
|
| } catch (error) {
|
| console.error('Error loading logs:', error);
|
| document.getElementById('logs-container').innerHTML = '<div style="color: var(--danger);">Error loading logs: ' + error.message + '</div>';
|
| }
|
| }
|
|
|
|
|
| async function loadDiscoveryStats() {
|
| try {
|
| const response = await fetch('/api/apl/summary');
|
| if (response.ok) {
|
| const data = await response.json();
|
| document.getElementById('discovery-found').textContent = data.total_active_providers || 0;
|
| document.getElementById('discovery-validated').textContent = (data.http_valid || 0) + (data.hf_valid || 0);
|
| document.getElementById('discovery-failed').textContent = (data.http_invalid || 0) + (data.hf_invalid || 0);
|
|
|
| if (data.timestamp) {
|
| document.getElementById('discovery-last').textContent = new Date(data.timestamp).toLocaleTimeString();
|
| }
|
| }
|
| } catch (error) {
|
| console.error('Error loading discovery stats:', error);
|
| }
|
| }
|
|
|
|
|
| async function runFullDiscovery() {
|
| const btn = document.getElementById('discovery-btn');
|
| btn.disabled = true;
|
| btn.textContent = '⏳ Discovering...';
|
|
|
| document.getElementById('discovery-progress').style.display = 'block';
|
|
|
| try {
|
| let progress = 0;
|
| const progressInterval = setInterval(() => {
|
| progress += 5;
|
| if (progress <= 95) {
|
| document.getElementById('discovery-progress-bar').style.width = progress + '%';
|
| document.getElementById('discovery-percent').textContent = progress + '%';
|
| }
|
| }, 200);
|
|
|
| const response = await fetch('/api/apl/run', { method: 'POST' });
|
|
|
| clearInterval(progressInterval);
|
| document.getElementById('discovery-progress-bar').style.width = '100%';
|
| document.getElementById('discovery-percent').textContent = '100%';
|
|
|
| if (response.ok) {
|
| const result = await response.json();
|
| showToast('Discovery completed successfully!', 'success');
|
| loadDiscoveryStats();
|
| } else {
|
| showToast('Discovery failed', 'error');
|
| }
|
| } catch (error) {
|
| console.error('Error during discovery:', error);
|
| showToast('Error: ' + error.message, 'error');
|
| } finally {
|
| btn.disabled = false;
|
| btn.textContent = '🚀 Run Full Discovery';
|
| setTimeout(() => {
|
| document.getElementById('discovery-progress').style.display = 'none';
|
| }, 2000);
|
| }
|
| }
|
|
|
|
|
| async function runAPLScan() {
|
| showToast('Running APL scan...', 'info');
|
|
|
| try {
|
| const response = await fetch('/api/apl/run', { method: 'POST' });
|
|
|
| if (response.ok) {
|
| showToast('APL scan completed!', 'success');
|
| loadDiscoveryStats();
|
| loadDashboardData();
|
| } else {
|
| showToast('APL scan failed', 'error');
|
| }
|
| } catch (error) {
|
| console.error('Error running APL:', error);
|
| showToast('Error: ' + error.message, 'error');
|
| }
|
| }
|
|
|
|
|
| async function runDiagnostics(autoFix) {
|
| showToast('Running diagnostics...', 'info');
|
|
|
| try {
|
| const response = await fetch(`/api/diagnostics/run?auto_fix=${autoFix}`, { method: 'POST' });
|
|
|
| if (response.ok) {
|
| const result = await response.json();
|
|
|
| let html = `
|
| <div class="card" style="background: rgba(16, 185, 129, 0.1); margin-top: 20px;">
|
| <h3>Diagnostics Results</h3>
|
| <p><strong>Issues Found:</strong> ${result.issues_found || 0}</p>
|
| <p><strong>Status:</strong> ${result.status || 'completed'}</p>
|
| ${autoFix ? `<p><strong>Fixes Applied:</strong> ${result.fixes_applied?.length || 0}</p>` : ''}
|
| </div>
|
| `;
|
|
|
| document.getElementById('diagnostics-output').innerHTML = html;
|
| showToast('Diagnostics completed', 'success');
|
| } else {
|
| showToast('Diagnostics failed', 'error');
|
| }
|
| } catch (error) {
|
| console.error('Error running diagnostics:', error);
|
| showToast('Error: ' + error.message, 'error');
|
| }
|
| }
|
|
|
|
|
| function showToast(message, type = 'info') {
|
| const toast = document.getElementById('toast');
|
| const toastMessage = document.getElementById('toast-message');
|
|
|
| toast.className = `toast ${type}`;
|
| toastMessage.textContent = message;
|
| toast.classList.add('show');
|
|
|
| setTimeout(() => {
|
| toast.classList.remove('show');
|
| }, 3000);
|
| }
|
|
|
| function refreshAllData() {
|
| showToast('Refreshing all data...', 'info');
|
| loadDashboardData();
|
| loadResources();
|
| }
|
|
|
| function refreshAnalytics() {
|
| showToast('Refreshing analytics...', 'info');
|
| loadAnalytics();
|
| }
|
|
|
| function refreshLogs() {
|
| loadLogs();
|
| }
|
|
|
| function filterLogs() {
|
| loadLogs();
|
| }
|
|
|
| function scanResources() {
|
| showToast('Scanning resources...', 'info');
|
| loadResources();
|
| }
|
|
|
| function fixDuplicates() {
|
| if (!confirm('Remove duplicate resources?')) return;
|
| showToast('Removing duplicates...', 'info');
|
| }
|
|
|
| function openAddResourceModal() {
|
| document.getElementById('add-resource-modal').classList.add('show');
|
| }
|
|
|
| function closeAddResourceModal() {
|
| document.getElementById('add-resource-modal').classList.remove('show');
|
| }
|
|
|
| async function addResource() {
|
| showToast('Adding resource...', 'info');
|
| closeAddResourceModal();
|
| }
|
|
|
| function testResource(id) {
|
| showToast(`Testing resource: ${id}`, 'info');
|
| }
|
|
|
| function editResource(id) {
|
| showToast(`Edit resource: ${id}`, 'info');
|
| }
|
|
|
| async function removeResource(id) {
|
| if (!confirm(`Remove resource: ${id}?`)) return;
|
| showToast('Resource removed', 'success');
|
| loadResources();
|
| }
|
|
|
| function validateAllResources() {
|
| showToast('Validating all resources...', 'info');
|
| }
|
|
|
| function refreshAllResources() {
|
| loadResources();
|
| }
|
|
|
| function removeInvalidResources() {
|
| if (!confirm('Remove all invalid resources?')) return;
|
| showToast('Removing invalid resources...', 'info');
|
| }
|
|
|
| function exportResources() {
|
| showToast('Exporting configuration...', 'info');
|
| }
|
|
|
| function importResources() {
|
| showToast('Import configuration...', 'info');
|
| }
|
|
|
| function exportAnalytics() {
|
| showToast('Exporting analytics...', 'info');
|
| }
|
|
|
| function exportLogs() {
|
| showToast('Exporting logs...', 'info');
|
| }
|
|
|
| function clearLogs() {
|
| if (!confirm('Clear all logs?')) return;
|
| showToast('Logs cleared', 'success');
|
| }
|
|
|
| function testConnections() {
|
| showToast('Testing connections...', 'info');
|
| }
|
|
|
| function clearCache() {
|
| if (!confirm('Clear cache?')) return;
|
| showToast('Cache cleared', 'success');
|
| }
|
|
|
| function discoverHFModels() {
|
| runFullDiscovery();
|
| }
|
|
|
| function discoverAPIs() {
|
| runFullDiscovery();
|
| }
|
|
|
|
|
| function startAutoRefresh() {
|
| setInterval(() => {
|
| const activeTab = document.querySelector('.tab-content.active').id;
|
| if (activeTab === 'tab-dashboard') {
|
| loadDashboardData();
|
| }
|
| }, 30000);
|
| }
|
| </script>
|
| </body>
|
| </html>
|
|
|