mirror of
				https://github.com/house-of-vanity/OutFleet.git
				synced 2025-10-24 17:29:08 +00:00 
			
		
		
		
	
		
			
	
	
		
			1289 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
		
		
			
		
	
	
			1289 lines
		
	
	
		
			55 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
|   | <!DOCTYPE html> | ||
|  | <html lang="en"> | ||
|  | <head> | ||
|  |     <meta charset="UTF-8"> | ||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||
|  |     <title>Xray Admin - Test Interface</title> | ||
|  |     <style> | ||
|  |         body { font-family: Arial, sans-serif; margin: 20px; background: #f5f5f5; } | ||
|  |         .container { max-width: 1200px; margin: 0 auto; } | ||
|  |         .section { background: white; margin: 20px 0; padding: 20px; border-radius: 8px; box-shadow: 0 2px 4px rgba(0,0,0,0.1); } | ||
|  |         h1 { color: #333; } | ||
|  |         h2 { color: #666; border-bottom: 1px solid #eee; padding-bottom: 10px; } | ||
|  |         table { width: 100%; border-collapse: collapse; margin: 10px 0; } | ||
|  |         th, td { border: 1px solid #ddd; padding: 8px; text-align: left; } | ||
|  |         th { background: #f9f9f9; } | ||
|  |         .btn { padding: 6px 12px; margin: 2px; border: none; border-radius: 4px; cursor: pointer; } | ||
|  |         .btn-primary { background: #007bff; color: white; } | ||
|  |         .btn-success { background: #28a745; color: white; } | ||
|  |         .btn-danger { background: #dc3545; color: white; } | ||
|  |         .btn-secondary { background: #6c757d; color: white; } | ||
|  |         .form-group { margin: 10px 0; } | ||
|  |         .form-group label { display: block; margin-bottom: 5px; font-weight: bold; } | ||
|  |         .form-group input, .form-group select { width: 300px; padding: 8px; border: 1px solid #ddd; border-radius: 4px; } | ||
|  |         /* Toast notifications */ | ||
|  |         .toast-container { | ||
|  |             position: fixed; | ||
|  |             top: 20px; | ||
|  |             right: 20px; | ||
|  |             z-index: 9999; | ||
|  |             pointer-events: none; | ||
|  |         } | ||
|  |         .toast { | ||
|  |             background: white; | ||
|  |             border-radius: 8px; | ||
|  |             box-shadow: 0 4px 12px rgba(0,0,0,0.15); | ||
|  |             margin-bottom: 10px; | ||
|  |             padding: 16px 20px; | ||
|  |             min-width: 300px; | ||
|  |             max-width: 400px; | ||
|  |             position: relative; | ||
|  |             transform: translateX(100%); | ||
|  |             transition: transform 0.3s ease-in-out, opacity 0.3s ease-in-out; | ||
|  |             opacity: 0; | ||
|  |             pointer-events: auto; | ||
|  |             border-left: 4px solid #007bff; | ||
|  |         } | ||
|  |         .toast.show { | ||
|  |             transform: translateX(0); | ||
|  |             opacity: 1; | ||
|  |         } | ||
|  |         .toast.success { | ||
|  |             border-left-color: #28a745; | ||
|  |         } | ||
|  |         .toast.error { | ||
|  |             border-left-color: #dc3545; | ||
|  |         } | ||
|  |         .toast-header { | ||
|  |             display: flex; | ||
|  |             justify-content: space-between; | ||
|  |             align-items: center; | ||
|  |             margin-bottom: 8px; | ||
|  |         } | ||
|  |         .toast-title { | ||
|  |             font-weight: bold; | ||
|  |             color: #333; | ||
|  |         } | ||
|  |         .toast-close { | ||
|  |             background: none; | ||
|  |             border: none; | ||
|  |             font-size: 18px; | ||
|  |             color: #999; | ||
|  |             cursor: pointer; | ||
|  |             padding: 0; | ||
|  |             margin-left: 10px; | ||
|  |         } | ||
|  |         .toast-close:hover { | ||
|  |             color: #666; | ||
|  |         } | ||
|  |         .toast-body { | ||
|  |             color: #666; | ||
|  |             line-height: 1.4; | ||
|  |         } | ||
|  |         .toast.success .toast-title { | ||
|  |             color: #155724; | ||
|  |         } | ||
|  |         .toast.error .toast-title { | ||
|  |             color: #721c24; | ||
|  |         } | ||
|  |         .tabs { border-bottom: 1px solid #ddd; margin-bottom: 20px; } | ||
|  |         .tab { display: inline-block; padding: 10px 20px; cursor: pointer; border-bottom: 2px solid transparent; } | ||
|  |         .tab.active { border-bottom-color: #007bff; background: #f8f9fa; } | ||
|  |         .tab-content { display: none; } | ||
|  |         .tab-content.active { display: block; } | ||
|  |         .loading { text-align: center; padding: 20px; color: #666; } | ||
|  |          | ||
|  |         /* Modal styles */ | ||
|  |         .modal { | ||
|  |             display: none; | ||
|  |             position: fixed; | ||
|  |             z-index: 10000; | ||
|  |             left: 0; | ||
|  |             top: 0; | ||
|  |             width: 100%; | ||
|  |             height: 100%; | ||
|  |             background-color: rgba(0,0,0,0.5); | ||
|  |         } | ||
|  |         .modal.show { | ||
|  |             display: flex; | ||
|  |             align-items: center; | ||
|  |             justify-content: center; | ||
|  |         } | ||
|  |         .modal-content { | ||
|  |             background: white; | ||
|  |             border-radius: 8px; | ||
|  |             padding: 20px; | ||
|  |             max-width: 600px; | ||
|  |             width: 90%; | ||
|  |             max-height: 80vh; | ||
|  |             overflow-y: auto; | ||
|  |             position: relative; | ||
|  |         } | ||
|  |         .modal-header { | ||
|  |             display: flex; | ||
|  |             justify-content: space-between; | ||
|  |             align-items: center; | ||
|  |             margin-bottom: 20px; | ||
|  |             border-bottom: 1px solid #eee; | ||
|  |             padding-bottom: 10px; | ||
|  |         } | ||
|  |         .modal-title { | ||
|  |             font-size: 18px; | ||
|  |             font-weight: bold; | ||
|  |             color: #333; | ||
|  |         } | ||
|  |         .modal-close { | ||
|  |             background: none; | ||
|  |             border: none; | ||
|  |             font-size: 24px; | ||
|  |             color: #999; | ||
|  |             cursor: pointer; | ||
|  |         } | ||
|  |         .modal-close:hover { | ||
|  |             color: #666; | ||
|  |         } | ||
|  |         .modal-body { | ||
|  |             margin-bottom: 20px; | ||
|  |         } | ||
|  |         .modal-footer { | ||
|  |             display: flex; | ||
|  |             justify-content: flex-end; | ||
|  |             gap: 10px; | ||
|  |             border-top: 1px solid #eee; | ||
|  |             padding-top: 15px; | ||
|  |         } | ||
|  |         .cert-details { | ||
|  |             font-family: monospace; | ||
|  |             background: #f8f9fa; | ||
|  |             border: 1px solid #e9ecef; | ||
|  |             border-radius: 4px; | ||
|  |             padding: 15px; | ||
|  |             white-space: pre-wrap; | ||
|  |             overflow-x: auto; | ||
|  |             max-height: 300px; | ||
|  |         } | ||
|  |     </style> | ||
|  | </head> | ||
|  | <body> | ||
|  |     <div class="container"> | ||
|  |         <h1>Xray Admin Panel - Test Interface</h1> | ||
|  |          | ||
|  |         <!-- Toast notifications container --> | ||
|  |         <div class="toast-container" id="toastContainer"></div> | ||
|  |          | ||
|  |         <div class="tabs"> | ||
|  |             <div class="tab active" onclick="showTab('dashboard')">Dashboard</div> | ||
|  |             <div class="tab" onclick="showTab('servers')">Servers</div> | ||
|  |             <div class="tab" onclick="showTab('templates')">Inbound Templates</div> | ||
|  |             <div class="tab" onclick="showTab('certificates')">Certificates</div> | ||
|  |             <div class="tab" onclick="showTab('inbounds')">Inbound Binding</div> | ||
|  |             <div class="tab" onclick="showTab('users')">Users</div> | ||
|  |         </div> | ||
|  |          | ||
|  |         <!-- Dashboard --> | ||
|  |         <div id="dashboard" class="tab-content active"> | ||
|  |             <div class="section"> | ||
|  |                 <h2>Statistics</h2> | ||
|  |                 <p>Servers: <span id="serverCount">Loading...</span></p> | ||
|  |                 <p>Templates: <span id="templateCount">Loading...</span></p> | ||
|  |                 <p>Certificates: <span id="certCount">Loading...</span></p> | ||
|  |                 <p>Users: <span id="userCount">Loading...</span></p> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |          | ||
|  |         <!-- Servers --> | ||
|  |         <div id="servers" class="tab-content"> | ||
|  |             <div class="section"> | ||
|  |                 <h2>Add Server</h2> | ||
|  |                 <form id="serverForm"> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="serverName" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Hostname:</label> | ||
|  |                         <input type="text" id="serverHostname" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>gRPC Port:</label> | ||
|  |                         <input type="number" id="serverPort" value="2053"> | ||
|  |                     </div> | ||
|  |                     <button type="submit" class="btn btn-primary">Add Server</button> | ||
|  |                 </form> | ||
|  |             </div> | ||
|  |              | ||
|  |             <div class="section"> | ||
|  |                 <h2>Servers List</h2> | ||
|  |                 <div id="serversList" class="loading">Loading...</div> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |          | ||
|  |         <!-- Templates --> | ||
|  |         <div id="templates" class="tab-content"> | ||
|  |             <div class="section"> | ||
|  |                 <h2>Add Template</h2> | ||
|  |                 <form id="templateForm"> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="templateName" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Protocol:</label> | ||
|  |                         <select id="templateProtocol" required> | ||
|  |                             <option value="vless">VLESS</option> | ||
|  |                             <option value="vmess">VMess</option> | ||
|  |                             <option value="trojan">Trojan</option> | ||
|  |                             <option value="shadowsocks">Shadowsocks</option> | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Default Port:</label> | ||
|  |                         <input type="number" id="templatePort" value="443" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="templateTls"> Requires TLS | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Configuration Template:</label> | ||
|  |                         <textarea id="templateConfig" rows="6" style="width: 300px;">{ | ||
|  |   "protocol": "{{protocol}}", | ||
|  |   "port": {{port}}, | ||
|  |   "settings": { | ||
|  |     "clients": [] | ||
|  |   } | ||
|  | }</textarea> | ||
|  |                     </div> | ||
|  |                     <button type="submit" class="btn btn-primary">Add Template</button> | ||
|  |                 </form> | ||
|  |             </div> | ||
|  |              | ||
|  |             <div class="section"> | ||
|  |                 <h2>Templates List</h2> | ||
|  |                 <div id="templatesList" class="loading">Loading...</div> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |          | ||
|  |         <!-- Certificates --> | ||
|  |         <div id="certificates" class="tab-content"> | ||
|  |             <div class="section"> | ||
|  |                 <h2>Add Certificate</h2> | ||
|  |                 <form id="certificateForm"> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="certName" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Domain:</label> | ||
|  |                         <input type="text" id="certDomain" placeholder="example.com" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Certificate Type:</label> | ||
|  |                         <select id="certType" required> | ||
|  |                             <option value="self_signed">Self-Signed</option> | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="certAutoRenew" checked> Auto Renew | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                     <button type="submit" class="btn btn-primary">Generate Certificate</button> | ||
|  |                 </form> | ||
|  |             </div> | ||
|  |              | ||
|  |             <div class="section"> | ||
|  |                 <h2>Certificates List</h2> | ||
|  |                 <div id="certificatesList" class="loading">Loading...</div> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |          | ||
|  |         <!-- Server Inbounds --> | ||
|  |         <div id="inbounds" class="tab-content"> | ||
|  |             <div class="section"> | ||
|  |                 <h2>Bind Template to Server</h2> | ||
|  |                 <form id="inboundForm"> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Server:</label> | ||
|  |                         <select id="inboundServer" required> | ||
|  |                             <option value="">Select Server...</option> | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Template:</label> | ||
|  |                         <select id="inboundTemplate" required> | ||
|  |                             <option value="">Select Template...</option> | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Port:</label> | ||
|  |                         <input type="number" id="inboundPort" value="443" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Certificate:</label> | ||
|  |                         <select id="inboundCertificate"> | ||
|  |                             <option value="">No Certificate</option> | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="inboundActive" checked> Active | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                     <button type="submit" class="btn btn-primary">Bind Template</button> | ||
|  |                 </form> | ||
|  |             </div> | ||
|  |              | ||
|  |             <div class="section"> | ||
|  |                 <h2>Server Inbounds</h2> | ||
|  |                 <div id="inboundsList" class="loading">Loading...</div> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |          | ||
|  |         <!-- Users --> | ||
|  |         <div id="users" class="tab-content"> | ||
|  |             <div class="section"> | ||
|  |                 <h2>Add User</h2> | ||
|  |                 <form id="userForm"> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="userName" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Comment:</label> | ||
|  |                         <input type="text" id="userComment"> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Telegram ID:</label> | ||
|  |                         <input type="number" id="userTelegram"> | ||
|  |                     </div> | ||
|  |                     <button type="submit" class="btn btn-primary">Add User</button> | ||
|  |                 </form> | ||
|  |             </div> | ||
|  |              | ||
|  |             <div class="section"> | ||
|  |                 <h2>Users List</h2> | ||
|  |                 <div id="usersList" class="loading">Loading...</div> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |     </div> | ||
|  |      | ||
|  |     <!-- Modal dialogs --> | ||
|  |     <div id="editModal" class="modal"> | ||
|  |         <div class="modal-content"> | ||
|  |             <div class="modal-header"> | ||
|  |                 <div class="modal-title" id="editModalTitle">Edit Item</div> | ||
|  |                 <button class="modal-close" onclick="closeModal('editModal')">×</button> | ||
|  |             </div> | ||
|  |             <div class="modal-body" id="editModalBody"> | ||
|  |                 <!-- Content will be dynamically loaded --> | ||
|  |             </div> | ||
|  |             <div class="modal-footer"> | ||
|  |                 <button class="btn btn-secondary" onclick="closeModal('editModal')">Cancel</button> | ||
|  |                 <button class="btn btn-primary" id="saveEditBtn" onclick="saveEdit()">Save</button> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |     </div> | ||
|  |      | ||
|  |     <div id="viewModal" class="modal"> | ||
|  |         <div class="modal-content"> | ||
|  |             <div class="modal-header"> | ||
|  |                 <div class="modal-title" id="viewModalTitle">View Details</div> | ||
|  |                 <button class="modal-close" onclick="closeModal('viewModal')">×</button> | ||
|  |             </div> | ||
|  |             <div class="modal-body" id="viewModalBody"> | ||
|  |                 <!-- Content will be dynamically loaded --> | ||
|  |             </div> | ||
|  |             <div class="modal-footer"> | ||
|  |                 <button class="btn btn-secondary" onclick="closeModal('viewModal')">Close</button> | ||
|  |             </div> | ||
|  |         </div> | ||
|  |     </div> | ||
|  |      | ||
|  |     <script> | ||
|  |         const API_BASE = '/api'; | ||
|  |          | ||
|  |         // Tab switching | ||
|  |         function showTab(tabName) { | ||
|  |             document.querySelectorAll('.tab').forEach(tab => tab.classList.remove('active')); | ||
|  |             document.querySelectorAll('.tab-content').forEach(content => content.classList.remove('active')); | ||
|  |              | ||
|  |             event.target.classList.add('active'); | ||
|  |             document.getElementById(tabName).classList.add('active'); | ||
|  |              | ||
|  |             loadTabData(tabName); | ||
|  |         } | ||
|  |          | ||
|  |         function loadTabData(tab) { | ||
|  |             switch(tab) { | ||
|  |                 case 'dashboard': loadDashboard(); break; | ||
|  |                 case 'servers': loadServers(); break; | ||
|  |                 case 'templates': loadTemplates(); break; | ||
|  |                 case 'certificates': loadCertificates(); break; | ||
|  |                 case 'inbounds': loadInbounds(); break; | ||
|  |                 case 'users': loadUsers(); break; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Dashboard | ||
|  |         async function loadDashboard() { | ||
|  |             try { | ||
|  |                 const [servers, templates, certificates, users] = await Promise.all([ | ||
|  |                     fetch(`${API_BASE}/servers`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/templates`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/certificates`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/users`).then(r => r.json()) | ||
|  |                 ]); | ||
|  |                  | ||
|  |                 document.getElementById('serverCount').textContent = servers.length; | ||
|  |                 document.getElementById('templateCount').textContent = templates.length; | ||
|  |                 document.getElementById('certCount').textContent = certificates.length; | ||
|  |                 document.getElementById('userCount').textContent = users.users ? users.users.length : users.length; | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading dashboard: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Servers | ||
|  |         async function loadServers() { | ||
|  |             try { | ||
|  |                 const servers = await fetch(`${API_BASE}/servers`).then(r => r.json()); | ||
|  |                  | ||
|  |                 if (servers.length === 0) { | ||
|  |                     document.getElementById('serversList').innerHTML = '<p>No servers found</p>'; | ||
|  |                     return; | ||
|  |                 } | ||
|  |                  | ||
|  |                 const table = ` | ||
|  |                     <table> | ||
|  |                         <tr><th>Name</th><th>Hostname</th><th>Port</th><th>Status</th><th>Actions</th></tr> | ||
|  |                         ${servers.map(s => ` | ||
|  |                             <tr> | ||
|  |                                 <td>${s.name}</td> | ||
|  |                                 <td>${s.hostname}</td> | ||
|  |                                 <td>${s.grpc_port}</td> | ||
|  |                                 <td>${s.status}</td> | ||
|  |                                 <td> | ||
|  |                                     <button class="btn btn-success" onclick="testServer('${s.id}')">Test</button> | ||
|  |                                     <button class="btn btn-primary" onclick="editServer('${s.id}')">Edit</button> | ||
|  |                                     <button class="btn btn-danger" onclick="deleteServer('${s.id}')">Delete</button> | ||
|  |                                 </td> | ||
|  |                             </tr> | ||
|  |                         `).join('')} | ||
|  |                     </table> | ||
|  |                 `; | ||
|  |                  | ||
|  |                 document.getElementById('serversList').innerHTML = table; | ||
|  |             } catch (error) { | ||
|  |                 document.getElementById('serversList').innerHTML = '<p>Error: ' + error.message + '</p>'; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Templates | ||
|  |         async function loadTemplates() { | ||
|  |             try { | ||
|  |                 const templates = await fetch(`${API_BASE}/templates`).then(r => r.json()); | ||
|  |                  | ||
|  |                 if (templates.length === 0) { | ||
|  |                     document.getElementById('templatesList').innerHTML = '<p>No templates found</p>'; | ||
|  |                     return; | ||
|  |                 } | ||
|  |                  | ||
|  |                 const table = ` | ||
|  |                     <table> | ||
|  |                         <tr><th>Name</th><th>Protocol</th><th>Port</th><th>TLS</th><th>Active</th><th>Actions</th></tr> | ||
|  |                         ${templates.map(t => ` | ||
|  |                             <tr> | ||
|  |                                 <td>${t.name}</td> | ||
|  |                                 <td>${t.protocol}</td> | ||
|  |                                 <td>${t.default_port}</td> | ||
|  |                                 <td>${t.requires_tls ? 'Yes' : 'No'}</td> | ||
|  |                                 <td>${t.is_active ? 'Yes' : 'No'}</td> | ||
|  |                                 <td> | ||
|  |                                     <button class="btn btn-primary" onclick="editTemplate('${t.id}')">Edit</button> | ||
|  |                                     <button class="btn btn-danger" onclick="deleteTemplate('${t.id}')">Delete</button> | ||
|  |                                 </td> | ||
|  |                             </tr> | ||
|  |                         `).join('')} | ||
|  |                     </table> | ||
|  |                 `; | ||
|  |                  | ||
|  |                 document.getElementById('templatesList').innerHTML = table; | ||
|  |             } catch (error) { | ||
|  |                 document.getElementById('templatesList').innerHTML = '<p>Error: ' + error.message + '</p>'; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Certificates | ||
|  |         async function loadCertificates() { | ||
|  |             try { | ||
|  |                 const certificates = await fetch(`${API_BASE}/certificates`).then(r => r.json()); | ||
|  |                  | ||
|  |                 if (certificates.length === 0) { | ||
|  |                     document.getElementById('certificatesList').innerHTML = '<p>No certificates found</p>'; | ||
|  |                     return; | ||
|  |                 } | ||
|  |                  | ||
|  |                 const table = ` | ||
|  |                     <table> | ||
|  |                         <tr><th>Name</th><th>Domain</th><th>Type</th><th>Expires</th><th>Auto Renew</th><th>Actions</th></tr> | ||
|  |                         ${certificates.map(c => ` | ||
|  |                             <tr> | ||
|  |                                 <td>${c.name}</td> | ||
|  |                                 <td>${c.domain}</td> | ||
|  |                                 <td>${c.cert_type}</td> | ||
|  |                                 <td>${new Date(c.expires_at).toLocaleDateString()}</td> | ||
|  |                                 <td>${c.auto_renew ? 'Yes' : 'No'}</td> | ||
|  |                                 <td> | ||
|  |                                     <button class="btn btn-secondary" onclick="viewCertificate('${c.id}')">View</button> | ||
|  |                                     <button class="btn btn-primary" onclick="editCertificate('${c.id}')">Edit</button> | ||
|  |                                     <button class="btn btn-danger" onclick="deleteCertificate('${c.id}')">Delete</button> | ||
|  |                                 </td> | ||
|  |                             </tr> | ||
|  |                         `).join('')} | ||
|  |                     </table> | ||
|  |                 `; | ||
|  |                  | ||
|  |                 document.getElementById('certificatesList').innerHTML = table; | ||
|  |             } catch (error) { | ||
|  |                 document.getElementById('certificatesList').innerHTML = '<p>Error: ' + error.message + '</p>'; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Users | ||
|  |         async function loadUsers() { | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/users`); | ||
|  |                 const data = await response.json(); | ||
|  |                 const users = data.users || data; | ||
|  |                  | ||
|  |                 if (users.length === 0) { | ||
|  |                     document.getElementById('usersList').innerHTML = '<p>No users found</p>'; | ||
|  |                     return; | ||
|  |                 } | ||
|  |                  | ||
|  |                 const table = ` | ||
|  |                     <table> | ||
|  |                         <tr><th>Name</th><th>Comment</th><th>Telegram</th><th>Created</th><th>Actions</th></tr> | ||
|  |                         ${users.map(u => ` | ||
|  |                             <tr> | ||
|  |                                 <td>${u.name}</td> | ||
|  |                                 <td>${u.comment || '-'}</td> | ||
|  |                                 <td>${u.telegram_id || '-'}</td> | ||
|  |                                 <td>${new Date(u.created_at).toLocaleDateString()}</td> | ||
|  |                                 <td> | ||
|  |                                     <button class="btn btn-primary" onclick="editUser('${u.id}')">Edit</button> | ||
|  |                                     <button class="btn btn-danger" onclick="deleteUser('${u.id}')">Delete</button> | ||
|  |                                 </td> | ||
|  |                             </tr> | ||
|  |                         `).join('')} | ||
|  |                     </table> | ||
|  |                 `; | ||
|  |                  | ||
|  |                 document.getElementById('usersList').innerHTML = table; | ||
|  |             } catch (error) { | ||
|  |                 document.getElementById('usersList').innerHTML = '<p>Error: ' + error.message + '</p>'; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Server Inbounds | ||
|  |         async function loadInbounds() { | ||
|  |             try { | ||
|  |                 // Load dropdown options | ||
|  |                 await loadInboundOptions(); | ||
|  |                  | ||
|  |                 // Load data for lookups | ||
|  |                 const [servers, templates, certificates] = await Promise.all([ | ||
|  |                     fetch(`${API_BASE}/servers`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/templates`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/certificates`).then(r => r.json()) | ||
|  |                 ]); | ||
|  |                  | ||
|  |                 // Create lookup maps | ||
|  |                 const serverMap = Object.fromEntries(servers.map(s => [s.id, s])); | ||
|  |                 const templateMap = Object.fromEntries(templates.map(t => [t.id, t])); | ||
|  |                 const certificateMap = Object.fromEntries(certificates.map(c => [c.id, c])); | ||
|  |                  | ||
|  |                 // Load inbounds list (we'll fetch all servers and their inbounds) | ||
|  |                 let allInbounds = []; | ||
|  |                 for (const server of servers) { | ||
|  |                     try { | ||
|  |                         const inbounds = await fetch(`${API_BASE}/servers/${server.id}/inbounds`).then(r => r.json()); | ||
|  |                         allInbounds = allInbounds.concat(inbounds.map(inbound => ({ | ||
|  |                             ...inbound, | ||
|  |                             server_name: server.name, | ||
|  |                             server_id: server.id, | ||
|  |                             template_name: templateMap[inbound.template_id]?.name || 'Unknown', | ||
|  |                             certificate_name: inbound.certificate_id ? certificateMap[inbound.certificate_id]?.name || 'Unknown' : 'None' | ||
|  |                         }))); | ||
|  |                     } catch (e) { | ||
|  |                         // Skip if server has no inbounds | ||
|  |                     } | ||
|  |                 } | ||
|  |                  | ||
|  |                 if (allInbounds.length === 0) { | ||
|  |                     document.getElementById('inboundsList').innerHTML = '<p>No inbound bindings found</p>'; | ||
|  |                     return; | ||
|  |                 } | ||
|  |                  | ||
|  |                 const table = ` | ||
|  |                     <table> | ||
|  |                         <tr><th>Server</th><th>Template</th><th>Port</th><th>Certificate</th><th>Status</th><th>Actions</th></tr> | ||
|  |                         ${allInbounds.map(inbound => ` | ||
|  |                             <tr> | ||
|  |                                 <td>${inbound.server_name}</td> | ||
|  |                                 <td>${inbound.template_name}</td> | ||
|  |                                 <td>${inbound.port}</td> | ||
|  |                                 <td>${inbound.certificate_name}</td> | ||
|  |                                 <td>${inbound.is_active ? 'Active' : 'Inactive'}</td> | ||
|  |                                 <td> | ||
|  |                                     <button class="btn btn-primary" onclick="editInbound('${inbound.server_id}', '${inbound.id}')">Edit</button> | ||
|  |                                     <button class="btn btn-danger" onclick="deleteInbound('${inbound.server_id}', '${inbound.id}')">Delete</button> | ||
|  |                                 </td> | ||
|  |                             </tr> | ||
|  |                         `).join('')} | ||
|  |                     </table> | ||
|  |                 `; | ||
|  |                  | ||
|  |                 document.getElementById('inboundsList').innerHTML = table; | ||
|  |             } catch (error) { | ||
|  |                 document.getElementById('inboundsList').innerHTML = '<p>Error: ' + error.message + '</p>'; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function loadInboundOptions() { | ||
|  |             try { | ||
|  |                 const [servers, templates, certificates] = await Promise.all([ | ||
|  |                     fetch(`${API_BASE}/servers`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/templates`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/certificates`).then(r => r.json()) | ||
|  |                 ]); | ||
|  |                  | ||
|  |                 // Store templates data globally for port loading | ||
|  |                 templatesData = templates; | ||
|  |                  | ||
|  |                 // Populate server dropdown | ||
|  |                 const serverSelect = document.getElementById('inboundServer'); | ||
|  |                 serverSelect.innerHTML = '<option value="">Select Server...</option>'; | ||
|  |                 servers.forEach(server => { | ||
|  |                     serverSelect.innerHTML += `<option value="${server.id}">${server.name} (${server.hostname})</option>`; | ||
|  |                 }); | ||
|  |                  | ||
|  |                 // Populate template dropdown | ||
|  |                 const templateSelect = document.getElementById('inboundTemplate'); | ||
|  |                 templateSelect.innerHTML = '<option value="">Select Template...</option>'; | ||
|  |                 templates.forEach(template => { | ||
|  |                     templateSelect.innerHTML += `<option value="${template.id}">${template.name} (${template.protocol})</option>`; | ||
|  |                 }); | ||
|  |                  | ||
|  |                 // Add event listener for template change | ||
|  |                 templateSelect.removeEventListener('change', onTemplateChange); // Remove if exists | ||
|  |                 templateSelect.addEventListener('change', onTemplateChange); | ||
|  |                  | ||
|  |                 // Populate certificate dropdown | ||
|  |                 const certSelect = document.getElementById('inboundCertificate'); | ||
|  |                 certSelect.innerHTML = '<option value="">No Certificate</option>'; | ||
|  |                 certificates.forEach(cert => { | ||
|  |                     certSelect.innerHTML += `<option value="${cert.id}">${cert.name} (${cert.domain})</option>`; | ||
|  |                 }); | ||
|  |             } catch (error) { | ||
|  |                 console.error('Error loading inbound options:', error); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Template change handler for automatic port loading | ||
|  |         function onTemplateChange(event) { | ||
|  |             const templateId = event.target.value; | ||
|  |             const portInput = document.getElementById('inboundPort'); | ||
|  |              | ||
|  |             if (templateId && templatesData.length > 0) { | ||
|  |                 const selectedTemplate = templatesData.find(t => t.id === templateId); | ||
|  |                 if (selectedTemplate) { | ||
|  |                     portInput.value = selectedTemplate.default_port; | ||
|  |                     console.log('Auto-loaded port', selectedTemplate.default_port, 'from template', selectedTemplate.name); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Server form | ||
|  |         document.getElementById('serverForm').addEventListener('submit', async (e) => { | ||
|  |             e.preventDefault(); | ||
|  |              | ||
|  |             const data = { | ||
|  |                 name: document.getElementById('serverName').value, | ||
|  |                 hostname: document.getElementById('serverHostname').value, | ||
|  |                 grpc_port: parseInt(document.getElementById('serverPort').value) | ||
|  |             }; | ||
|  |              | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/servers`, { | ||
|  |                     method: 'POST', | ||
|  |                     headers: { 'Content-Type': 'application/json' }, | ||
|  |                     body: JSON.stringify(data) | ||
|  |                 }); | ||
|  |                  | ||
|  |                 if (response.ok) { | ||
|  |                     showStatus('Server added successfully', 'success'); | ||
|  |                     document.getElementById('serverForm').reset(); | ||
|  |                     loadServers(); | ||
|  |                 } else { | ||
|  |                     showStatus('Error adding server', 'error'); | ||
|  |                 } | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         }); | ||
|  |          | ||
|  |         // Template form | ||
|  |         document.getElementById('templateForm').addEventListener('submit', async (e) => { | ||
|  |             e.preventDefault(); | ||
|  |              | ||
|  |             const data = { | ||
|  |                 name: document.getElementById('templateName').value, | ||
|  |                 protocol: document.getElementById('templateProtocol').value, | ||
|  |                 default_port: parseInt(document.getElementById('templatePort').value), | ||
|  |                 requires_tls: document.getElementById('templateTls').checked, | ||
|  |                 config_template: document.getElementById('templateConfig').value | ||
|  |             }; | ||
|  |              | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/templates`, { | ||
|  |                     method: 'POST', | ||
|  |                     headers: { 'Content-Type': 'application/json' }, | ||
|  |                     body: JSON.stringify(data) | ||
|  |                 }); | ||
|  |                  | ||
|  |                 if (response.ok) { | ||
|  |                     showStatus('Template added successfully', 'success'); | ||
|  |                     document.getElementById('templateForm').reset(); | ||
|  |                     loadTemplates(); | ||
|  |                 } else { | ||
|  |                     showStatus('Error adding template', 'error'); | ||
|  |                 } | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         }); | ||
|  |          | ||
|  |         // Certificate form | ||
|  |         document.getElementById('certificateForm').addEventListener('submit', async (e) => { | ||
|  |             e.preventDefault(); | ||
|  |              | ||
|  |             const data = { | ||
|  |                 name: document.getElementById('certName').value, | ||
|  |                 domain: document.getElementById('certDomain').value, | ||
|  |                 cert_type: document.getElementById('certType').value, | ||
|  |                 auto_renew: document.getElementById('certAutoRenew').checked | ||
|  |             }; | ||
|  |              | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/certificates`, { | ||
|  |                     method: 'POST', | ||
|  |                     headers: { 'Content-Type': 'application/json' }, | ||
|  |                     body: JSON.stringify(data) | ||
|  |                 }); | ||
|  |                  | ||
|  |                 if (response.ok) { | ||
|  |                     showStatus('Certificate generated successfully', 'success'); | ||
|  |                     document.getElementById('certificateForm').reset(); | ||
|  |                     loadCertificates(); | ||
|  |                 } else { | ||
|  |                     showStatus('Error generating certificate', 'error'); | ||
|  |                 } | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         }); | ||
|  |          | ||
|  |         // Inbound form | ||
|  |         document.getElementById('inboundForm').addEventListener('submit', async (e) => { | ||
|  |             e.preventDefault(); | ||
|  |              | ||
|  |             const serverId = document.getElementById('inboundServer').value; | ||
|  |             const data = { | ||
|  |                 template_id: document.getElementById('inboundTemplate').value, | ||
|  |                 port: parseInt(document.getElementById('inboundPort').value), | ||
|  |                 certificate_id: document.getElementById('inboundCertificate').value || null, | ||
|  |                 is_active: document.getElementById('inboundActive').checked | ||
|  |             }; | ||
|  |              | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/servers/${serverId}/inbounds`, { | ||
|  |                     method: 'POST', | ||
|  |                     headers: { 'Content-Type': 'application/json' }, | ||
|  |                     body: JSON.stringify(data) | ||
|  |                 }); | ||
|  |                  | ||
|  |                 if (response.ok) { | ||
|  |                     showStatus('Server inbound created successfully', 'success'); | ||
|  |                     document.getElementById('inboundForm').reset(); | ||
|  |                     loadInbounds(); | ||
|  |                 } else { | ||
|  |                     showStatus('Error creating server inbound', 'error'); | ||
|  |                 } | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         }); | ||
|  |          | ||
|  |         // User form | ||
|  |         document.getElementById('userForm').addEventListener('submit', async (e) => { | ||
|  |             e.preventDefault(); | ||
|  |              | ||
|  |             const data = { | ||
|  |                 name: document.getElementById('userName').value | ||
|  |             }; | ||
|  |              | ||
|  |             const comment = document.getElementById('userComment').value; | ||
|  |             const telegram = document.getElementById('userTelegram').value; | ||
|  |              | ||
|  |             if (comment) data.comment = comment; | ||
|  |             if (telegram) data.telegram_id = parseInt(telegram); | ||
|  |              | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/users`, { | ||
|  |                     method: 'POST', | ||
|  |                     headers: { 'Content-Type': 'application/json' }, | ||
|  |                     body: JSON.stringify(data) | ||
|  |                 }); | ||
|  |                  | ||
|  |                 if (response.ok) { | ||
|  |                     showStatus('User added successfully', 'success'); | ||
|  |                     document.getElementById('userForm').reset(); | ||
|  |                     loadUsers(); | ||
|  |                 } else { | ||
|  |                     showStatus('Error adding user', 'error'); | ||
|  |                 } | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         }); | ||
|  |          | ||
|  |         // Actions | ||
|  |         async function testServer(id) { | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/servers/${id}/test`, { method: 'POST' }); | ||
|  |                 const result = await response.json(); | ||
|  |                 showStatus(result.connected ? 'Connection OK' : 'Connection failed', result.connected ? 'success' : 'error'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Test error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function deleteServer(id) { | ||
|  |             if (!confirm('Delete server?')) return; | ||
|  |              | ||
|  |             try { | ||
|  |                 await fetch(`${API_BASE}/servers/${id}`, { method: 'DELETE' }); | ||
|  |                 showStatus('Server deleted', 'success'); | ||
|  |                 loadServers(); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Delete error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function deleteUser(id) { | ||
|  |             if (!confirm('Delete user?')) return; | ||
|  |              | ||
|  |             try { | ||
|  |                 await fetch(`${API_BASE}/users/${id}`, { method: 'DELETE' }); | ||
|  |                 showStatus('User deleted', 'success'); | ||
|  |                 loadUsers(); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Delete error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function deleteTemplate(id) { | ||
|  |             if (!confirm('Delete template?')) return; | ||
|  |              | ||
|  |             try { | ||
|  |                 await fetch(`${API_BASE}/templates/${id}`, { method: 'DELETE' }); | ||
|  |                 showStatus('Template deleted', 'success'); | ||
|  |                 loadTemplates(); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Delete error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function deleteCertificate(id) { | ||
|  |             if (!confirm('Delete certificate?')) return; | ||
|  |              | ||
|  |             try { | ||
|  |                 await fetch(`${API_BASE}/certificates/${id}`, { method: 'DELETE' }); | ||
|  |                 showStatus('Certificate deleted', 'success'); | ||
|  |                 loadCertificates(); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Delete error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function deleteInbound(serverId, inboundId) { | ||
|  |             if (!confirm('Delete server inbound?')) return; | ||
|  |              | ||
|  |             try { | ||
|  |                 await fetch(`${API_BASE}/servers/${serverId}/inbounds/${inboundId}`, { method: 'DELETE' }); | ||
|  |                 showStatus('Server inbound deleted', 'success'); | ||
|  |                 loadInbounds(); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Delete error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         function showStatus(message, type) { | ||
|  |             showToast(message, type); | ||
|  |         } | ||
|  |          | ||
|  |         function showToast(message, type = 'info') { | ||
|  |             const container = document.getElementById('toastContainer'); | ||
|  |             const toastId = 'toast-' + Date.now(); | ||
|  |              | ||
|  |             const typeTitle = { | ||
|  |                 'success': 'Success', | ||
|  |                 'error': 'Error', | ||
|  |                 'info': 'Info' | ||
|  |             }; | ||
|  |              | ||
|  |             const toast = document.createElement('div'); | ||
|  |             toast.className = `toast ${type}`; | ||
|  |             toast.id = toastId; | ||
|  |             toast.innerHTML = ` | ||
|  |                 <div class="toast-header"> | ||
|  |                     <div class="toast-title">${typeTitle[type] || 'Notification'}</div> | ||
|  |                     <button class="toast-close" onclick="closeToast('${toastId}')">×</button> | ||
|  |                 </div> | ||
|  |                 <div class="toast-body">${message}</div> | ||
|  |             `; | ||
|  |              | ||
|  |             container.appendChild(toast); | ||
|  |              | ||
|  |             // Trigger animation | ||
|  |             setTimeout(() => { | ||
|  |                 toast.classList.add('show'); | ||
|  |             }, 10); | ||
|  |              | ||
|  |             // Auto-remove after 4 seconds | ||
|  |             setTimeout(() => { | ||
|  |                 closeToast(toastId); | ||
|  |             }, 4000); | ||
|  |         } | ||
|  |          | ||
|  |         function closeToast(toastId) { | ||
|  |             const toast = document.getElementById(toastId); | ||
|  |             if (toast) { | ||
|  |                 toast.classList.remove('show'); | ||
|  |                 setTimeout(() => { | ||
|  |                     if (toast.parentNode) { | ||
|  |                         toast.parentNode.removeChild(toast); | ||
|  |                     } | ||
|  |                 }, 300); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Modal management | ||
|  |         function showModal(modalId) { | ||
|  |             document.getElementById(modalId).classList.add('show'); | ||
|  |         } | ||
|  |          | ||
|  |         function closeModal(modalId) { | ||
|  |             document.getElementById(modalId).classList.remove('show'); | ||
|  |         } | ||
|  |          | ||
|  |         // Global variables for edit state | ||
|  |         let currentEditType = ''; | ||
|  |         let currentEditId = ''; | ||
|  |         let currentEditServerId = ''; | ||
|  |          | ||
|  |         // Global template data for port loading | ||
|  |         let templatesData = []; | ||
|  |          | ||
|  |         // Edit functions | ||
|  |         async function editServer(id) { | ||
|  |             try { | ||
|  |                 const server = await fetch(`${API_BASE}/servers/${id}`).then(r => r.json()); | ||
|  |                 currentEditType = 'server'; | ||
|  |                 currentEditId = id; | ||
|  |                  | ||
|  |                 document.getElementById('editModalTitle').textContent = 'Edit Server'; | ||
|  |                 document.getElementById('editModalBody').innerHTML = ` | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="editName" value="${server.name}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Hostname:</label> | ||
|  |                         <input type="text" id="editHostname" value="${server.hostname}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>gRPC Port:</label> | ||
|  |                         <input type="number" id="editPort" value="${server.grpc_port}" required> | ||
|  |                     </div> | ||
|  |                 `; | ||
|  |                 showModal('editModal'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading server: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function editUser(id) { | ||
|  |             try { | ||
|  |                 const user = await fetch(`${API_BASE}/users/${id}`).then(r => r.json()); | ||
|  |                 currentEditType = 'user'; | ||
|  |                 currentEditId = id; | ||
|  |                  | ||
|  |                 document.getElementById('editModalTitle').textContent = 'Edit User'; | ||
|  |                 document.getElementById('editModalBody').innerHTML = ` | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="editName" value="${user.name}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Comment:</label> | ||
|  |                         <input type="text" id="editComment" value="${user.comment || ''}"> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Telegram ID:</label> | ||
|  |                         <input type="number" id="editTelegram" value="${user.telegram_id || ''}"> | ||
|  |                     </div> | ||
|  |                 `; | ||
|  |                 showModal('editModal'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading user: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function editTemplate(id) { | ||
|  |             try { | ||
|  |                 const template = await fetch(`${API_BASE}/templates/${id}`).then(r => r.json()); | ||
|  |                 currentEditType = 'template'; | ||
|  |                 currentEditId = id; | ||
|  |                  | ||
|  |                 document.getElementById('editModalTitle').textContent = 'Edit Template'; | ||
|  |                 document.getElementById('editModalBody').innerHTML = ` | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="editName" value="${template.name}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Protocol:</label> | ||
|  |                         <select id="editProtocol" required> | ||
|  |                             <option value="vless" ${template.protocol === 'vless' ? 'selected' : ''}>VLESS</option> | ||
|  |                             <option value="vmess" ${template.protocol === 'vmess' ? 'selected' : ''}>VMess</option> | ||
|  |                             <option value="trojan" ${template.protocol === 'trojan' ? 'selected' : ''}>Trojan</option> | ||
|  |                             <option value="shadowsocks" ${template.protocol === 'shadowsocks' ? 'selected' : ''}>Shadowsocks</option> | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Default Port:</label> | ||
|  |                         <input type="number" id="editPort" value="${template.default_port}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="editTls" ${template.requires_tls ? 'checked' : ''}> Requires TLS | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="editActive" ${template.is_active ? 'checked' : ''}> Active | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                 `; | ||
|  |                 showModal('editModal'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading template: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function editCertificate(id) { | ||
|  |             try { | ||
|  |                 const cert = await fetch(`${API_BASE}/certificates/${id}`).then(r => r.json()); | ||
|  |                 currentEditType = 'certificate'; | ||
|  |                 currentEditId = id; | ||
|  |                  | ||
|  |                 document.getElementById('editModalTitle').textContent = 'Edit Certificate'; | ||
|  |                 document.getElementById('editModalBody').innerHTML = ` | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Name:</label> | ||
|  |                         <input type="text" id="editName" value="${cert.name}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Domain:</label> | ||
|  |                         <input type="text" id="editDomain" value="${cert.domain}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="editAutoRenew" ${cert.auto_renew ? 'checked' : ''}> Auto Renew | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                 `; | ||
|  |                 showModal('editModal'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading certificate: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function editInbound(serverId, inboundId) { | ||
|  |             try { | ||
|  |                 const response = await fetch(`${API_BASE}/servers/${serverId}/inbounds/${inboundId}`); | ||
|  |                 if (!response.ok) { | ||
|  |                     throw new Error(`HTTP ${response.status}: ${response.statusText}`); | ||
|  |                 } | ||
|  |                 const inbound = await response.json(); | ||
|  |                 const [templates, certificates] = await Promise.all([ | ||
|  |                     fetch(`${API_BASE}/templates`).then(r => r.json()), | ||
|  |                     fetch(`${API_BASE}/certificates`).then(r => r.json()) | ||
|  |                 ]); | ||
|  |                  | ||
|  |                 currentEditType = 'inbound'; | ||
|  |                 currentEditId = inboundId; | ||
|  |                 currentEditServerId = serverId; | ||
|  |                  | ||
|  |                 document.getElementById('editModalTitle').textContent = 'Edit Inbound Binding'; | ||
|  |                 document.getElementById('editModalBody').innerHTML = ` | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Template:</label> | ||
|  |                         <select id="editTemplate" required> | ||
|  |                             ${templates.map(t =>  | ||
|  |                                 `<option value="${t.id}" ${t.id === inbound.template_id ? 'selected' : ''}>${t.name}</option>` | ||
|  |                             ).join('')} | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Port:</label> | ||
|  |                         <input type="number" id="editPort" value="${inbound.port}" required> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label>Certificate:</label> | ||
|  |                         <select id="editCertificate"> | ||
|  |                             <option value="">No Certificate</option> | ||
|  |                             ${certificates.map(c =>  | ||
|  |                                 `<option value="${c.id}" ${c.id === inbound.certificate_id ? 'selected' : ''}>${c.name}</option>` | ||
|  |                             ).join('')} | ||
|  |                         </select> | ||
|  |                     </div> | ||
|  |                     <div class="form-group"> | ||
|  |                         <label> | ||
|  |                             <input type="checkbox" id="editActive" ${inbound.is_active ? 'checked' : ''}> Active | ||
|  |                         </label> | ||
|  |                     </div> | ||
|  |                 `; | ||
|  |                 showModal('editModal'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading inbound: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function viewCertificate(id) { | ||
|  |             try { | ||
|  |                 const cert = await fetch(`${API_BASE}/certificates/${id}/details`).then(r => r.json()); | ||
|  |                  | ||
|  |                 document.getElementById('viewModalTitle').textContent = 'Certificate Details'; | ||
|  |                 document.getElementById('viewModalBody').innerHTML = ` | ||
|  |                     <h4>Basic Information</h4> | ||
|  |                     <p><strong>Name:</strong> ${cert.name}</p> | ||
|  |                     <p><strong>Domain:</strong> ${cert.domain}</p> | ||
|  |                     <p><strong>Type:</strong> ${cert.cert_type}</p> | ||
|  |                     <p><strong>Auto Renew:</strong> ${cert.auto_renew ? 'Yes' : 'No'}</p> | ||
|  |                     <p><strong>Created:</strong> ${new Date(cert.created_at).toLocaleString()}</p> | ||
|  |                     <p><strong>Expires:</strong> ${new Date(cert.expires_at).toLocaleString()}</p> | ||
|  |                      | ||
|  |                     <h4>Certificate PEM</h4> | ||
|  |                     <div class="cert-details">${cert.certificate_pem || 'Not available'}</div> | ||
|  |                      | ||
|  |                     <h4>Private Key</h4> | ||
|  |                     <div class="cert-details">${cert.has_private_key ? '[Hidden for security]' : 'Not available'}</div> | ||
|  |                 `; | ||
|  |                 showModal('viewModal'); | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error loading certificate: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         async function saveEdit() { | ||
|  |             try { | ||
|  |                 let url, data; | ||
|  |                  | ||
|  |                 switch (currentEditType) { | ||
|  |                     case 'server': | ||
|  |                         url = `${API_BASE}/servers/${currentEditId}`; | ||
|  |                         data = { | ||
|  |                             name: document.getElementById('editName').value, | ||
|  |                             hostname: document.getElementById('editHostname').value, | ||
|  |                             grpc_port: parseInt(document.getElementById('editPort').value) | ||
|  |                         }; | ||
|  |                         break; | ||
|  |                          | ||
|  |                     case 'user': | ||
|  |                         url = `${API_BASE}/users/${currentEditId}`; | ||
|  |                         data = { | ||
|  |                             name: document.getElementById('editName').value, | ||
|  |                             comment: document.getElementById('editComment').value || null, | ||
|  |                             telegram_id: document.getElementById('editTelegram').value ? parseInt(document.getElementById('editTelegram').value) : null | ||
|  |                         }; | ||
|  |                         break; | ||
|  |                          | ||
|  |                     case 'template': | ||
|  |                         url = `${API_BASE}/templates/${currentEditId}`; | ||
|  |                         data = { | ||
|  |                             name: document.getElementById('editName').value, | ||
|  |                             protocol: document.getElementById('editProtocol').value, | ||
|  |                             default_port: parseInt(document.getElementById('editPort').value), | ||
|  |                             requires_tls: document.getElementById('editTls').checked, | ||
|  |                             is_active: document.getElementById('editActive').checked | ||
|  |                         }; | ||
|  |                         break; | ||
|  |                          | ||
|  |                     case 'certificate': | ||
|  |                         url = `${API_BASE}/certificates/${currentEditId}`; | ||
|  |                         data = { | ||
|  |                             name: document.getElementById('editName').value, | ||
|  |                             domain: document.getElementById('editDomain').value, | ||
|  |                             auto_renew: document.getElementById('editAutoRenew').checked | ||
|  |                         }; | ||
|  |                         break; | ||
|  |                          | ||
|  |                     case 'inbound': | ||
|  |                         url = `${API_BASE}/servers/${currentEditServerId}/inbounds/${currentEditId}`; | ||
|  |                         data = { | ||
|  |                             port_override: parseInt(document.getElementById('editPort').value), | ||
|  |                             certificate_id: document.getElementById('editCertificate').value || null, | ||
|  |                             is_active: document.getElementById('editActive').checked | ||
|  |                         }; | ||
|  |                         break; | ||
|  |                          | ||
|  |                     default: | ||
|  |                         throw new Error('Unknown edit type'); | ||
|  |                 } | ||
|  |                  | ||
|  |                 const response = await fetch(url, { | ||
|  |                     method: 'PUT', | ||
|  |                     headers: { 'Content-Type': 'application/json' }, | ||
|  |                     body: JSON.stringify(data) | ||
|  |                 }); | ||
|  |                  | ||
|  |                 if (response.ok) { | ||
|  |                     showStatus(`${currentEditType.charAt(0).toUpperCase() + currentEditType.slice(1)} updated successfully`, 'success'); | ||
|  |                     closeModal('editModal'); | ||
|  |                      | ||
|  |                     // Reload appropriate data | ||
|  |                     switch (currentEditType) { | ||
|  |                         case 'server': loadServers(); break; | ||
|  |                         case 'user': loadUsers(); break; | ||
|  |                         case 'template': loadTemplates(); break; | ||
|  |                         case 'certificate': loadCertificates(); break; | ||
|  |                         case 'inbound': loadInbounds(); break; | ||
|  |                     } | ||
|  |                 } else { | ||
|  |                     showStatus('Error updating ' + currentEditType, 'error'); | ||
|  |                 } | ||
|  |             } catch (error) { | ||
|  |                 showStatus('Error: ' + error.message, 'error'); | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         // Initialize | ||
|  |         loadDashboard(); | ||
|  |     </script> | ||
|  | </body> | ||
|  | </html> |