|
|
<!DOCTYPE html> |
|
|
<html lang="pt-BR"> |
|
|
<head> |
|
|
<meta charset="UTF-8"> |
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"> |
|
|
<title>Sistema de Testes Massivos - AgentGraph</title> |
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/css/bootstrap.min.css" rel="stylesheet"> |
|
|
<link href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css" rel="stylesheet"> |
|
|
<link href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700;800&display=swap" rel="stylesheet"> |
|
|
<style> |
|
|
|
|
|
:root { |
|
|
--primary-color: #2563eb; |
|
|
--primary-dark: #1d4ed8; |
|
|
--secondary-color: #64748b; |
|
|
--success-color: #059669; |
|
|
--warning-color: #d97706; |
|
|
--danger-color: #dc2626; |
|
|
--info-color: #0891b2; |
|
|
|
|
|
--bg-primary: #f8fafc; |
|
|
--bg-secondary: #f1f5f9; |
|
|
--bg-dark: #0f172a; |
|
|
--bg-card: #ffffff; |
|
|
|
|
|
--text-primary: #1e293b; |
|
|
--text-secondary: #64748b; |
|
|
--text-muted: #94a3b8; |
|
|
|
|
|
--border-color: #e2e8f0; |
|
|
--border-radius: 12px; |
|
|
--shadow-sm: 0 1px 2px 0 rgb(0 0 0 / 0.05); |
|
|
--shadow-md: 0 4px 6px -1px rgb(0 0 0 / 0.1); |
|
|
--shadow-lg: 0 10px 15px -3px rgb(0 0 0 / 0.1); |
|
|
} |
|
|
|
|
|
|
|
|
body { |
|
|
font-family: 'Inter', -apple-system, BlinkMacSystemFont, sans-serif; |
|
|
background: var(--bg-primary); |
|
|
color: var(--text-primary); |
|
|
} |
|
|
|
|
|
.container-fluid { |
|
|
padding: 0; |
|
|
} |
|
|
|
|
|
|
|
|
.sidebar { |
|
|
background: linear-gradient(135deg, var(--bg-dark), #1e293b); |
|
|
min-height: 100vh; |
|
|
border-right: 1px solid var(--border-color); |
|
|
box-shadow: var(--shadow-lg); |
|
|
padding: 2rem 1.5rem; |
|
|
} |
|
|
|
|
|
.sidebar h4 { |
|
|
color: white; |
|
|
font-weight: 800; |
|
|
margin-bottom: 2rem; |
|
|
font-size: 1.5rem; |
|
|
} |
|
|
|
|
|
.sidebar h6 { |
|
|
color: #cbd5e1; |
|
|
font-weight: 600; |
|
|
margin-bottom: 1rem; |
|
|
margin-top: 2rem; |
|
|
font-size: 0.875rem; |
|
|
text-transform: uppercase; |
|
|
letter-spacing: 0.05em; |
|
|
} |
|
|
|
|
|
.sidebar .form-control, |
|
|
.sidebar .form-select { |
|
|
background: rgba(255, 255, 255, 0.1); |
|
|
border: 1px solid rgba(255, 255, 255, 0.2); |
|
|
color: white; |
|
|
border-radius: 8px; |
|
|
} |
|
|
|
|
|
.sidebar .form-control::placeholder { |
|
|
color: rgba(255, 255, 255, 0.6); |
|
|
} |
|
|
|
|
|
.sidebar .form-control:focus, |
|
|
.sidebar .form-select:focus { |
|
|
background: rgba(255, 255, 255, 0.15); |
|
|
border-color: var(--primary-color); |
|
|
box-shadow: 0 0 0 3px rgb(37 99 235 / 0.2); |
|
|
color: white; |
|
|
} |
|
|
|
|
|
|
|
|
.main-content { |
|
|
background: var(--bg-primary); |
|
|
min-height: 100vh; |
|
|
padding: 2rem; |
|
|
} |
|
|
|
|
|
|
|
|
.dashboard-header { |
|
|
background: var(--bg-card); |
|
|
border-radius: var(--border-radius); |
|
|
padding: 2rem; |
|
|
margin-bottom: 2rem; |
|
|
border: 1px solid var(--border-color); |
|
|
box-shadow: var(--shadow-sm); |
|
|
} |
|
|
|
|
|
.dashboard-header h2 { |
|
|
color: var(--text-primary); |
|
|
font-weight: 800; |
|
|
margin: 0; |
|
|
font-size: 2rem; |
|
|
} |
|
|
|
|
|
|
|
|
.metric-card { |
|
|
background: linear-gradient(135deg, var(--primary-color), var(--info-color)); |
|
|
color: white; |
|
|
border-radius: var(--border-radius); |
|
|
padding: 2rem; |
|
|
margin: 1rem 0; |
|
|
box-shadow: var(--shadow-md); |
|
|
transition: all 0.3s ease; |
|
|
border: none; |
|
|
} |
|
|
|
|
|
.metric-card:hover { |
|
|
transform: translateY(-2px); |
|
|
box-shadow: var(--shadow-lg); |
|
|
} |
|
|
|
|
|
.metric-value { |
|
|
font-size: 3rem; |
|
|
font-weight: 800; |
|
|
line-height: 1; |
|
|
margin-bottom: 0.5rem; |
|
|
} |
|
|
|
|
|
.metric-label { |
|
|
font-size: 0.875rem; |
|
|
font-weight: 600; |
|
|
opacity: 0.9; |
|
|
text-transform: uppercase; |
|
|
letter-spacing: 0.05em; |
|
|
} |
|
|
|
|
|
.test-group-card { |
|
|
background: var(--bg-card); |
|
|
border: 1px solid var(--border-color); |
|
|
border-radius: var(--border-radius); |
|
|
padding: 2rem; |
|
|
margin: 1.5rem 0; |
|
|
box-shadow: var(--shadow-sm); |
|
|
transition: all 0.3s ease; |
|
|
} |
|
|
|
|
|
.test-group-card:hover { |
|
|
box-shadow: var(--shadow-md); |
|
|
border-color: var(--primary-color); |
|
|
transform: translateY(-1px); |
|
|
} |
|
|
|
|
|
|
|
|
.status-badge { |
|
|
padding: 0.75rem 1.5rem; |
|
|
border-radius: 50px; |
|
|
font-weight: 700; |
|
|
font-size: 0.875rem; |
|
|
display: inline-flex; |
|
|
align-items: center; |
|
|
gap: 0.5rem; |
|
|
box-shadow: var(--shadow-sm); |
|
|
text-transform: uppercase; |
|
|
letter-spacing: 0.05em; |
|
|
} |
|
|
|
|
|
.status-idle { |
|
|
background: var(--secondary-color); |
|
|
color: white; |
|
|
} |
|
|
.status-running { |
|
|
background: var(--success-color); |
|
|
color: white; |
|
|
animation: pulse 2s infinite; |
|
|
} |
|
|
.status-completed { |
|
|
background: var(--primary-color); |
|
|
color: white; |
|
|
} |
|
|
.status-error { |
|
|
background: var(--danger-color); |
|
|
color: white; |
|
|
} |
|
|
|
|
|
@keyframes pulse { |
|
|
0%, 100% { opacity: 1; } |
|
|
50% { opacity: 0.8; } |
|
|
} |
|
|
|
|
|
|
|
|
.progress-container { |
|
|
display: none; |
|
|
margin: 2rem 0; |
|
|
background: var(--bg-card); |
|
|
border-radius: var(--border-radius); |
|
|
padding: 2rem; |
|
|
border: 1px solid var(--border-color); |
|
|
box-shadow: var(--shadow-md); |
|
|
} |
|
|
|
|
|
.results-container { |
|
|
display: none; |
|
|
margin: 2rem 0; |
|
|
} |
|
|
|
|
|
|
|
|
.btn { |
|
|
border-radius: 10px; |
|
|
font-weight: 600; |
|
|
padding: 0.875rem 2rem; |
|
|
border: none; |
|
|
transition: all 0.3s ease; |
|
|
box-shadow: var(--shadow-sm); |
|
|
text-transform: uppercase; |
|
|
letter-spacing: 0.05em; |
|
|
font-size: 0.875rem; |
|
|
} |
|
|
|
|
|
.btn:hover { |
|
|
transform: translateY(-2px); |
|
|
box-shadow: var(--shadow-md); |
|
|
} |
|
|
|
|
|
.btn-primary { |
|
|
background: linear-gradient(135deg, var(--primary-color), var(--primary-dark)); |
|
|
border: none; |
|
|
} |
|
|
|
|
|
.btn-success { |
|
|
background: linear-gradient(135deg, var(--success-color), #047857); |
|
|
} |
|
|
|
|
|
.btn-warning { |
|
|
background: linear-gradient(135deg, var(--warning-color), #b45309); |
|
|
} |
|
|
|
|
|
.btn-danger { |
|
|
background: linear-gradient(135deg, var(--danger-color), #b91c1c); |
|
|
} |
|
|
|
|
|
|
|
|
.progress { |
|
|
height: 16px; |
|
|
border-radius: 8px; |
|
|
background: var(--bg-secondary); |
|
|
overflow: hidden; |
|
|
box-shadow: inset 0 1px 2px rgba(0,0,0,0.1); |
|
|
} |
|
|
|
|
|
.progress-bar { |
|
|
background: linear-gradient(90deg, var(--primary-color), var(--info-color)); |
|
|
border-radius: 8px; |
|
|
transition: width 0.6s ease; |
|
|
position: relative; |
|
|
overflow: hidden; |
|
|
} |
|
|
|
|
|
.progress-bar::after { |
|
|
content: ''; |
|
|
position: absolute; |
|
|
top: 0; |
|
|
left: 0; |
|
|
right: 0; |
|
|
bottom: 0; |
|
|
background: linear-gradient(45deg, transparent 35%, rgba(255,255,255,0.2) 50%, transparent 65%); |
|
|
animation: shimmer 2s infinite; |
|
|
} |
|
|
|
|
|
@keyframes shimmer { |
|
|
0% { transform: translateX(-100%); } |
|
|
100% { transform: translateX(100%); } |
|
|
} |
|
|
|
|
|
|
|
|
.alert { |
|
|
border-radius: var(--border-radius); |
|
|
border: none; |
|
|
box-shadow: var(--shadow-sm); |
|
|
padding: 1.5rem; |
|
|
} |
|
|
|
|
|
|
|
|
.form-control, .form-select { |
|
|
border-radius: 10px; |
|
|
border: 2px solid var(--border-color); |
|
|
padding: 0.875rem 1rem; |
|
|
transition: all 0.3s ease; |
|
|
font-weight: 500; |
|
|
} |
|
|
|
|
|
.form-control:focus, .form-select:focus { |
|
|
border-color: var(--primary-color); |
|
|
box-shadow: 0 0 0 4px rgb(37 99 235 / 0.1); |
|
|
transform: translateY(-1px); |
|
|
} |
|
|
|
|
|
|
|
|
.badge { |
|
|
font-weight: 700; |
|
|
padding: 0.5rem 1rem; |
|
|
border-radius: 8px; |
|
|
font-size: 0.75rem; |
|
|
text-transform: uppercase; |
|
|
letter-spacing: 0.05em; |
|
|
} |
|
|
|
|
|
|
|
|
.fas, .far { |
|
|
margin-right: 0.5rem; |
|
|
} |
|
|
|
|
|
|
|
|
@media (max-width: 768px) { |
|
|
.main-content { |
|
|
padding: 1rem; |
|
|
} |
|
|
|
|
|
.metric-value { |
|
|
font-size: 2rem; |
|
|
} |
|
|
|
|
|
.sidebar { |
|
|
padding: 1rem; |
|
|
} |
|
|
} |
|
|
</style> |
|
|
</head> |
|
|
<body> |
|
|
<div class="container-fluid"> |
|
|
<div class="row"> |
|
|
|
|
|
<div class="col-md-3 sidebar"> |
|
|
<h4><i class="fas fa-flask"></i> Sistema de Testes</h4> |
|
|
|
|
|
|
|
|
<div class="mb-4"> |
|
|
<h6><i class="fas fa-question-circle"></i> Pergunta do Teste</h6> |
|
|
<textarea id="testQuestion" class="form-control" rows="3" |
|
|
placeholder="Digite a pergunta que será testada..."></textarea> |
|
|
<button id="createSession" class="btn btn-primary btn-sm mt-2 w-100"> |
|
|
<i class="fas fa-plus"></i> Criar Sessão |
|
|
</button> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="groupConfig" style="display: none;"> |
|
|
<h6><i class="fas fa-cogs"></i> Configurar Grupo</h6> |
|
|
|
|
|
<div class="mb-3"> |
|
|
<label class="form-label">Modelo SQL Agent</label> |
|
|
<select id="sqlModel" class="form-select form-select-sm"> |
|
|
{% for name, value in available_models.items() %} |
|
|
<option value="{{ value }}">{{ name }}</option> |
|
|
{% endfor %} |
|
|
</select> |
|
|
</div> |
|
|
|
|
|
<div class="mb-3"> |
|
|
<div class="form-check"> |
|
|
<input class="form-check-input" type="checkbox" id="enableProcessing"> |
|
|
<label class="form-check-label" for="enableProcessing"> |
|
|
Ativar Processing Agent |
|
|
</label> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
<div id="processingModelDiv" class="mb-3" style="display: none;"> |
|
|
<label class="form-label">Modelo Processing Agent</label> |
|
|
<select id="processingModel" class="form-select form-select-sm"> |
|
|
{% for name, value in available_models.items() %} |
|
|
<option value="{{ value }}">{{ name }}</option> |
|
|
{% endfor %} |
|
|
</select> |
|
|
</div> |
|
|
|
|
|
<div class="mb-3"> |
|
|
<label class="form-label">Número de Iterações</label> |
|
|
<input type="number" id="iterations" class="form-control form-control-sm" |
|
|
value="5" min="1" max="100"> |
|
|
</div> |
|
|
|
|
|
<button id="addGroup" class="btn btn-success btn-sm w-100 mb-2"> |
|
|
<i class="fas fa-plus"></i> Adicionar Grupo |
|
|
</button> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="validationConfig" style="display: none;"> |
|
|
<hr> |
|
|
<h6><i class="fas fa-check-circle"></i> Validação</h6> |
|
|
|
|
|
<div class="mb-3"> |
|
|
<label class="form-label">Método</label> |
|
|
<select id="validationMethod" class="form-select form-select-sm"> |
|
|
<option value="llm">LLM (Automático)</option> |
|
|
<option value="keyword">Palavra-chave</option> |
|
|
</select> |
|
|
</div> |
|
|
|
|
|
<div id="keywordDiv" class="mb-3" style="display: none;"> |
|
|
<label class="form-label">Conteúdo Esperado</label> |
|
|
<input type="text" id="expectedContent" class="form-control form-control-sm" |
|
|
placeholder="Texto que deve aparecer na resposta"> |
|
|
</div> |
|
|
|
|
|
<button id="runTests" class="btn btn-warning btn-sm w-100"> |
|
|
<i class="fas fa-play"></i> Executar Testes |
|
|
</button> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div class="col-md-9 main-content"> |
|
|
<div class="dashboard-header"> |
|
|
<div class="d-flex justify-content-between align-items-center"> |
|
|
<h2><i class="fas fa-chart-line"></i> Dashboard de Testes</h2> |
|
|
<div id="sessionStatus" class="status-badge status-idle"> |
|
|
<i class="fas fa-circle"></i> Aguardando |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="sessionInfo" class="alert alert-info" style="display: none;"> |
|
|
<h6><i class="fas fa-info-circle"></i> Informações da Sessão</h6> |
|
|
<div id="sessionDetails"></div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="testGroups"> |
|
|
<h5><i class="fas fa-layer-group"></i> Grupos de Teste</h5> |
|
|
<div id="groupsList"> |
|
|
<p class="text-muted">Nenhum grupo configurado ainda.</p> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="progressContainer" class="progress-container"> |
|
|
<h5><i class="fas fa-tasks"></i> Progresso dos Testes</h5> |
|
|
<div class="progress mb-3"> |
|
|
<div id="progressBar" class="progress-bar progress-bar-striped progress-bar-animated" |
|
|
role="progressbar" style="width: 0%"></div> |
|
|
</div> |
|
|
<div id="progressDetails" class="row"> |
|
|
<div class="col-md-3"> |
|
|
<div class="metric-card"> |
|
|
<div class="metric-value" id="completedTests">0</div> |
|
|
<div class="metric-label">Testes Concluídos</div> |
|
|
</div> |
|
|
</div> |
|
|
<div class="col-md-3"> |
|
|
<div class="metric-card"> |
|
|
<div class="metric-value" id="totalTests">0</div> |
|
|
<div class="metric-label">Total de Testes</div> |
|
|
</div> |
|
|
</div> |
|
|
<div class="col-md-3"> |
|
|
<div class="metric-card"> |
|
|
<div class="metric-value" id="currentGroup">-</div> |
|
|
<div class="metric-label">Grupo Atual</div> |
|
|
</div> |
|
|
</div> |
|
|
<div class="col-md-3"> |
|
|
<div class="metric-card"> |
|
|
<div class="metric-value" id="estimatedTime">-</div> |
|
|
<div class="metric-label">Tempo Restante</div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div class="card mt-3"> |
|
|
<div class="card-header"> |
|
|
<h6><i class="fas fa-stop-circle"></i> Controles de Teste</h6> |
|
|
</div> |
|
|
<div class="card-body"> |
|
|
<div class="row"> |
|
|
<div class="col-md-6"> |
|
|
<div id="currentTestInfo" class="mb-3"> |
|
|
<strong>Teste Atual:</strong> |
|
|
<div id="currentTestDetails" class="text-muted">Nenhum teste em execução</div> |
|
|
</div> |
|
|
<div id="runningTestsInfo" class="mb-3"> |
|
|
<strong>Testes em Execução:</strong> |
|
|
<span id="runningTestsCount" class="badge bg-primary">0</span> |
|
|
</div> |
|
|
</div> |
|
|
<div class="col-md-6"> |
|
|
<div class="d-grid gap-2"> |
|
|
<button id="cancelCurrentBtn" class="btn btn-warning btn-sm" onclick="cancelCurrentTest()"> |
|
|
<i class="fas fa-stop"></i> Cancelar Teste Atual |
|
|
</button> |
|
|
<button id="skipStuckBtn" class="btn btn-danger btn-sm" onclick="skipStuckTests()"> |
|
|
<i class="fas fa-forward"></i> Pular Testes Travados (>2min) |
|
|
</button> |
|
|
<button id="cancelAllBtn" class="btn btn-dark btn-sm" onclick="cancelAllTests()"> |
|
|
<i class="fas fa-ban"></i> Cancelar Todos os Testes |
|
|
</button> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="runningTestsList" class="mt-3" style="display: none;"> |
|
|
<h6>Testes em Execução:</h6> |
|
|
<div id="runningTestsContainer" class="list-group"> |
|
|
|
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<div id="resultsContainer" class="results-container"> |
|
|
<div class="d-flex justify-content-between align-items-center mb-3"> |
|
|
<h5><i class="fas fa-chart-bar"></i> Resultados</h5> |
|
|
<button id="downloadCsv" class="btn btn-outline-primary btn-sm"> |
|
|
<i class="fas fa-download"></i> Baixar CSV |
|
|
</button> |
|
|
</div> |
|
|
|
|
|
|
|
|
<ul class="nav nav-tabs" id="resultsTabs" role="tablist"> |
|
|
<li class="nav-item" role="presentation"> |
|
|
<button class="nav-link active" id="summary-tab" data-bs-toggle="tab" |
|
|
data-bs-target="#summary" type="button" role="tab"> |
|
|
<i class="fas fa-chart-pie"></i> Resumo |
|
|
</button> |
|
|
</li> |
|
|
<li class="nav-item" role="presentation"> |
|
|
<button class="nav-link" id="groups-tab" data-bs-toggle="tab" |
|
|
data-bs-target="#groups" type="button" role="tab"> |
|
|
<i class="fas fa-layer-group"></i> Por Grupo |
|
|
</button> |
|
|
</li> |
|
|
<li class="nav-item" role="presentation"> |
|
|
<button class="nav-link" id="individual-tab" data-bs-toggle="tab" |
|
|
data-bs-target="#individual" type="button" role="tab"> |
|
|
<i class="fas fa-list"></i> Individual |
|
|
</button> |
|
|
</li> |
|
|
</ul> |
|
|
|
|
|
<div class="tab-content" id="resultsTabContent"> |
|
|
<div class="tab-pane fade show active" id="summary" role="tabpanel"> |
|
|
<div id="summaryContent" class="p-3"></div> |
|
|
</div> |
|
|
<div class="tab-pane fade" id="groups" role="tabpanel"> |
|
|
<div id="groupsContent" class="p-3"></div> |
|
|
</div> |
|
|
<div class="tab-pane fade" id="individual" role="tabpanel"> |
|
|
<div id="individualContent" class="p-3"></div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
</div> |
|
|
|
|
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/js/bootstrap.bundle.min.js"></script> |
|
|
<script src="{{ url_for('static', filename='js/app.js') }}"></script> |
|
|
</body> |
|
|
</html> |
|
|
|