| {% extends "base.html" %} {% block title %}错误日志管理 - Gemini Balance{% |
| endblock %} {% block head_extra_styles %} |
| <style> |
| |
| .styled-table th { |
| position: sticky; |
| top: 0; |
| background-color: rgba(249, 250, 251, 0.95) !important; |
| color: #374151 !important; |
| z-index: 10; |
| border-bottom: 1px solid rgba(0, 0, 0, 0.08); |
| } |
| .styled-table tbody tr:hover { |
| background-color: rgba(59, 130, 246, 0.05) !important; |
| } |
| .styled-table td { |
| padding: 12px 20px; |
| vertical-align: middle; |
| white-space: nowrap; |
| overflow: hidden; |
| text-overflow: ellipsis; |
| max-width: 250px; |
| color: #d1d5db; |
| border-bottom: 1px solid rgba(0, 0, 0, 0.08); |
| } |
| .styled-table td:nth-child(4) { |
| white-space: nowrap; |
| } |
| .btn-view-details { |
| background-color: #3b82f6 !important; |
| color: #ffffff !important; |
| padding: 6px 12px; |
| border-radius: 6px; |
| font-weight: 500; |
| transition: all 0.2s ease-in-out; |
| border: 1px solid #2563eb !important; |
| } |
| .btn-view-details:hover { |
| background-color: #2563eb !important; |
| color: #ffffff !important; |
| box-shadow: 0 4px 8px rgba(59, 130, 246, 0.3) !important; |
| transform: translateY(-1px) !important; |
| } |
| @media (max-width: 768px) { |
| .search-container { |
| grid-template-columns: 1fr; |
| } |
| } |
| |
| input[type="text"], |
| input[type="datetime-local"], |
| select, |
| button { |
| height: 36px !important; |
| } |
| .form-input-themed { |
| background-color: rgba(255, 255, 255, 0.95) !important; |
| border: 1px solid rgba(0, 0, 0, 0.12) !important; |
| color: #374151 !important; |
| box-shadow: none !important; |
| outline: none !important; |
| } |
| |
| input[type="datetime-local"], |
| select#pageSize { |
| background-color: rgba(255, 255, 255, 0.95) !important; |
| border: 1px solid rgba(0, 0, 0, 0.12) !important; |
| color: #374151 !important; |
| box-shadow: none !important; |
| outline: none !important; |
| } |
| .form-input-themed::placeholder, |
| input[type="datetime-local"]::placeholder { |
| color: #9ca3af !important; |
| } |
| .form-input-themed:focus, |
| input[type="datetime-local"]:focus, |
| select#pageSize:focus { |
| border-color: #3b82f6 !important; |
| box-shadow: none !important; |
| outline: none !important; |
| } |
| select#pageSize { |
| |
| background-color: rgba(255, 255, 255, 0.95) !important; |
| border: 1px solid rgba(0, 0, 0, 0.12) !important; |
| color: #374151 !important; |
| background-image: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 20 20'%3e%3cpath stroke='%236b7280' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M6 8l4 4 4-4'/%3e%3c/svg%3e") !important; |
| appearance: none !important; |
| padding: 0.5rem 2.5rem 0.5rem 0.75rem !important; |
| background-repeat: no-repeat !important; |
| background-position: right 0.6rem center !important; |
| background-size: 1.2em 1.2em !important; |
| border-radius: 0.375rem !important; |
| font-weight: 500 !important; |
| height: auto !important; |
| min-height: 2.25rem !important; |
| line-height: 1.25 !important; |
| display: inline-flex !important; |
| align-items: center !important; |
| vertical-align: middle !important; |
| box-shadow: none !important; |
| cursor: pointer !important; |
| } |
| |
| select#pageSize:focus { |
| border-color: #3b82f6 !important; |
| box-shadow: none !important; |
| outline: none !important; |
| } |
| |
| select#pageSize option { |
| background-color: rgba(255, 255, 255, 0.98) !important; |
| color: #374151 !important; |
| padding: 8px !important; |
| font-weight: 500 !important; |
| } |
| |
| .date-range-container { |
| display: flex; |
| align-items: center; |
| gap: 0.5rem; |
| } |
| |
| @media (max-width: 640px) { |
| input[type="datetime-local"] { |
| min-width: 0; |
| width: 100%; |
| } |
| } |
| label { |
| color: #e2e8f0 !important; |
| font-weight: 500; |
| } |
| |
| |
| .nav-link { |
| transition: all 0.2s ease-in-out; |
| } |
| |
| .nav-link:hover { |
| background-color: rgba(59, 130, 246, 0.1) !important; |
| transform: translateY(-2px); |
| box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); |
| } |
| |
| |
| .rounded-2xl[style*="background-color: rgba(80, 60, 160"], |
| .rounded-xl[style*="background-color: rgba(70, 50, 150"] { |
| background-color: rgba(255, 255, 255, 0.95) !important; |
| border-color: rgba(0, 0, 0, 0.08) !important; |
| } |
| |
| |
| .nav-link { |
| background-color: rgba(229, 231, 235, 0.8) !important; |
| color: #374151 !important; |
| transition: all 0.2s ease-in-out; |
| position: relative; |
| z-index: 1; |
| } |
| |
| .nav-link:hover { |
| background-color: rgba(59, 130, 246, 0.1) !important; |
| transform: scale(1.02); |
| box-shadow: 0 6px 12px rgba(0, 0, 0, 0.15); |
| } |
| |
| |
| .nav-buttons-container { |
| padding-top: 0.5rem; |
| padding-bottom: 0.75rem; |
| } |
| |
| |
| .main-nav-btn:hover { |
| transform: scale(1.02) !important; |
| box-shadow: 0 8px 16px rgba(59, 130, 246, 0.3) !important; |
| } |
| |
| .bg-violet-600 { |
| background-color: #3b82f6 !important; |
| } |
| |
| |
| .text-transparent.bg-clip-text.bg-gradient-to-r.from-violet-400.to-pink-400 { |
| background: none !important; |
| color: #1f2937 !important; |
| -webkit-background-clip: unset !important; |
| background-clip: unset !important; |
| } |
| |
| |
| .text-gray-100 { |
| color: #1f2937 !important; |
| } |
| |
| .border-violet-300 { |
| border-color: #d1d5db !important; |
| } |
| |
| .text-violet-400 { |
| color: #3b82f6 !important; |
| } |
| |
| |
| .pagination-text, .text-gray-300 { |
| color: #374151 !important; |
| font-weight: 500; |
| } |
| |
| |
| .bg-violet-600, .bg-violet-700 { |
| background-color: #3b82f6 !important; |
| } |
| |
| .hover\\:bg-violet-700:hover { |
| background-color: #2563eb !important; |
| } |
| |
| |
| .bg-blue-600, button.bg-blue-600 { |
| background-color: #3b82f6 !important; |
| } |
| |
| .bg-blue-600:hover, button.bg-blue-600:hover, |
| .hover\\:bg-blue-700:hover { |
| background-color: #2563eb !important; |
| } |
| |
| |
| .bg-red-600, button.bg-red-600, |
| .bg-red-700, button.bg-red-700, |
| .bg-red-800, button.bg-red-800 { |
| background-color: #f87171 !important; |
| } |
| |
| .bg-red-600:hover, button.bg-red-600:hover, |
| .bg-red-700:hover, button.bg-red-700:hover, |
| .bg-red-800:hover, button.bg-red-800:hover, |
| .hover\\:bg-red-700:hover, .hover\\:bg-red-800:hover { |
| background-color: #ef4444 !important; |
| } |
| |
| |
| .text-violet-500 { |
| color: #3b82f6 !important; |
| } |
| |
| .focus\\:ring-violet-500:focus { |
| --tw-ring-color: rgba(59, 130, 246, 0.2) !important; |
| } |
| |
| .focus\\:border-violet-400:focus { |
| border-color: #3b82f6 !important; |
| } |
| |
| |
| .border-violet-400 { |
| border-color: #3b82f6 !important; |
| } |
| |
| |
| ul.pagination a { |
| |
| display: inline-flex; |
| align-items: center; |
| justify-content: center; |
| |
| |
| background-color: rgba(255, 255, 255, 0.9) !important; |
| color: #374151 !important; |
| border: 1px solid rgba(0, 0, 0, 0.08) !important; |
| text-shadow: none; |
| min-width: 36px; |
| text-align: center; |
| |
| transition: background-color 0.15s ease-in-out, |
| border-color 0.15s ease-in-out, color 0.15s ease-in-out; |
| } |
| |
| ul.pagination a:hover:not(.active):not(.disabled) { |
| |
| background-color: rgba(229, 231, 235, 1) !important; |
| border-color: rgba(0, 0, 0, 0.12) !important; |
| color: #374151 !important; |
| } |
| |
| ul.pagination a.active { |
| |
| background-color: #3b82f6 !important; |
| border-color: #2563eb !important; |
| color: #ffffff !important; |
| font-weight: 600; |
| cursor: default; |
| } |
| |
| ul.pagination a.disabled { |
| |
| background-color: rgba(249, 250, 251, 0.5) !important; |
| color: rgba(156, 163, 175, 0.8) !important; |
| border-color: rgba(0, 0, 0, 0.04) !important; |
| cursor: not-allowed; |
| pointer-events: none; |
| text-shadow: none; |
| } |
| |
| |
| .modal .w-full.max-w-6xl[style*="background-color: rgba(70, 50, 150"], |
| .modal .w-full.max-w-md[style*="background-color: rgba(70, 50, 150"] { |
| background-color: rgba(255, 255, 255, 0.98) !important; |
| color: #374151 !important; |
| border-color: rgba(0, 0, 0, 0.08) !important; |
| box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1), 0 10px 10px -5px rgba(0, 0, 0, 0.04) !important; |
| } |
| |
| |
| .modal .text-gray-100, .modal h2.text-gray-100 { |
| color: #1f2937 !important; |
| font-weight: 600 !important; |
| } |
| |
| |
| .modal .text-violet-200, .modal h6.text-violet-200 { |
| color: #3b82f6 !important; |
| font-weight: 600 !important; |
| } |
| |
| |
| .modal .text-gray-300 { |
| color: #6b7280 !important; |
| } |
| |
| .modal .text-gray-300:hover { |
| color: #374151 !important; |
| } |
| |
| |
| .modal [style*="background-color: rgba(80, 60, 160"] { |
| background-color: rgba(249, 250, 251, 0.95) !important; |
| border: 1px solid rgba(0, 0, 0, 0.08) !important; |
| } |
| |
| |
| .modal pre, .modal code { |
| background-color: rgba(243, 244, 246, 0.95) !important; |
| color: #374151 !important; |
| border: 1px solid rgba(0, 0, 0, 0.08) !important; |
| } |
| |
| |
| .modal .text-gray-200, .modal p.text-gray-200 { |
| color: #374151 !important; |
| } |
| |
| |
| .modal .text-red-300 { |
| color: #dc2626 !important; |
| font-weight: 500 !important; |
| } |
| |
| |
| .modal .bg-gray-500, .modal button.bg-gray-500 { |
| background-color: #6b7280 !important; |
| color: #ffffff !important; |
| border: 1px solid #4b5563 !important; |
| } |
| |
| .modal .bg-gray-500:hover, .modal button.bg-gray-500:hover { |
| background-color: #4b5563 !important; |
| transform: translateY(-1px) !important; |
| box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1) !important; |
| } |
| |
| |
| .modal .bg-red-600, .modal button.bg-red-600 { |
| background-color: #dc2626 !important; |
| color: #ffffff !important; |
| border: 1px solid #b91c1c !important; |
| } |
| |
| .modal .bg-red-600:hover, .modal button.bg-red-600:hover { |
| background-color: #b91c1c !important; |
| transform: translateY(-1px) !important; |
| box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1) !important; |
| } |
| |
| |
| .modal .copy-btn { |
| background-color: rgba(107, 114, 128, 0.8) !important; |
| color: #ffffff !important; |
| border: 1px solid rgba(75, 85, 99, 0.8) !important; |
| } |
| |
| .modal .copy-btn:hover { |
| background-color: rgba(75, 85, 99, 0.9) !important; |
| transform: translateY(-1px) !important; |
| } |
| |
| |
| .styled-table tbody td { |
| color: #374151 !important; |
| } |
| |
| .styled-table tbody tr:hover td { |
| color: #1f2937 !important; |
| } |
| |
| |
| .styled-table thead th { |
| color: #374151 !important; |
| } |
| |
| |
| .text-gray-300 { |
| color: #6b7280 !important; |
| } |
| |
| .text-gray-400 { |
| color: #9ca3af !important; |
| } |
| |
| |
| .pagination-text { |
| color: #374151 !important; |
| display: inline-flex !important; |
| align-items: center !important; |
| line-height: 1.25 !important; |
| vertical-align: middle !important; |
| } |
| |
| |
| .pagination-container { |
| display: flex !important; |
| align-items: center !important; |
| gap: 0.5rem !important; |
| } |
| |
| .pagination-container > * { |
| display: inline-flex !important; |
| align-items: center !important; |
| vertical-align: middle !important; |
| } |
| |
| |
| .bg-blue-500, .bg-blue-600, .bg-blue-700, |
| .bg-red-500, .bg-red-600, .bg-red-700, .bg-red-800, |
| .bg-green-500, .bg-green-600, .bg-green-700, |
| .bg-sky-500, .bg-sky-600, .bg-sky-700, |
| .bg-purple-500, .bg-purple-600, .bg-purple-700, |
| .bg-violet-500, .bg-violet-600, .bg-violet-700 { |
| color: #ffffff !important; |
| } |
| |
| |
| .bg-blue-500 *, .bg-blue-600 *, .bg-blue-700 *, |
| .bg-red-500 *, .bg-red-600 *, .bg-red-700 *, .bg-red-800 *, |
| .bg-green-500 *, .bg-green-600 *, .bg-green-700 *, |
| .bg-sky-500 *, .bg-sky-600 *, .bg-sky-700 *, |
| .bg-purple-500 *, .bg-purple-600 *, .bg-purple-700 *, |
| .bg-violet-500 *, .bg-violet-600 *, .bg-violet-700 * { |
| color: inherit !important; |
| } |
| |
| |
| .focus\\:border-primary-500:focus, |
| .focus\\:border-primary-600:focus { |
| border-color: #3b82f6 !important; |
| } |
| |
| .focus\\:ring-primary-200:focus, |
| .focus\\:ring-primary-300:focus { |
| --tw-ring-color: rgba(59, 130, 246, 0.2) !important; |
| } |
| |
| |
| .text-primary-600, .text-primary-500 { |
| color: #3b82f6 !important; |
| } |
| |
| .bg-primary-600, .bg-primary-500 { |
| background-color: #3b82f6 !important; |
| } |
| |
| .bg-primary-700:hover, .hover\\:bg-primary-700:hover { |
| background-color: #2563eb !important; |
| } |
| |
| |
| |
| .bg-red-600, button.bg-red-600, |
| .bg-red-700, button.bg-red-700 { |
| background-color: #f87171 !important; |
| } |
| |
| .bg-red-600:hover, button.bg-red-600:hover, |
| .bg-red-700:hover, button.bg-red-700:hover, |
| .hover\\:bg-red-700:hover { |
| background-color: #ef4444 !important; |
| } |
| |
| |
| .bg-gray-500, button.bg-gray-500, |
| .bg-gray-600, button.bg-gray-600 { |
| background-color: #e5e7eb !important; |
| color: #374151 !important; |
| } |
| |
| .bg-gray-500:hover, button.bg-gray-500:hover, |
| .bg-gray-600:hover, button.bg-gray-600:hover, |
| .hover\\:bg-gray-600:hover { |
| background-color: #d1d5db !important; |
| color: #374151 !important; |
| } |
| |
| |
| .bg-blue-600, button.bg-blue-600, |
| .bg-blue-700, button.bg-blue-700 { |
| background-color: #3b82f6 !important; |
| } |
| |
| .bg-blue-600:hover, button.bg-blue-600:hover, |
| .bg-blue-700:hover, button.bg-blue-700:hover, |
| .hover\\:bg-blue-700:hover { |
| background-color: #2563eb !important; |
| } |
| |
| |
| .bg-sky-600, button.bg-sky-600, |
| .bg-sky-700, button.bg-sky-700 { |
| background-color: #3b82f6 !important; |
| } |
| |
| .bg-sky-600:hover, button.bg-sky-600:hover, |
| .bg-sky-700:hover, button.bg-sky-700:hover, |
| .hover\\:bg-sky-700:hover { |
| background-color: #2563eb !important; |
| } |
| |
| |
| #deleteConfirmModal .bg-red-600, #deleteConfirmModal button.bg-red-600 { |
| background-color: #f87171 !important; |
| color: #ffffff !important; |
| display: flex !important; |
| align-items: center !important; |
| justify-content: center !important; |
| } |
| |
| #deleteConfirmModal .bg-red-600:hover, #deleteConfirmModal button.bg-red-600:hover { |
| background-color: #ef4444 !important; |
| } |
| |
| #deleteConfirmModal .bg-gray-500, #deleteConfirmModal button.bg-gray-500 { |
| background-color: #e5e7eb !important; |
| color: #374151 !important; |
| display: flex !important; |
| align-items: center !important; |
| justify-content: center !important; |
| } |
| |
| #deleteConfirmModal .bg-gray-500:hover, #deleteConfirmModal button.bg-gray-500:hover { |
| background-color: #d1d5db !important; |
| color: #374151 !important; |
| } |
| |
| |
| .modal button { |
| display: flex !important; |
| align-items: center !important; |
| justify-content: center !important; |
| text-align: center !important; |
| } |
| </style> |
| {% endblock %} {% block content %} |
| <div class="container mx-auto px-4"> |
| <div |
| class="rounded-2xl shadow-xl p-6 md:p-8" |
| style=" |
| background-color: rgba(80, 60, 160, 0.3); |
| backdrop-filter: blur(10px); |
| -webkit-backdrop-filter: blur(10px); |
| border: 1px solid rgba(150, 130, 230, 0.3); |
| " |
| > |
| <h1 |
| class="text-3xl font-extrabold text-center text-gray-800 mb-4" |
| > |
| <img |
| src="/static/icons/logo.png" |
| alt="Gemini Balance Logo" |
| class="h-9 inline-block align-middle mr-2" |
| /> |
| Gemini Balance - 错误日志 |
| </h1> |
|
|
| |
| <div class="nav-buttons-container flex justify-center mb-8 overflow-x-auto gap-2"> |
| <a |
| href="/config" |
| class="nav-link whitespace-nowrap flex items-center justify-center gap-2 px-6 py-3 font-medium rounded-lg text-gray-700 hover:text-gray-900 transition-all duration-200" |
| style="background-color: rgba(229, 231, 235, 0.8)" |
| > |
| <i class="fas fa-cog"></i> 配置编辑 |
| </a> |
| <a |
| href="/keys" |
| class="nav-link whitespace-nowrap flex items-center justify-center gap-2 px-6 py-3 font-medium rounded-lg text-gray-700 hover:text-gray-900 transition-all duration-200" |
| style="background-color: rgba(229, 231, 235, 0.8)" |
| > |
| <i class="fas fa-tachometer-alt"></i> 监控面板 |
| </a> |
| <a |
| href="/logs" |
| class="main-nav-btn whitespace-nowrap flex items-center justify-center gap-2 px-6 py-3 font-medium rounded-lg shadow-md hover:shadow-lg transition-all duration-200" |
| style="background-color: #3b82f6 !important; color: #ffffff !important;" |
| > |
| <i class="fas fa-exclamation-triangle"></i> 错误日志 |
| </a> |
| </div> |
|
|
| |
| <div |
| class="rounded-xl p-6 shadow-lg animate-fade-in" |
| style=" |
| background-color: rgba(70, 50, 150, 0.5); |
| backdrop-filter: blur(5px); |
| -webkit-backdrop-filter: blur(5px); |
| border: 1px solid rgba(0, 0, 0, 0.08); |
| " |
| > |
| <h2 |
| class="text-xl font-bold mb-6 pb-3 border-b flex items-center gap-2 text-gray-100 border-violet-300 border-opacity-30" |
| > |
| <i class="fas fa-bug text-violet-400"></i> 错误日志列表 |
| </h2> |
|
|
| |
| <div |
| class="grid grid-cols-1 lg:grid-cols-[1fr_auto] items-center gap-4 mb-6" |
| > |
| <div |
| class="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-3 w-full" |
| > |
| <input |
| type="text" |
| id="keySearch" |
| placeholder="搜索密钥 (部分)" |
| class="px-3 py-1 rounded-lg form-input-themed" |
| /> |
| <input |
| type="text" |
| id="errorSearch" |
| placeholder="搜索错误类型/日志" |
| class="px-3 py-1 rounded-lg form-input-themed" |
| /> |
| <input |
| type="text" |
| id="errorCodeSearch" |
| placeholder="搜索错误码" |
| class="px-3 py-1 rounded-lg form-input-themed" |
| /> |
| <div |
| class="grid grid-cols-1 sm:grid-cols-2 gap-2 col-span-1 sm:col-span-2 lg:col-span-3 mt-2" |
| > |
| <div class="flex items-center gap-2"> |
| <label class="text-sm text-gray-300 whitespace-nowrap" |
| >开始时间:</label |
| > |
| <input |
| type="datetime-local" |
| id="startDate" |
| class="px-3 py-1 rounded-lg text-sm w-full form-input-themed" |
| /> |
| </div> |
| <div class="flex items-center gap-2"> |
| <label class="text-sm text-gray-300 whitespace-nowrap" |
| >结束时间:</label |
| > |
| <input |
| type="datetime-local" |
| id="endDate" |
| class="px-3 py-1 rounded-lg text-sm w-full form-input-themed" |
| /> |
| </div> |
| </div> |
| </div> |
| <div class="flex items-center gap-3 flex-shrink-0"> |
| <button |
| id="searchBtn" |
| class="flex items-center justify-center px-4 py-1.5 bg-blue-600 hover:bg-blue-700 text-white rounded-lg font-medium transition-all duration-200 shadow-sm hover:shadow-md whitespace-nowrap" |
| > |
| <i class="fas fa-search mr-1.5"></i>搜索 |
| </button> |
| <button |
| id="copySelectedKeysBtn" |
| class="flex items-center justify-center px-4 py-1.5 bg-sky-600 hover:bg-sky-700 text-white rounded-lg font-medium transition-all duration-200 disabled:opacity-50 disabled:cursor-not-allowed shadow-sm hover:shadow-md whitespace-nowrap" |
| disabled |
| > |
| <i class="far fa-copy mr-1.5"></i>复制 |
| </button> |
| <button |
| id="deleteSelectedBtn" |
| class="flex items-center justify-center px-4 py-1.5 bg-red-600 hover:bg-red-700 text-white rounded-lg font-medium transition-all duration-200 disabled:opacity-50 disabled:cursor-not-allowed shadow-sm hover:shadow-md whitespace-nowrap" |
| disabled |
| > |
| <i class="fas fa-trash-alt mr-1.5"></i>删除 |
| </button> |
| <button |
| id="deleteAllLogsBtn" |
| class="flex items-center justify-center px-4 py-1.5 bg-red-700 hover:bg-red-800 text-white rounded-lg font-medium transition-all duration-200 shadow-sm hover:shadow-md whitespace-nowrap" |
| title="清空所有错误日志" |
| > |
| <i class="fas fa-dumpster-fire mr-1.5"></i>清空全部 |
| </button> |
| </div> |
| </div> |
|
|
| |
| <div |
| class="overflow-x-auto rounded-lg border mb-6" |
| style="border-color: rgba(0, 0, 0, 0.12)" |
| > |
| <table class="styled-table w-full min-w-full text-sm"> |
| <thead> |
| <tr class="text-left"> |
| <th |
| class="px-3 py-3 font-semibold rounded-tl-lg w-12 text-center" |
| > |
| <input |
| type="checkbox" |
| id="selectAllCheckbox" |
| class="form-checkbox h-4 w-4 text-blue-500 border-gray-500 rounded focus:ring-blue-500 bg-transparent" |
| /> |
| </th> |
| <th |
| class="px-5 py-3 font-semibold text-gray-700 cursor-pointer" |
| id="sortById" |
| > |
| ID <i class="fas fa-sort ml-1"></i> |
| </th> |
| <th class="px-5 py-3 font-semibold text-gray-700">Gemini密钥</th> |
| <th class="px-5 py-3 font-semibold text-gray-700">错误类型</th> |
| <th class="px-5 py-3 font-semibold text-gray-700">错误码</th> |
| <th class="px-5 py-3 font-semibold text-gray-700">模型名称</th> |
| <th class="px-5 py-3 font-semibold text-gray-700">请求时间</th> |
| <th |
| class="px-5 py-3 font-semibold text-gray-700 rounded-tr-lg text-center" |
| > |
| 操作 |
| </th> |
| </tr> |
| </thead> |
| <tbody |
| id="errorLogsTable" |
| class="divide-y" |
| style="border-color: rgba(0, 0, 0, 0.08)" |
| > |
| |
| </tbody> |
| </table> |
| </div> |
|
|
| |
| <div |
| id="loadingIndicator" |
| class="flex items-center justify-center p-8 hidden" |
| > |
| <div |
| class="animate-spin rounded-full h-12 w-12 border-b-2 border-blue-500" |
| ></div> |
| <p class="ml-4 text-lg text-gray-600 font-medium">加载中,请稍候...</p> |
| </div> |
|
|
| <div id="noDataMessage" class="text-center py-12 text-gray-600 hidden"> |
| <i class="fas fa-inbox text-5xl mb-3"></i> |
| <p class="text-lg">暂无错误日志数据</p> |
| </div> |
|
|
| <div |
| id="errorMessage" |
| class="p-4 rounded-lg font-medium text-center hidden" |
| style="background-color: rgba(220, 38, 38, 0.2); color: #fca5a5" |
| > |
| <i class="fas fa-exclamation-circle mr-2"></i> |
| 加载错误日志失败,请稍后重试。 |
| </div> |
|
|
| |
| <div |
| class="flex flex-col sm:flex-row justify-between items-center mt-6 gap-4" |
| > |
| <div class="pagination-container text-sm text-gray-700"> |
| <label for="pageSize" class="font-medium pagination-text" |
| >每页显示:</label |
| > |
| <select |
| id="pageSize" |
| class="rounded-md border focus:ring focus:border-blue-500 px-2 py-1 text-sm" |
| > |
| <option value="10">10</option> |
| <option value="20" selected>20</option> |
| <option value="50">50</option> |
| <option value="100">100</option> |
| </select> |
| <span class="pagination-text">条</span> |
| </div> |
| <div class="flex items-center gap-4"> |
| <ul class="pagination flex items-center gap-1" id="pagination"> |
| |
| </ul> |
| <div class="flex items-center gap-1"> |
| <input |
| type="number" |
| id="pageInput" |
| min="1" |
| class="w-16 px-2 py-1 rounded-md text-sm form-input-themed" |
| placeholder="页码" |
| /> |
| <button |
| id="goToPageBtn" |
| class="px-3 py-1 bg-blue-600 hover:bg-blue-700 text-white text-sm rounded-md transition" |
| > |
| 跳转 |
| </button> |
| </div> |
| </div> |
| </div> |
| </div> |
| </div> |
| </div> |
|
|
| |
| <div class="scroll-buttons"> |
| <button class="scroll-button" onclick="scrollToTop()" title="回到顶部"> |
| <i class="fas fa-chevron-up"></i> |
| </button> |
| <button class="scroll-button" onclick="scrollToBottom()" title="滚动到底部"> |
| <i class="fas fa-chevron-down"></i> |
| </button> |
| </div> |
|
|
| |
| <div id="notification" class="notification"></div> |
| |
|
|
| |
| <div id="logDetailModal" class="modal"> |
| <div |
| class="w-full max-w-6xl mx-auto rounded-2xl shadow-2xl overflow-hidden animate-fade-in" |
| style=" |
| background-color: #ffffff; |
| color: #374151; |
| border: 1px solid #e5e7eb; |
| " |
| > |
| <div class="p-6"> |
| <div |
| class="flex justify-between items-center pb-4 mb-4" |
| style="border-bottom: 1px solid #e5e7eb" |
| > |
| <h2 class="text-xl font-bold text-gray-800">错误日志详情</h2> |
| <button |
| id="closeLogDetailModalBtn" |
| class="text-gray-600 hover:text-gray-800 text-xl" |
| > |
| × |
| </button> |
| </div> |
|
|
| <div class="space-y-4 max-h-[60vh] overflow-y-auto p-1"> |
| <div |
| class="p-4 rounded-lg relative group" |
| style="background-color: #f8fafc" |
| > |
| <h6 class="text-sm font-semibold text-blue-600 mb-1"> |
| Gemini密钥: |
| </h6> |
| <pre |
| id="modalGeminiKey" |
| class="font-mono text-sm p-3 rounded overflow-x-auto" |
| style="background-color: #f1f5f9; color: #374151" |
| ></pre> |
| <button |
| class="copy-btn absolute top-2 right-2 hover:bg-gray-200 text-gray-600 p-1.5 rounded opacity-0 group-hover:opacity-100 transition-opacity" |
| style="background-color: #e2e8f0" |
| data-target="modalGeminiKey" |
| title="复制密钥" |
| > |
| <i class="far fa-copy"></i> |
| </button> |
| </div> |
|
|
| <div |
| class="p-4 rounded-lg relative group" |
| style="background-color: #f8fafc" |
| > |
| <h6 class="text-sm font-semibold text-blue-600 mb-1">错误类型:</h6> |
| <p id="modalErrorType" class="text-red-600 font-medium pr-8"></p> |
| <button |
| class="copy-btn absolute top-2 right-2 hover:bg-gray-200 text-gray-600 p-1.5 rounded opacity-0 group-hover:opacity-100 transition-opacity" |
| style="background-color: #e2e8f0" |
| data-target="modalErrorType" |
| title="复制错误类型" |
| > |
| <i class="far fa-copy"></i> |
| </button> |
| </div> |
|
|
| <div |
| class="p-4 rounded-lg relative group" |
| style="background-color: #f8fafc" |
| > |
| <h6 class="text-sm font-semibold text-blue-600 mb-1">错误日志:</h6> |
| <pre |
| id="modalErrorLog" |
| class="font-mono text-sm p-3 rounded overflow-x-auto whitespace-pre-wrap" |
| style="background-color: #f1f5f9; color: #374151" |
| ></pre> |
| <button |
| class="copy-btn absolute top-2 right-2 hover:bg-gray-200 text-gray-600 p-1.5 rounded opacity-0 group-hover:opacity-100 transition-opacity" |
| style="background-color: #e2e8f0" |
| data-target="modalErrorLog" |
| title="复制错误日志" |
| > |
| <i class="far fa-copy"></i> |
| </button> |
| </div> |
|
|
| <div |
| class="p-4 rounded-lg relative group" |
| style="background-color: #f8fafc" |
| > |
| <h6 class="text-sm font-semibold text-blue-600 mb-1">请求消息:</h6> |
| <pre |
| id="modalRequestMsg" |
| class="font-mono text-sm p-3 rounded overflow-x-auto whitespace-pre-wrap" |
| style="background-color: #f1f5f9; color: #374151" |
| ></pre> |
| <button |
| class="copy-btn absolute top-2 right-2 hover:bg-gray-200 text-gray-600 p-1.5 rounded opacity-0 group-hover:opacity-100 transition-opacity" |
| style="background-color: #e2e8f0" |
| data-target="modalRequestMsg" |
| title="复制请求消息" |
| > |
| <i class="far fa-copy"></i> |
| </button> |
| </div> |
|
|
| <div |
| class="p-4 rounded-lg relative group" |
| style="background-color: #f8fafc" |
| > |
| <h6 class="text-sm font-semibold text-blue-600 mb-1">模型名称:</h6> |
| <p id="modalModelName" class="font-medium pr-8 text-gray-700"></p> |
| <button |
| class="copy-btn absolute top-2 right-2 hover:bg-gray-200 text-gray-600 p-1.5 rounded opacity-0 group-hover:opacity-100 transition-opacity" |
| style="background-color: #e2e8f0" |
| data-target="modalModelName" |
| title="复制模型名称" |
| > |
| <i class="far fa-copy"></i> |
| </button> |
| </div> |
|
|
| <div |
| class="p-4 rounded-lg relative group" |
| style="background-color: #f8fafc" |
| > |
| <h6 class="text-sm font-semibold text-blue-600 mb-1">请求时间:</h6> |
| <p id="modalRequestTime" class="font-medium pr-8 text-gray-700"></p> |
| <button |
| class="copy-btn absolute top-2 right-2 hover:bg-gray-200 text-gray-600 p-1.5 rounded opacity-0 group-hover:opacity-100 transition-opacity" |
| style="background-color: #e2e8f0" |
| data-target="modalRequestTime" |
| title="复制请求时间" |
| > |
| <i class="far fa-copy"></i> |
| </button> |
| </div> |
| </div> |
|
|
| <div |
| class="flex justify-end mt-6 pt-4" |
| style="border-top: 1px solid #e5e7eb" |
| > |
| <button |
| type="button" |
| id="closeModalFooterBtn" |
| class="bg-gray-500 hover:bg-gray-600 text-white px-6 py-2 rounded-lg font-medium transition" |
| > |
| 关闭 |
| </button> |
| </div> |
| </div> |
| </div> |
| </div> |
|
|
| |
| <div id="deleteConfirmModal" class="modal"> |
| <div |
| class="w-full max-w-md mx-auto rounded-xl shadow-xl overflow-hidden animate-fade-in" |
| style=" |
| background-color: #ffffff; |
| color: #374151; |
| border: 1px solid #e5e7eb; |
| " |
| > |
| <div class="p-6"> |
| <div |
| class="flex justify-between items-center pb-3 mb-4" |
| style="border-bottom: 1px solid #e5e7eb" |
| > |
| <h2 class="text-lg font-semibold text-gray-800">确认删除</h2> |
| <button |
| id="closeDeleteConfirmModalBtn" |
| class="text-gray-600 hover:text-gray-800 text-xl" |
| > |
| × |
| </button> |
| </div> |
| <p id="deleteConfirmMessage" class="text-gray-700 mb-6"> |
| 你确定要删除选中的项目吗?此操作不可恢复! |
| </p> |
| <div class="flex justify-end gap-3"> |
| <button |
| id="cancelDeleteBtn" |
| type="button" |
| class="bg-gray-500 hover:bg-gray-600 text-white px-5 py-2 rounded-lg font-medium transition" |
| > |
| 取消 |
| </button> |
| <button |
| id="confirmDeleteBtn" |
| type="button" |
| class="bg-red-600 hover:bg-red-700 text-white px-5 py-2 rounded-lg font-medium transition" |
| > |
| 确认删除 |
| </button> |
| </div> |
| </div> |
| </div> |
| </div> |
| {% endblock %} {% block body_scripts %} |
| <script src="/static/js/error_logs.js"></script> |
| <script> |
| |
| |
| |
| </script> |
| {% endblock %} |
|
|