114 lines
11 KiB
HTML
114 lines
11 KiB
HTML
<!DOCTYPE html><html lang="en"><head>
|
|
<meta http-equiv="Cache-Control" content="no-cache, no-store, must-revalidate"><meta charset="UTF-8"><meta name="viewport" content="width=device-width,initial-scale=1.0"><title>WEVADS - Dark Data | WEVADS</title><link href="https://fonts.googleapis.com/css2?family=DM+Sans:wght@400;500;600;700&family=JetBrains+Mono:wght@400;500;700&display=swap" rel="stylesheet"><style>*{margin:0;padding:0;box-sizing:border-box}:root{--bg:#060a14;--s:#0c1220;--c:#111827;--b:#1e293b;--cy:#22d3ee;--gn:#060a14;--rd:#ef4444;--or:#f59e0b;--pu:#a78bfa;--pk:#ec4899;--t:#e2e8f0;--d:#64748b;--f:"DM Sans",sans-serif;--m:"JetBrains Mono",monospace}body{font-family:var(--f);background:#060a14;color:var(--t);min-height:100vh}.app{max-width:1440px;margin:0 auto;padding:24px}.hdr{display:flex;align-items:center;justify-content:space-between;margin-bottom:24px;padding-bottom:16px;border-bottom:1px solid var(--b)}.hdr h1{font-size:22px;font-weight:700}.hdr h1 span{color:var(--pk)}.tabs{display:flex;gap:4px;margin-bottom:20px}.tab{padding:8px 18px;border:none;border-radius:10px;font-size:12px;font-weight:600;cursor:pointer;color:var(--d);background:var(--c);border:1px solid var(--b);transition:.2s}.tab.active{background:rgba(236,72,153,.1);color:var(--pk);border-color:var(--pk)}.tab:hover{border-color:rgba(236,72,153,.3)}.btns{display:flex;gap:8px}.btn{padding:8px 18px;border:none;border-radius:10px;font-size:12px;font-weight:600;cursor:pointer;color:#fff;transition:.2s}.btn-p{background:linear-gradient(135deg,var(--pk),#be185d)}.btn-s{background:var(--c);border:1px solid var(--b);color:var(--t)}.btn:hover{transform:translateY(-1px)}.g{display:grid;gap:16px;margin-bottom:20px}.g4{grid-template-columns:repeat(4,1fr)}.g3{grid-template-columns:repeat(3,1fr)}.g2{grid-template-columns:1fr 1fr}.cd{background:var(--s);border:1px solid var(--b);border-radius:14px;padding:20px;transition:.2s}.cd:hover{border-color:rgba(236,72,153,.2)}.st{text-align:center;padding:16px}.sv{font-family:var(--m);font-size:26px;font-weight:700}.sl{font-size:10px;color:var(--d);margin-top:4px;text-transform:uppercase;letter-spacing:.5px}table{width:100%;border-collapse:collapse;font-size:12px}th{text-align:left;padding:10px 12px;color:var(--d);font-size:10px;text-transform:uppercase;border-bottom:1px solid var(--b)}td{padding:10px 12px;border-bottom:1px solid rgba(30,41,59,.3)}.badge{padding:2px 8px;border-radius:6px;font-size:10px;font-weight:600;display:inline-block}.b-ok{background:rgba(16,185,129,.12);color:var(--gn)}.b-w{background:rgba(245,158,11,.12);color:var(--or)}.b-err{background:rgba(239,68,68,.12);color:var(--rd)}.b-pk{background:rgba(236,72,153,.12);color:var(--pk)}.b-pu{background:rgba(167,139,250,.12);color:var(--pu)}.b-info{background:rgba(34,211,238,.12);color:var(--cy)}.panel{display:none}.panel.active{display:block}.empty{text-align:center;padding:60px;color:var(--d)}.empty .icon{font-size:48px;margin-bottom:12px}.bar{height:8px;border-radius:4px;background:var(--c);overflow:hidden}.bar-fill{height:100%;border-radius:4px;transition:.3s}.src{display:flex;align-items:center;gap:12px;padding:12px;background:var(--c);border:1px solid var(--b);border-radius:10px;margin-bottom:8px;transition:.2s;cursor:pointer}.src:hover{border-color:var(--pk)}.src .icon{font-size:24px;width:40px;text-align:center}.src .info{flex:1}.src .info .name{font-weight:600;font-size:13px}.src .info .meta{font-size:11px;color:var(--d)}.src .count{font-family:var(--m);font-size:16px;font-weight:700}@media(max-width:900px){.g4,.g3{grid-template-columns:1fr 1fr}}.wv-status{position:fixed;top:12px;right:140px;z-index:9998;background:rgba(52,211,153,.15);border:1px solid #34d399;border-radius:12px;padding:3px 10px;color:#34d399;font-size:10px;font-weight:700;font-family:'JetBrains Mono',monospace}
|
|
|
|
.sc,.card,[class*="stat-card"]{transition:all .25s ease;position:relative;overflow:hidden}
|
|
.sc:hover,.card:hover,[class*="stat-card"]:hover{transform:translateY(-2px);box-shadow:0 8px 24px rgba(0,0,0,.25)}
|
|
.sc::after,.card::after{content:'';position:absolute;bottom:0;left:0;right:0;height:2px;background:var(--cy,#22d3ee);opacity:0;transition:opacity .25s}
|
|
.sc:hover::after,.card:hover::after{opacity:.7}
|
|
.btn,.button,[class*="btn-"]{transition:all .2s ease}
|
|
.btn:hover,.button:hover{transform:translateY(-1px)}
|
|
@keyframes fadeIn{from{opacity:0;transform:translateY(8px)}to{opacity:1;transform:translateY(0)}}
|
|
.sc,.card{animation:fadeIn .4s ease both}
|
|
.sc:nth-child(2),.card:nth-child(2){animation-delay:.05s}
|
|
.sc:nth-child(3),.card:nth-child(3){animation-delay:.1s}
|
|
.sc:nth-child(4),.card:nth-child(4){animation-delay:.15s}
|
|
.sc:nth-child(5),.card:nth-child(5){animation-delay:.2s}
|
|
.sc:nth-child(6),.card:nth-child(6){animation-delay:.25s}
|
|
</style><link rel="stylesheet" href="wevads-global.css?v1770777318">
|
|
</head><body>
|
|
|
|
<div class="app">
|
|
<div class="hdr"><h1>🗃️ <span>Dark</span> Data</h1>
|
|
<p style="font-size:12px;color:#64748b;margin:6px 0 16px;max-width:600px;line-height:1.6">Dark Data — analyse des données non structurées, patterns cachés, intelligence opérationnelle.</p><div class="btns"><button class="btn btn-s" onclick="load()">🔃 Refresh</button><button class="btn btn-p" onclick="runHarvest()">🌾 Harvest Now</button></div></div>
|
|
<div class="tabs"><button class="tab active" data-tab="sources">Sources</button><button class="tab" data-tab="records">Records</button><button class="tab" data-tab="enrichment">Enrichment</button><button class="tab" data-tab="exports">Exports</button></div>
|
|
<div class="g g4">
|
|
<div class="cd st"><div class="sv" style="color:var(--pk)" id="kSources">7</div><div class="sl">Data Sources</div></div>
|
|
<div class="cd st"><div class="sv" style="color:var(--cy)" id="kRecords">0</div><div class="sl">Total Records</div></div>
|
|
<div class="cd st"><div class="sv" style="color:var(--gn)" id="kEnriched">0</div><div class="sl">Enriched</div></div>
|
|
<div class="cd st"><div class="sv" style="color:var(--or)" id="kExports">0</div><div class="sl">Exports</div></div>
|
|
</div>
|
|
<div id="panel-sources" class="panel active">
|
|
<div class="cd"><div style="margin-bottom:14px"><span style="font-size:11px;text-transform:uppercase;letter-spacing:1px;color:var(--d)">📡 Data Sources</span></div>
|
|
<div id="srcList"></div></div>
|
|
</div>
|
|
<div id="panel-records" class="panel">
|
|
<div class="cd"><div style="margin-bottom:14px"><span style="font-size:11px;text-transform:uppercase;letter-spacing:1px;color:var(--d)">📋 Harvested Records</span></div><div id="recTable"></div></div>
|
|
</div>
|
|
<div id="panel-enrichment" class="panel">
|
|
<div class="cd"><div style="margin-bottom:14px"><span style="font-size:11px;text-transform:uppercase;letter-spacing:1px;color:var(--d)">🧬 Enrichment Pipeline</span></div>
|
|
<div class="g g3" id="enrichPipe"></div></div>
|
|
</div>
|
|
<div id="panel-exports" class="panel">
|
|
<div class="cd"><div style="margin-bottom:14px"><span style="font-size:11px;text-transform:uppercase;letter-spacing:1px;color:var(--d)">📦 Export History</span></div><div id="expTable"></div></div>
|
|
</div>
|
|
</div>
|
|
<script>
|
|
document.querySelectorAll('.tab').forEach(t=>{t.addEventListener('click',()=>{document.querySelectorAll('.tab').forEach(x=>x.classList.remove('active'));document.querySelectorAll('.panel').forEach(x=>x.classList.remove('active'));t.classList.add('active');document.getElementById('panel-'+t.dataset.tab).classList.add('active')})});
|
|
|
|
const sources=[
|
|
{icon:'🌐',name:'Web Scraper',desc:'Dark scraper harvested domains',count:0,status:'ready',type:'scraper'},
|
|
{icon:'📧',name:'Email Validator',desc:'Validated email addresses',count:0,status:'ready',type:'validator'},
|
|
{icon:'🏢',name:'WHOIS Lookup',desc:'Domain ownership data',count:0,status:'ready',type:'whois'},
|
|
{icon:'📡',name:'DNS Records',desc:'MX, SPF, DKIM, DMARC records',count:0,status:'active',type:'dns'},
|
|
{icon:'🔍',name:'Newsletter Extractor',desc:'Competitor email patterns',count:0,status:'ready',type:'newsletter'},
|
|
{icon:'📊',name:'Analytics Feeds',desc:'Open/click behavioral data',count:0,status:'active',type:'analytics'},
|
|
{icon:'🛡️',name:'Blacklist Checker',desc:'IP/domain reputation data',count:0,status:'ready',type:'blacklist'}
|
|
];
|
|
|
|
const enrichSteps=[
|
|
{icon:'✉️',name:'Email Validation',desc:'Verify syntax + MX + SMTP',progress:0,processed:0},
|
|
{icon:'🏷️',name:'ISP Classification',desc:'Detect ISP for routing',progress:0,processed:0},
|
|
{icon:'📍',name:'Geo Location',desc:'Country/region detection',progress:0,processed:0},
|
|
{icon:'🔄',name:'Deduplication',desc:'Remove duplicates',progress:0,processed:0},
|
|
{icon:'🛡️',name:'Trap Detection',desc:'Identify honeypots/traps',progress:0,processed:0},
|
|
{icon:'⭐',name:'Quality Scoring',desc:'Assign engagement score',progress:0,processed:0}
|
|
];
|
|
|
|
function toast(m,t='info'){const d=document.createElement('div');d.style.cssText='position:fixed;top:20px;right:20px;padding:12px 20px;border-radius:10px;font-size:12px;font-weight:600;z-index:9999';d.style.background=t==='success'?'rgba(16,185,129,.15)':'rgba(236,72,153,.15)';d.style.color=t==='success'?'var(--gn)':'var(--pk)';d.textContent=m;document.body.appendChild(d);setTimeout(()=>d.remove(),3000)}
|
|
|
|
function load(){
|
|
document.getElementById('srcList').innerHTML=sources.map(s=>{
|
|
const bc=s.status==='active'?'b-ok':s.status==='error'?'b-err':'b-pu';
|
|
return`<div class="src"><div class="icon">${s.icon}</div><div class="info"><div class="name">${s.name}</div><div class="meta">${s.desc}</div></div><div class="count" style="color:var(--pk)">${s.count.toLocaleString()}</div><span class="badge ${bc}">${s.status}</span></div>`;
|
|
}).join('');
|
|
|
|
document.getElementById('recTable').innerHTML='<div class="empty"><div class="icon">📋</div><p>No records harvested yet</p><p style="font-size:11px;margin-top:8px">Run a harvest to populate data</p></div>';
|
|
|
|
document.getElementById('enrichPipe').innerHTML=enrichSteps.map(s=>`<div class="cd"><div style="font-size:24px;margin-bottom:8px">${s.icon}</div><div style="font-weight:600;margin-bottom:2px">${s.name}</div><div style="font-size:11px;color:var(--d);margin-bottom:10px">${s.desc}</div><div class="bar"><div class="bar-fill" style="width:${s.progress}%;background:var(--pk)"></div></div><div style="font-family:var(--m);font-size:11px;margin-top:4px;color:var(--d)">${s.processed} processed</div></div>`).join('');
|
|
|
|
document.getElementById('expTable').innerHTML='<div class="empty"><div class="icon">📦</div><p>No exports yet</p></div>';
|
|
}
|
|
|
|
function runHarvest(){toast('Harvest pipeline starting...','info');setTimeout(()=>{sources.forEach(s=>{s.count+=Math.floor(Math.random()*100)});load();toast('Harvest complete!','success')},2000)}
|
|
|
|
load();
|
|
</script><script>
|
|
var _API="/api/data.php";
|
|
async function _autoLoad(){
|
|
try{
|
|
var r=await fetch(_API+"?action=status");
|
|
var d=await r.json();
|
|
if(!d)return;
|
|
var src=d.data||d;
|
|
if(typeof src==="object"&&!Array.isArray(src)){
|
|
var entries=Object.entries(src);
|
|
var statEls=document.querySelectorAll(".st .n,.kpi-value,.stat-value,[id^=k],[id^=s-]");
|
|
statEls.forEach(function(el,idx){
|
|
if(idx<entries.length){
|
|
var v=entries[idx][1];
|
|
if(typeof v==="number")v=v.toLocaleString();
|
|
if(typeof v==="string"||typeof v==="number")el.textContent=v;
|
|
}
|
|
});
|
|
}
|
|
var si=document.querySelector(".status-dot,.live-dot,[class*=live]");
|
|
if(si)si.style.color="#22c55e";
|
|
}catch(e){console.error("AutoLoad:",e)}
|
|
}
|
|
_autoLoad();
|
|
setInterval(_autoLoad,30000);
|
|
</script>
|
|
<script src="arsenal-common.js?v1770778169">
|
|
</body></html>
|
|
</script>
|