You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2054 lines
86 KiB

<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>数据缓冲平台</title>
<style>
:root {
--primary: #4f46e5;
--primary-light: #818cf8;
--primary-dark: #3730a3;
--success: #10b981;
--warning: #f59e0b;
--danger: #ef4444;
--info: #3b82f6;
--gray-50: #f9fafb;
--gray-100: #f3f4f6;
--gray-200: #e5e7eb;
--gray-300: #d1d5db;
--gray-400: #9ca3af;
--gray-500: #6b7280;
--gray-600: #4b5563;
--gray-700: #374151;
--gray-800: #1f2937;
--gray-900: #111827;
--sidebar-width: 240px;
--header-height: 64px;
--radius: 8px;
--shadow: 0 1px 3px rgba(0,0,0,0.1), 0 1px 2px rgba(0,0,0,0.06);
--shadow-md: 0 4px 6px -1px rgba(0,0,0,0.1), 0 2px 4px -1px rgba(0,0,0,0.06);
}
* { margin: 0; padding: 0; box-sizing: border-box; }
body {
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', sans-serif;
background: var(--gray-50);
color: var(--gray-800);
line-height: 1.5;
}
/* Sidebar */
.sidebar {
position: fixed;
left: 0;
top: 0;
bottom: 0;
width: var(--sidebar-width);
background: var(--gray-900);
color: white;
z-index: 100;
display: flex;
flex-direction: column;
transition: transform 0.3s;
}
.sidebar-brand {
height: var(--header-height);
display: flex;
align-items: center;
padding: 0 20px;
border-bottom: 1px solid rgba(255,255,255,0.1);
}
.sidebar-brand svg { width: 28px; height: 28px; margin-right: 12px; color: var(--primary-light); }
.sidebar-brand h1 { font-size: 16px; font-weight: 600; letter-spacing: -0.5px; }
.sidebar-nav { flex: 1; padding: 16px 12px; overflow-y: auto; }
.nav-item {
display: flex;
align-items: center;
padding: 10px 12px;
border-radius: var(--radius);
cursor: pointer;
transition: all 0.2s;
margin-bottom: 4px;
color: var(--gray-400);
text-decoration: none;
}
.nav-item:hover { background: rgba(255,255,255,0.08); color: white; }
.nav-item.active { background: var(--primary); color: white; }
.nav-item svg { width: 20px; height: 20px; margin-right: 12px; flex-shrink: 0; }
.nav-item span { font-size: 14px; font-weight: 500; }
.nav-divider { height: 1px; background: rgba(255,255,255,0.1); margin: 12px 0; }
/* Main Content */
.main {
margin-left: var(--sidebar-width);
min-height: 100vh;
display: flex;
flex-direction: column;
}
.header {
height: var(--header-height);
background: white;
border-bottom: 1px solid var(--gray-200);
display: flex;
align-items: center;
justify-content: space-between;
padding: 0 32px;
position: sticky;
top: 0;
z-index: 50;
}
.header h2 { font-size: 20px; font-weight: 600; color: var(--gray-900); }
.header-actions { display: flex; align-items: center; gap: 12px; }
.content { flex: 1; padding: 32px; overflow-y: auto; min-height: calc(100vh - var(--header-height)); }
/* Page Sections */
.page { display: none; }
.page.active { display: block; }
/* Cards */
.card {
background: white;
border-radius: var(--radius);
box-shadow: var(--shadow);
padding: 24px;
margin-bottom: 24px;
}
.card-header {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 20px;
padding-bottom: 16px;
border-bottom: 1px solid var(--gray-100);
}
.card-title { font-size: 16px; font-weight: 600; color: var(--gray-900); }
.card-subtitle { font-size: 13px; color: var(--gray-500); margin-top: 2px; }
/* Stats Grid */
.stats-grid {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 20px;
margin-bottom: 24px;
}
.stat-card {
background: white;
border-radius: var(--radius);
box-shadow: var(--shadow);
padding: 20px;
display: flex;
align-items: flex-start;
gap: 16px;
}
.stat-icon {
width: 48px;
height: 48px;
border-radius: 12px;
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
}
.stat-icon.blue { background: #eff6ff; color: var(--info); }
.stat-icon.green { background: #ecfdf5; color: var(--success); }
.stat-icon.yellow { background: #fffbeb; color: var(--warning); }
.stat-icon.red { background: #fef2f2; color: var(--danger); }
.stat-icon svg { width: 24px; height: 24px; }
.stat-content { flex: 1; }
.stat-value { font-size: 28px; font-weight: 700; color: var(--gray-900); line-height: 1.2; }
.stat-label { font-size: 13px; color: var(--gray-500); margin-top: 4px; }
/* Buttons */
.btn {
display: inline-flex;
align-items: center;
gap: 8px;
padding: 8px 16px;
border: none;
border-radius: var(--radius);
cursor: pointer;
font-size: 14px;
font-weight: 500;
transition: all 0.2s;
text-decoration: none;
}
.btn svg { width: 16px; height: 16px; }
.btn-primary { background: var(--primary); color: white; }
.btn-primary:hover { background: var(--primary-dark); }
.btn-success { background: var(--success); color: white; }
.btn-success:hover { background: #059669; }
.btn-warning { background: var(--warning); color: white; }
.btn-warning:hover { background: #d97706; }
.btn-danger { background: var(--danger); color: white; }
.btn-danger:hover { background: #dc2626; }
.btn-outline {
background: white;
color: var(--gray-700);
border: 1px solid var(--gray-300);
}
.btn-outline:hover { background: var(--gray-50); border-color: var(--gray-400); }
.btn:disabled { opacity: 0.5; cursor: not-allowed; }
.btn-sm { padding: 6px 12px; font-size: 13px; }
/* Form */
.form-group { margin-bottom: 20px; }
.form-label { display: block; margin-bottom: 6px; font-size: 14px; font-weight: 500; color: var(--gray-700); }
.form-input, .form-select {
width: 100%;
padding: 10px 12px;
border: 1px solid var(--gray-300);
border-radius: var(--radius);
font-size: 14px;
transition: border-color 0.2s;
background: white;
}
.form-input:focus, .form-select:focus {
outline: none;
border-color: var(--primary);
box-shadow: 0 0 0 3px rgba(79, 70, 229, 0.1);
}
.form-row {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 16px;
align-items: end;
}
/* Upload Area */
.upload-zone {
border: 2px dashed var(--gray-300);
border-radius: var(--radius);
padding: 40px;
text-align: center;
cursor: pointer;
transition: all 0.2s;
background: var(--gray-50);
}
.upload-zone:hover, .upload-zone.dragover {
border-color: var(--primary);
background: #f0f0ff;
}
.upload-zone svg { width: 48px; height: 48px; color: var(--gray-400); margin-bottom: 12px; }
.upload-zone p { color: var(--gray-600); }
.upload-zone .hint { color: var(--gray-400); font-size: 13px; margin-top: 8px; }
.upload-zone input[type="file"] { display: none; }
/* Table */
.table-container { overflow-x: auto; }
table {
width: 100%;
border-collapse: collapse;
}
th, td {
padding: 12px 16px;
text-align: left;
border-bottom: 1px solid var(--gray-200);
font-size: 14px;
}
th {
background: var(--gray-50);
font-weight: 600;
color: var(--gray-600);
font-size: 13px;
text-transform: uppercase;
letter-spacing: 0.5px;
}
tr:hover { background: var(--gray-50); }
/* Badge */
.badge {
display: inline-flex;
align-items: center;
padding: 4px 10px;
border-radius: 9999px;
font-size: 12px;
font-weight: 500;
}
.badge-success { background: #d1fae5; color: #065f46; }
.badge-warning { background: #fef3c7; color: #92400e; }
.badge-danger { background: #fee2e2; color: #991b1b; }
.badge-info { background: #dbeafe; color: #1e40af; }
.badge-gray { background: var(--gray-100); color: var(--gray-600); }
/* Symbol Grid */
.symbol-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(180px, 1fr));
gap: 12px;
margin-top: 16px;
max-height: 500px;
overflow-y: auto;
padding-right: 8px;
}
.symbol-card {
background: var(--gray-50);
border: 1px solid var(--gray-200);
border-radius: var(--radius);
padding: 14px;
text-align: center;
transition: all 0.2s;
position: relative;
}
.symbol-card:hover { border-color: var(--primary-light); box-shadow: var(--shadow-md); }
.symbol-name { font-weight: 600; color: var(--gray-800); font-size: 14px; }
.symbol-code { font-family: 'SF Mono', 'Cascadia Code', monospace; font-size: 13px; color: var(--primary); margin-top: 4px; }
.symbol-actions {
display: flex;
gap: 6px;
margin-top: 10px;
justify-content: center;
}
.symbol-actions .btn-sm {
padding: 4px 8px;
font-size: 12px;
}
.add-symbol-form {
background: var(--gray-50);
border: 2px dashed var(--gray-300);
border-radius: var(--radius);
padding: 20px;
margin-top: 16px;
}
.add-symbol-form .form-row {
grid-template-columns: 1fr 1fr auto;
gap: 12px;
align-items: end;
}
.highlight {
background: #fef08a;
padding: 0 2px;
border-radius: 2px;
}
/* Modal */
.modal-overlay {
display: none;
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.5);
z-index: 1000;
align-items: center;
justify-content: center;
}
.modal-overlay.active {
display: flex;
}
.modal {
background: white;
border-radius: var(--radius);
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.1);
max-width: 600px;
width: 90%;
max-height: 80vh;
display: flex;
flex-direction: column;
}
.modal-header {
padding: 20px 24px;
border-bottom: 1px solid var(--gray-200);
display: flex;
align-items: center;
justify-content: space-between;
}
.modal-title {
font-size: 18px;
font-weight: 600;
color: var(--gray-900);
}
.modal-close {
background: none;
border: none;
cursor: pointer;
padding: 4px;
color: var(--gray-500);
}
.modal-close:hover {
color: var(--gray-700);
}
.modal-body {
padding: 24px;
overflow-y: auto;
flex: 1;
}
.modal-footer {
padding: 16px 24px;
border-top: 1px solid var(--gray-200);
display: flex;
gap: 12px;
justify-content: flex-end;
}
.symbol-select-list {
max-height: 400px;
overflow-y: auto;
}
.symbol-select-item {
display: flex;
align-items: center;
padding: 10px 12px;
border-radius: var(--radius);
cursor: pointer;
transition: background 0.2s;
}
.symbol-select-item:hover {
background: var(--gray-50);
}
.symbol-select-item input[type="checkbox"] {
margin-right: 12px;
width: 18px;
height: 18px;
cursor: pointer;
}
.symbol-select-info {
flex: 1;
}
.symbol-select-name {
font-weight: 500;
color: var(--gray-800);
}
.symbol-select-code {
font-family: 'SF Mono', 'Cascadia Code', monospace;
font-size: 13px;
color: var(--primary);
}
/* K-line Chart */
.chart-container {
background: white;
border-radius: var(--radius);
padding: 20px;
margin-top: 20px;
}
.chart-header {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 16px;
padding-bottom: 12px;
border-bottom: 1px solid var(--gray-200);
}
.chart-title {
font-size: 16px;
font-weight: 600;
color: var(--gray-900);
}
.period-tabs {
display: flex;
gap: 8px;
}
.period-tab {
padding: 6px 14px;
border-radius: var(--radius);
cursor: pointer;
font-size: 13px;
font-weight: 500;
background: var(--gray-100);
color: var(--gray-600);
border: 1px solid transparent;
transition: all 0.2s;
}
.period-tab:hover {
background: var(--gray-200);
}
.period-tab.active {
background: var(--primary);
color: white;
}
#klineChart {
width: 100%;
height: 500px;
}
/* Log */
.log-panel {
background: var(--gray-900);
color: #e5e7eb;
border-radius: var(--radius);
padding: 16px;
font-family: 'SF Mono', 'Cascadia Code', 'Courier New', monospace;
font-size: 13px;
max-height: 400px;
overflow-y: auto;
line-height: 1.8;
}
.log-panel .log-line { margin: 2px 0; }
.log-panel .log-time { color: var(--gray-500); }
.log-panel .log-success { color: var(--success); }
.log-panel .log-error { color: var(--danger); }
.log-panel .log-info { color: var(--info); }
/* Progress */
.progress-bar {
width: 100%;
height: 6px;
background: var(--gray-200);
border-radius: 9999px;
overflow: hidden;
margin: 16px 0;
}
.progress-fill {
height: 100%;
background: var(--primary);
border-radius: 9999px;
transition: width 0.3s;
}
/* Toast */
.toast-container {
position: fixed;
top: 80px;
right: 32px;
z-index: 200;
display: flex;
flex-direction: column;
gap: 12px;
}
.toast {
background: white;
border-radius: var(--radius);
box-shadow: var(--shadow-md);
padding: 16px 20px;
display: flex;
align-items: center;
gap: 12px;
min-width: 320px;
animation: slideIn 0.3s;
border-left: 4px solid;
}
.toast.success { border-color: var(--success); }
.toast.error { border-color: var(--danger); }
.toast.info { border-color: var(--info); }
@keyframes slideIn {
from { transform: translateX(100%); opacity: 0; }
to { transform: translateX(0); opacity: 1; }
}
/* Empty State */
.empty-state {
text-align: center;
padding: 60px 20px;
color: var(--gray-400);
}
.empty-state svg { width: 64px; height: 64px; margin-bottom: 16px; }
.empty-state p { font-size: 15px; }
/* Tabs */
.tabs {
display: flex;
border-bottom: 2px solid var(--gray-200);
margin-bottom: 24px;
}
.tab {
padding: 12px 20px;
cursor: pointer;
font-size: 14px;
font-weight: 500;
color: var(--gray-500);
border-bottom: 2px solid transparent;
margin-bottom: -2px;
transition: all 0.2s;
}
.tab:hover { color: var(--gray-700); }
.tab.active { color: var(--primary); border-color: var(--primary); }
.tab-content { display: none; }
.tab-content.active { display: block; }
/* Spinner */
.spinner {
width: 20px;
height: 20px;
border: 2px solid rgba(255,255,255,0.3);
border-top-color: white;
border-radius: 50%;
animation: spin 0.6s linear infinite;
}
@keyframes spin { to { transform: rotate(360deg); } }
/* Responsive */
@media (max-width: 768px) {
.sidebar { transform: translateX(-100%); }
.sidebar.open { transform: translateX(0); }
.main { margin-left: 0; }
.content { padding: 20px; }
.form-row { grid-template-columns: 1fr; }
}
</style>
</head>
<body>
<!-- Sidebar -->
<aside class="sidebar" id="sidebar">
<div class="sidebar-brand">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
<path d="M3 3v18h18"/><path d="M18 17V9"/><path d="M13 17V5"/><path d="M8 17v-3"/>
</svg>
<h1>数据缓冲平台</h1>
</div>
<nav class="sidebar-nav">
<a class="nav-item active" data-page="dashboard">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><rect x="3" y="3" width="7" height="7"/><rect x="14" y="3" width="7" height="7"/><rect x="3" y="14" width="7" height="7"/><rect x="14" y="14" width="7" height="7"/></svg>
<span>仪表盘</span>
</a>
<a class="nav-item" data-page="config">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/></svg>
<span>品种配置</span>
</a>
<div class="nav-divider"></div>
<a class="nav-item" data-page="fetch">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg>
<span>数据获取</span>
</a>
<a class="nav-item" data-page="tasks">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
<span>定时任务</span>
</a>
<div class="nav-divider"></div>
<a class="nav-item" data-page="cache">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"/><polyline points="3.27 6.96 12 12.01 20.73 6.96"/><line x1="12" y1="22.08" x2="12" y2="12"/></svg>
<span>缓存查询</span>
</a>
<a class="nav-item" data-page="logs">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><line x1="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/></svg>
<span>运行日志</span>
</a>
</nav>
</aside>
<!-- Main -->
<div class="main">
<header class="header">
<h2 id="pageTitle">仪表盘</h2>
<div class="header-actions">
<button class="btn btn-outline btn-sm" onclick="location.reload()">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 4 23 10 17 10"/><path d="M20.49 15a9 9 0 1 1-2.12-9.36L23 10"/></svg>
刷新
</button>
<a href="/docs" target="_blank" class="btn btn-outline btn-sm">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/></svg>
API 文档
</a>
</div>
</header>
<div class="content">
<!-- Dashboard -->
<div class="page active" id="page-dashboard">
<div class="stats-grid" id="statsGrid">
<div class="stat-card">
<div class="stat-icon blue">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"/></svg>
</div>
<div class="stat-content">
<div class="stat-value" id="statSymbols">-</div>
<div class="stat-label">配置品种数</div>
</div>
</div>
<div class="stat-card">
<div class="stat-icon green">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M22 11.08V12a10 10 0 1 1-5.93-9.14"/><polyline points="22 4 12 14.01 9 11.01"/></svg>
</div>
<div class="stat-content">
<div class="stat-value" id="statCached">-</div>
<div class="stat-label">已缓存品种</div>
</div>
</div>
<div class="stat-card">
<div class="stat-icon yellow">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
</div>
<div class="stat-content">
<div class="stat-value" id="statTasks">-</div>
<div class="stat-label">运行中任务</div>
</div>
</div>
<div class="stat-card">
<div class="stat-icon red">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg>
</div>
<div class="stat-content">
<div class="stat-value" id="statFresh">-</div>
<div class="stat-label">缓存新鲜度</div>
</div>
</div>
</div>
<div class="card">
<div class="card-header">
<div>
<div class="card-title">快捷操作</div>
</div>
</div>
<div style="display: flex; gap: 12px; flex-wrap: wrap;">
<button class="btn btn-primary" onclick="navigateTo('config')">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="17 8 12 3 7 8"/><line x1="12" y1="3" x2="12" y2="15"/></svg>
上传配置
</button>
<button class="btn btn-success" onclick="navigateTo('fetch')">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg>
批量获取
</button>
<button class="btn btn-warning" onclick="navigateTo('tasks')">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
定时任务
</button>
<button class="btn btn-outline" onclick="navigateTo('cache')">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="11" cy="11" r="8"/><line x1="21" y1="21" x2="16.65" y2="16.65"/></svg>
查询缓存
</button>
</div>
</div>
<div class="card">
<div class="card-header">
<div class="card-title">最近采集记录</div>
</div>
<div id="recentFetchLog">
<div class="empty-state">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><line x1="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/></svg>
<p>暂无采集记录</p>
</div>
</div>
</div>
</div>
<!-- Config -->
<div class="page" id="page-config">
<div class="card">
<div class="card-header">
<div>
<div class="card-title">上传品种配置</div>
<div class="card-subtitle">支持拖拽或点击上传 JSON 配置文件</div>
</div>
</div>
<div class="upload-zone" id="uploadZone">
<input type="file" id="fileInput" accept=".json">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="17 8 12 3 7 8"/><line x1="12" y1="3" x2="12" y2="15"/></svg>
<p>点击或拖拽文件到此处</p>
<p class="hint">JSON 格式: {"futures": {"沪银": "AG2606"}, "stock": {...}}</p>
</div>
</div>
<div class="card">
<div class="card-header">
<div>
<div class="card-title">当前配置品种</div>
<div class="card-subtitle" id="configCount">加载中...</div>
</div>
<div style="display: flex; gap: 12px; align-items: center;">
<div style="position: relative;">
<input class="form-input" id="symbolSearch" placeholder="搜索品种..." style="width: 200px; padding-left: 36px;" oninput="filterSymbols(this.value)">
<svg style="position: absolute; left: 10px; top: 50%; transform: translateY(-50%); width: 16px; height: 16px; color: var(--gray-400);" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="11" cy="11" r="8"/><line x1="21" y1="21" x2="16.65" y2="16.65"/></svg>
</div>
<button class="btn btn-outline btn-sm" onclick="loadConfig()">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 4 23 10 17 10"/><path d="M20.49 15a9 9 0 1 1-2.12-9.36L23 10"/></svg>
刷新
</button>
</div>
</div>
<div id="configDisplay">
<div class="tabs" id="configTabs">
<div class="tab active" data-tab="futures">期货品种</div>
<div class="tab" data-tab="stock">股票品种</div>
</div>
<div class="tab-content active" id="tab-futures"></div>
<div class="tab-content" id="tab-stock"></div>
</div>
<div class="add-symbol-form">
<div class="card-title" style="margin-bottom: 16px; font-size: 15px;">手动添加品种</div>
<div class="form-row">
<div class="form-group" style="margin-bottom: 0;">
<label class="form-label">品种名称</label>
<input class="form-input" id="newSymbolName" placeholder="如: 沪银">
</div>
<div class="form-group" style="margin-bottom: 0;">
<label class="form-label">品种代码</label>
<input class="form-input" id="newSymbolCode" placeholder="如: AG2606">
</div>
<div class="form-group" style="margin-bottom: 0;">
<label class="form-label">类型</label>
<select class="form-select" id="newSymbolType">
<option value="futures">期货</option>
<option value="stock">股票</option>
</select>
</div>
<div class="form-group" style="margin-bottom: 0;">
<button class="btn btn-primary" onclick="addSymbol()" style="height: 42px;">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><line x1="12" y1="5" x2="12" y2="19"/><line x1="5" y1="12" x2="19" y2="12"/></svg>
添加
</button>
</div>
</div>
</div>
</div>
</div>
<!-- Fetch -->
<div class="page" id="page-fetch">
<div class="card">
<div class="card-header">
<div>
<div class="card-title">批量获取行情数据</div>
<div class="card-subtitle">智能缓存:已存在且有效的数据不会重复请求</div>
</div>
</div>
<div class="form-row">
<div class="form-group">
<label class="form-label">数据类型</label>
<select class="form-select" id="fetchDataType">
<option value="futures">期货</option>
<option value="stock">股票</option>
</select>
</div>
<div class="form-group">
<label class="form-label">周期(逗号分隔)</label>
<input class="form-input" id="fetchPeriods" value="5min,15min,30min,60min,daily">
</div>
<div class="form-group">
<button class="btn btn-success" id="btnBatchFetch" onclick="batchFetchAll()" style="height: 42px;">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg>
一键批量获取
</button>
</div>
</div>
<div id="fetchProgress" class="hidden">
<div class="progress-bar"><div class="progress-fill" id="progressFill" style="width: 0%"></div></div>
<p id="progressText" style="text-align: center; color: var(--gray-500); font-size: 14px;"></p>
</div>
<div id="fetchResult" class="hidden" style="margin-top: 20px;"></div>
</div>
<div class="card">
<div class="card-header">
<div class="card-title">单个品种查询</div>
</div>
<div class="form-row">
<div class="form-group">
<label class="form-label">品种代码</label>
<input class="form-input" id="querySymbol" placeholder="如: AG2606">
</div>
<div class="form-group">
<label class="form-label">周期</label>
<select class="form-select" id="queryPeriod">
<option value="">全部周期</option>
<option value="5min">5分钟</option>
<option value="15min">15分钟</option>
<option value="30min">30分钟</option>
<option value="60min">60分钟</option>
<option value="daily">日线</option>
</select>
</div>
<div class="form-group">
<button class="btn btn-primary" onclick="queryCache()" style="height: 42px;">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="11" cy="11" r="8"/><line x1="21" y1="21" x2="16.65" y2="16.65"/></svg>
查询缓存
</button>
</div>
</div>
<div id="queryResult" class="hidden" style="margin-top: 20px;"></div>
</div>
</div>
<!-- Tasks -->
<div class="page" id="page-tasks">
<div class="card">
<div class="card-header">
<div>
<div class="card-title">批量创建定时任务</div>
<div class="card-subtitle">为配置中的所有品种自动创建定时采集任务</div>
</div>
</div>
<div class="form-row">
<div class="form-group">
<label class="form-label">数据类型</label>
<select class="form-select" id="taskDataType">
<option value="futures">期货</option>
<option value="stock">股票</option>
</select>
</div>
<div class="form-group">
<label class="form-label">采集周期</label>
<input class="form-input" id="taskPeriods" value="5min,15min,60min">
</div>
<div class="form-group">
<label class="form-label">轮询间隔(秒)</label>
<input class="form-input" type="number" id="taskInterval" value="300" min="30" max="86400">
</div>
<div class="form-group">
<button class="btn btn-warning" id="btnBatchTasks" onclick="batchCreateTasks()" style="height: 42px;">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
批量创建任务
</button>
</div>
</div>
</div>
<div class="card">
<div class="card-header">
<div class="card-title">任务列表</div>
<button class="btn btn-outline btn-sm" onclick="loadTasks()">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="23 4 23 10 17 10"/><path d="M20.49 15a9 9 0 1 1-2.12-9.36L23 10"/></svg>
刷新
</button>
</div>
<div class="table-container" id="taskTable">
<div class="empty-state">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
<p>暂无定时任务</p>
</div>
</div>
</div>
</div>
<!-- Cache -->
<div class="page" id="page-cache">
<div class="card">
<div class="card-header">
<div class="card-title">缓存状态查询</div>
</div>
<div class="form-row">
<div class="form-group">
<label class="form-label">品种代码</label>
<input class="form-input" id="cacheSymbol" placeholder="如: AG2606">
</div>
<div class="form-group">
<button class="btn btn-primary" onclick="checkCacheStatus()" style="height: 42px;">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"/></svg>
查看状态
</button>
</div>
</div>
<div id="cacheStatusResult" class="hidden" style="margin-top: 20px;"></div>
</div>
</div>
<!-- Logs -->
<div class="page" id="page-logs">
<div class="card">
<div class="card-header">
<div class="card-title">运行日志</div>
<button class="btn btn-outline btn-sm" onclick="clearLogs()">清空</button>
</div>
<div class="log-panel" id="logPanel">
<div class="log-line"><span class="log-time">[系统]</span> <span class="log-info">平台已就绪,等待操作...</span></div>
</div>
</div>
</div>
</div>
</div>
<!-- Toast Container -->
<div class="toast-container" id="toastContainer"></div>
<!-- Batch Fetch Modal -->
<div class="modal-overlay" id="batchFetchModal">
<div class="modal">
<div class="modal-header">
<div class="modal-title">选择要获取的合约</div>
<button class="modal-close" onclick="closeBatchFetchModal()">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="width: 24px; height: 24px;">
<line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/>
</svg>
</button>
</div>
<div class="modal-body">
<div style="margin-bottom: 12px;">
<div style="position: relative;">
<input class="form-input" id="symbolSearchModal" placeholder="搜索合约..." style="width: 100%; padding-left: 36px;" oninput="filterModalSymbols(this.value)">
<svg style="position: absolute; left: 10px; top: 50%; transform: translateY(-50%); width: 16px; height: 16px; color: var(--gray-400);" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="11" cy="11" r="8"/><line x1="21" y1="21" x2="16.65" y2="16.65"/></svg>
</div>
</div>
<div style="margin-bottom: 16px; display: flex; gap: 12px; align-items: center;">
<label style="display: flex; align-items: center; cursor: pointer;">
<input type="checkbox" id="selectAllSymbols" onchange="toggleSelectAll()" style="margin-right: 8px; width: 18px; height: 18px; cursor: pointer;">
<span style="font-weight: 500;">全选</span>
</label>
<span id="selectedCount" style="color: var(--gray-500); font-size: 14px;">已选择 0 个合约</span>
</div>
<div class="symbol-select-list" id="symbolSelectList"></div>
</div>
<div class="modal-footer">
<button class="btn btn-outline" onclick="closeBatchFetchModal()">取消</button>
<button class="btn btn-success" id="btnConfirmBatchFetch" onclick="confirmBatchFetch()">
<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="width: 16px; height: 16px;"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg>
开始获取
</button>
</div>
</div>
</div>
<script src="https://cdn.jsdelivr.net/npm/echarts@5.4.3/dist/echarts.min.js"></script>
<script>
const API = '/api/v1';
let currentConfig = {};
let logs = [];
let currentSearchTerm = '';
let selectedSymbolsForFetch = [];
let modalSymbolsData = [];
// Navigation
function navigateTo(page) {
document.querySelectorAll('.page').forEach(p => p.classList.remove('active'));
document.querySelectorAll('.nav-item').forEach(n => n.classList.remove('active'));
document.getElementById(`page-${page}`).classList.add('active');
document.querySelector(`.nav-item[data-page="${page}"]`).classList.add('active');
const titles = {
dashboard: '仪表盘',
config: '品种配置',
fetch: '数据获取',
tasks: '定时任务',
cache: '缓存查询',
logs: '运行日志'
};
document.getElementById('pageTitle').textContent = titles[page] || page;
}
document.querySelectorAll('.nav-item').forEach(item => {
item.addEventListener('click', () => navigateTo(item.dataset.page));
});
// Config tabs
document.querySelectorAll('#configTabs .tab').forEach(tab => {
tab.addEventListener('click', () => {
document.querySelectorAll('#configTabs .tab').forEach(t => t.classList.remove('active'));
document.querySelectorAll('#configDisplay .tab-content').forEach(c => c.classList.remove('active'));
tab.classList.add('active');
document.getElementById(`tab-${tab.dataset.tab}`).classList.add('active');
});
});
// Toast
function showToast(message, type = 'info') {
const container = document.getElementById('toastContainer');
const toast = document.createElement('div');
toast.className = `toast ${type}`;
toast.innerHTML = `
<span style="font-size: 18px;">${type === 'success' ? '✅' : type === 'error' ? '❌' : ''}</span>
<span style="flex: 1; font-size: 14px;">${message}</span>
`;
container.appendChild(toast);
setTimeout(() => toast.remove(), 4000);
}
// Log
function addLog(msg, level = 'info') {
const time = new Date().toLocaleTimeString();
logs.push({ time, msg, level });
const panel = document.getElementById('logPanel');
const line = document.createElement('div');
line.className = 'log-line';
line.innerHTML = `<span class="log-time">[${time}]</span> <span class="log-${level}">${msg}</span>`;
panel.appendChild(line);
panel.scrollTop = panel.scrollHeight;
}
function clearLogs() {
document.getElementById('logPanel').innerHTML = '';
logs = [];
addLog('日志已清空', 'info');
}
// Upload
const uploadZone = document.getElementById('uploadZone');
const fileInput = document.getElementById('fileInput');
uploadZone.addEventListener('click', () => fileInput.click());
uploadZone.addEventListener('dragover', e => { e.preventDefault(); uploadZone.classList.add('dragover'); });
uploadZone.addEventListener('dragleave', () => uploadZone.classList.remove('dragover'));
uploadZone.addEventListener('drop', e => {
e.preventDefault();
uploadZone.classList.remove('dragover');
if (e.dataTransfer.files.length) handleUpload(e.dataTransfer.files[0]);
});
fileInput.addEventListener('change', () => { if (fileInput.files.length) handleUpload(fileInput.files[0]); });
async function handleUpload(file) {
addLog(`上传文件: ${file.name}`);
const formData = new FormData();
formData.append('file', file);
try {
const res = await fetch(`${API}/config/upload`, { method: 'POST', body: formData });
const data = await res.json();
if (res.ok) {
addLog(`上传成功: 期货 ${data.futures_symbols} 个, 股票 ${data.stock_symbols} 个`, 'success');
showToast('配置文件上传成功', 'success');
loadConfig();
} else {
addLog(`上传失败: ${data.detail}`, 'error');
showToast(data.detail, 'error');
}
} catch (e) {
addLog(`上传异常: ${e.message}`, 'error');
}
}
// Load Config
async function loadConfig() {
try {
const res = await fetch(`${API}/config`);
currentConfig = await res.json();
renderConfig();
updateDashboard();
} catch (e) {
addLog(`加载配置失败: ${e.message}`, 'error');
}
}
function renderConfig() {
const futures = currentConfig.futures || {};
const stock = currentConfig.stock || {};
const futuresCount = Object.keys(futures).length;
const stockCount = Object.keys(stock).length;
document.getElementById('configCount').textContent = `期货 ${futuresCount} 个, 股票 ${stockCount} 个`;
const renderGrid = (items, type) => {
let filteredItems = items;
if (currentSearchTerm) {
const searchTerm = currentSearchTerm.toLowerCase();
filteredItems = {};
for (const [name, code] of Object.entries(items)) {
if (name.toLowerCase().includes(searchTerm) || code.toLowerCase().includes(searchTerm)) {
filteredItems[name] = code;
}
}
}
if (!Object.keys(filteredItems).length) {
return currentSearchTerm
? `<div class="empty-state"><p>未找到匹配 "${currentSearchTerm}" 的品种</p></div>`
: '<div class="empty-state"><p>暂无品种</p></div>';
}
const highlightText = (text) => {
if (!currentSearchTerm) return text;
const regex = new RegExp(`(${currentSearchTerm})`, 'gi');
return text.replace(regex, '<span class="highlight">$1</span>');
};
return `<div class="symbol-grid">${Object.entries(filteredItems).map(([name, code]) =>
`<div class="symbol-card">
<div class="symbol-name">${highlightText(name)}</div>
<div class="symbol-code">${highlightText(code)}</div>
<div class="symbol-actions">
<button class="btn btn-primary btn-sm" onclick="editSymbol('${type}', '${code}', '${name}')">修改</button>
<button class="btn btn-danger btn-sm" onclick="deleteSymbol('${type}', '${code}')">删除</button>
</div>
</div>`
).join('')}</div>`;
};
document.getElementById('tab-futures').innerHTML = renderGrid(futures, 'futures');
document.getElementById('tab-stock').innerHTML = renderGrid(stock, 'stock');
}
function filterSymbols(searchTerm) {
currentSearchTerm = searchTerm.trim();
renderConfig();
}
async function addSymbol() {
const name = document.getElementById('newSymbolName').value.trim();
const code = document.getElementById('newSymbolCode').value.trim();
const type = document.getElementById('newSymbolType').value;
if (!name || !code) {
return showToast('请填写品种名称和代码', 'error');
}
const symbols = type === 'futures' ? (currentConfig.futures || {}) : (currentConfig.stock || {});
if (symbols[name]) {
return showToast('品种名称已存在', 'error');
}
symbols[name] = code;
const fullConfig = {
futures: currentConfig.futures || {},
stock: currentConfig.stock || {}
};
fullConfig[type] = symbols;
try {
const res = await fetch(`${API}/config/upload`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(fullConfig),
});
if (res.ok) {
showToast('品种添加成功', 'success');
document.getElementById('newSymbolName').value = '';
document.getElementById('newSymbolCode').value = '';
loadConfig();
} else {
const data = await res.json();
showToast(data.detail || '添加失败', 'error');
}
} catch (e) {
showToast(`添加失败: ${e.message}`, 'error');
}
}
async function editSymbol(type, oldCode, oldName) {
const newName = prompt('请输入新的品种名称:', oldName);
if (!newName || newName === oldName) return;
const newCode = prompt('请输入新的品种代码:', oldCode);
if (!newCode) return;
const symbols = type === 'futures' ? (currentConfig.futures || {}) : (currentConfig.stock || {});
delete symbols[oldName];
symbols[newName] = newCode;
const fullConfig = {
futures: currentConfig.futures || {},
stock: currentConfig.stock || {}
};
fullConfig[type] = symbols;
try {
const res = await fetch(`${API}/config/upload`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(fullConfig),
});
if (res.ok) {
showToast('品种修改成功', 'success');
loadConfig();
} else {
const data = await res.json();
showToast(data.detail || '修改失败', 'error');
}
} catch (e) {
showToast(`修改失败: ${e.message}`, 'error');
}
}
async function deleteSymbol(type, code) {
if (!confirm('确定删除此品种?')) return;
const symbols = type === 'futures' ? (currentConfig.futures || {}) : (currentConfig.stock || {});
const name = Object.keys(symbols).find(k => symbols[k] === code);
if (name) {
delete symbols[name];
}
const fullConfig = {
futures: currentConfig.futures || {},
stock: currentConfig.stock || {}
};
fullConfig[type] = symbols;
try {
const res = await fetch(`${API}/config/upload`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(fullConfig),
});
if (res.ok) {
showToast('品种删除成功', 'success');
loadConfig();
} else {
const data = await res.json();
showToast(data.detail || '删除失败', 'error');
}
} catch (e) {
showToast(`删除失败: ${e.message}`, 'error');
}
}
// Dashboard
async function updateDashboard() {
const futures = currentConfig.futures || {};
document.getElementById('statSymbols').textContent = Object.keys(futures).length || '-';
try {
const tasksRes = await fetch(`${API}/tasks`);
const tasksData = await tasksRes.json();
const running = tasksData.tasks.filter(t => t.running).length;
document.getElementById('statTasks').textContent = running || '0';
// Count cached symbols
let cachedCount = 0;
for (const code of Object.values(futures)) {
const res = await fetch(`${API}/data/cache-status/${code}`);
const status = await res.json();
if (status.cached_periods && status.cached_periods.length) cachedCount++;
}
document.getElementById('statCached').textContent = cachedCount;
document.getElementById('statFresh').textContent = cachedCount > 0 ? `${Math.round(cachedCount / Object.keys(futures).length * 100)}%` : '-';
} catch (e) {}
}
// Batch Fetch
async function batchFetchAll() {
const dataType = document.getElementById('fetchDataType').value;
if (!currentConfig[dataType] || Object.keys(currentConfig[dataType]).length === 0) {
showToast(`没有可用的${dataType === 'futures' ? '期货' : '股票'}品种配置`, 'error');
return;
}
openBatchFetchModal(dataType);
}
function openBatchFetchModal(dataType) {
const symbols = currentConfig[dataType] || {};
modalSymbolsData = Object.entries(symbols).map(([name, code]) => ({ name, code }));
renderModalSymbols();
document.getElementById('selectAllSymbols').checked = true;
updateSelectedCount();
document.getElementById('batchFetchModal').classList.add('active');
}
function renderModalSymbols(filterText = '') {
const listContainer = document.getElementById('symbolSelectList');
const filterLower = filterText.toLowerCase();
const filteredSymbols = filterText
? modalSymbolsData.filter(s =>
s.name.toLowerCase().includes(filterLower) ||
s.code.toLowerCase().includes(filterLower)
)
: modalSymbolsData;
let html = '';
filteredSymbols.forEach(({ name, code }) => {
html += `
<label class="symbol-select-item">
<input type="checkbox" value="${code}" data-name="${name}" checked onchange="updateSelectedCount()">
<div class="symbol-select-info">
<div class="symbol-select-name">${highlightTextInModal(name, filterText)}</div>
<div class="symbol-select-code">${highlightTextInModal(code, filterText)}</div>
</div>
</label>
`;
});
if (filteredSymbols.length === 0) {
html = `<div class="empty-state" style="padding: 40px 20px;"><p>未找到匹配 "${filterText}" 的合约</p></div>`;
}
listContainer.innerHTML = html;
updateSelectedCount();
}
function highlightTextInModal(text, searchTerm) {
if (!searchTerm) return text;
const regex = new RegExp(`(${searchTerm})`, 'gi');
return text.replace(regex, '<span class="highlight">$1</span>');
}
function filterModalSymbols(searchTerm) {
renderModalSymbols(searchTerm.trim());
}
function closeBatchFetchModal() {
document.getElementById('batchFetchModal').classList.remove('active');
}
function toggleSelectAll() {
const selectAll = document.getElementById('selectAllSymbols').checked;
const checkboxes = document.querySelectorAll('#symbolSelectList input[type="checkbox"]');
checkboxes.forEach(cb => cb.checked = selectAll);
updateSelectedCount();
}
function updateSelectedCount() {
const checkboxes = document.querySelectorAll('#symbolSelectList input[type="checkbox"]:checked');
const count = checkboxes.length;
const total = document.querySelectorAll('#symbolSelectList input[type="checkbox"]').length;
document.getElementById('selectedCount').textContent = `已选择 ${count}/${total} 个合约`;
const selectAll = document.getElementById('selectAllSymbols');
if (count === total) {
selectAll.checked = true;
} else if (count === 0) {
selectAll.checked = false;
} else {
selectAll.checked = false;
}
}
async function confirmBatchFetch() {
const checkboxes = document.querySelectorAll('#symbolSelectList input[type="checkbox"]:checked');
if (checkboxes.length === 0) {
showToast('请至少选择一个合约', 'error');
return;
}
closeBatchFetchModal();
const dataType = document.getElementById('fetchDataType').value;
const periods = document.getElementById('fetchPeriods').value;
const btn = document.getElementById('btnBatchFetch');
const selectedSymbols = Array.from(checkboxes).map(cb => ({
code: cb.value,
name: cb.dataset.name
}));
btn.disabled = true;
btn.innerHTML = '<div class="spinner"></div> 正在获取...';
const progressDiv = document.getElementById('fetchProgress');
const progressFill = document.getElementById('progressFill');
const progressText = document.getElementById('progressText');
const resultDiv = document.getElementById('fetchResult');
progressDiv.classList.remove('hidden');
progressFill.style.width = '0%';
progressText.textContent = `准备获取 ${selectedSymbols.length} 个合约...`;
resultDiv.classList.add('hidden');
addLog(`开始批量获取 [${dataType}] 数据,周期: ${periods},合约数: ${selectedSymbols.length}`);
const totalSymbols = selectedSymbols.length;
let processedCount = 0;
let successCount = 0;
let failedCount = 0;
let cachedCount = 0;
const results = [];
for (const symbol of selectedSymbols) {
try {
progressText.textContent = `正在获取: ${symbol.name} (${symbol.code}) [${processedCount + 1}/${totalSymbols}]`;
const res = await fetch(`${API}/config/batch-fetch-all`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
data_type: dataType,
periods: periods,
selected_symbols: symbol.code
}),
});
const data = await res.json();
addLog(`API返回 [${symbol.code}]: 成功=${data.success?.length || 0}, 失败=${data.failed?.length || 0}, 缓存=${data.cached?.length || 0}`, 'info');
processedCount++;
const progress = (processedCount / totalSymbols) * 100;
progressFill.style.width = `${progress}%`;
if (data.success && data.success.length > 0) {
successCount++;
const detail = data.details && data.details[symbol.code];
const isCached = detail && detail.source === 'cache';
if (isCached) {
cachedCount++;
}
results.push({
name: symbol.name,
code: symbol.code,
status: isCached ? 'cached' : 'success',
statusText: isCached ? '缓存命中' : '已更新',
source: detail ? detail.source : '-',
price: detail && detail.current_price ? detail.current_price.toFixed(2) : '-',
candleCount: detail && detail.timeframes && detail.timeframes.length > 0 ?
detail.timeframes.reduce((sum, tf) => sum + tf.candle_count, 0) : 0
});
} else if (data.failed && data.failed.length > 0) {
failedCount++;
results.push({
name: symbol.name,
code: symbol.code,
status: 'failed',
statusText: '失败',
source: '-',
price: '-',
error: data.failed[0].error || '未知错误'
});
}
} catch (e) {
processedCount++;
failedCount++;
const progress = (processedCount / totalSymbols) * 100;
progressFill.style.width = `${progress}%`;
results.push({
name: symbol.name,
code: symbol.code,
status: 'failed',
statusText: '异常',
source: '-',
price: '-',
error: e.message
});
}
}
progressFill.style.width = '100%';
progressText.textContent = `完成! 成功: ${successCount}, 失败: ${failedCount}, 缓存: ${cachedCount}`;
addLog(`批量获取完成: 成功 ${successCount}, 失败 ${failedCount}, 缓存 ${cachedCount}`,
failedCount === 0 ? 'success' : 'error');
showToast(`获取完成: ${successCount} 成功, ${failedCount} 失败, ${cachedCount} 缓存`, failedCount === 0 ? 'success' : 'error');
let html = `
<div style="margin-bottom: 16px; padding: 12px; background: var(--gray-50); border-radius: var(--radius);">
<div style="display: flex; gap: 24px; font-size: 14px;">
<span>总计: <strong style="color: var(--gray-900);">${totalSymbols}</strong></span>
<span>成功: <strong style="color: var(--success);">${successCount}</strong></span>
<span>失败: <strong style="color: var(--danger);">${failedCount}</strong></span>
<span>缓存: <strong style="color: var(--info);">${cachedCount}</strong></span>
</div>
</div>
<div class="table-container">
<table>
<thead><tr><th>品种</th><th>代码</th><th>状态</th><th>K线数</th><th>来源</th><th>最新价</th>${failedCount > 0 ? '<th>错误信息</th>' : ''}</tr></thead>
<tbody>
`;
for (const item of results) {
const statusBadge = item.status === 'cached'
? `<span class="badge badge-info">${item.statusText}</span>`
: item.status === 'success'
? `<span class="badge badge-success">${item.statusText}</span>`
: `<span class="badge badge-danger">${item.statusText}</span>`;
html += `<tr>
<td>${item.name}</td>
<td><code style="color: var(--primary);">${item.code}</code></td>
<td>${statusBadge}</td>
<td>${item.candleCount || '-'}</td>
<td><span class="badge badge-gray">${item.source}</span></td>
<td style="font-weight: 600;">${item.price}</td>
${failedCount > 0 ? `<td style="color: var(--danger); font-size: 13px;">${item.error || '-'}</td>` : ''}
</tr>`;
}
html += '</tbody></table></div>';
resultDiv.innerHTML = html;
resultDiv.classList.remove('hidden');
updateDashboard();
setTimeout(() => {
btn.disabled = false;
btn.innerHTML = '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="width:16px;height:16px"><polyline points="23 6 13.5 15.5 8.5 10.5 1 18"/><polyline points="17 6 23 6 23 12"/></svg> 一键批量获取';
}, 2000);
}
// Query Cache
let currentQueryData = null;
let klineChart = null;
async function queryCache() {
const symbol = document.getElementById('querySymbol').value.trim();
const period = document.getElementById('queryPeriod').value;
if (!symbol) return showToast('请输入品种代码', 'error');
addLog(`查询缓存: ${symbol} ${period || '全部'}`);
try {
const url = period ? `${API}/data/latest/${symbol}/${period}` : `${API}/data/latest/${symbol}`;
const res = await fetch(url);
const data = await res.json();
if (!res.ok) {
showToast(data.detail || '未找到缓存数据', 'error');
return;
}
addLog(`查询成功: ${symbol}, 缓存 ${data.timeframes ? data.timeframes.length : 0} 个周期`, 'success');
currentQueryData = data;
if (!data.timeframes || data.timeframes.length === 0) {
document.getElementById('queryResult').innerHTML = '<div class="empty-state"><p>暂无K线数据</p></div>';
document.getElementById('queryResult').classList.remove('hidden');
return;
}
renderKlineChart(data.timeframes[0], symbol);
} catch (e) {
showToast(`查询失败: ${e.message}`, 'error');
}
}
function renderKlineChart(timeframe, symbol) {
const resultContainer = document.getElementById('queryResult');
let html = `
<div class="chart-container">
<div class="chart-header">
<div class="chart-title">${symbol} K线图</div>
${timeframe.period === '__all__' || currentQueryData.timeframes.length > 1 ? `
<div class="period-tabs" id="periodTabs">
${currentQueryData.timeframes.map(tf => `
<div class="period-tab ${tf.period === timeframe.period ? 'active' : ''}"
onclick="switchPeriod('${tf.period}')">
${getPeriodLabel(tf.period)}
</div>
`).join('')}
</div>
` : ''}
</div>
<div id="klineChart"></div>
</div>
`;
resultContainer.innerHTML = html;
resultContainer.classList.remove('hidden');
setTimeout(() => initECharts(timeframe, symbol), 100);
}
function initECharts(timeframe, symbol) {
const chartDom = document.getElementById('klineChart');
if (!chartDom) return;
if (klineChart) {
klineChart.dispose();
}
klineChart = echarts.init(chartDom);
const candles = timeframe.candles || [];
if (candles.length === 0) {
chartDom.innerHTML = '<div class="empty-state"><p>该周期暂无数据</p></div>';
return;
}
const dates = candles.map(c => c.datetime);
const data = candles.map(c => [c.open, c.close, c.low, c.high]);
const volumes = candles.map((c, i) => ({
value: c.volume || 0,
index: i,
close: c.close,
open: c.open
}));
const upColor = '#ef4444';
const downColor = '#10b981';
const option = {
animation: true,
legend: {
bottom: 10,
left: 'center',
data: ['K线', 'MA5', 'MA10', 'MA20', 'MA60']
},
tooltip: {
trigger: 'axis',
axisPointer: {
type: 'cross',
crossStyle: {
color: '#999'
},
link: { xAxisIndex: 'all' }
},
formatter: function(params) {
if (!params || params.length === 0) return '';
const date = params[0].axisValue;
let result = `<div style="font-weight: 600; margin-bottom: 8px;">${date}</div>`;
params.forEach(param => {
if (param.seriesName === 'K线' && param.data) {
const [open, close, low, high] = param.data;
const change = ((close - open) / open * 100).toFixed(2);
result += `
<div>开: ${open} | 收: ${close}</div>
<div>低: ${low} | 高: ${high}</div>
<div>涨跌: ${change}%</div>
`;
} else if (param.seriesName.startsWith('MA')) {
result += `<div>${param.seriesName}: ${param.data}</div>`;
} else if (param.seriesName === '成交量') {
result += `<div>成交量: ${param.data}</div>`;
}
});
return result;
}
},
grid: [
{ left: 60, right: 40, height: '60%' },
{ left: 60, right: 40, top: '75%', height: '15%' }
],
xAxis: [
{
type: 'category',
data: dates,
gridIndex: 0,
axisLine: { lineStyle: { color: '#8392A5' } },
axisLabel: { show: false }
},
{
type: 'category',
data: dates,
gridIndex: 1,
axisLine: { lineStyle: { color: '#8392A5' } },
axisLabel: {
show: true,
formatter: function(value) {
return value.substring(5);
}
}
}
],
yAxis: [
{
scale: true,
gridIndex: 0,
splitArea: { show: true },
axisLine: { lineStyle: { color: '#8392A5' } },
splitLine: { lineStyle: { color: '#E8EEF4' } }
},
{
scale: true,
gridIndex: 1,
splitNumber: 2,
axisLine: { show: false },
axisTick: { show: false },
axisLabel: { show: false },
splitLine: { show: false }
}
],
dataZoom: [
{
type: 'inside',
xAxisIndex: [0, 1],
start: Math.max(0, 100 - 100 * 100 / candles.length),
end: 100
},
{
show: true,
xAxisIndex: [0, 1],
type: 'slider',
bottom: 10,
start: Math.max(0, 100 - 100 * 100 / candles.length),
end: 100,
height: 20
}
],
series: [
{
name: 'K线',
type: 'candlestick',
data: data,
itemStyle: {
color: upColor,
color0: downColor,
borderColor: upColor,
borderColor0: downColor
}
},
{
name: 'MA5',
type: 'line',
data: calculateMA(candles, 5),
smooth: true,
lineStyle: { width: 1 }
},
{
name: 'MA10',
type: 'line',
data: calculateMA(candles, 10),
smooth: true,
lineStyle: { width: 1 }
},
{
name: 'MA20',
type: 'line',
data: calculateMA(candles, 20),
smooth: true,
lineStyle: { width: 1 }
},
{
name: 'MA60',
type: 'line',
data: calculateMA(candles, 60),
smooth: true,
lineStyle: { width: 1 }
},
{
name: '成交量',
type: 'bar',
xAxisIndex: 1,
yAxisIndex: 1,
data: volumes.map(v => {
return {
value: v.value,
itemStyle: {
color: v.close >= v.open ? upColor : downColor
}
};
})
}
]
};
klineChart.setOption(option);
window.addEventListener('resize', () => {
if (klineChart) klineChart.resize();
});
}
function calculateMA(candles, days) {
const result = [];
for (let i = 0; i < candles.length; i++) {
if (i < days - 1) {
result.push('-');
} else {
let sum = 0;
for (let j = 0; j < days; j++) {
sum += candles[i - j].close;
}
result.push((sum / days).toFixed(2));
}
}
return result;
}
function getPeriodLabel(period) {
const map = {
'5min': '5分钟',
'15min': '15分钟',
'30min': '30分钟',
'60min': '60分钟',
'daily': '日线',
'__all__': '全部'
};
return map[period] || period;
}
function switchPeriod(period) {
if (!currentQueryData || !currentQueryData.timeframes) return;
const timeframe = currentQueryData.timeframes.find(tf => tf.period === period);
if (!timeframe) return;
const symbol = document.getElementById('querySymbol').value.trim();
renderKlineChart(timeframe, symbol);
}
// Batch Tasks
async function batchCreateTasks() {
const dataType = document.getElementById('taskDataType').value;
const periods = document.getElementById('taskPeriods').value;
const interval = document.getElementById('taskInterval').value;
const btn = document.getElementById('btnBatchTasks');
btn.disabled = true;
btn.innerHTML = '<div class="spinner"></div> 创建中...';
addLog(`批量创建定时任务: ${dataType}, 间隔 ${interval}s`);
try {
const res = await fetch(`${API}/config/batch-tasks`, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ data_type: dataType, periods, interval_seconds: parseInt(interval) }),
});
const data = await res.json();
addLog(`任务创建完成: 成功 ${data.created.length}, 失败 ${data.failed.length}`,
data.failed.length === 0 ? 'success' : 'error');
showToast(`成功创建 ${data.created.length} 个定时任务`, 'success');
loadTasks();
} catch (e) {
addLog(`创建任务异常: ${e.message}`, 'error');
} finally {
btn.disabled = false;
btn.innerHTML = '<svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="width:16px;height:16px"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg> 批量创建任务';
}
}
// Load Tasks
async function loadTasks() {
try {
const res = await fetch(`${API}/tasks`);
const data = await res.json();
if (!data.tasks.length) {
document.getElementById('taskTable').innerHTML = '<div class="empty-state"><svg viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg><p>暂无定时任务</p></div>';
return;
}
let html = `<table>
<thead><tr><th>ID</th><th>品种</th><th>周期</th><th>间隔</th><th>状态</th><th>最后执行</th><th>操作</th></tr></thead>
<tbody>`;
for (const t of data.tasks) {
const statusBadge = t.running
? '<span class="badge badge-success">运行中</span>'
: t.enabled
? '<span class="badge badge-warning">已停止</span>'
: '<span class="badge badge-gray">已禁用</span>';
html += `<tr>
<td>${t.id}</td>
<td><code style="color: var(--primary);">${t.symbol}</code></td>
<td>${t.periods.join(', ')}</td>
<td>${t.interval_seconds}s</td>
<td>${statusBadge}</td>
<td>${t.last_run ? new Date(t.last_run).toLocaleString() : '-'}</td>
<td>
${t.running
? `<button class="btn btn-warning btn-sm" onclick="stopTask(${t.id})">停止</button>`
: `<button class="btn btn-success btn-sm" onclick="startTask(${t.id})">启动</button>`
}
<button class="btn btn-danger btn-sm" onclick="deleteTask(${t.id})">删除</button>
</td>
</tr>`;
}
html += '</tbody></table>';
document.getElementById('taskTable').innerHTML = html;
} catch (e) {
addLog(`加载任务失败: ${e.message}`, 'error');
}
}
async function stopTask(id) {
await fetch(`${API}/tasks/${id}/stop`, { method: 'POST' });
showToast('任务已停止', 'success');
loadTasks();
}
async function startTask(id) {
await fetch(`${API}/tasks/${id}/start`, { method: 'POST' });
showToast('任务已启动', 'success');
loadTasks();
}
async function deleteTask(id) {
if (!confirm('确定删除此任务?')) return;
await fetch(`${API}/tasks/${id}`, { method: 'DELETE' });
showToast('任务已删除', 'success');
loadTasks();
}
// Cache Status
async function checkCacheStatus() {
const symbol = document.getElementById('cacheSymbol').value.trim();
if (!symbol) return showToast('请输入品种代码', 'error');
addLog(`查询缓存状态: ${symbol}`);
try {
const res = await fetch(`${API}/data/cache-status/${symbol}`);
const data = await res.json();
if (data.status === 'no_data') {
showToast(`${symbol} 无缓存数据`, 'info');
document.getElementById('cacheStatusResult').innerHTML = `<div class="empty-state"><p>品种 ${symbol} 暂无缓存数据</p></div>`;
document.getElementById('cacheStatusResult').classList.remove('hidden');
return;
}
let html = `<div class="table-container"><table>
<thead><tr><th>周期</th><th>K线数</th><th>获取时间</th><th>数据年龄</th><th>新鲜度</th></tr></thead><tbody>`;
for (const p of data.cached_periods) {
const fresh = p.is_fresh;
html += `<tr>
<td><span class="badge badge-info">${p.period}</span></td>
<td>${p.candle_count}</td>
<td>${new Date(p.fetched_at).toLocaleString()}</td>
<td>${p.age_seconds}秒</td>
<td><span class="badge ${fresh ? 'badge-success' : 'badge-warning'}">${fresh ? '新鲜' : '过期'}</span></td>
</tr>`;
}
html += '</tbody></table></div>';
document.getElementById('cacheStatusResult').innerHTML = html;
document.getElementById('cacheStatusResult').classList.remove('hidden');
addLog(`${symbol} 缓存状态: ${data.cached_periods.length} 个周期`, 'success');
} catch (e) {
showToast(`查询失败: ${e.message}`, 'error');
}
}
// Init
loadConfig();
loadTasks();
</script>
</body>
</html>