Критические фиксы: - main.py: монтировать /static из web/static/ (CSS не грузился совсем) - api/scans.py: filtered total count (был всегда общий, игнорируя фильтры) - web/routes.py: исправлен VALID_SORT_FIELDS (отсутствовали ключи packages) - web/routes.py: filtered total count для web scans list - package_detail.html: f.data.X вместо f.X (findings не отображались) Чистка мёртвого кода: - config.py: удалён _parse_repos и nexus_repositories (не использовались) - web/routes.py: удалён completed_scans/failed_scans (не отображались) - удалён мёртвый guarddog_nexus/static/style.css (67-байтный стаб) Качество кода: - web/routes.py: Jinja2 Environment кэшируется на уровне модуля - Вынесен дублирующийся JS в web/static/app.js - Вынесены дублирующиеся inline-стили в CSS-классы - Исправлен duplicate class attribute в списках - Удалены гигантские SVG из empty states Тесты: - 20 новых edge-case тестов (CSV export, search/filter/sort, 404, pagination) - Добавлен sample_flagged_scan fixture - Итого: 50 тестов, все зелёные
62 lines
2.7 KiB
HTML
62 lines
2.7 KiB
HTML
{% extends "base.html" %}
|
|
{% block title %}Scan #{{ scan.id }} — GuardDog Nexus{% endblock %}
|
|
{% block breadcrumbs %}
|
|
<div class="breadcrumbs">
|
|
<a href="/">Home</a>
|
|
<span class="separator">/</span>
|
|
<a href="/scans">Scans</a>
|
|
<span class="separator">/</span>
|
|
<span>Scan #{{ scan.id }}</span>
|
|
</div>
|
|
{% endblock %}
|
|
{% block content %}
|
|
<h1>Scan #{{ scan.id }}</h1>
|
|
|
|
<table>
|
|
<tr><td><strong>Package</strong></td><td><a href="/packages/{{ scan.package_name }}/{{ scan.package_version }}">{{ scan.package_name }}</a></td></tr>
|
|
<tr><td><strong>Version</strong></td><td>{{ scan.package_version }}</td></tr>
|
|
<tr><td><strong>Ecosystem</strong></td><td>{{ scan.ecosystem }}</td></tr>
|
|
<tr><td><strong>Repository</strong></td><td>{{ scan.repository }}</td></tr>
|
|
<tr><td><strong>Status</strong></td><td>
|
|
{% if scan.status == 'scanning' %}<span class="status-scanning"><span class="spinner"></span>scanning</span>{% else %}<span class="status-{{ scan.status }}">{{ scan.status }}</span>{% endif %}
|
|
</td></tr>
|
|
<tr><td><strong>SHA256</strong></td><td><code>{{ scan.sha256 or '-' }}</code></td></tr>
|
|
<tr><td><strong>Started</strong></td><td>{{ scan.started_at.isoformat() if scan.started_at }}</td></tr>
|
|
<tr><td><strong>Finished</strong></td><td>{{ scan.finished_at.isoformat() if scan.finished_at }}</td></tr>
|
|
{% if scan.error_message %}<tr><td><strong>Error</strong></td><td><span class="flagged">{{ scan.error_message }}</span></td></tr>{% endif %}
|
|
</table>
|
|
|
|
<div class="finding-header-row">
|
|
<h2>Findings ({{ scan.findings|length }})</h2>
|
|
{% if scan.findings|length > 1 %}
|
|
<button class="toggle-all-btn" onclick="toggleFindings()">Collapse All</button>
|
|
{% endif %}
|
|
</div>
|
|
|
|
{% if scan.findings %}
|
|
<div id="findings-container">
|
|
{% for f in scan.findings %}
|
|
<details class="finding-card {{ f.data.severity }}" data-finding-id="{{ f.id }}">
|
|
<summary class="finding-summary">
|
|
<strong class="severity-{{ f.data.severity }}">[{{ f.data.severity }}]</strong>
|
|
<strong>{{ f.data.rule }}</strong>
|
|
{% if f.data.location %}<small> @ {{ f.data.location }}</small>{% endif %}
|
|
<span class="finding-summary-hint">click to expand</span>
|
|
</summary>
|
|
<div class="finding-details">
|
|
<p>{{ f.data.message }}</p>
|
|
{% if f.data.code %}
|
|
<div class="code-toolbar">
|
|
<button class="copy-btn" onclick="copyCode(this, 'code-{{ f.id }}')">Copy</button>
|
|
</div>
|
|
<pre><code id="code-{{ f.id }}">{{ f.data.code }}</code></pre>
|
|
{% endif %}
|
|
</div>
|
|
</details>
|
|
{% endfor %}
|
|
</div>
|
|
{% else %}
|
|
<p class="empty-state">No findings — package looks clean.</p>
|
|
{% endif %}
|
|
{% endblock %}
|