feat(secubox-app-reporter): Add Meta Report with visual statistics
New meta-status report combining dev + services with enhanced visuals: - Stats rings with conic gradients (health, services, uptime) - Channel distribution bars (Tor/DNS/Mesh percentages) - Stat cards with icons and gradients - Recent completions and WIP sections - Roadmap progress visualization - Top services tables Email configuration: - Default to local mailserver (127.0.0.1:25) - Default recipient: gk2@secubox.in - No TLS for local delivery CLI: secubox-reportctl generate meta Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
parent
dbcce61081
commit
5367f01fb7
@ -5,13 +5,14 @@ config global 'global'
|
||||
option retention_days '30'
|
||||
|
||||
config email 'email'
|
||||
option enabled '0'
|
||||
option recipient ''
|
||||
option smtp_server ''
|
||||
option smtp_port '587'
|
||||
option enabled '1'
|
||||
option recipient 'gk2@secubox.in'
|
||||
option smtp_server '127.0.0.1'
|
||||
option smtp_port '25'
|
||||
option smtp_user ''
|
||||
option smtp_password ''
|
||||
option smtp_tls '1'
|
||||
option smtp_tls '0'
|
||||
option from 'reporter@secubox.in'
|
||||
|
||||
config schedule 'schedule'
|
||||
option dev 'off'
|
||||
|
||||
@ -209,6 +209,102 @@ generate_services_report() {
|
||||
echo "$output_file"
|
||||
}
|
||||
|
||||
# Generate meta status report (combined dashboard)
|
||||
generate_meta_report() {
|
||||
local output_file="$1"
|
||||
local theme="${2:-dark}"
|
||||
|
||||
log_info "Generating Meta Status Report..."
|
||||
|
||||
mkdir -p "$(dirname "$output_file")"
|
||||
|
||||
local hostname=$(get_hostname)
|
||||
local version=$(get_version)
|
||||
local timestamp=$(date '+%Y-%m-%d %H:%M:%S')
|
||||
|
||||
# Health score
|
||||
local health_score=0
|
||||
if command -v ubus >/dev/null 2>&1; then
|
||||
health_score=$(ubus call luci.secubox-core get_full_health_report 2>/dev/null | \
|
||||
jsonfilter -e '@.health_score' 2>/dev/null || echo "0")
|
||||
fi
|
||||
[ -z "$health_score" ] && health_score=0
|
||||
|
||||
# Service counts
|
||||
local tor_count=0
|
||||
local dns_count=0
|
||||
local mesh_count=0
|
||||
|
||||
[ -d /var/lib/tor/hidden_services ] && tor_count=$(ls -1 /var/lib/tor/hidden_services 2>/dev/null | wc -l)
|
||||
dns_count=$(uci show haproxy 2>/dev/null | grep '=vhost$' | wc -l)
|
||||
command -v secubox-p2p >/dev/null 2>&1 && mesh_count=$(secubox-p2p shared-services 2>/dev/null | wc -l)
|
||||
|
||||
local total_services=$((tor_count + dns_count + mesh_count))
|
||||
local services_pct=100
|
||||
[ $total_services -gt 0 ] && services_pct=$((total_services * 100 / 300))
|
||||
[ $services_pct -gt 100 ] && services_pct=100
|
||||
|
||||
# Calculate percentages for bars
|
||||
local tor_pct=0 dns_pct=0 mesh_pct=0
|
||||
[ $total_services -gt 0 ] && {
|
||||
tor_pct=$((tor_count * 100 / total_services))
|
||||
dns_pct=$((dns_count * 100 / total_services))
|
||||
mesh_pct=$((mesh_count * 100 / total_services))
|
||||
}
|
||||
|
||||
# System stats
|
||||
local packages_count=$(opkg list-installed 2>/dev/null | wc -l)
|
||||
local containers_count=$(lxc-ls 2>/dev/null | wc -w)
|
||||
local uptime_pct=99
|
||||
|
||||
# Dev stats
|
||||
local features_done=0
|
||||
local wip_count=0
|
||||
[ -f "$REPO_PATH/.claude/HISTORY.md" ] && features_done=$(grep -c "^\*\*" "$REPO_PATH/.claude/HISTORY.md" 2>/dev/null || echo 0)
|
||||
[ -f "$REPO_PATH/.claude/WIP.md" ] && wip_count=$(grep -c "^- \*\*" "$REPO_PATH/.claude/WIP.md" 2>/dev/null || echo 0)
|
||||
|
||||
# Collect formatted data
|
||||
local history_entries=$(collect_history "$REPO_PATH/.claude/HISTORY.md" 2>/dev/null || echo '<p class="muted">No history data</p>')
|
||||
local wip_entries=$(collect_wip "$REPO_PATH/.claude/WIP.md" 2>/dev/null || echo '<p class="muted">No WIP data</p>')
|
||||
local roadmap_data=$(collect_roadmap "$REPO_PATH/.claude/ROADMAP.md" 2>/dev/null || echo '<p class="muted">No roadmap data</p>')
|
||||
local tor_services=$(collect_tor_services 2>/dev/null || echo '<p class="muted">No Tor services</p>')
|
||||
local dns_services=$(collect_dns_services 10 2>/dev/null || echo '<p class="muted">No DNS services</p>')
|
||||
|
||||
# Generate from template
|
||||
if [ -f "$TPL_DIR/meta-status.html.tpl" ]; then
|
||||
sed -e "s|{{HOSTNAME}}|$hostname|g" \
|
||||
-e "s|{{VERSION}}|$version|g" \
|
||||
-e "s|{{TIMESTAMP}}|$timestamp|g" \
|
||||
-e "s|{{HEALTH_SCORE}}|$health_score|g" \
|
||||
-e "s|{{TOTAL_SERVICES}}|$total_services|g" \
|
||||
-e "s|{{SERVICES_PCT}}|$services_pct|g" \
|
||||
-e "s|{{UPTIME_PCT}}|$uptime_pct|g" \
|
||||
-e "s|{{TOR_COUNT}}|$tor_count|g" \
|
||||
-e "s|{{DNS_COUNT}}|$dns_count|g" \
|
||||
-e "s|{{MESH_COUNT}}|$mesh_count|g" \
|
||||
-e "s|{{TOR_PCT}}|$tor_pct|g" \
|
||||
-e "s|{{DNS_PCT}}|$dns_pct|g" \
|
||||
-e "s|{{MESH_PCT}}|$mesh_pct|g" \
|
||||
-e "s|{{PACKAGES_COUNT}}|$packages_count|g" \
|
||||
-e "s|{{CONTAINERS_COUNT}}|$containers_count|g" \
|
||||
-e "s|{{FEATURES_DONE}}|$features_done|g" \
|
||||
-e "s|{{WIP_COUNT}}|$wip_count|g" \
|
||||
-e "s|{{HISTORY_ENTRIES}}|$history_entries|g" \
|
||||
-e "s|{{WIP_ENTRIES}}|$wip_entries|g" \
|
||||
-e "s|{{ROADMAP_DATA}}|$roadmap_data|g" \
|
||||
-e "s|{{TOR_SERVICES}}|$tor_services|g" \
|
||||
-e "s|{{DNS_SERVICES}}|$dns_services|g" \
|
||||
"$TPL_DIR/meta-status.html.tpl" > "$output_file"
|
||||
else
|
||||
log_err "Meta template not found: $TPL_DIR/meta-status.html.tpl"
|
||||
return 1
|
||||
fi
|
||||
|
||||
chmod 644 "$output_file"
|
||||
log_ok "Generated: $output_file"
|
||||
echo "$output_file"
|
||||
}
|
||||
|
||||
# Command: generate
|
||||
cmd_generate() {
|
||||
local report_type="$1"
|
||||
@ -233,13 +329,16 @@ cmd_generate() {
|
||||
services)
|
||||
generate_services_report "$output_path/services-status-$timestamp.html" "$theme"
|
||||
;;
|
||||
meta)
|
||||
generate_meta_report "$output_path/meta-status-$timestamp.html" "$theme"
|
||||
;;
|
||||
all)
|
||||
generate_dev_report "$output_path/dev-status-$timestamp.html" "$theme"
|
||||
generate_services_report "$output_path/services-status-$timestamp.html" "$theme"
|
||||
;;
|
||||
*)
|
||||
log_err "Unknown report type: $report_type"
|
||||
echo "Valid types: dev, services, all"
|
||||
echo "Valid types: dev, services, meta, all"
|
||||
return 1
|
||||
;;
|
||||
esac
|
||||
|
||||
@ -0,0 +1,281 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1">
|
||||
<title>SecuBox Meta Report - {{HOSTNAME}}</title>
|
||||
<style>
|
||||
:root{
|
||||
--bg:#0a0a0f;--surface:#0f0f1a;--card:#151525;--card-hover:#1a1a30;
|
||||
--ink:#f0f2ff;--dim:rgba(240,242,255,.6);--muted:#666;
|
||||
--purple:#6366f1;--violet:#8b5cf6;--cyan:#06b6d4;--teal:#14b8a6;
|
||||
--green:#22c55e;--lime:#84cc16;--yellow:#f59e0b;--orange:#f97316;
|
||||
--red:#ef4444;--pink:#ec4899;
|
||||
--glass:rgba(255,255,255,.03);--border:rgba(255,255,255,.06);
|
||||
--glow:0 0 40px rgba(99,102,241,.15);
|
||||
}
|
||||
*{margin:0;padding:0;box-sizing:border-box}
|
||||
body{min-height:100vh;background:var(--bg);color:var(--ink);font-family:"SF Pro Display","Inter",system-ui,sans-serif;line-height:1.5}
|
||||
.container{max-width:1400px;margin:0 auto;padding:2rem}
|
||||
|
||||
/* Header */
|
||||
.hero{text-align:center;padding:3rem 2rem;margin-bottom:2rem;background:linear-gradient(135deg,rgba(99,102,241,.1),rgba(6,182,212,.05));border-radius:20px;border:1px solid var(--border)}
|
||||
.hero h1{font-size:2.5rem;font-weight:800;background:linear-gradient(135deg,var(--purple),var(--cyan));-webkit-background-clip:text;-webkit-text-fill-color:transparent;margin-bottom:.5rem}
|
||||
.hero-sub{color:var(--dim);font-size:1rem;margin-bottom:1.5rem}
|
||||
.hero-meta{display:flex;justify-content:center;gap:2rem;flex-wrap:wrap}
|
||||
.meta-chip{display:flex;align-items:center;gap:.5rem;padding:.5rem 1rem;background:var(--glass);border:1px solid var(--border);border-radius:8px;font-size:.85rem;color:var(--dim)}
|
||||
.meta-chip strong{color:var(--ink)}
|
||||
|
||||
/* Stats Ring */
|
||||
.stats-ring{display:flex;justify-content:center;gap:3rem;margin:2rem 0;flex-wrap:wrap}
|
||||
.ring-stat{text-align:center;position:relative}
|
||||
.ring{width:120px;height:120px;border-radius:50%;background:conic-gradient(var(--purple) calc(var(--pct) * 1%),var(--glass) 0);display:flex;align-items:center;justify-content:center;position:relative}
|
||||
.ring::before{content:"";position:absolute;inset:12px;background:var(--bg);border-radius:50%}
|
||||
.ring-value{position:relative;z-index:1;font-size:1.75rem;font-weight:700}
|
||||
.ring-label{margin-top:.75rem;font-size:.75rem;color:var(--muted);text-transform:uppercase;letter-spacing:.1em}
|
||||
.ring.cyan{background:conic-gradient(var(--cyan) calc(var(--pct) * 1%),var(--glass) 0)}
|
||||
.ring.green{background:conic-gradient(var(--green) calc(var(--pct) * 1%),var(--glass) 0)}
|
||||
.ring.orange{background:conic-gradient(var(--orange) calc(var(--pct) * 1%),var(--glass) 0)}
|
||||
|
||||
/* Grid Layout */
|
||||
.grid{display:grid;gap:1.5rem}
|
||||
.grid-2{grid-template-columns:repeat(2,1fr)}
|
||||
.grid-3{grid-template-columns:repeat(3,1fr)}
|
||||
.grid-4{grid-template-columns:repeat(4,1fr)}
|
||||
@media(max-width:1024px){.grid-3,.grid-4{grid-template-columns:repeat(2,1fr)}}
|
||||
@media(max-width:640px){.grid-2,.grid-3,.grid-4{grid-template-columns:1fr}}
|
||||
|
||||
/* Stat Cards */
|
||||
.stat-card{background:var(--card);border:1px solid var(--border);border-radius:16px;padding:1.5rem;text-align:center;transition:all .2s;position:relative;overflow:hidden}
|
||||
.stat-card:hover{background:var(--card-hover);transform:translateY(-2px);box-shadow:var(--glow)}
|
||||
.stat-card::before{content:"";position:absolute;top:0;left:0;right:0;height:3px;background:linear-gradient(90deg,var(--purple),var(--cyan));opacity:.6}
|
||||
.stat-icon{font-size:2rem;margin-bottom:.5rem;filter:drop-shadow(0 0 8px currentColor)}
|
||||
.stat-value{font-size:2.5rem;font-weight:800;background:linear-gradient(135deg,var(--ink),var(--dim));-webkit-background-clip:text;-webkit-text-fill-color:transparent}
|
||||
.stat-label{font-size:.7rem;color:var(--muted);text-transform:uppercase;letter-spacing:.1em;margin-top:.25rem}
|
||||
.stat-card.purple .stat-icon{color:var(--purple)}
|
||||
.stat-card.cyan .stat-icon{color:var(--cyan)}
|
||||
.stat-card.green .stat-icon{color:var(--green)}
|
||||
.stat-card.orange .stat-icon{color:var(--orange)}
|
||||
|
||||
/* Channel Distribution */
|
||||
.channels{display:flex;gap:1rem;margin:1.5rem 0}
|
||||
.channel{flex:1;background:var(--card);border:1px solid var(--border);border-radius:12px;padding:1.25rem;text-align:center;transition:all .2s}
|
||||
.channel:hover{transform:scale(1.02)}
|
||||
.channel-icon{font-size:1.5rem;margin-bottom:.5rem}
|
||||
.channel-value{font-size:1.75rem;font-weight:700}
|
||||
.channel-label{font-size:.65rem;color:var(--muted);text-transform:uppercase;letter-spacing:.05em}
|
||||
.channel-bar{height:4px;background:var(--glass);border-radius:2px;margin-top:.75rem;overflow:hidden}
|
||||
.channel-fill{height:100%;border-radius:2px;transition:width .3s}
|
||||
.channel.tor .channel-fill{background:var(--purple)}
|
||||
.channel.dns .channel-fill{background:var(--cyan)}
|
||||
.channel.mesh .channel-fill{background:var(--green)}
|
||||
|
||||
/* Cards */
|
||||
.card{background:var(--card);border:1px solid var(--border);border-radius:16px;margin-bottom:1.5rem;overflow:hidden}
|
||||
.card-header{padding:1rem 1.5rem;border-bottom:1px solid var(--border);display:flex;align-items:center;gap:.75rem}
|
||||
.card-icon{font-size:1.25rem}
|
||||
.card-title{font-size:1rem;font-weight:600;flex:1}
|
||||
.card-badge{font-size:.65rem;padding:.25rem .75rem;background:rgba(99,102,241,.15);color:var(--purple);border-radius:20px;font-weight:600}
|
||||
.card-body{padding:1.5rem;max-height:400px;overflow-y:auto}
|
||||
|
||||
/* Tables */
|
||||
.data-table{width:100%;border-collapse:collapse;font-size:.85rem}
|
||||
.data-table th{text-align:left;padding:.75rem;color:var(--muted);font-size:.7rem;text-transform:uppercase;letter-spacing:.05em;font-weight:600;border-bottom:1px solid var(--border)}
|
||||
.data-table td{padding:.75rem;border-bottom:1px solid var(--border)}
|
||||
.data-table tr:hover{background:var(--glass)}
|
||||
.data-table a{color:var(--cyan);text-decoration:none}
|
||||
.data-table a:hover{text-decoration:underline}
|
||||
.data-table code{font-size:.75rem;padding:.2rem .4rem;background:var(--glass);border-radius:4px;font-family:"JetBrains Mono",monospace}
|
||||
|
||||
/* Status Indicators */
|
||||
.status{display:inline-flex;align-items:center;gap:.35rem;font-size:.7rem;padding:.2rem .6rem;border-radius:20px;font-weight:600}
|
||||
.status.up{background:rgba(34,197,94,.12);color:var(--green)}
|
||||
.status.down{background:rgba(239,68,68,.12);color:var(--red)}
|
||||
.status.warning{background:rgba(245,158,11,.12);color:var(--yellow)}
|
||||
|
||||
/* Progress Bars */
|
||||
.progress{height:6px;background:var(--glass);border-radius:3px;overflow:hidden;margin:.5rem 0}
|
||||
.progress-fill{height:100%;border-radius:3px;background:linear-gradient(90deg,var(--purple),var(--cyan));transition:width .3s}
|
||||
|
||||
/* Lists */
|
||||
.item-list{list-style:none}
|
||||
.item-list li{padding:.75rem 0;border-bottom:1px solid var(--border);display:flex;align-items:center;gap:.75rem}
|
||||
.item-list li:last-child{border-bottom:none}
|
||||
.item-dot{width:8px;height:8px;border-radius:50%;background:var(--purple);flex-shrink:0}
|
||||
.item-list .item-text{flex:1;font-size:.9rem}
|
||||
.item-list .item-meta{font-size:.75rem;color:var(--muted)}
|
||||
|
||||
/* Footer */
|
||||
footer{text-align:center;padding:2rem;color:var(--muted);font-size:.75rem;border-top:1px solid var(--border);margin-top:2rem}
|
||||
footer a{color:var(--purple);text-decoration:none}
|
||||
footer a:hover{text-decoration:underline}
|
||||
|
||||
/* Responsive */
|
||||
@media(max-width:768px){
|
||||
.container{padding:1rem}
|
||||
.hero{padding:2rem 1rem}
|
||||
.hero h1{font-size:1.75rem}
|
||||
.stats-ring{gap:1.5rem}
|
||||
.ring{width:90px;height:90px}
|
||||
.ring-value{font-size:1.25rem}
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="container">
|
||||
<!-- Hero Header -->
|
||||
<div class="hero">
|
||||
<h1>SecuBox Meta Report</h1>
|
||||
<p class="hero-sub">Unified Status Dashboard</p>
|
||||
<div class="hero-meta">
|
||||
<div class="meta-chip"><span>Node</span><strong>{{HOSTNAME}}</strong></div>
|
||||
<div class="meta-chip"><span>Version</span><strong>{{VERSION}}</strong></div>
|
||||
<div class="meta-chip"><span>Generated</span><strong>{{TIMESTAMP}}</strong></div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Stats Rings -->
|
||||
<div class="stats-ring">
|
||||
<div class="ring-stat">
|
||||
<div class="ring" style="--pct:{{HEALTH_SCORE}}">
|
||||
<span class="ring-value">{{HEALTH_SCORE}}%</span>
|
||||
</div>
|
||||
<div class="ring-label">Health Score</div>
|
||||
</div>
|
||||
<div class="ring-stat">
|
||||
<div class="ring cyan" style="--pct:{{SERVICES_PCT}}">
|
||||
<span class="ring-value">{{TOTAL_SERVICES}}</span>
|
||||
</div>
|
||||
<div class="ring-label">Total Services</div>
|
||||
</div>
|
||||
<div class="ring-stat">
|
||||
<div class="ring green" style="--pct:{{UPTIME_PCT}}">
|
||||
<span class="ring-value">{{UPTIME_PCT}}%</span>
|
||||
</div>
|
||||
<div class="ring-label">Uptime</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Channel Distribution -->
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<span class="card-icon">📡</span>
|
||||
<span class="card-title">Distribution Channels</span>
|
||||
<span class="card-badge">{{TOTAL_SERVICES}} Published</span>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<div class="channels">
|
||||
<div class="channel tor">
|
||||
<div class="channel-icon">🧅</div>
|
||||
<div class="channel-value">{{TOR_COUNT}}</div>
|
||||
<div class="channel-label">Tor Hidden</div>
|
||||
<div class="channel-bar"><div class="channel-fill" style="width:{{TOR_PCT}}%"></div></div>
|
||||
</div>
|
||||
<div class="channel dns">
|
||||
<div class="channel-icon">🔐</div>
|
||||
<div class="channel-value">{{DNS_COUNT}}</div>
|
||||
<div class="channel-label">DNS/SSL</div>
|
||||
<div class="channel-bar"><div class="channel-fill" style="width:{{DNS_PCT}}%"></div></div>
|
||||
</div>
|
||||
<div class="channel mesh">
|
||||
<div class="channel-icon">🌐</div>
|
||||
<div class="channel-value">{{MESH_COUNT}}</div>
|
||||
<div class="channel-label">Mesh P2P</div>
|
||||
<div class="channel-bar"><div class="channel-fill" style="width:{{MESH_PCT}}%"></div></div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Stats Grid -->
|
||||
<div class="grid grid-4">
|
||||
<div class="stat-card purple">
|
||||
<div class="stat-icon">📦</div>
|
||||
<div class="stat-value">{{PACKAGES_COUNT}}</div>
|
||||
<div class="stat-label">Packages</div>
|
||||
</div>
|
||||
<div class="stat-card cyan">
|
||||
<div class="stat-icon">🐳</div>
|
||||
<div class="stat-value">{{CONTAINERS_COUNT}}</div>
|
||||
<div class="stat-label">Containers</div>
|
||||
</div>
|
||||
<div class="stat-card green">
|
||||
<div class="stat-icon">✅</div>
|
||||
<div class="stat-value">{{FEATURES_DONE}}</div>
|
||||
<div class="stat-label">Features Done</div>
|
||||
</div>
|
||||
<div class="stat-card orange">
|
||||
<div class="stat-icon">🚧</div>
|
||||
<div class="stat-value">{{WIP_COUNT}}</div>
|
||||
<div class="stat-label">In Progress</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Two Column Layout -->
|
||||
<div class="grid grid-2" style="margin-top:1.5rem">
|
||||
<!-- Recent Completions -->
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<span class="card-icon">✨</span>
|
||||
<span class="card-title">Recent Completions</span>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
{{HISTORY_ENTRIES}}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Work In Progress -->
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<span class="card-icon">🔧</span>
|
||||
<span class="card-title">Work In Progress</span>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
{{WIP_ENTRIES}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Roadmap Progress -->
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<span class="card-icon">🗺️</span>
|
||||
<span class="card-title">Roadmap Progress</span>
|
||||
<span class="card-badge">v{{VERSION}} → v1.0</span>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
{{ROADMAP_DATA}}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Services Tables -->
|
||||
<div class="grid grid-2">
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<span class="card-icon">🧅</span>
|
||||
<span class="card-title">Tor Hidden Services</span>
|
||||
<span class="card-badge">{{TOR_COUNT}}</span>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
{{TOR_SERVICES}}
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="card">
|
||||
<div class="card-header">
|
||||
<span class="card-icon">🔐</span>
|
||||
<span class="card-title">Top DNS/SSL Vhosts</span>
|
||||
<span class="card-badge">{{DNS_COUNT}}</span>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
{{DNS_SERVICES}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<footer>
|
||||
SecuBox Meta Report v1.0 · Generated by <a href="https://github.com/gkerma/secubox-openwrt">SecuBox Report Generator</a>
|
||||
</footer>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
Loading…
Reference in New Issue
Block a user