|
|
<!DOCTYPE html> |
|
|
<html lang="en"> |
|
|
<head> |
|
|
<meta charset="UTF-8"> |
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"> |
|
|
<title>Source Pool Management - Crypto API Monitor</title> |
|
|
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700;800&display=swap" rel="stylesheet"> |
|
|
<style> |
|
|
:root { |
|
|
--bg-primary: #f8fafc; |
|
|
--bg-secondary: #ffffff; |
|
|
--bg-card: #ffffff; |
|
|
--bg-hover: #f1f5f9; |
|
|
--text-primary: #0f172a; |
|
|
--text-secondary: #475569; |
|
|
--text-muted: #94a3b8; |
|
|
--accent-primary: #3b82f6; |
|
|
--accent-secondary: #8b5cf6; |
|
|
--success: #10b981; |
|
|
--success-bg: #d1fae5; |
|
|
--warning: #f59e0b; |
|
|
--warning-bg: #fef3c7; |
|
|
--danger: #ef4444; |
|
|
--danger-bg: #fee2e2; |
|
|
--info: #06b6d4; |
|
|
--info-bg: #cffafe; |
|
|
--border: #e2e8f0; |
|
|
--shadow: 0 4px 12px rgba(0, 0, 0, 0.08); |
|
|
--radius: 12px; |
|
|
} |
|
|
|
|
|
* { |
|
|
margin: 0; |
|
|
padding: 0; |
|
|
box-sizing: border-box; |
|
|
} |
|
|
|
|
|
body { |
|
|
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; |
|
|
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
|
|
color: var(--text-primary); |
|
|
line-height: 1.6; |
|
|
min-height: 100vh; |
|
|
padding: 20px; |
|
|
} |
|
|
|
|
|
.container { |
|
|
max-width: 1600px; |
|
|
margin: 0 auto; |
|
|
} |
|
|
|
|
|
.header { |
|
|
background: var(--bg-secondary); |
|
|
border-radius: var(--radius); |
|
|
padding: 24px; |
|
|
margin-bottom: 24px; |
|
|
box-shadow: var(--shadow); |
|
|
} |
|
|
|
|
|
.header h1 { |
|
|
font-size: 28px; |
|
|
font-weight: 800; |
|
|
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
|
|
-webkit-background-clip: text; |
|
|
-webkit-text-fill-color: transparent; |
|
|
margin-bottom: 8px; |
|
|
} |
|
|
|
|
|
.header p { |
|
|
color: var(--text-muted); |
|
|
font-size: 14px; |
|
|
} |
|
|
|
|
|
.header-actions { |
|
|
display: flex; |
|
|
gap: 12px; |
|
|
margin-top: 16px; |
|
|
} |
|
|
|
|
|
.btn { |
|
|
padding: 10px 20px; |
|
|
border-radius: 8px; |
|
|
border: none; |
|
|
font-weight: 600; |
|
|
cursor: pointer; |
|
|
transition: all 0.3s; |
|
|
font-size: 14px; |
|
|
} |
|
|
|
|
|
.btn-primary { |
|
|
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); |
|
|
color: white; |
|
|
} |
|
|
|
|
|
.btn-primary:hover { |
|
|
transform: translateY(-2px); |
|
|
box-shadow: 0 4px 12px rgba(102, 126, 234, 0.4); |
|
|
} |
|
|
|
|
|
.btn-secondary { |
|
|
background: var(--bg-hover); |
|
|
color: var(--text-primary); |
|
|
} |
|
|
|
|
|
.btn-danger { |
|
|
background: var(--danger); |
|
|
color: white; |
|
|
} |
|
|
|
|
|
.btn-sm { |
|
|
padding: 6px 12px; |
|
|
font-size: 12px; |
|
|
} |
|
|
|
|
|
.pools-grid { |
|
|
display: grid; |
|
|
grid-template-columns: repeat(auto-fill, minmax(400px, 1fr)); |
|
|
gap: 20px; |
|
|
margin-bottom: 24px; |
|
|
} |
|
|
|
|
|
.pool-card { |
|
|
background: var(--bg-card); |
|
|
border-radius: var(--radius); |
|
|
padding: 20px; |
|
|
box-shadow: var(--shadow); |
|
|
transition: all 0.3s; |
|
|
} |
|
|
|
|
|
.pool-card:hover { |
|
|
transform: translateY(-4px); |
|
|
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.12); |
|
|
} |
|
|
|
|
|
.pool-header { |
|
|
display: flex; |
|
|
justify-content: space-between; |
|
|
align-items: flex-start; |
|
|
margin-bottom: 16px; |
|
|
} |
|
|
|
|
|
.pool-title { |
|
|
font-size: 18px; |
|
|
font-weight: 700; |
|
|
color: var(--text-primary); |
|
|
margin-bottom: 4px; |
|
|
} |
|
|
|
|
|
.pool-category { |
|
|
display: inline-block; |
|
|
padding: 4px 12px; |
|
|
border-radius: 999px; |
|
|
background: var(--info-bg); |
|
|
color: var(--info); |
|
|
font-size: 12px; |
|
|
font-weight: 600; |
|
|
} |
|
|
|
|
|
.pool-stats { |
|
|
display: grid; |
|
|
grid-template-columns: repeat(2, 1fr); |
|
|
gap: 12px; |
|
|
margin: 16px 0; |
|
|
} |
|
|
|
|
|
.stat-item { |
|
|
background: var(--bg-hover); |
|
|
padding: 12px; |
|
|
border-radius: 8px; |
|
|
} |
|
|
|
|
|
.stat-label { |
|
|
font-size: 12px; |
|
|
color: var(--text-muted); |
|
|
margin-bottom: 4px; |
|
|
} |
|
|
|
|
|
.stat-value { |
|
|
font-size: 20px; |
|
|
font-weight: 700; |
|
|
color: var(--text-primary); |
|
|
} |
|
|
|
|
|
.pool-members { |
|
|
margin-top: 16px; |
|
|
} |
|
|
|
|
|
.member-item { |
|
|
display: flex; |
|
|
justify-content: space-between; |
|
|
align-items: center; |
|
|
padding: 10px; |
|
|
background: var(--bg-hover); |
|
|
border-radius: 8px; |
|
|
margin-bottom: 8px; |
|
|
} |
|
|
|
|
|
.member-name { |
|
|
font-weight: 600; |
|
|
color: var(--text-primary); |
|
|
} |
|
|
|
|
|
.member-stats { |
|
|
display: flex; |
|
|
gap: 12px; |
|
|
font-size: 12px; |
|
|
color: var(--text-muted); |
|
|
} |
|
|
|
|
|
.status-indicator { |
|
|
width: 10px; |
|
|
height: 10px; |
|
|
border-radius: 50%; |
|
|
display: inline-block; |
|
|
margin-right: 8px; |
|
|
} |
|
|
|
|
|
.status-online { |
|
|
background: var(--success); |
|
|
} |
|
|
|
|
|
.status-warning { |
|
|
background: var(--warning); |
|
|
} |
|
|
|
|
|
.status-offline { |
|
|
background: var(--danger); |
|
|
} |
|
|
|
|
|
.rotation-history { |
|
|
background: var(--bg-card); |
|
|
border-radius: var(--radius); |
|
|
padding: 20px; |
|
|
box-shadow: var(--shadow); |
|
|
} |
|
|
|
|
|
.history-item { |
|
|
display: flex; |
|
|
align-items: center; |
|
|
padding: 12px; |
|
|
border-left: 3px solid var(--accent-primary); |
|
|
background: var(--bg-hover); |
|
|
border-radius: 4px; |
|
|
margin-bottom: 12px; |
|
|
} |
|
|
|
|
|
.history-time { |
|
|
font-size: 12px; |
|
|
color: var(--text-muted); |
|
|
margin-bottom: 4px; |
|
|
} |
|
|
|
|
|
.history-desc { |
|
|
font-size: 14px; |
|
|
color: var(--text-primary); |
|
|
} |
|
|
|
|
|
.modal { |
|
|
display: none; |
|
|
position: fixed; |
|
|
top: 0; |
|
|
left: 0; |
|
|
width: 100%; |
|
|
height: 100%; |
|
|
background: rgba(0, 0, 0, 0.5); |
|
|
z-index: 1000; |
|
|
align-items: center; |
|
|
justify-content: center; |
|
|
} |
|
|
|
|
|
.modal.active { |
|
|
display: flex; |
|
|
} |
|
|
|
|
|
.modal-content { |
|
|
background: var(--bg-secondary); |
|
|
border-radius: var(--radius); |
|
|
padding: 32px; |
|
|
max-width: 600px; |
|
|
width: 90%; |
|
|
max-height: 90vh; |
|
|
overflow-y: auto; |
|
|
} |
|
|
|
|
|
.modal-header { |
|
|
font-size: 24px; |
|
|
font-weight: 700; |
|
|
margin-bottom: 24px; |
|
|
} |
|
|
|
|
|
.form-group { |
|
|
margin-bottom: 20px; |
|
|
} |
|
|
|
|
|
.form-label { |
|
|
display: block; |
|
|
font-weight: 600; |
|
|
margin-bottom: 8px; |
|
|
color: var(--text-primary); |
|
|
} |
|
|
|
|
|
.form-input, .form-select, .form-textarea { |
|
|
width: 100%; |
|
|
padding: 12px; |
|
|
border: 2px solid var(--border); |
|
|
border-radius: 8px; |
|
|
font-family: inherit; |
|
|
font-size: 14px; |
|
|
transition: all 0.3s; |
|
|
} |
|
|
|
|
|
.form-input:focus, .form-select:focus, .form-textarea:focus { |
|
|
outline: none; |
|
|
border-color: var(--accent-primary); |
|
|
} |
|
|
|
|
|
.form-textarea { |
|
|
resize: vertical; |
|
|
min-height: 100px; |
|
|
} |
|
|
|
|
|
.alert { |
|
|
padding: 12px 16px; |
|
|
border-radius: 8px; |
|
|
margin-bottom: 16px; |
|
|
font-size: 14px; |
|
|
} |
|
|
|
|
|
.alert-success { |
|
|
background: var(--success-bg); |
|
|
color: var(--success); |
|
|
border-left: 4px solid var(--success); |
|
|
} |
|
|
|
|
|
.alert-error { |
|
|
background: var(--danger-bg); |
|
|
color: var(--danger); |
|
|
border-left: 4px solid var(--danger); |
|
|
} |
|
|
|
|
|
.badge { |
|
|
display: inline-block; |
|
|
padding: 4px 10px; |
|
|
border-radius: 999px; |
|
|
font-size: 12px; |
|
|
font-weight: 600; |
|
|
} |
|
|
|
|
|
.badge-success { |
|
|
background: var(--success-bg); |
|
|
color: var(--success); |
|
|
} |
|
|
|
|
|
.badge-warning { |
|
|
background: var(--warning-bg); |
|
|
color: var(--warning); |
|
|
} |
|
|
|
|
|
.badge-danger { |
|
|
background: var(--danger-bg); |
|
|
color: var(--danger); |
|
|
} |
|
|
|
|
|
.rate-limit-bar { |
|
|
height: 8px; |
|
|
background: var(--bg-hover); |
|
|
border-radius: 4px; |
|
|
overflow: hidden; |
|
|
margin-top: 4px; |
|
|
} |
|
|
|
|
|
.rate-limit-fill { |
|
|
height: 100%; |
|
|
background: linear-gradient(90deg, var(--success) 0%, var(--warning) 50%, var(--danger) 100%); |
|
|
transition: width 0.3s; |
|
|
} |
|
|
</style> |
|
|
</head> |
|
|
<body> |
|
|
<div class="container"> |
|
|
<div class="header"> |
|
|
<h1>🔄 Source Pool Management</h1> |
|
|
<p>Intelligent API source rotation and failover management</p> |
|
|
<div class="header-actions"> |
|
|
<button class="btn btn-primary" onclick="showCreatePoolModal()">➕ Create New Pool</button> |
|
|
<button class="btn btn-secondary" onclick="loadPools()">🔄 Refresh</button> |
|
|
<button class="btn btn-secondary" onclick="window.location.href='index.html'">← Back to Dashboard</button> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
<div id="alertContainer"></div> |
|
|
|
|
|
<div id="poolsContainer" class="pools-grid"> |
|
|
|
|
|
</div> |
|
|
|
|
|
<div class="rotation-history" style="margin-top: 24px;"> |
|
|
<h2 style="margin-bottom: 16px;">Recent Rotation Events</h2> |
|
|
<div id="historyContainer"> |
|
|
|
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="createPoolModal" class="modal"> |
|
|
<div class="modal-content"> |
|
|
<h2 class="modal-header">Create New Source Pool</h2> |
|
|
<form id="createPoolForm"> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Pool Name</label> |
|
|
<input type="text" class="form-input" id="poolName" required> |
|
|
</div> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Category</label> |
|
|
<select class="form-select" id="poolCategory" required> |
|
|
<option value="market_data">Market Data</option> |
|
|
<option value="blockchain_explorers">Blockchain Explorers</option> |
|
|
<option value="news">News</option> |
|
|
<option value="sentiment">Sentiment</option> |
|
|
<option value="onchain_analytics">On-Chain Analytics</option> |
|
|
<option value="rpc_nodes">RPC Nodes</option> |
|
|
</select> |
|
|
</div> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Rotation Strategy</label> |
|
|
<select class="form-select" id="rotationStrategy" required> |
|
|
<option value="round_robin">Round Robin</option> |
|
|
<option value="least_used">Least Used</option> |
|
|
<option value="priority">Priority Based</option> |
|
|
<option value="weighted">Weighted</option> |
|
|
</select> |
|
|
</div> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Description</label> |
|
|
<textarea class="form-textarea" id="poolDescription"></textarea> |
|
|
</div> |
|
|
<div style="display: flex; gap: 12px; justify-content: flex-end;"> |
|
|
<button type="button" class="btn btn-secondary" onclick="closeCreatePoolModal()">Cancel</button> |
|
|
<button type="submit" class="btn btn-primary">Create Pool</button> |
|
|
</div> |
|
|
</form> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="addMemberModal" class="modal"> |
|
|
<div class="modal-content"> |
|
|
<h2 class="modal-header">Add Provider to Pool</h2> |
|
|
<form id="addMemberForm"> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Provider</label> |
|
|
<select class="form-select" id="memberProvider" required> |
|
|
|
|
|
</select> |
|
|
</div> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Priority (higher = better)</label> |
|
|
<input type="number" class="form-input" id="memberPriority" value="1" min="1" max="10"> |
|
|
</div> |
|
|
<div class="form-group"> |
|
|
<label class="form-label">Weight</label> |
|
|
<input type="number" class="form-input" id="memberWeight" value="1" min="1" max="100"> |
|
|
</div> |
|
|
<div style="display: flex; gap: 12px; justify-content: flex-end;"> |
|
|
<button type="button" class="btn btn-secondary" onclick="closeAddMemberModal()">Cancel</button> |
|
|
<button type="submit" class="btn btn-primary">Add Member</button> |
|
|
</div> |
|
|
</form> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
<script> |
|
|
const API_BASE = window.location.origin; |
|
|
let currentPoolId = null; |
|
|
let allProviders = []; |
|
|
|
|
|
|
|
|
document.addEventListener('DOMContentLoaded', () => { |
|
|
loadPools(); |
|
|
loadProviders(); |
|
|
}); |
|
|
|
|
|
async function loadPools() { |
|
|
try { |
|
|
const response = await fetch(`${API_BASE}/api/pools`); |
|
|
const data = await response.json(); |
|
|
|
|
|
const container = document.getElementById('poolsContainer'); |
|
|
container.innerHTML = ''; |
|
|
|
|
|
if (data.pools.length === 0) { |
|
|
container.innerHTML = '<p style="grid-column: 1/-1; text-align: center; color: var(--text-muted);">No pools configured. Create your first pool to get started.</p>'; |
|
|
return; |
|
|
} |
|
|
|
|
|
data.pools.forEach(pool => { |
|
|
container.appendChild(createPoolCard(pool)); |
|
|
}); |
|
|
} catch (error) { |
|
|
console.error('Error loading pools:', error); |
|
|
showAlert('Failed to load pools', 'error'); |
|
|
} |
|
|
} |
|
|
|
|
|
function createPoolCard(pool) { |
|
|
const card = document.createElement('div'); |
|
|
card.className = 'pool-card'; |
|
|
|
|
|
const currentProvider = pool.current_provider |
|
|
? `<div style="margin-bottom: 12px;"> |
|
|
<span class="status-indicator status-online"></span> |
|
|
Current: <strong>${pool.current_provider.name}</strong> |
|
|
</div>` |
|
|
: '<div style="margin-bottom: 12px; color: var(--text-muted);">No active provider</div>'; |
|
|
|
|
|
const membersHTML = pool.members.map(member => { |
|
|
const successRate = member.success_rate || 0; |
|
|
const statusClass = successRate >= 90 ? 'status-online' : successRate >= 70 ? 'status-warning' : 'status-offline'; |
|
|
|
|
|
let rateLimitHTML = ''; |
|
|
if (member.rate_limit) { |
|
|
const percentage = member.rate_limit.percentage; |
|
|
rateLimitHTML = ` |
|
|
<div style="margin-top: 4px;"> |
|
|
<div style="font-size: 11px; color: var(--text-muted);"> |
|
|
${member.rate_limit.usage}/${member.rate_limit.limit} (${percentage}%) |
|
|
</div> |
|
|
<div class="rate-limit-bar"> |
|
|
<div class="rate-limit-fill" style="width: ${percentage}%"></div> |
|
|
</div> |
|
|
</div> |
|
|
`; |
|
|
} |
|
|
|
|
|
return ` |
|
|
<div class="member-item"> |
|
|
<div> |
|
|
<span class="status-indicator ${statusClass}"></span> |
|
|
<span class="member-name">${member.provider_name}</span> |
|
|
<div class="member-stats"> |
|
|
<span>Used: ${member.use_count}</span> |
|
|
<span>Success: ${successRate.toFixed(1)}%</span> |
|
|
<span>Priority: ${member.priority}</span> |
|
|
</div> |
|
|
${rateLimitHTML} |
|
|
</div> |
|
|
</div> |
|
|
`; |
|
|
}).join(''); |
|
|
|
|
|
card.innerHTML = ` |
|
|
<div class="pool-header"> |
|
|
<div> |
|
|
<div class="pool-title">${pool.pool_name}</div> |
|
|
<span class="pool-category">${pool.category}</span> |
|
|
</div> |
|
|
<div style="display: flex; gap: 8px;"> |
|
|
<button class="btn btn-sm btn-secondary" onclick="addMember(${pool.pool_id})">➕</button> |
|
|
<button class="btn btn-sm btn-primary" onclick="rotatePool(${pool.pool_id})">🔄</button> |
|
|
<button class="btn btn-sm btn-danger" onclick="deletePool(${pool.pool_id}, '${pool.pool_name}')">🗑️</button> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
${currentProvider} |
|
|
|
|
|
<div class="pool-stats"> |
|
|
<div class="stat-item"> |
|
|
<div class="stat-label">Strategy</div> |
|
|
<div class="stat-value" style="font-size: 14px;">${pool.rotation_strategy}</div> |
|
|
</div> |
|
|
<div class="stat-item"> |
|
|
<div class="stat-label">Total Rotations</div> |
|
|
<div class="stat-value">${pool.total_rotations}</div> |
|
|
</div> |
|
|
<div class="stat-item"> |
|
|
<div class="stat-label">Members</div> |
|
|
<div class="stat-value">${pool.members.length}</div> |
|
|
</div> |
|
|
<div class="stat-item"> |
|
|
<div class="stat-label">Status</div> |
|
|
<div class="stat-value"> |
|
|
<span class="badge ${pool.enabled ? 'badge-success' : 'badge-danger'}"> |
|
|
${pool.enabled ? 'Enabled' : 'Disabled'} |
|
|
</span> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
<div class="pool-members"> |
|
|
<div style="font-weight: 600; margin-bottom: 12px;">Pool Members</div> |
|
|
${membersHTML || '<div style="color: var(--text-muted); font-size: 14px;">No members</div>'} |
|
|
</div> |
|
|
`; |
|
|
|
|
|
return card; |
|
|
} |
|
|
|
|
|
async function loadProviders() { |
|
|
try { |
|
|
const response = await fetch(`${API_BASE}/api/providers`); |
|
|
const providers = await response.json(); |
|
|
allProviders = providers; |
|
|
|
|
|
const select = document.getElementById('memberProvider'); |
|
|
select.innerHTML = providers.map(p => |
|
|
`<option value="${p.id}">${p.name} (${p.category})</option>` |
|
|
).join(''); |
|
|
} catch (error) { |
|
|
console.error('Error loading providers:', error); |
|
|
} |
|
|
} |
|
|
|
|
|
function showCreatePoolModal() { |
|
|
document.getElementById('createPoolModal').classList.add('active'); |
|
|
} |
|
|
|
|
|
function closeCreatePoolModal() { |
|
|
document.getElementById('createPoolModal').classList.remove('active'); |
|
|
document.getElementById('createPoolForm').reset(); |
|
|
} |
|
|
|
|
|
function showAddMemberModal() { |
|
|
document.getElementById('addMemberModal').classList.add('active'); |
|
|
} |
|
|
|
|
|
function closeAddMemberModal() { |
|
|
document.getElementById('addMemberModal').classList.remove('active'); |
|
|
document.getElementById('addMemberForm').reset(); |
|
|
} |
|
|
|
|
|
function addMember(poolId) { |
|
|
currentPoolId = poolId; |
|
|
showAddMemberModal(); |
|
|
} |
|
|
|
|
|
document.getElementById('createPoolForm').addEventListener('submit', async (e) => { |
|
|
e.preventDefault(); |
|
|
|
|
|
const data = { |
|
|
name: document.getElementById('poolName').value, |
|
|
category: document.getElementById('poolCategory').value, |
|
|
rotation_strategy: document.getElementById('rotationStrategy').value, |
|
|
description: document.getElementById('poolDescription').value |
|
|
}; |
|
|
|
|
|
try { |
|
|
const response = await fetch(`${API_BASE}/api/pools`, { |
|
|
method: 'POST', |
|
|
headers: { 'Content-Type': 'application/json' }, |
|
|
body: JSON.stringify(data) |
|
|
}); |
|
|
|
|
|
if (response.ok) { |
|
|
showAlert('Pool created successfully', 'success'); |
|
|
closeCreatePoolModal(); |
|
|
loadPools(); |
|
|
} else { |
|
|
const error = await response.json(); |
|
|
showAlert(error.detail || 'Failed to create pool', 'error'); |
|
|
} |
|
|
} catch (error) { |
|
|
console.error('Error creating pool:', error); |
|
|
showAlert('Failed to create pool', 'error'); |
|
|
} |
|
|
}); |
|
|
|
|
|
document.getElementById('addMemberForm').addEventListener('submit', async (e) => { |
|
|
e.preventDefault(); |
|
|
|
|
|
const data = { |
|
|
provider_id: parseInt(document.getElementById('memberProvider').value), |
|
|
priority: parseInt(document.getElementById('memberPriority').value), |
|
|
weight: parseInt(document.getElementById('memberWeight').value) |
|
|
}; |
|
|
|
|
|
try { |
|
|
const response = await fetch(`${API_BASE}/api/pools/${currentPoolId}/members`, { |
|
|
method: 'POST', |
|
|
headers: { 'Content-Type': 'application/json' }, |
|
|
body: JSON.stringify(data) |
|
|
}); |
|
|
|
|
|
if (response.ok) { |
|
|
showAlert('Member added successfully', 'success'); |
|
|
closeAddMemberModal(); |
|
|
loadPools(); |
|
|
} else { |
|
|
const error = await response.json(); |
|
|
showAlert(error.detail || 'Failed to add member', 'error'); |
|
|
} |
|
|
} catch (error) { |
|
|
console.error('Error adding member:', error); |
|
|
showAlert('Failed to add member', 'error'); |
|
|
} |
|
|
}); |
|
|
|
|
|
async function rotatePool(poolId) { |
|
|
try { |
|
|
const response = await fetch(`${API_BASE}/api/pools/${poolId}/rotate`, { |
|
|
method: 'POST', |
|
|
headers: { 'Content-Type': 'application/json' }, |
|
|
body: JSON.stringify({ reason: 'manual' }) |
|
|
}); |
|
|
|
|
|
if (response.ok) { |
|
|
const result = await response.json(); |
|
|
showAlert(`Rotated to ${result.provider_name}`, 'success'); |
|
|
loadPools(); |
|
|
} else { |
|
|
const error = await response.json(); |
|
|
showAlert(error.detail || 'Failed to rotate', 'error'); |
|
|
} |
|
|
} catch (error) { |
|
|
console.error('Error rotating pool:', error); |
|
|
showAlert('Failed to rotate pool', 'error'); |
|
|
} |
|
|
} |
|
|
|
|
|
async function deletePool(poolId, poolName) { |
|
|
if (!confirm(`Are you sure you want to delete pool "${poolName}"?`)) { |
|
|
return; |
|
|
} |
|
|
|
|
|
try { |
|
|
const response = await fetch(`${API_BASE}/api/pools/${poolId}`, { |
|
|
method: 'DELETE' |
|
|
}); |
|
|
|
|
|
if (response.ok) { |
|
|
showAlert('Pool deleted successfully', 'success'); |
|
|
loadPools(); |
|
|
} else { |
|
|
const error = await response.json(); |
|
|
showAlert(error.detail || 'Failed to delete pool', 'error'); |
|
|
} |
|
|
} catch (error) { |
|
|
console.error('Error deleting pool:', error); |
|
|
showAlert('Failed to delete pool', 'error'); |
|
|
} |
|
|
} |
|
|
|
|
|
function showAlert(message, type) { |
|
|
const container = document.getElementById('alertContainer'); |
|
|
const alert = document.createElement('div'); |
|
|
alert.className = `alert alert-${type}`; |
|
|
alert.textContent = message; |
|
|
container.appendChild(alert); |
|
|
|
|
|
setTimeout(() => { |
|
|
alert.remove(); |
|
|
}, 5000); |
|
|
} |
|
|
|
|
|
|
|
|
document.querySelectorAll('.modal').forEach(modal => { |
|
|
modal.addEventListener('click', (e) => { |
|
|
if (e.target === modal) { |
|
|
modal.classList.remove('active'); |
|
|
} |
|
|
}); |
|
|
}); |
|
|
</script> |
|
|
</body> |
|
|
</html> |
|
|
|