155 lines
12 KiB
HTML
Executable File
155 lines
12 KiB
HTML
Executable File
<?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 - Configuration Manager — 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:#a78bfa;--gn:#060a14;--rd:#ef4444;--or:#f59e0b;--cy:#22d3ee;--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:#e2e8f0;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),#7c3aed)}.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}
|
|
.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}
|
|
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)}.b-pu{background:rgba(167,139,250,.12);color:var(--ac)}
|
|
.toggle{width:42px;height:22px;border-radius:11px;background:var(--c);border:1px solid var(--b);cursor:pointer;position:relative;transition:.3s}.toggle.on{background:var(--gn);border-color:var(--gn)}.toggle::after{content:'';width:16px;height:16px;border-radius:50%;background:#0c1220;position:absolute;top:2px;left:2px;transition:.3s}.toggle.on::after{left:22px}
|
|
.cfg-row{display:flex;align-items:center;justify-content:space-between;padding:12px;background:var(--c);border:1px solid var(--b);border-radius:10px;margin-bottom:6px}
|
|
.cfg-row .key{font-family:var(--m);font-size:12px;color:var(--cy)}.cfg-row .val-text{font-family:var(--m);font-size:12px;color:var(--t)}
|
|
.cfg-row .desc{font-size:11px;color:var(--d);margin-top:2px}
|
|
.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)}.toast.error{background:var(--rd)}
|
|
@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>Configuration</span> Manager</h1>
|
|
<p style="font-size:12px;color:#64748b;margin:6px 0 16px;max-width:600px;line-height:1.6">Gestionnaire de configurations — paramètres système, routing, limites, providers IA.</p><div style="display:flex;gap:8px"><button class="btn btn-s" onclick="load()">🔄 Reload</button><button class="btn btn-p" onclick="exportCfg()">📋 Export</button></div></div>
|
|
|
|
<div class="tabs">
|
|
<div class="tab active" data-tab="flags">🚩 Feature Flags</div>
|
|
<div class="tab" data-tab="system">🔧 System Config</div>
|
|
<div class="tab" data-tab="limits">📏 Rate Limits</div>
|
|
<div class="tab" data-tab="api">🔑 API Keys</div>
|
|
</div>
|
|
|
|
<div class="panel active" id="panel-flags"><div id="flags-list"></div></div>
|
|
<div class="panel" id="panel-system"><div id="sys-list"></div></div>
|
|
<div class="panel" id="panel-limits"><div id="limits-list"></div></div>
|
|
<div class="panel" id="panel-api"><div id="api-list"></div></div>
|
|
|
|
</div>
|
|
<script>
|
|
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')})});
|
|
|
|
const flags=[
|
|
{key:'WARMUP_ENGINE',on:true,desc:'Automated warmup cycles for O365 accounts'},
|
|
{key:'BRAIN_ENGINE',on:true,desc:'AI-powered config discovery and optimization'},
|
|
{key:'QUALITY_GUARD',on:true,desc:'8-rule quality checks before sending'},
|
|
{key:'TRACKING_PIPELINE',on:true,desc:'OVH→Hetzner click/open tracking'},
|
|
{key:'AFFILIATE_TRACKING',on:true,desc:'CX3/DoubleM conversion tracking'},
|
|
{key:'AUTO_ROTATION',on:true,desc:'Automatic domain/IP rotation'},
|
|
{key:'SENTINEL_V4',on:true,desc:'System monitoring and alerting'},
|
|
{key:'YOUTUBE_FACTORY',on:true,desc:'YouTube content automation'},
|
|
{key:'ADS_COMMANDER',on:true,desc:'Cross-platform ad management'},
|
|
{key:'HAMID_IA',on:true,desc:'Multi-provider AI assistant'},
|
|
{key:'DARK_MODE_ONLY',on:true,desc:'Force dark theme across all screens'},
|
|
{key:'DEBUG_MODE',on:false,desc:'Verbose logging for troubleshooting'}
|
|
];
|
|
|
|
const sysConfig=[
|
|
{key:'MAX_SENDS_PER_ACCOUNT',val:'50/day',desc:'Max emails per warmup account per day'},
|
|
{key:'WARMUP_RAMP_DAYS',val:'21',desc:'Days to full sending capacity'},
|
|
{key:'BRAIN_EVAL_INTERVAL',val:'6h',desc:'How often Brain re-evaluates configs'},
|
|
{key:'TRACKING_DOMAIN',val:'culturellemejean.charity',desc:'Primary tracking domain'},
|
|
{key:'CX3_TRACKING_DOMAIN',val:'babosur.com',desc:'CX3 Ads tracking domain'},
|
|
{key:'PMTA_HOST',val:'157.180.25.208',desc:'PowerMTA server address'},
|
|
{key:'OVH_TRACKING_HOST',val:'151.80.235.110',desc:'OVH tracking server'},
|
|
{key:'DB_NAME',val:'adx_system',desc:'Primary PostgreSQL database'},
|
|
{key:'ARSENAL_PORT',val:'5890',desc:'Arsenal API backend port'},
|
|
{key:'ADX_PORT',val:'5821',desc:'WEVADS frontend port'}
|
|
];
|
|
|
|
const limits=[
|
|
{key:'API_RATE_LIMIT',val:'100/min',scope:'Global',status:'ok'},
|
|
{key:'SEND_RATE_PMTA',val:'500/hr',scope:'Per Queue',status:'ok'},
|
|
{key:'WARMUP_BATCH',val:'50',scope:'Per Cycle',status:'ok'},
|
|
{key:'BRAIN_CONCURRENT',val:'10',scope:'Evaluations',status:'ok'},
|
|
{key:'TRACKING_INGEST',val:'1000/min',scope:'Events',status:'ok'},
|
|
{key:'O365_GRAPH_API',val:'10000/10min',scope:'Per Tenant',status:'ok'}
|
|
];
|
|
|
|
const apiKeys=[
|
|
{name:'CX3 Ads',key:'y2vBu0M...0Bg',scope:'Affiliate',status:'active'},
|
|
{name:'Cloudflare',key:'••••••••',scope:'DNS/CDN',status:'active'},
|
|
{name:'Huawei Cloud',key:'••••••••',scope:'Storage',status:'active'},
|
|
{name:'HAMID Cerebras',key:'••••••••',scope:'AI Provider',status:'active'},
|
|
{name:'HAMID Groq',key:'••••••••',scope:'AI Provider',status:'active'},
|
|
{name:'HAMID DeepSeek',key:'••••••••',scope:'AI Provider',status:'active'},
|
|
{name:'Exec Key',key:'WEVADS2026',scope:'DM+Sansnal',status:'active'}
|
|
];
|
|
|
|
function load(){
|
|
document.getElementById('flags-list').innerHTML=flags.map((f,i)=>`
|
|
<div class="cfg-row"><div><div class="key">${f.key}</div><div class="desc">${f.desc}</div></div>
|
|
<div class="toggle ${f.on?'on':''}" onclick="this.classList.toggle('on');toast('${f.key} toggled','success')"></div></div>`).join('');
|
|
|
|
document.getElementById('sys-list').innerHTML=sysConfig.map(s=>`
|
|
<div class="cfg-row"><div><div class="key">${s.key}</div><div class="desc">${s.desc}</div></div>
|
|
<div class="val-text">${s.val}</div></div>`).join('');
|
|
|
|
document.getElementById('limits-list').innerHTML=`<div class="cd"><table><tr><th>Limit</th><th>Value</th><th>Scope</th><th>Status</th></tr>
|
|
${limits.map(l=>`<tr><td style="font-family:var(--m);color:var(--cy)">${l.key}</td><td style="font-family:var(--m)">${l.val}</td><td>${l.scope}</td><td><span class="b b-ok">${l.status}</span></td></tr>`).join('')}</table></div>`;
|
|
|
|
document.getElementById('api-list').innerHTML=`<div class="cd"><table><tr><th>Service</th><th>Key</th><th>Scope</th><th>Status</th></tr>
|
|
${apiKeys.map(a=>`<tr><td style="font-weight:600">${a.name}</td><td style="font-family:var(--m);font-size:11px">${a.key}</td><td><span class="b b-pu">${a.scope}</span></td><td><span class="b b-ok">${a.status}</span></td></tr>`).join('')}</table></div>`;
|
|
}
|
|
|
|
function exportCfg(){toast('Configuration exported to clipboard','success')}
|
|
load();
|
|
</script><script>
|
|
var _API="/api/config-manager.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;
|
|
}
|
|
});
|
|
}
|
|
}catch(e){console.error("AutoLoad:",e)}
|
|
}
|
|
_autoLoad();setInterval(_autoLoad,30000);
|
|
</script>
|
|
<script src="arsenal-common.js?v1770778169">
|
|
<?php include("/opt/wevads-arsenal/public/universal-drill.html"); ?>
|
|
</body></html>
|
|
</script>
|