Files
autarch/web/templates/targets.html

588 lines
38 KiB
HTML
Raw Normal View History

2026-03-13 15:17:15 -07:00
{% extends "base.html" %}
{% block title %}Investigations - AUTARCH{% endblock %}
2026-03-13 15:17:15 -07:00
{% block content %}
<div class="page-header" style="display:flex;align-items:center;gap:1rem;flex-wrap:wrap">
<div>
<h1>Investigations</h1>
2026-03-13 15:17:15 -07:00
<p style="margin:0;font-size:0.85rem;color:var(--text-secondary)">
Investigation profiles for IPs, domains, threat actors, and incidents. Build detailed profiles with GeoIP, traceroute, DNS, WHOIS, and custom fields.
2026-03-13 15:17:15 -07:00
</p>
</div>
<div style="margin-left:auto;display:flex;gap:0.5rem;flex-wrap:wrap">
<button class="btn btn-sm btn-primary" onclick="toggleAddForm()">+ New Investigation</button>
<button class="btn btn-sm" onclick="exportTargets()">Export</button>
<label class="btn btn-sm" style="cursor:pointer">Import<input type="file" accept=".json" style="display:none" onchange="importTargets(this)"></label>
2026-03-13 15:17:15 -07:00
</div>
</div>
<!-- Add Investigation Form -->
2026-03-13 15:17:15 -07:00
<div id="add-form" class="section" style="display:none">
<h2>New Investigation Profile</h2>
<!-- Basic Info -->
<fieldset style="border:1px solid var(--border);border-radius:var(--radius);padding:0.75rem;margin-bottom:0.75rem">
<legend style="font-size:0.85rem;font-weight:600;color:var(--accent);padding:0 0.5rem">Basic Info</legend>
<div style="display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:0.75rem">
<div class="form-group" style="margin:0"><label>Host / IP <span style="color:var(--danger)">*</span></label><input type="text" id="add-host" placeholder="192.168.1.1 or example.com"></div>
<div class="form-group" style="margin:0"><label>Name / Label</label><input type="text" id="add-name" placeholder="Investigation name"></div>
<div class="form-group" style="margin:0"><label>Type</label><select id="add-type"><option value="ip">IP</option><option value="cidr">CIDR</option><option value="domain">Domain</option><option value="url">URL</option><option value="email">Email</option><option value="actor">Threat Actor</option><option value="incident">Incident</option></select></div>
<div class="form-group" style="margin:0"><label>Status</label><select id="add-status"><option value="active">Active</option><option value="pending">Pending</option><option value="completed">Completed</option><option value="escalated">Escalated</option><option value="closed">Closed</option></select></div>
<div class="form-group" style="margin:0"><label>Threat Level</label><select id="add-threat"><option value="unknown">Unknown</option><option value="low">Low</option><option value="medium">Medium</option><option value="high">High</option><option value="critical">Critical</option></select></div>
<div class="form-group" style="margin:0"><label>OS</label><select id="add-os"><option>Unknown</option><option>Linux</option><option>Windows</option><option>macOS</option><option>Android</option><option>iOS</option><option>Network Device</option></select></div>
<div class="form-group" style="margin:0"><label>Source</label><input type="text" id="add-source" placeholder="fail2ban, IDS, manual"></div>
<div class="form-group" style="margin:0"><label>Tags</label><input type="text" id="add-tags" placeholder="brute-force, ssh, external"></div>
2026-03-13 15:17:15 -07:00
</div>
</fieldset>
<!-- Network Info -->
<fieldset style="border:1px solid var(--border);border-radius:var(--radius);padding:0.75rem;margin-bottom:0.75rem">
<legend style="font-size:0.85rem;font-weight:600;color:var(--accent);padding:0 0.5rem">Network</legend>
<div style="display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:0.75rem">
<div class="form-group" style="margin:0"><label>IPv4</label><input type="text" id="add-ipv4" placeholder="192.168.1.1"></div>
<div class="form-group" style="margin:0"><label>IPv6</label><input type="text" id="add-ipv6" placeholder="::1"></div>
<div class="form-group" style="margin:0"><label>Domain</label><input type="text" id="add-domain" placeholder="example.com"></div>
<div class="form-group" style="margin:0"><label>Reverse DNS</label><input type="text" id="add-rdns" placeholder="host.isp.net"></div>
<div class="form-group" style="margin:0"><label>MAC Address</label><input type="text" id="add-mac" placeholder="aa:bb:cc:dd:ee:ff"></div>
<div class="form-group" style="margin:0"><label>Hostname</label><input type="text" id="add-hostname" placeholder="server01"></div>
<div class="form-group" style="margin:0"><label>Ports / Services</label><input type="text" id="add-ports" placeholder="22/ssh, 80/http, 443/https"></div>
2026-03-13 15:17:15 -07:00
</div>
</fieldset>
<!-- GeoIP -->
<fieldset style="border:1px solid var(--border);border-radius:var(--radius);padding:0.75rem;margin-bottom:0.75rem">
<legend style="font-size:0.85rem;font-weight:600;color:var(--accent);padding:0 0.5rem">GeoIP</legend>
<div style="display:grid;grid-template-columns:repeat(auto-fill,minmax(200px,1fr));gap:0.75rem">
<div class="form-group" style="margin:0"><label>Country</label><input type="text" id="add-geo-country" placeholder="US"></div>
<div class="form-group" style="margin:0"><label>City</label><input type="text" id="add-geo-city" placeholder="New York"></div>
<div class="form-group" style="margin:0"><label>ISP</label><input type="text" id="add-geo-isp" placeholder="Comcast"></div>
<div class="form-group" style="margin:0"><label>ASN</label><input type="text" id="add-geo-asn" placeholder="AS7922"></div>
<div class="form-group" style="margin:0"><label>Coordinates</label><input type="text" id="add-geo-coords" placeholder="40.7128,-74.0060"></div>
2026-03-13 15:17:15 -07:00
</div>
</fieldset>
<!-- Intel -->
<fieldset style="border:1px solid var(--border);border-radius:var(--radius);padding:0.75rem;margin-bottom:0.75rem">
<legend style="font-size:0.85rem;font-weight:600;color:var(--accent);padding:0 0.5rem">Intelligence</legend>
<div style="display:grid;grid-template-columns:repeat(auto-fill,minmax(250px,1fr));gap:0.75rem">
<div class="form-group" style="margin:0"><label>DNS Records</label><textarea id="add-dns" rows="2" placeholder="A: 1.2.3.4&#10;MX: mail.example.com" style="font-family:monospace;font-size:0.78rem"></textarea></div>
<div class="form-group" style="margin:0"><label>Email(s)</label><input type="text" id="add-email" placeholder="admin@example.com, abuse@isp.net"></div>
<div class="form-group" style="margin:0"><label>Username(s)</label><input type="text" id="add-usernames" placeholder="root, admin, attacker123"></div>
<div class="form-group" style="margin:0"><label>Vulnerabilities</label><textarea id="add-vulns" rows="2" placeholder="CVE-2024-1234, open SSH" style="font-family:monospace;font-size:0.78rem"></textarea></div>
2026-03-13 15:17:15 -07:00
</div>
<div class="form-group" style="margin:0.5rem 0 0"><label>Traceroute</label><textarea id="add-traceroute" rows="3" placeholder="Paste traceroute output" style="font-family:monospace;font-size:0.78rem;width:100%"></textarea></div>
<div class="form-group" style="margin:0.5rem 0 0"><label>WHOIS</label><textarea id="add-whois" rows="3" placeholder="Paste WHOIS output" style="font-family:monospace;font-size:0.78rem;width:100%"></textarea></div>
<div class="form-group" style="margin:0.5rem 0 0"><label>Notes</label><textarea id="add-notes" rows="3" placeholder="Investigation notes, observations, timeline..." style="font-size:0.82rem;width:100%"></textarea></div>
</fieldset>
<!-- Custom Fields -->
<fieldset style="border:1px solid var(--border);border-radius:var(--radius);padding:0.75rem;margin-bottom:0.75rem">
<legend style="font-size:0.85rem;font-weight:600;color:var(--accent);padding:0 0.5rem">Custom Fields</legend>
<div id="custom-fields-container"></div>
<button class="btn btn-sm" onclick="addCustomField()" style="margin-top:0.5rem">+ Add Field</button>
</fieldset>
<div style="display:flex;gap:0.5rem;margin-top:0.75rem">
<button class="btn btn-primary" onclick="addTarget()">Create Investigation</button>
2026-03-13 15:17:15 -07:00
<button class="btn btn-sm" onclick="toggleAddForm()">Cancel</button>
<span id="add-status-msg" style="font-size:0.82rem;color:var(--text-secondary);align-self:center"></span>
</div>
</div>
<!-- Investigation Reports -->
<div class="section">
<div style="display:flex;justify-content:space-between;align-items:center;margin-bottom:0.75rem">
<h2 style="margin:0">Investigation Reports</h2>
<button class="btn btn-sm" onclick="loadIRs()">Refresh</button>
</div>
<div id="ir-list" style="margin-bottom:0.5rem"><p style="color:var(--text-muted)">Loading reports...</p></div>
</div>
2026-03-13 15:17:15 -07:00
<!-- Filter / Search -->
<div class="section" style="padding:0.6rem 1rem">
<div style="display:flex;gap:0.75rem;align-items:center;flex-wrap:wrap">
<input type="text" id="filter-input" placeholder="Filter by host, name, tag…"
oninput="filterTargets()" style="flex:1;max-width:320px">
<select id="filter-status" onchange="filterTargets()" style="width:150px">
<option value="">All Statuses</option>
<option value="active">Active</option>
<option value="pending">Pending</option>
<option value="completed">Completed</option>
<option value="out-of-scope">Out of Scope</option>
</select>
<select id="filter-type" onchange="filterTargets()" style="width:140px">
<option value="">All Types</option>
<option value="ip">IP</option>
<option value="cidr">CIDR</option>
<option value="domain">Domain</option>
<option value="url">URL</option>
</select>
<span id="filter-count" style="font-size:0.8rem;color:var(--text-secondary)"></span>
</div>
</div>
<!-- Targets Table -->
<div class="section" style="padding:0">
<div style="overflow-x:auto">
<table id="targets-table" style="width:100%;border-collapse:collapse;font-size:0.85rem">
<thead>
<tr style="border-bottom:2px solid var(--border)">
<th style="padding:0.6rem 1rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">#</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Host</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Name</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Type</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">OS</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Status</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Ports</th>
<th style="padding:0.6rem 0.75rem;text-align:left;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Tags</th>
<th style="padding:0.6rem 0.75rem;text-align:right;font-size:0.75rem;text-transform:uppercase;letter-spacing:0.05em;color:var(--text-secondary)">Actions</th>
</tr>
</thead>
<tbody id="targets-body">
{% if targets %}
{% for t in targets %}
<tr class="target-row" data-id="{{ t.id }}"
data-host="{{ t.host }}" data-name="{{ t.name }}"
data-type="{{ t.type }}" data-status="{{ t.status }}"
data-tags="{{ t.tags | join(',') }}"
style="border-bottom:1px solid var(--border)">
<td style="padding:0.55rem 1rem;color:var(--text-muted);font-size:0.75rem">{{ loop.index }}</td>
<td style="padding:0.55rem 0.75rem;font-family:monospace;font-weight:600">{{ t.host }}</td>
<td style="padding:0.55rem 0.75rem;color:var(--text-secondary)">{{ t.name if t.name != t.host else '' }}</td>
<td style="padding:0.55rem 0.75rem"><span class="type-badge type-{{ t.type }}">{{ t.type }}</span></td>
<td style="padding:0.55rem 0.75rem;color:var(--text-secondary);font-size:0.8rem">{{ t.os }}</td>
<td style="padding:0.55rem 0.75rem">
<span class="status-badge status-{{ t.status | replace('-','_') }}"
onclick="cycleStatus('{{ t.id }}', this)"
title="Click to cycle status" style="cursor:pointer">{{ t.status }}</span>
</td>
<td style="padding:0.55rem 0.75rem;font-family:monospace;font-size:0.78rem;color:var(--text-secondary)">{{ t.ports or '—' }}</td>
<td style="padding:0.55rem 0.75rem">
{% for tag in t.tags %}<span class="tag-chip">{{ tag }}</span>{% endfor %}
</td>
<td style="padding:0.55rem 0.75rem;text-align:right;white-space:nowrap">
<button class="btn btn-xs" onclick="copyHost('{{ t.host }}')" title="Copy host">&#x2398;</button>
<a href="/osint?target={{ t.host | urlencode }}" class="btn btn-xs" title="OSINT scan">&#x1F50D;</a>
<button class="btn btn-xs" onclick="toggleEdit('{{ t.id }}')" title="Edit">&#x270E;</button>
<button class="btn btn-xs btn-danger" onclick="deleteTarget('{{ t.id }}')" title="Delete">&#x2715;</button>
</td>
</tr>
<!-- Edit row (hidden) -->
<tr id="edit-{{ t.id }}" style="display:none;background:var(--bg-card)">
<td colspan="9" style="padding:0.75rem 1rem;border-bottom:2px solid var(--accent)">
<div style="display:grid;grid-template-columns:repeat(auto-fill,minmax(160px,1fr));gap:0.5rem 0.75rem">
<div class="form-group" style="margin-bottom:0">
<label style="font-size:0.72rem">Host</label>
<input type="text" id="e-host-{{ t.id }}" value="{{ t.host }}">
</div>
<div class="form-group" style="margin-bottom:0">
<label style="font-size:0.72rem">Name</label>
<input type="text" id="e-name-{{ t.id }}" value="{{ t.name }}">
</div>
<div class="form-group" style="margin-bottom:0">
<label style="font-size:0.72rem">Type</label>
<select id="e-type-{{ t.id }}">
<option value="ip" {{ 'selected' if t.type=='ip' }}>IP</option>
<option value="cidr" {{ 'selected' if t.type=='cidr' }}>CIDR</option>
<option value="domain" {{ 'selected' if t.type=='domain' }}>Domain</option>
<option value="url" {{ 'selected' if t.type=='url' }}>URL</option>
</select>
</div>
<div class="form-group" style="margin-bottom:0">
<label style="font-size:0.72rem">OS</label>
<select id="e-os-{{ t.id }}">
{% for os in ['Unknown','Linux','Windows','macOS','Android','iOS','Network Device','Other'] %}
<option {{ 'selected' if t.os==os }}>{{ os }}</option>
{% endfor %}
</select>
</div>
<div class="form-group" style="margin-bottom:0">
<label style="font-size:0.72rem">Status</label>
<select id="e-status-{{ t.id }}">
{% for s in ['active','pending','completed','out-of-scope'] %}
<option value="{{ s }}" {{ 'selected' if t.status==s }}>{{ s }}</option>
{% endfor %}
</select>
</div>
<div class="form-group" style="margin-bottom:0">
<label style="font-size:0.72rem">Ports</label>
<input type="text" id="e-ports-{{ t.id }}" value="{{ t.ports }}">
</div>
<div class="form-group" style="margin-bottom:0;grid-column:span 2">
<label style="font-size:0.72rem">Tags (comma-separated)</label>
<input type="text" id="e-tags-{{ t.id }}" value="{{ t.tags | join(', ') }}">
</div>
<div class="form-group" style="margin-bottom:0;grid-column:span 2">
<label style="font-size:0.72rem">Notes</label>
<input type="text" id="e-notes-{{ t.id }}" value="{{ t.notes }}">
</div>
</div>
<div style="display:flex;gap:0.5rem;margin-top:0.6rem">
<button class="btn btn-primary btn-sm" onclick="saveEdit('{{ t.id }}')">Save</button>
<button class="btn btn-sm" onclick="toggleEdit('{{ t.id }}')">Cancel</button>
</div>
</td>
</tr>
{% endfor %}
{% else %}
<tr id="empty-row">
<td colspan="9" style="padding:2rem;text-align:center;color:var(--text-muted)">
No targets yet — click <strong>+ Add Target</strong> to add your first one.
</td>
</tr>
{% endif %}
</tbody>
</table>
</div>
</div>
<style>
/* Status badges */
.status-badge {
display:inline-block; padding:2px 8px; border-radius:10px;
font-size:0.72rem; font-weight:600; text-transform:uppercase; letter-spacing:0.04em;
}
.status-active { background:rgba(52,199,89,0.15); color:#34c759; border:1px solid rgba(52,199,89,0.3); }
.status-pending { background:rgba(255,159,10,0.15); color:#ff9f0a; border:1px solid rgba(255,159,10,0.3); }
.status-completed { background:rgba(10,132,255,0.15); color:#0a84ff; border:1px solid rgba(10,132,255,0.3); }
.status-out_of_scope{ background:rgba(142,142,147,0.15);color:#8e8e93; border:1px solid rgba(142,142,147,0.3); }
/* Type badges */
.type-badge {
display:inline-block; padding:1px 6px; border-radius:3px;
font-size:0.7rem; font-weight:600; text-transform:uppercase;
}
.type-ip { background:#1a3a6e; color:#5ac8fa; }
.type-cidr { background:#3a1a6e; color:#bf5af2; }
.type-domain { background:#1a6e2e; color:#30d158; }
.type-url { background:#6e3a1a; color:#ff9f0a; }
/* Tags */
.tag-chip {
display:inline-block; margin:1px 2px; padding:1px 6px;
background:rgba(var(--accent-rgb,10,132,255),0.12);
color:var(--accent); border-radius:3px;
font-size:0.7rem; font-weight:500;
}
/* Micro button */
.btn-xs {
padding:2px 7px; font-size:0.75rem; border-radius:3px;
background:var(--bg-input); border:1px solid var(--border);
color:var(--text-secondary); cursor:pointer;
}
.btn-xs:hover { background:var(--hover); color:var(--text-primary); }
.btn-xs.btn-danger:hover { background:rgba(255,59,48,0.15); color:#ff3b30; border-color:#ff3b30; }
/* Row hover */
.target-row:hover { background:var(--hover); }
</style>
<script>
var _targets = {{ targets | tojson }};
var _STATUS_CYCLE = ['active','pending','completed','out-of-scope'];
// ── Add form toggle ────────────────────────────────────────────────────────────
function toggleAddForm() {
var f = document.getElementById('add-form');
f.style.display = f.style.display === 'none' ? '' : 'none';
if (f.style.display !== 'none') document.getElementById('add-host').focus();
}
// ── Add target ────────────────────────────────────────────────────────────────
var _customFieldCount = 0;
function addCustomField() {
_customFieldCount++;
var container = document.getElementById('custom-fields-container');
var row = document.createElement('div');
row.style.cssText = 'display:flex;gap:0.5rem;align-items:flex-start;margin-bottom:0.5rem;flex-wrap:wrap';
row.innerHTML = '<input type="text" class="cf-name" placeholder="Field name" style="width:150px;padding:0.3rem 0.5rem;border-radius:var(--radius);border:1px solid var(--border);background:var(--bg-input);color:inherit;font-size:0.8rem">'
+ '<select class="cf-type" style="width:130px;padding:0.3rem 0.5rem;border-radius:var(--radius);border:1px solid var(--border);background:var(--bg-input);color:inherit;font-size:0.8rem">'
+ '<option value="text">Text</option><option value="ipv4">IPv4</option><option value="ipv6">IPv6</option>'
+ '<option value="domain">Domain</option><option value="dns">DNS Record</option>'
+ '<option value="email">Email</option><option value="username">Username</option>'
+ '<option value="url">URL</option><option value="mac">MAC Address</option>'
+ '<option value="hash">Hash</option><option value="misc">Misc (4000 char)</option></select>'
+ '<input type="text" class="cf-value" placeholder="Value" style="flex:1;min-width:200px;padding:0.3rem 0.5rem;border-radius:var(--radius);border:1px solid var(--border);background:var(--bg-input);color:inherit;font-size:0.8rem">'
+ '<button class="btn btn-sm" onclick="this.parentElement.remove()" style="padding:2px 8px;color:var(--danger)">X</button>';
container.appendChild(row);
}
function _getVal(id) { var e = document.getElementById(id); return e ? (e.value||'').trim() : ''; }
2026-03-13 15:17:15 -07:00
function addTarget() {
var host = _getVal('add-host');
2026-03-13 15:17:15 -07:00
if (!host) { document.getElementById('add-host').focus(); return; }
var msg = document.getElementById('add-status-msg');
msg.textContent = 'Saving…';
// Collect custom fields
var customFields = [];
document.querySelectorAll('#custom-fields-container > div').forEach(function(row) {
var name = row.querySelector('.cf-name').value.trim();
var type = row.querySelector('.cf-type').value;
var value = row.querySelector('.cf-value').value.trim();
if (name && value) customFields.push({name: name, type: type, value: value});
});
2026-03-13 15:17:15 -07:00
fetch('/targets/add', {
method: 'POST',
headers: {'Content-Type': 'application/json'},
body: JSON.stringify({
host: host,
name: _getVal('add-name'),
type: _getVal('add-type'),
os: _getVal('add-os'),
status: _getVal('add-status'),
threat_level: _getVal('add-threat'),
source: _getVal('add-source'),
ports: _getVal('add-ports'),
tags: _getVal('add-tags'),
notes: _getVal('add-notes'),
ipv4: _getVal('add-ipv4'),
ipv6: _getVal('add-ipv6'),
domain: _getVal('add-domain'),
rdns: _getVal('add-rdns'),
mac_address: _getVal('add-mac'),
hostname: _getVal('add-hostname'),
services: _getVal('add-ports'),
geo_country: _getVal('add-geo-country'),
geo_city: _getVal('add-geo-city'),
geo_isp: _getVal('add-geo-isp'),
geo_asn: _getVal('add-geo-asn'),
geo_coords: _getVal('add-geo-coords'),
dns_records: _getVal('add-dns'),
email: _getVal('add-email'),
usernames: _getVal('add-usernames'),
vulns: _getVal('add-vulns'),
traceroute: _getVal('add-traceroute'),
whois: _getVal('add-whois'),
custom_fields: customFields,
2026-03-13 15:17:15 -07:00
})
}).then(function(r){ return r.json(); })
.then(function(d) {
if (!d.ok) { msg.textContent = 'Error: ' + d.error; return; }
msg.textContent = '✓ Added';
// Clear form
['add-host','add-name','add-ports','add-tags','add-notes'].forEach(function(id){
document.getElementById(id).value = '';
});
document.getElementById('add-type').value = 'ip';
document.getElementById('add-os').value = 'Unknown';
document.getElementById('add-status').value = 'active';
// Reload page to show new row
location.reload();
}).catch(function(e){ msg.textContent = 'Error: ' + e.message; });
}
// ── Delete ─────────────────────────────────────────────────────────────────────
function deleteTarget(id) {
if (!confirm('Delete this target?')) return;
fetch('/targets/delete/' + id, {method:'POST'})
.then(function(r){ return r.json(); })
.then(function(d){ if (d.ok) { location.reload(); } else { alert(d.error); } });
}
// ── Edit toggle ────────────────────────────────────────────────────────────────
function toggleEdit(id) {
var row = document.getElementById('edit-' + id);
row.style.display = row.style.display === 'none' ? '' : 'none';
}
function saveEdit(id) {
fetch('/targets/update/' + id, {
method: 'POST',
headers: {'Content-Type': 'application/json'},
body: JSON.stringify({
host: document.getElementById('e-host-' + id).value,
name: document.getElementById('e-name-' + id).value,
type: document.getElementById('e-type-' + id).value,
os: document.getElementById('e-os-' + id).value,
status: document.getElementById('e-status-'+ id).value,
ports: document.getElementById('e-ports-' + id).value,
tags: document.getElementById('e-tags-' + id).value,
notes: document.getElementById('e-notes-' + id).value,
})
}).then(function(r){ return r.json(); })
.then(function(d){ if (d.ok) { location.reload(); } else { alert(d.error); } });
}
// ── Status cycle ───────────────────────────────────────────────────────────────
function cycleStatus(id, badgeEl) {
var cur = badgeEl.textContent.trim();
var idx = _STATUS_CYCLE.indexOf(cur);
var next = _STATUS_CYCLE[(idx + 1) % _STATUS_CYCLE.length];
fetch('/targets/status/' + id, {
method: 'POST',
headers: {'Content-Type': 'application/json'},
body: JSON.stringify({status: next})
}).then(function(r){ return r.json(); })
.then(function(d){
if (!d.ok) { alert(d.error); return; }
badgeEl.textContent = next;
badgeEl.className = 'status-badge status-' + next.replace('-','_');
var row = badgeEl.closest('tr');
if (row) row.dataset.status = next;
filterTargets();
});
}
// ── Copy host ─────────────────────────────────────────────────────────────────
function copyHost(host) {
navigator.clipboard.writeText(host).then(function(){
// brief visual feedback handled by browser
});
}
// ── Filter ─────────────────────────────────────────────────────────────────────
function filterTargets() {
var q = document.getElementById('filter-input').value.toLowerCase();
var status = document.getElementById('filter-status').value;
var type = document.getElementById('filter-type').value;
var rows = document.querySelectorAll('#targets-body .target-row');
var shown = 0;
rows.forEach(function(row) {
var host = (row.dataset.host || '').toLowerCase();
var name = (row.dataset.name || '').toLowerCase();
var tags = (row.dataset.tags || '').toLowerCase();
var rtype = row.dataset.type || '';
var rstat = row.dataset.status|| '';
var matchQ = !q || host.includes(q) || name.includes(q) || tags.includes(q);
var matchS = !status || rstat === status;
var matchT = !type || rtype === type;
var visible = matchQ && matchS && matchT;
row.style.display = visible ? '' : 'none';
// Hide corresponding edit row too
var editRow = document.getElementById('edit-' + row.dataset.id);
if (editRow && !visible) editRow.style.display = 'none';
if (visible) shown++;
});
document.getElementById('filter-count').textContent =
shown + ' of ' + rows.length + ' targets';
}
// ── Export ─────────────────────────────────────────────────────────────────────
function exportTargets() {
window.location.href = '/targets/export';
}
// ── Import ─────────────────────────────────────────────────────────────────────
function importTargets(input) {
var file = input.files[0];
if (!file) return;
var reader = new FileReader();
reader.onload = function(e) {
var data;
try { data = JSON.parse(e.target.result); } catch(err) { alert('Invalid JSON: ' + err.message); return; }
var targets = Array.isArray(data) ? data : (data.targets || []);
fetch('/targets/import', {
method: 'POST',
headers: {'Content-Type': 'application/json'},
body: JSON.stringify({targets: targets})
}).then(function(r){ return r.json(); })
.then(function(d){
if (d.ok) {
alert('Imported ' + d.added + ' new targets (' + d.total + ' total).');
location.reload();
} else { alert('Import error: ' + d.error); }
});
};
reader.readAsText(file);
input.value = '';
}
// Init filter count on load
document.addEventListener('DOMContentLoaded', function(){
var rows = document.querySelectorAll('#targets-body .target-row');
if (rows.length > 0) {
document.getElementById('filter-count').textContent = rows.length + ' targets';
}
// Enter key in add form
document.getElementById('add-notes').addEventListener('keypress', function(e){
if (e.key === 'Enter') addTarget();
});
loadIRs();
2026-03-13 15:17:15 -07:00
});
function loadIRs() {
fetch('/targets/ir').then(function(r){ return r.json(); }).then(function(d) {
var el = document.getElementById('ir-list');
if (!d.ok || !d.reports || !d.reports.length) {
el.innerHTML = '<p style="color:var(--text-muted);font-size:0.85rem">No investigation reports yet. Use HAL\'s "Report Only" or "Let HAL Fix It + IR" buttons after a scan.</p>';
return;
}
var html = '<table class="data-table" style="font-size:0.82rem;width:100%"><thead><tr>'
+ '<th>ID</th><th>Title</th><th>Risk</th><th>Status</th><th>Source</th><th>Created</th><th>Actions</th>'
+ '</tr></thead><tbody>';
d.reports.forEach(function(r) {
var riskColors = {critical:'#ff3b30',high:'#ff6b35',medium:'#f59e0b',low:'#8ec07c',clean:'#34c759',unknown:'#888'};
var rc = riskColors[r.risk_level] || '#888';
var halBadge = r.created_by_hal ? ' <span style="background:var(--accent);color:#000;font-size:0.6rem;padding:1px 4px;border-radius:2px;font-weight:700">HAL</span>' : '';
var statusColor = r.status === 'closed' ? 'var(--text-muted)' : r.status === 'open' ? 'var(--accent)' : '#f59e0b';
var created = r.created_at ? new Date(r.created_at).toLocaleString() : '';
html += '<tr>'
+ '<td><strong style="font-family:monospace;color:var(--accent)">' + escapeHtml(r.id) + '</strong>' + halBadge + '</td>'
+ '<td>' + escapeHtml(r.title || '') + '</td>'
+ '<td><span style="color:' + rc + ';font-weight:600;text-transform:uppercase;font-size:0.75rem">' + escapeHtml(r.risk_level || 'unknown') + '</span></td>'
+ '<td><span style="color:' + statusColor + '">' + escapeHtml(r.status || '') + '</span></td>'
+ '<td style="font-size:0.75rem">' + escapeHtml(r.source || '') + '</td>'
+ '<td style="font-size:0.75rem">' + escapeHtml(created) + '</td>'
+ '<td style="white-space:nowrap">'
+ '<button class="btn btn-sm" style="font-size:0.65rem;padding:1px 6px;margin-right:3px" onclick="viewIR(\'' + escapeHtml(r.id) + '\')">View</button>'
+ '<button class="btn btn-sm" style="font-size:0.65rem;padding:1px 6px;margin-right:3px" onclick="loadIRToHal(\'' + escapeHtml(r.id) + '\')">Send to HAL</button>'
+ '<button class="btn btn-sm" style="font-size:0.65rem;padding:1px 6px;color:var(--danger);border-color:var(--danger)" onclick="deleteIR(\'' + escapeHtml(r.id) + '\')">Delete</button>'
+ '</td></tr>';
});
html += '</tbody></table>';
el.innerHTML = html;
}).catch(function() {
document.getElementById('ir-list').innerHTML = '<p style="color:var(--danger)">Failed to load reports.</p>';
});
}
function viewIR(irId) {
fetch('/targets/ir/' + irId).then(function(r){ return r.json(); }).then(function(d) {
if (!d.ok) { alert('IR not found'); return; }
var ir = d.ir;
var w = window.open('', '_blank', 'width=700,height=600');
w.document.write('<html><head><title>' + ir.id + '</title>'
+ '<style>body{background:#1a1a2e;color:#e0e0e0;font-family:system-ui;padding:20px;font-size:14px} '
+ 'h1{color:#00ff41;font-size:1.2rem} h2{color:#5ac8fa;font-size:1rem;margin-top:1rem} '
+ 'pre{background:#12122a;padding:10px;border-radius:6px;white-space:pre-wrap;font-size:0.82rem;border:1px solid #333} '
+ '.badge{display:inline-block;padding:2px 8px;border-radius:3px;font-size:0.75rem;font-weight:700}</style></head><body>');
w.document.write('<h1>' + ir.id + (ir.created_by_hal ? ' <span class="badge" style="background:#00ff41;color:#000">HAL</span>' : '') + '</h1>');
w.document.write('<div><strong>Title:</strong> ' + (ir.title||'') + '</div>');
w.document.write('<div><strong>Status:</strong> ' + (ir.status||'') + ' &middot; <strong>Risk:</strong> ' + (ir.risk_level||'') + ' &middot; <strong>Source:</strong> ' + (ir.source||'') + '</div>');
w.document.write('<div><strong>Created:</strong> ' + (ir.created_at||'') + '</div>');
if (ir.ip) w.document.write('<div><strong>IP:</strong> ' + ir.ip + '</div>');
if (ir.analysis) { w.document.write('<h2>Analysis</h2><pre>' + ir.analysis.replace(/</g,'&lt;') + '</pre>'); }
if (ir.fix_attempted) {
w.document.write('<h2>Fix Attempted</h2><pre>' + (ir.fix_results||'').replace(/</g,'&lt;') + '</pre>');
}
if (ir.notes) { w.document.write('<h2>Notes</h2><pre>' + ir.notes.replace(/</g,'&lt;') + '</pre>'); }
w.document.write('</body></html>');
w.document.close();
});
}
function loadIRToHal(irId) {
fetch('/targets/ir/' + irId + '/load-to-hal', {method:'POST'}).then(function(r){return r.json()}).then(function(d) {
if (d.ok) {
// Open HAL panel and show the loaded IR
var panel = document.getElementById('hal-panel');
if (panel) panel.style.display = 'flex';
halAppendStyled('status', 'Loaded IR ' + d.ir.id + ' into HAL memory');
halAppendStyled('bot', 'IR ' + d.ir.id + ': ' + (d.ir.title||'') + '\nRisk: ' + (d.ir.risk_level||'unknown') + '\nStatus: ' + (d.ir.status||'') + '\n\nAnalysis loaded. Ask me to continue working on this investigation.');
halScroll();
} else {
alert('Failed to load IR: ' + (d.error||''));
}
});
}
function deleteIR(irId) {
if (!confirm('Delete investigation report ' + irId + '?')) return;
fetch('/targets/ir/' + irId + '/delete', {method:'POST'}).then(function(r){return r.json()}).then(function(d) {
loadIRs();
});
}
2026-03-13 15:17:15 -07:00
</script>
{% endblock %}