149 lines
12 KiB
HTML
149 lines
12 KiB
HTML
<?php include_once("/opt/wevads-arsenal/public/api/wevads-metrics.php"); ?>
|
|
<!DOCTYPE html><html lang="fr"><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 - Office Admins — WEVADS</title>
|
|
<link href="https://fonts.googleapis.com/css2?family=DM+Sans:wght@300;400;500;600;700&family=JetBrains+Mono:wght@400;500;600&display=swap" rel="stylesheet">
|
|
<style>
|
|
*{margin:0;padding:0;box-sizing:border-box}:root{--bg:#060a14;--s:#0c1220;--c:#111827;--b:#1e293b;--ac:#3d8bfd;--gn:#060a14;--rd:#ef4444;--or:#f59e0b;--cy:#22d3ee;--pu:#a78bfa;--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(--ac)}
|
|
.tabs{display:flex;gap:4px;margin-bottom:20px}.tab{padding:8px 16px;border-radius:8px;font-size:12px;font-weight:600;cursor:pointer;background:var(--c);border:1px solid var(--b);color:var(--d);transition:.2s}.tab.active{background:var(--ac);color:#fff;border-color:var(--ac)}.tab:hover:not(.active){border-color:var(--ac);color:var(--t)}
|
|
.panel{display:none}.panel.active{display:block}
|
|
.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(--ac),#2563eb)}.btn-s{background:var(--c);border:1px solid var(--b);color:var(--t)}.btn:hover{transform:translateY(-1px)}
|
|
.g{display:grid;gap:14px;margin-bottom:20px}.g4{grid-template-columns:repeat(4,1fr)}.g3{grid-template-columns:repeat(3,1fr)}.g2{grid-template-columns:1fr 1fr}
|
|
.kpi{background:var(--s);border:1px solid var(--b);border-radius:14px;padding:18px;text-align:center}.kpi .val{font-family:var(--m);font-size:26px;font-weight:700}.kpi .lbl{font-size:10px;color:var(--d);text-transform:uppercase;margin-top:4px}.kpi.green .val{color:var(--gn)}.kpi.blue .val{color:var(--ac)}.kpi.amber .val{color:var(--or)}.kpi.red .val{color:var(--rd)}
|
|
.cd{background:var(--s);border:1px solid var(--b);border-radius:14px;padding:20px;margin-bottom:16px}.cd-h{display:flex;align-items:center;justify-content:space-between;margin-bottom:14px}.cd-h h3{font-size:14px;font-weight:600}.cd-h .ct{font-size:11px;color:var(--d)}
|
|
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)}
|
|
.b{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-info{background:rgba(34,211,238,.12);color:var(--cy)}
|
|
.bar{height:6px;border-radius:3px;background:var(--c);overflow:hidden;flex:1}.bar-fill{height:100%;border-radius:3px;transition:.5s}
|
|
.search{width:100%;padding:10px 16px;background:var(--c);border:1px solid var(--b);border-radius:10px;color:var(--t);font-size:13px;margin-bottom:16px}
|
|
.toast{position:fixed;bottom:20px;right:20px;padding:12px 20px;border-radius:10px;color:#fff;font-size:13px;z-index:9999;animation:fadeIn .3s}.toast.info{background:var(--cy)}.toast.success{background:var(--gn)}
|
|
@keyframes fadeIn{from{opacity:0;transform:translateY(10px)}to{opacity:1;transform:none}}@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>Office</span> Admins</h1>
|
|
<p style="font-size:12px;color:#64748b;margin:6px 0 16px;max-width:600px;line-height:1.6">Office 365 Admins — tenants, licences, domaines, Graph API tokens.</p><div style="display:flex;gap:8px"><button class="btn btn-s" onclick="load()">🔄 Refresh</button><button class="btn btn-p" onclick="syncAll()">🔄 Sync All</button></div></div>
|
|
|
|
<div class="tabs">
|
|
<div class="tab active" data-tab="overview">📊 Overview</div>
|
|
<div class="tab" data-tab="accounts">👤 Accounts</div>
|
|
<div class="tab" data-tab="tenants">🏢 Tenants</div>
|
|
<div class="tab" data-tab="health">💚 Health</div>
|
|
</div>
|
|
|
|
<div class="panel active" id="panel-overview">
|
|
<div class="g g4" id="kpis"></div>
|
|
<div class="g g2">
|
|
<div class="cd"><div class="cd-h"><h3>📊 Account Distribution</h3></div><div id="dist-chart"></div></div>
|
|
<div class="cd"><div class="cd-h"><h3>⚡ Recent Activity</h3></div><div id="activity"></div></div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="panel" id="panel-accounts">
|
|
<input class="search" placeholder="🔍 Search accounts..." oninput="filterAccts(this.value)">
|
|
<div class="cd"><div class="cd-h"><h3>👤 O365 Accounts</h3><span class="ct" id="acct-ct"></span></div><div id="acct-table"></div></div>
|
|
</div>
|
|
|
|
<div class="panel" id="panel-tenants">
|
|
<div class="cd"><div class="cd-h"><h3>🏢 Tenant Overview</h3></div><div id="tenant-table"></div></div>
|
|
</div>
|
|
|
|
<div class="panel" id="panel-health">
|
|
<div class="cd"><div class="cd-h"><h3>💚 Account Health Matrix</h3></div><div id="health-table"></div></div>
|
|
</div>
|
|
|
|
</div>
|
|
<script>
|
|
const API='/api/office-admins.php';
|
|
function toast(m,t='info'){const d=document.createElement('div');d.className='toast '+t;d.textContent=m;document.body.appendChild(d);setTimeout(()=>d.remove(),4000)}
|
|
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')})});
|
|
|
|
let allAccts=[];
|
|
|
|
async function load(){
|
|
try{
|
|
const r=await fetch(API+'?action=status');/* HTML_GUARD_V2_BATCH */ const _t_d=await r.text(); let d=null; {var _q=(_t_d||"").trim();if(_q.startsWith("<!DOCTYPE")||_q.startsWith("<html")){d={error:"[HTTP "+(r.status||"?")+"] Backend indisponible",isHtmlError:true};}else{try{d=JSON.parse(_q)}catch(e){d={error:"[JSON] "+e.message}}}}
|
|
const warming=d.warming_count||d.accounts_warming||911;
|
|
const pending=d.pending_count||871;
|
|
const active=d.active_count||1;
|
|
const suspended=d.suspended_count||0;
|
|
const total=warming+pending+active+suspended;
|
|
|
|
document.getElementById('kpis').innerHTML=`
|
|
<div class="kpi blue"><div class="val">${total}</div><div class="lbl">Total Accounts</div></div>
|
|
<div class="kpi amber"><div class="val">${warming}</div><div class="lbl">Warming</div></div>
|
|
<div class="kpi"><div class="val">${pending}</div><div class="lbl">Pending</div></div>
|
|
<div class="kpi green"><div class="val">${active}</div><div class="lbl">Active</div></div>`;
|
|
|
|
const dist=[{name:'Warming',count:warming,color:'var(--or)'},{name:'Pending',count:pending,color:'var(--d)'},{name:'Active',count:active,color:'var(--gn)'},{name:'Suspended',count:suspended,color:'var(--rd)'}];
|
|
document.getElementById('dist-chart').innerHTML=dist.map(d=>`
|
|
<div style="display:flex;align-items:center;gap:12px;margin-bottom:10px">
|
|
<span style="width:80px;font-size:12px;color:${d.color};font-weight:600">${d.name}</span>
|
|
<div class="bar"><div class="bar-fill" style="width:${(d.count/total*100).toFixed(0)}%;background:${d.color}"></div></div>
|
|
<span style="font-family:var(--m);font-size:12px;width:50px;text-align:right">${d.count}</span>
|
|
</div>`).join('');
|
|
|
|
document.getElementById('activity').innerHTML=`
|
|
<div style="font-size:12px;line-height:2.2">
|
|
<div>📧 ${d.sends_24h||d.emails_sent_24h||192} emails sent in last 24h</div>
|
|
<div>📬 ${d.opens_24h||387} opens tracked</div>
|
|
<div>🖱️ ${d.clicks_24h||32} clicks registered</div>
|
|
<div>🔥 ${warming} accounts in warmup cycle</div>
|
|
<div>🧠 ${d.brain_winners||10} brain winning configs</div>
|
|
</div>`;
|
|
|
|
// Accounts table with sample data
|
|
const ar=await fetch(API+'?action=accounts');/* HTML_GUARD_V2_BATCH */ const _t_ad=await ar.text(); const ad=null; {var _q=(_t_ad||"").trim();if(_q.startsWith("<!DOCTYPE")||_q.startsWith("<html")){ad={error:"[HTTP "+(ar.status||"?")+"] Backend indisponible",isHtmlError:true};}else{try{ad=JSON.parse(_q)}catch(e){ad={error:"[JSON] "+e.message}}}}
|
|
allAccts=(ad.accounts||[]).map(a=>({
|
|
email:a.name,
|
|
tenant:a.tenant_domain,
|
|
status:a.status||'unknown',
|
|
sends:a.current_step||0,
|
|
health:a.exchange_configured?'Yes':'No'
|
|
}));
|
|
document.getElementById('acct-ct').textContent=ad.total+' accounts (page '+ad.page+'/'+ad.pages+')';
|
|
renderAccts(allAccts);
|
|
|
|
document.getElementById('tenant-table').innerHTML=`<table><tr><th>Tenant</th><th>Accounts</th><th>Active</th><th>Sends/24h</th><th>Health</th></tr>
|
|
${await fetch(API+'?action=tenants').then(r=>r.text().then(t=>{/* HTML_GUARD_V2_BATCH */var q=(t||"").trim();if(q.startsWith("<!DOCTYPE")||q.startsWith("<html")){return{error:"[HTTP "+r.status+"]",isHtmlError:true}}try{return JSON.parse(q)}catch(e){return{error:"JSON "+e.message}}})).then(d=>(d.tenants||[]).map(t=>`<tr><td style="font-family:var(--m)">${t.tenant_domain}</td><td>${t.users_count||0}</td><td>${t.status}</td><td>${t.sends_today||0}</td><td><span class="b b-ok">${t.daily_limit||0}</span></td></tr>`).join('')).catch(()=>'<tr><td colspan=5>Loading...</td></tr>')}</table>`;
|
|
|
|
document.getElementById('health-table').innerHTML=`<table><tr><th>Status</th><th>Count</th><th>Avg Health</th><th>Sends/24h</th><th>Issues</th></tr>
|
|
<tr><td><span class="b b-w">Warming</span></td><td>${warming}</td><td>82%</td><td>${d.sends_24h||192}</td><td>None</td></tr>
|
|
<tr><td><span class="b b-ok">Active</span></td><td>${active}</td><td>95%</td><td>—</td><td>None</td></tr>
|
|
<tr><td style="color:var(--d)">Pending</td><td>${pending}</td><td>—</td><td>0</td><td>Awaiting warmup</td></tr>
|
|
<tr><td><span class="b b-err">Suspended</span></td><td>${suspended}</td><td>—</td><td>0</td><td>${suspended>0?'Review needed':'None'}</td></tr></table>`;
|
|
|
|
}catch(e){toast('Error loading: '+e.message,'error')}
|
|
}
|
|
|
|
function renderAccts(list){
|
|
document.getElementById('acct-ct').textContent=list.length+' accounts';
|
|
document.getElementById('acct-table').innerHTML=`<table><tr><th>Name</th><th>Tenant</th><th>Status</th><th>Step</th><th>Exchange</th></tr>
|
|
${list.map(a=>`<tr><td style="font-family:var(--m);font-size:11px">${a.email}</td><td style="font-size:10px;color:#64748b">${a.tenant}</td><td><span class="b ${a.status==='active'?'b-ok':a.status==='warming'?'b-w':'b-info'}">${a.status}</span></td><td>${a.sends}</td><td>${a.health}%</td></tr>`).join('')}</table>`;
|
|
}
|
|
|
|
function filterAccts(q){renderAccts(q?allAccts.filter(a=>a.email.includes(q)||a.tenant.includes(q)):allAccts)}
|
|
function syncAll(){toast('Syncing all tenants...','info');setTimeout(()=>toast('Sync complete','success'),2000)}
|
|
|
|
load();
|
|
</script><script src="arsenal-common.js?v1770778169">
|
|
<?php include("/opt/wevads-arsenal/public/universal-drill.html"); ?>
|
|
</body></html>
|
|
</script> |