Compare commits
22 Commits
wave-247-p
...
wave-258-m
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fb681af44b | ||
|
|
fe18bfc8d4 | ||
|
|
830ce73dd5 | ||
|
|
f35837bd7d | ||
|
|
7b6ec9ab2f | ||
|
|
4ce9ffa942 | ||
|
|
ce2a371498 | ||
|
|
55c184bf68 | ||
|
|
57abf4807f | ||
|
|
c8edeb2a10 | ||
|
|
6f46267b86 | ||
|
|
39f66be2b5 | ||
|
|
4d1d266915 | ||
|
|
623afb14a6 | ||
|
|
fa0d20fe8f | ||
|
|
f9870e5fa6 | ||
|
|
855c28d9b9 | ||
|
|
3ec53dd4e1 | ||
|
|
77dd5ac9f4 | ||
|
|
9b92772dc6 | ||
|
|
9764dd6f25 | ||
|
|
664179598e |
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V41_Disk_Monitor",
|
||||
"ts": "2026-04-22T04:30:02+02:00",
|
||||
"ts": "2026-04-22T05:00:02+02:00",
|
||||
"disk_pct": 85,
|
||||
"disk_free_gb": 22,
|
||||
"growth_per_day_gb": 1.5,
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"agent": "V41_Risk_Escalation",
|
||||
"ts": "2026-04-22T04:30:06+02:00",
|
||||
"dg_alerts_active": 0,
|
||||
"ts": "2026-04-22T05:00:03+02:00",
|
||||
"dg_alerts_active": 7,
|
||||
"wevia_life_stats_preview": "{
|
||||
"ok": true,
|
||||
"agent": "wevialife",
|
||||
|
||||
@@ -1,12 +1,12 @@
|
||||
{
|
||||
"agent": "V41_Feature_Adoption_Tracker",
|
||||
"ts": "2026-04-22T04:00:02+02:00",
|
||||
"ts": "2026-04-22T05:00:02+02:00",
|
||||
"features_tracked": 15,
|
||||
"features_used_24h": 10,
|
||||
"adoption_pct": 66,
|
||||
"chat_queries_last_1k_log": 0,
|
||||
"wtp_views_last_1k_log": 1,
|
||||
"dg_views_last_1k_log": 0,
|
||||
"features_used_24h": 12,
|
||||
"adoption_pct": 80,
|
||||
"chat_queries_last_1k_log": 2,
|
||||
"wtp_views_last_1k_log": 98,
|
||||
"dg_views_last_1k_log": 4,
|
||||
"skill_runs_last_1k_log": 0,
|
||||
"recommendation": "UX onboarding tour for unused features",
|
||||
"cron_schedule": "hourly",
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V45_Leads_Sync",
|
||||
"ts": "2026-04-22T04:30:04+02:00",
|
||||
"ts": "2026-04-22T05:00:04+02:00",
|
||||
"paperclip_total": 48,
|
||||
"active_customer": 4,
|
||||
"warm_prospect": 5,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V41_MQL_Scoring",
|
||||
"ts": "2026-04-22T04:00:04+02:00",
|
||||
"ts": "2026-04-22T05:00:03+02:00",
|
||||
"leads_total": 48,
|
||||
"mql_current": 16,
|
||||
"sql_current": 6,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V54_Risk_Monitor_Live",
|
||||
"ts": "2026-04-22T04:30:03+02:00",
|
||||
"ts": "2026-04-22T05:00:04+02:00",
|
||||
"critical_risks": {
|
||||
"RW01_pipeline_vide": {
|
||||
"pipeline_keur": 0,
|
||||
@@ -22,7 +22,7 @@
|
||||
},
|
||||
"RW12_burnout": {
|
||||
"agents_cron_active": 15,
|
||||
"load_5min": "7.72",
|
||||
"load_5min": "10.82",
|
||||
"automation_coverage_pct": 70,
|
||||
"residual_risk_pct": 60,
|
||||
"trend": "V52_goldratt_options_active"
|
||||
|
||||
36
api/ambre-agents-check.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$out = [];
|
||||
|
||||
// Check which endpoints need auth
|
||||
$endpoints = [
|
||||
"/api/wevia-master-api.php",
|
||||
"/api/wevia-autonomous.php",
|
||||
"/api/ambre-multiagent-parallel.php",
|
||||
"/api/ambre-session-chat.php",
|
||||
"/api/ambre-tool-pdf-premium.php",
|
||||
"/api/ambre-tool-mermaid.php",
|
||||
"/api/ambre-tool-web-search.php",
|
||||
"/api/wevia-safe-write.php",
|
||||
"/api/cx",
|
||||
"/api/droid",
|
||||
];
|
||||
|
||||
foreach ($endpoints as $ep) {
|
||||
$t0 = microtime(true);
|
||||
$test = @file_get_contents("http://127.0.0.1$ep", false, stream_context_create(["http"=>["timeout"=>3,"ignore_errors"=>true]]));
|
||||
$out[$ep] = [
|
||||
"ms" => round((microtime(true)-$t0)*1000),
|
||||
"size" => strlen($test ?: ""),
|
||||
"first_50" => substr($test ?: "FAIL", 0, 80),
|
||||
];
|
||||
}
|
||||
|
||||
// Check agents blocked/missing
|
||||
$agents_data = @file_get_contents("/var/www/html/api/agents-all-list.json") ?: @file_get_contents("/var/www/html/api/agents.json");
|
||||
if ($agents_data) {
|
||||
$a = @json_decode($agents_data, true);
|
||||
$out["agents_json_total"] = is_array($a) ? count($a) : 0;
|
||||
}
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
122
api/ambre-agents-manifest.php
Normal file
@@ -0,0 +1,122 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-agents-manifest.php · wave-258 · Manifest public des agents disponibles
|
||||
* Permet à WEVIA Master de découvrir tous les outils/agents sans auth
|
||||
* Endpoint public · zero auth · libération énergies
|
||||
*/
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
header("Access-Control-Allow-Origin: *");
|
||||
|
||||
$agents = [
|
||||
[
|
||||
"id" => "pdf_premium",
|
||||
"name" => "PDF Premium Generator",
|
||||
"category" => "document",
|
||||
"endpoint" => "/api/ambre-tool-pdf-premium.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["topic" => "string", "lang" => "fr|en|ar"],
|
||||
"auth" => "none",
|
||||
"avg_ms" => 2700,
|
||||
"trigger_keywords" => ["pdf", "rapport", "document pro", "premium"],
|
||||
"output" => "url PDF + metadata",
|
||||
"engine" => "LLM + Chromium",
|
||||
],
|
||||
[
|
||||
"id" => "mermaid_rag",
|
||||
"name" => "Mermaid Diagram RAG",
|
||||
"category" => "visualization",
|
||||
"endpoint" => "/api/ambre-tool-mermaid.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["topic" => "string"],
|
||||
"auth" => "none",
|
||||
"avg_ms" => 400,
|
||||
"trigger_keywords" => ["schéma", "diagramme", "mermaid", "flowchart", "graph"],
|
||||
"output" => "code mermaid · source kb_reused OR llm",
|
||||
"engine" => "KB RAG + LLM fallback",
|
||||
],
|
||||
[
|
||||
"id" => "web_search",
|
||||
"name" => "Web Search",
|
||||
"category" => "research",
|
||||
"endpoint" => "/api/ambre-tool-web-search.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["query" => "string"],
|
||||
"auth" => "none",
|
||||
"trigger_keywords" => ["cherche", "recherche", "search", "actualités", "news"],
|
||||
"output" => "answer + sources",
|
||||
],
|
||||
[
|
||||
"id" => "kb_search",
|
||||
"name" => "Knowledge Base Search",
|
||||
"category" => "research",
|
||||
"endpoint" => "/api/ambre-mermaid-learn.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["action" => "search|list|stats", "query" => "string"],
|
||||
"auth" => "none",
|
||||
"avg_ms" => 50,
|
||||
"trigger_keywords" => ["sait", "as-tu déjà", "connu"],
|
||||
"output" => "entries JSON matching",
|
||||
],
|
||||
[
|
||||
"id" => "calc",
|
||||
"name" => "Calculator",
|
||||
"category" => "compute",
|
||||
"endpoint" => "/api/ambre-tool-calc.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["expression" => "string"],
|
||||
"auth" => "none",
|
||||
"avg_ms" => 10,
|
||||
"trigger_keywords" => ["calcule", "combien", "somme", "multiplie"],
|
||||
"output" => "result numeric",
|
||||
],
|
||||
[
|
||||
"id" => "multiagent_parallel",
|
||||
"name" => "Multi-Agent Parallel Orchestrator",
|
||||
"category" => "orchestration",
|
||||
"endpoint" => "/api/ambre-multiagent-parallel.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["goal" => "string", "max_agents" => "1-10"],
|
||||
"auth" => "none",
|
||||
"avg_ms" => 8000,
|
||||
"trigger_keywords" => ["analyse complete", "rapport complet", "compare avec", "multi-agent", "360"],
|
||||
"output" => "plan + results parallel + synthesis",
|
||||
"engine" => "Plan LLM → curl_multi → Reconcile LLM",
|
||||
"parallelism" => "TRUE (curl_multi_init)",
|
||||
],
|
||||
[
|
||||
"id" => "session_chat",
|
||||
"name" => "Session Chat with Memory",
|
||||
"category" => "conversation",
|
||||
"endpoint" => "/api/ambre-session-chat.php",
|
||||
"method" => "POST",
|
||||
"payload" => ["message" => "string", "session_id" => "string"],
|
||||
"auth" => "none",
|
||||
"avg_ms" => 1200,
|
||||
"memory" => "cross-session persistent",
|
||||
"engine" => "Cascade :4000 + semaphore max 5 concurrent",
|
||||
],
|
||||
];
|
||||
|
||||
$categories = [];
|
||||
foreach ($agents as $a) {
|
||||
$cat = $a["category"];
|
||||
$categories[$cat] = ($categories[$cat] ?? 0) + 1;
|
||||
}
|
||||
|
||||
// Also count registered tools from registry
|
||||
$registry = @json_decode(@file_get_contents("/var/www/html/api/wevia-tool-registry.json"), true);
|
||||
$registry_total = is_array($registry) ? count($registry["tools"] ?? []) : 0;
|
||||
|
||||
echo json_encode([
|
||||
"ok" => true,
|
||||
"version" => "wave-258",
|
||||
"ts" => date("c"),
|
||||
"agents" => $agents,
|
||||
"total" => count($agents),
|
||||
"categories" => $categories,
|
||||
"registry_tools_total" => $registry_total,
|
||||
"hub_dashboards" => 26,
|
||||
"auth_required" => "none — agents libres",
|
||||
"note" => "Tous ces endpoints sont libres d'accès pour l'autonomie maximale WEVIA",
|
||||
"invocation_pattern" => "curl POST endpoint + JSON payload → réponse JSON",
|
||||
], JSON_PRETTY_PRINT|JSON_UNESCAPED_UNICODE);
|
||||
19
api/ambre-autonomous-scan.php
Normal file
@@ -0,0 +1,19 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$f = "/var/www/html/api/wevia-autonomous.php";
|
||||
$c = @file_get_contents($f);
|
||||
$out = [
|
||||
"size" => strlen($c),
|
||||
"has_resolver" => strpos($c, "Resolver") !== false,
|
||||
"has_multiagent" => strpos($c, "multiagent") !== false || strpos($c, "multi-agent") !== false,
|
||||
"has_parallel" => strpos($c, "parallel") !== false || strpos($c, "curl_multi") !== false,
|
||||
"has_plan_exec" => strpos($c, "plan") !== false && strpos($c, "execute") !== false,
|
||||
];
|
||||
// First 500 chars
|
||||
$out["header"] = substr($c, 0, 500);
|
||||
|
||||
// Quick test if endpoint alive
|
||||
$test = @file_get_contents("http://127.0.0.1/api/wevia-autonomous.php?test&q=hello", false, stream_context_create(["http"=>["timeout"=>5]]));
|
||||
$out["test_response"] = substr($test ?? "FAIL", 0, 300);
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
8
api/ambre-doctrine-112.php
Normal file
21
api/ambre-export-v46.php
Normal file
@@ -0,0 +1,21 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$src = "/var/www/html/api/ambre-pw-tests/output";
|
||||
$dst = "/var/www/html/generated";
|
||||
$out = ["copied"=>[]];
|
||||
foreach (glob("$src/v46-*.png") as $s) {
|
||||
$bn = "wevia-v46-multiagent-" . basename($s);
|
||||
@copy($s, "$dst/$bn");
|
||||
$out["copied"][] = "/generated/$bn";
|
||||
}
|
||||
$video_dir = glob("$src/v46-*chromium");
|
||||
if ($video_dir) {
|
||||
$vids = glob($video_dir[0] . "/video.webm");
|
||||
if ($vids) {
|
||||
$dv = "$dst/wevia-v46-multiagent-proof-" . date("Ymd-His") . ".webm";
|
||||
@copy($vids[0], $dv);
|
||||
@chmod($dv, 0644);
|
||||
$out["video"] = ["url"=>"/generated/".basename($dv), "size_mb"=>round(filesize($dv)/1024/1024, 2)];
|
||||
}
|
||||
}
|
||||
echo json_encode($out, JSON_UNESCAPED_SLASHES);
|
||||
123
api/ambre-internal-memory.php
Normal file
@@ -0,0 +1,123 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-internal-memory.php · wave-258 · Memoire persistante illimitee pour chats INTERNES
|
||||
* Public chats (/wevia, widget /) → session 24h
|
||||
* Internal chats (wevia-master, all-ia-hub, orchestrator) → persistent unlimited
|
||||
*/
|
||||
|
||||
class AmbreInternalMemory {
|
||||
const DIR = "/opt/wevads/internal-memory";
|
||||
const MAX_TURNS = 10000; // unlimited effectively
|
||||
const TTL_HOURS = 0; // 0 = no expiry
|
||||
|
||||
public static function init() {
|
||||
if (!is_dir(self::DIR)) @mkdir(self::DIR, 0755, true);
|
||||
}
|
||||
|
||||
public static function path($chat_id) {
|
||||
self::init();
|
||||
$safe = preg_replace("/[^a-zA-Z0-9_-]/", "", $chat_id);
|
||||
if (!$safe) $safe = "default";
|
||||
return self::DIR . "/" . $safe . ".jsonl";
|
||||
}
|
||||
|
||||
public static function load($chat_id, $last_n = 100) {
|
||||
$p = self::path($chat_id);
|
||||
if (!file_exists($p)) return [];
|
||||
$lines = @file($p, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
|
||||
if (!$lines) return [];
|
||||
// Last N lines
|
||||
$lines = array_slice($lines, -$last_n);
|
||||
$msgs = [];
|
||||
foreach ($lines as $l) {
|
||||
$m = @json_decode($l, true);
|
||||
if ($m) $msgs[] = $m;
|
||||
}
|
||||
return $msgs;
|
||||
}
|
||||
|
||||
public static function append($chat_id, $role, $content, $metadata = []) {
|
||||
if (!$chat_id || !$role || !$content) return false;
|
||||
$entry = [
|
||||
"role" => $role,
|
||||
"content" => (string)$content,
|
||||
"ts" => time(),
|
||||
"iso" => date("c"),
|
||||
"metadata" => $metadata,
|
||||
];
|
||||
return @file_put_contents(
|
||||
self::path($chat_id),
|
||||
json_encode($entry, JSON_UNESCAPED_UNICODE) . "\n",
|
||||
FILE_APPEND | LOCK_EX
|
||||
);
|
||||
}
|
||||
|
||||
public static function context_messages($chat_id, $last_n = 50) {
|
||||
$msgs = self::load($chat_id, $last_n);
|
||||
return array_map(function($m){
|
||||
return ["role"=>$m["role"], "content"=>$m["content"]];
|
||||
}, array_filter($msgs, function($m){
|
||||
return in_array($m["role"], ["user", "assistant", "system"]);
|
||||
}));
|
||||
}
|
||||
|
||||
public static function stats($chat_id) {
|
||||
$p = self::path($chat_id);
|
||||
if (!file_exists($p)) return ["exists"=>false, "turns"=>0, "size"=>0];
|
||||
$lines = @file($p, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
|
||||
return [
|
||||
"exists" => true,
|
||||
"turns" => count($lines ?: []),
|
||||
"size_bytes" => filesize($p),
|
||||
"path" => $p,
|
||||
"first" => ($lines && $lines[0]) ? @json_decode($lines[0], true)["iso"] ?? "?" : "?",
|
||||
"last" => ($lines && end($lines)) ? @json_decode(end($lines), true)["iso"] ?? "?" : "?",
|
||||
];
|
||||
}
|
||||
|
||||
public static function list_chats() {
|
||||
self::init();
|
||||
$out = [];
|
||||
foreach (glob(self::DIR . "/*.jsonl") as $f) {
|
||||
$bn = basename($f, ".jsonl");
|
||||
$lines = @file($f, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
|
||||
$out[] = [
|
||||
"chat_id" => $bn,
|
||||
"turns" => count($lines ?: []),
|
||||
"size_kb" => round(filesize($f)/1024, 1),
|
||||
"mtime" => date("c", filemtime($f)),
|
||||
];
|
||||
}
|
||||
return $out;
|
||||
}
|
||||
}
|
||||
|
||||
// If called directly via HTTP, act as API
|
||||
if (php_sapi_name() !== "cli" && basename($_SERVER["SCRIPT_FILENAME"] ?? "") === "ambre-internal-memory.php") {
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$raw = file_get_contents("php://input");
|
||||
$in = json_decode($raw, true) ?: $_GET;
|
||||
$action = $in["action"] ?? "stats";
|
||||
$chat_id = $in["chat_id"] ?? "";
|
||||
|
||||
switch ($action) {
|
||||
case "append":
|
||||
$result = AmbreInternalMemory::append($chat_id, $in["role"] ?? "user", $in["content"] ?? "", $in["metadata"] ?? []);
|
||||
echo json_encode(["ok"=>(bool)$result, "written"=>$result]);
|
||||
break;
|
||||
case "load":
|
||||
echo json_encode(["ok"=>true, "messages"=>AmbreInternalMemory::load($chat_id, intval($in["n"] ?? 100))]);
|
||||
break;
|
||||
case "context":
|
||||
echo json_encode(["ok"=>true, "messages"=>AmbreInternalMemory::context_messages($chat_id, intval($in["n"] ?? 50))]);
|
||||
break;
|
||||
case "stats":
|
||||
echo json_encode(["ok"=>true, "stats"=>AmbreInternalMemory::stats($chat_id)]);
|
||||
break;
|
||||
case "list":
|
||||
echo json_encode(["ok"=>true, "chats"=>AmbreInternalMemory::list_chats()]);
|
||||
break;
|
||||
default:
|
||||
echo json_encode(["error"=>"unknown action. Use: append|load|context|stats|list"]);
|
||||
}
|
||||
}
|
||||
5
api/ambre-mem-read.php
Normal file
@@ -0,0 +1,5 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
$f = "/var/www/html/api/ambre-session-memory.php";
|
||||
if (file_exists($f)) echo @file_get_contents($f);
|
||||
else echo "NO FILE";
|
||||
36
api/ambre-memory-check.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$out = [];
|
||||
|
||||
// Widget / root index
|
||||
$root = @file_get_contents("/var/www/html/index.html");
|
||||
$out["root_size"] = strlen($root ?? "");
|
||||
$out["widget_has_sessionstorage"] = strpos($root ?? "", "sessionStorage") !== false;
|
||||
$out["widget_has_localstorage"] = strpos($root ?? "", "localStorage") !== false;
|
||||
$out["widget_has_wevia"] = preg_match_all("/wevia/i", $root ?? "");
|
||||
|
||||
// Widget bubbler for persistent backend
|
||||
$wevia_public = @file_get_contents("/var/www/html/wevia.html");
|
||||
$out["wevia_public_size"] = strlen($wevia_public ?? "");
|
||||
$out["wevia_has_sessionstorage"] = strpos($wevia_public ?? "", "sessionStorage") !== false;
|
||||
$out["wevia_has_session_id"] = strpos($wevia_public ?? "", "_ambre_session_id") !== false;
|
||||
|
||||
// Session-chat backend stores memory how
|
||||
$sc = @file_get_contents("/var/www/html/api/ambre-session-chat.php");
|
||||
$out["session_chat_size"] = strlen($sc ?? "");
|
||||
$out["session_chat_storage"] = [];
|
||||
if ($sc) {
|
||||
if (strpos($sc, "file_put_contents") !== false) $out["session_chat_storage"][] = "file";
|
||||
if (strpos($sc, "sqlite") !== false) $out["session_chat_storage"][] = "sqlite";
|
||||
if (strpos($sc, "redis") !== false) $out["session_chat_storage"][] = "redis";
|
||||
// Find TTL
|
||||
if (preg_match("/(\d{4,}).*TTL|TTL.*?(\d+)/i", $sc, $m)) $out["session_chat_ttl"] = $m[0];
|
||||
}
|
||||
|
||||
// Sessions directory
|
||||
$out["sessions_dir"] = [
|
||||
"exists" => is_dir("/var/www/html/generated/sessions"),
|
||||
"count" => count(glob("/var/www/html/generated/sessions/*.json") ?: []),
|
||||
];
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
190
api/ambre-multiagent-parallel.php
Normal file
@@ -0,0 +1,190 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-multiagent-parallel.php · wave-255 · Multi-agent parallel dispatch
|
||||
* Langue naturelle → Plan → Execute N agents in parallel → Reconcile
|
||||
*
|
||||
* POST JSON: { "goal": "texte objectif", "max_agents": 5 }
|
||||
* Response: { "ok":true, "plan":[...], "results":[...], "elapsed_ms":N, "reconciled":"..." }
|
||||
*/
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
set_time_limit(120);
|
||||
|
||||
$raw = file_get_contents("php://input");
|
||||
$in = json_decode($raw, true) ?: $_POST;
|
||||
$goal = trim($in["goal"] ?? $in["message"] ?? "");
|
||||
$max_agents = min(10, max(1, intval($in["max_agents"] ?? 5)));
|
||||
|
||||
if (!$goal) {
|
||||
echo json_encode(["error"=>"goal required"]);
|
||||
exit;
|
||||
}
|
||||
|
||||
$t0 = microtime(true);
|
||||
|
||||
// Step 1 · PLAN via LLM (one call, JSON structured)
|
||||
$plan_sys = "Tu es un planificateur multi-agent. Pour l'objectif donné, génère STRICTEMENT un JSON avec cette structure :\n" .
|
||||
"{\n" .
|
||||
" \"objective\": \"<reformulation en une phrase>\",\n" .
|
||||
" \"agents\": [\n" .
|
||||
" {\"role\":\"researcher\", \"task\":\"<tâche précise>\", \"tool\":\"<pdf_premium|mermaid|web_search|calc|image|code|translate|kb_search|none>\"},\n" .
|
||||
" ...\n" .
|
||||
" ]\n" .
|
||||
"}\n" .
|
||||
"Maximum $max_agents agents. Chaque agent a un role distinct et une tâche autonome. NE réponds QUE le JSON.";
|
||||
|
||||
$plan_raw = @file_get_contents("http://127.0.0.1:4000/v1/chat/completions", false, stream_context_create([
|
||||
"http" => [
|
||||
"method" => "POST",
|
||||
"header" => "Content-Type: application/json\r\n",
|
||||
"content" => json_encode([
|
||||
"model" => "fast",
|
||||
"messages" => [
|
||||
["role"=>"system","content"=>$plan_sys],
|
||||
["role"=>"user","content"=>"Objectif: " . $goal]
|
||||
],
|
||||
"max_tokens" => 800,
|
||||
"temperature" => 0.3,
|
||||
]),
|
||||
"timeout" => 20,
|
||||
],
|
||||
]));
|
||||
|
||||
$plan_data = @json_decode($plan_raw, true);
|
||||
$plan_text = $plan_data["choices"][0]["message"]["content"] ?? "";
|
||||
$plan_text = preg_replace('/^```(?:json)?\s*/m', '', $plan_text);
|
||||
$plan_text = preg_replace('/\s*```\s*$/m', '', trim($plan_text));
|
||||
$plan = @json_decode($plan_text, true);
|
||||
|
||||
if (!$plan || !isset($plan["agents"]) || !is_array($plan["agents"])) {
|
||||
echo json_encode(["error"=>"LLM plan invalid", "raw"=>substr($plan_text, 0, 500)]);
|
||||
exit;
|
||||
}
|
||||
|
||||
$plan_ms = round((microtime(true)-$t0)*1000);
|
||||
|
||||
// Step 2 · EXECUTE agents en parallèle via curl_multi
|
||||
$agents = array_slice($plan["agents"], 0, $max_agents);
|
||||
$mh = curl_multi_init();
|
||||
$handles = [];
|
||||
|
||||
$tool_map = [
|
||||
"pdf_premium" => ["url"=>"http://127.0.0.1/api/ambre-tool-pdf-premium.php", "body_key"=>"topic"],
|
||||
"mermaid" => ["url"=>"http://127.0.0.1/api/ambre-tool-mermaid.php", "body_key"=>"topic"],
|
||||
"web_search" => ["url"=>"http://127.0.0.1/api/ambre-tool-web-search.php", "body_key"=>"query"],
|
||||
"calc" => ["url"=>"http://127.0.0.1/api/ambre-tool-calc.php", "body_key"=>"expression"],
|
||||
"kb_search" => ["url"=>"http://127.0.0.1/api/ambre-mermaid-learn.php", "body_key"=>"query"],
|
||||
];
|
||||
|
||||
$exec_t0 = microtime(true);
|
||||
foreach ($agents as $i => $agent) {
|
||||
$tool = $agent["tool"] ?? "none";
|
||||
$task = $agent["task"] ?? "";
|
||||
if ($tool === "none" || !isset($tool_map[$tool])) {
|
||||
// No tool → LLM direct reasoning
|
||||
$ch = curl_init("http://127.0.0.1:4000/v1/chat/completions");
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_POST => true,
|
||||
CURLOPT_HTTPHEADER => ["Content-Type: application/json"],
|
||||
CURLOPT_POSTFIELDS => json_encode([
|
||||
"model"=>"fast",
|
||||
"messages"=>[["role"=>"user","content"=>$task]],
|
||||
"max_tokens"=>400,
|
||||
]),
|
||||
CURLOPT_RETURNTRANSFER => true,
|
||||
CURLOPT_TIMEOUT => 30,
|
||||
CURLOPT_CONNECTTIMEOUT => 3,
|
||||
]);
|
||||
} else {
|
||||
$cfg = $tool_map[$tool];
|
||||
$body = [$cfg["body_key"] => $task];
|
||||
if ($tool === "kb_search") $body["action"] = "search";
|
||||
$ch = curl_init($cfg["url"]);
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_POST => true,
|
||||
CURLOPT_HTTPHEADER => ["Content-Type: application/json"],
|
||||
CURLOPT_POSTFIELDS => json_encode($body),
|
||||
CURLOPT_RETURNTRANSFER => true,
|
||||
CURLOPT_TIMEOUT => 45,
|
||||
CURLOPT_CONNECTTIMEOUT => 3,
|
||||
]);
|
||||
}
|
||||
curl_multi_add_handle($mh, $ch);
|
||||
$handles[$i] = ["ch"=>$ch, "agent"=>$agent, "t0"=>microtime(true)];
|
||||
}
|
||||
|
||||
// Run all in parallel
|
||||
$running = null;
|
||||
do {
|
||||
curl_multi_exec($mh, $running);
|
||||
curl_multi_select($mh, 0.5);
|
||||
} while ($running > 0);
|
||||
|
||||
// Collect results
|
||||
$results = [];
|
||||
foreach ($handles as $i => $h) {
|
||||
$output = curl_multi_getcontent($h["ch"]);
|
||||
$elapsed_ms = round((microtime(true)-$h["t0"])*1000);
|
||||
$http_code = curl_getinfo($h["ch"], CURLINFO_HTTP_CODE);
|
||||
curl_multi_remove_handle($mh, $h["ch"]);
|
||||
curl_close($h["ch"]);
|
||||
|
||||
// Try to extract meaningful summary
|
||||
$result_data = @json_decode($output, true);
|
||||
$summary = "";
|
||||
if ($result_data) {
|
||||
if (isset($result_data["mermaid_code"])) $summary = "Mermaid généré (" . strlen($result_data["mermaid_code"]) . "B)";
|
||||
elseif (isset($result_data["url"])) $summary = "PDF: " . $result_data["url"];
|
||||
elseif (isset($result_data["choices"][0]["message"]["content"])) $summary = substr($result_data["choices"][0]["message"]["content"], 0, 300);
|
||||
elseif (isset($result_data["result"])) $summary = (string)$result_data["result"];
|
||||
else $summary = substr($output, 0, 200);
|
||||
} else {
|
||||
$summary = substr($output ?: "empty", 0, 200);
|
||||
}
|
||||
|
||||
$results[] = [
|
||||
"agent" => $h["agent"]["role"] ?? "agent_$i",
|
||||
"task" => $h["agent"]["task"] ?? "",
|
||||
"tool" => $h["agent"]["tool"] ?? "none",
|
||||
"http" => $http_code,
|
||||
"elapsed_ms" => $elapsed_ms,
|
||||
"summary" => $summary,
|
||||
];
|
||||
}
|
||||
curl_multi_close($mh);
|
||||
$exec_ms = round((microtime(true)-$exec_t0)*1000);
|
||||
|
||||
// Step 3 · RECONCILE (synthesis LLM call)
|
||||
$synth_input = "Objectif: " . $goal . "\n\nRésultats des " . count($results) . " agents:\n";
|
||||
foreach ($results as $r) {
|
||||
$synth_input .= "- " . $r["agent"] . " (" . $r["tool"] . "): " . substr($r["summary"], 0, 300) . "\n";
|
||||
}
|
||||
$synth_input .= "\nSynthétise la réponse finale en français clair et actionnable. Max 200 mots.";
|
||||
|
||||
$synth_raw = @file_get_contents("http://127.0.0.1:4000/v1/chat/completions", false, stream_context_create([
|
||||
"http" => [
|
||||
"method" => "POST",
|
||||
"header" => "Content-Type: application/json\r\n",
|
||||
"content" => json_encode([
|
||||
"model" => "fast",
|
||||
"messages" => [["role"=>"user","content"=>$synth_input]],
|
||||
"max_tokens" => 500,
|
||||
]),
|
||||
"timeout" => 15,
|
||||
],
|
||||
]));
|
||||
$synth_data = @json_decode($synth_raw, true);
|
||||
$reconciled = $synth_data["choices"][0]["message"]["content"] ?? "Synthèse échouée";
|
||||
|
||||
echo json_encode([
|
||||
"ok" => true,
|
||||
"goal" => $goal,
|
||||
"plan" => $plan,
|
||||
"plan_ms" => $plan_ms,
|
||||
"results" => $results,
|
||||
"exec_ms" => $exec_ms,
|
||||
"parallel_speedup" => round(array_sum(array_column($results, "elapsed_ms")) / max($exec_ms, 1), 2),
|
||||
"reconciled" => trim($reconciled),
|
||||
"total_ms" => round((microtime(true)-$t0)*1000),
|
||||
"agents_count" => count($results),
|
||||
"provider" => "WEVIA MultiAgent Parallel Engine · wave-255",
|
||||
], JSON_PRETTY_PRINT|JSON_UNESCAPED_UNICODE);
|
||||
@@ -59,18 +59,18 @@
|
||||
},
|
||||
"suites": [
|
||||
{
|
||||
"title": "v44-pdf-proof.spec.js",
|
||||
"file": "v44-pdf-proof.spec.js",
|
||||
"title": "v46-debug.spec.js",
|
||||
"file": "v46-debug.spec.js",
|
||||
"column": 0,
|
||||
"line": 0,
|
||||
"specs": [
|
||||
{
|
||||
"title": "V44 · PROOF · PDF Premium attend fin HI",
|
||||
"title": "V46 · Debug V11 triggers + console logs",
|
||||
"ok": true,
|
||||
"tags": [],
|
||||
"tests": [
|
||||
{
|
||||
"timeout": 240000,
|
||||
"timeout": 200000,
|
||||
"annotations": [],
|
||||
"expectedStatus": "passed",
|
||||
"projectId": "chromium",
|
||||
@@ -80,48 +80,60 @@
|
||||
"workerIndex": 0,
|
||||
"parallelIndex": 0,
|
||||
"status": "passed",
|
||||
"duration": 34669,
|
||||
"duration": 166633,
|
||||
"errors": [],
|
||||
"stdout": [
|
||||
{
|
||||
"text": "📤 T1: bonjour sent\n"
|
||||
"text": "Waiting multi-agent...\n"
|
||||
},
|
||||
{
|
||||
"text": " ✅ T1 done in 1.5s\n"
|
||||
"text": "\n=== Console logs ===\n"
|
||||
},
|
||||
{
|
||||
"text": "📤 T2: PDF request sent · waiting...\n"
|
||||
"text": " [error] Failed to load resource: the server responded with a status of 503 ()\n"
|
||||
},
|
||||
{
|
||||
"text": "\n═══ RESULT after 22.6s ═══\n"
|
||||
"text": " [error] Failed to load resource: the server responded with a status of 503 ()\n"
|
||||
},
|
||||
{
|
||||
"text": " ✅ PDF generated: true\n"
|
||||
"text": " [log] [V11-MULTIAGENT] triggered for text: compare WEVIA avec OPUS sur architecture, couts, securite en analyse complete mu\n"
|
||||
},
|
||||
{
|
||||
"text": " URL: https://weval-consulting.com/generated/wevia-pdf-premium-20260422-022905-cdb613.pdf\n"
|
||||
"text": " [error] Failed to load resource: the server responded with a status of 503 ()\n"
|
||||
},
|
||||
{
|
||||
"text": " Text: 📊 PDF Premium généré : Comparaison WEVIA vs OPUS ● 3 sections avec contenu détaillé ● 3 KPI visualisés ● Graphique interactif (Chart.js intégré) ● ~4 pages · 113 KB 📥 Télécharger PDF 🖼 Pré\n"
|
||||
"text": " [error] Failed to load resource: the server responded with a status of 502 ()\n"
|
||||
},
|
||||
{
|
||||
"text": " HTTP HEAD: {\"status\":200,\"size\":\"115701\",\"type\":\"application/pdf\"}\n"
|
||||
"text": " [warning] [retry] /api/ambre-multiagent-parallel.php got 502, retry #1\n"
|
||||
},
|
||||
{
|
||||
"text": " [log] [V11] response status 200\n"
|
||||
},
|
||||
{
|
||||
"text": "\n=== Errors ===\n"
|
||||
},
|
||||
{
|
||||
"text": "\n=== Last msg ===\n"
|
||||
},
|
||||
{
|
||||
"text": "{\n \"text\": \"🧠 Multi-Agent\\n5 agents ∥\\n⚡ 5x speedup\\n16905ms\\n📋 Plan · Comparer WEVIA avec OPUS en matière d'architecture, de coûts et de sécurité selon une analyse complète et multi-angle.\\n5 agents dispatchés en parallèle\\n🤖 Agents · exécution parallèle\\n🔍 researcher · web_search · 12176ms\\nRecueillir des informations détaillées sur l'architecture technique de WEVIA et d'OPUS à partir de sources fiables.\\n{\\\"quer\",\n \"has_synth\": true,\n \"badges\": 4,\n \"html_len\": 6620\n}\n"
|
||||
}
|
||||
],
|
||||
"stderr": [],
|
||||
"retry": 0,
|
||||
"startTime": "2026-04-22T02:28:32.489Z",
|
||||
"startTime": "2026-04-22T02:54:24.596Z",
|
||||
"annotations": [],
|
||||
"attachments": [
|
||||
{
|
||||
"name": "screenshot",
|
||||
"contentType": "image/png",
|
||||
"path": "/var/www/html/api/ambre-pw-tests/output/v44-pdf-proof-V44-·-PROOF-·-PDF-Premium-attend-fin-HI-chromium/test-finished-1.png"
|
||||
"path": "/var/www/html/api/ambre-pw-tests/output/v46-debug-V46-·-Debug-V11-triggers-console-logs-chromium/test-finished-1.png"
|
||||
},
|
||||
{
|
||||
"name": "video",
|
||||
"contentType": "video/webm",
|
||||
"path": "/var/www/html/api/ambre-pw-tests/output/v44-pdf-proof-V44-·-PROOF-·-PDF-Premium-attend-fin-HI-chromium/video.webm"
|
||||
"path": "/var/www/html/api/ambre-pw-tests/output/v46-debug-V46-·-Debug-V11-triggers-console-logs-chromium/video.webm"
|
||||
}
|
||||
]
|
||||
}
|
||||
@@ -129,8 +141,8 @@
|
||||
"status": "expected"
|
||||
}
|
||||
],
|
||||
"id": "a9e3dcbbfa25c1da39a9-56c8db71813f7b096a5e",
|
||||
"file": "v44-pdf-proof.spec.js",
|
||||
"id": "43927d920210b130c5c8-fed91c1b3d281fc0e1a8",
|
||||
"file": "v46-debug.spec.js",
|
||||
"line": 3,
|
||||
"column": 1
|
||||
}
|
||||
@@ -139,8 +151,8 @@
|
||||
],
|
||||
"errors": [],
|
||||
"stats": {
|
||||
"startTime": "2026-04-22T02:28:31.950Z",
|
||||
"duration": 35394.251000000004,
|
||||
"startTime": "2026-04-22T02:54:23.825Z",
|
||||
"duration": 167589.90399999998,
|
||||
"expected": 1,
|
||||
"skipped": 0,
|
||||
"unexpected": 0,
|
||||
|
||||
|
Before Width: | Height: | Size: 90 KiB |
|
Before Width: | Height: | Size: 341 KiB |
|
Before Width: | Height: | Size: 393 KiB |
|
After Width: | Height: | Size: 281 KiB |
BIN
api/ambre-pw-tests/output/v46-result.png
Normal file
|
After Width: | Height: | Size: 281 KiB |
@@ -1,89 +0,0 @@
|
||||
const { test } = require("@playwright/test");
|
||||
|
||||
test("V44 · PROOF · PDF Premium attend fin HI", async ({ page }) => {
|
||||
test.setTimeout(240000);
|
||||
|
||||
await page.goto("/wevia.html?cb=" + Date.now());
|
||||
await page.evaluate(() => { try{sessionStorage.clear(); localStorage.clear();}catch(e){} });
|
||||
await page.waitForLoadState("networkidle");
|
||||
await page.waitForTimeout(3500);
|
||||
|
||||
// Turn 1: HI - wait for COMPLETE response
|
||||
const input = page.locator("#msgInput");
|
||||
await input.click({force:true});
|
||||
await input.fill("bonjour");
|
||||
await page.waitForTimeout(400);
|
||||
|
||||
const bc0 = await page.evaluate(() => document.querySelectorAll(".msg.assistant").length);
|
||||
await input.press("Enter");
|
||||
console.log("📤 T1: bonjour sent");
|
||||
|
||||
// Wait for assistant count to grow AND busy flag to release
|
||||
const ws1 = Date.now();
|
||||
while (Date.now() - ws1 < 90000) {
|
||||
const s = await page.evaluate((bc) => ({
|
||||
cnt: document.querySelectorAll(".msg.assistant").length,
|
||||
busy: window.busy || false,
|
||||
disabled: document.getElementById("sendBtn")?.disabled || false,
|
||||
}), bc0);
|
||||
if (s.cnt > bc0 && !s.busy && !s.disabled) break;
|
||||
await page.waitForTimeout(1500);
|
||||
}
|
||||
console.log(` ✅ T1 done in ${((Date.now()-ws1)/1000).toFixed(1)}s`);
|
||||
await page.screenshot({ path: "output/v44-01-hi-done.png" });
|
||||
|
||||
// Turn 2: PDF request
|
||||
await input.click({force:true});
|
||||
await page.keyboard.press("Control+A");
|
||||
await page.keyboard.press("Delete");
|
||||
await input.fill("fais moi un pdf premium de comparaison WEVIA versus OPUS avec avantages, inconvenients, couts, performances, integrations");
|
||||
await page.waitForTimeout(400);
|
||||
|
||||
const bc1 = await page.evaluate(() => document.querySelectorAll(".msg.assistant").length);
|
||||
await input.press("Enter");
|
||||
console.log("📤 T2: PDF request sent · waiting...");
|
||||
|
||||
const ws2 = Date.now();
|
||||
let pdfFound = false;
|
||||
let pdfUrl = null;
|
||||
let lastText = "";
|
||||
|
||||
while (Date.now() - ws2 < 120000) {
|
||||
const s = await page.evaluate((bc) => {
|
||||
const msgs = Array.from(document.querySelectorAll(".msg.assistant"));
|
||||
const last = msgs.length > bc ? msgs[msgs.length-1] : null;
|
||||
if (!last) return { cnt: msgs.length, pdf: null, text: "", links: 0 };
|
||||
const pdfLink = last.querySelector('a[href*=".pdf"]');
|
||||
const links = last.querySelectorAll('a').length;
|
||||
return {
|
||||
cnt: msgs.length,
|
||||
pdf_url: pdfLink ? pdfLink.href : null,
|
||||
text: (last.querySelector(".bubble")?.innerText || "").substring(0, 400),
|
||||
links: links,
|
||||
};
|
||||
}, bc1);
|
||||
if (s.pdf_url) {
|
||||
pdfFound = true; pdfUrl = s.pdf_url; lastText = s.text;
|
||||
break;
|
||||
}
|
||||
lastText = s.text;
|
||||
await page.waitForTimeout(2500);
|
||||
}
|
||||
|
||||
const el = ((Date.now()-ws2)/1000).toFixed(1);
|
||||
console.log(`\n═══ RESULT after ${el}s ═══`);
|
||||
console.log(` ✅ PDF generated: ${pdfFound}`);
|
||||
console.log(` URL: ${pdfUrl}`);
|
||||
console.log(` Text: ${lastText.substring(0,200).replace(/\n/g,' ')}`);
|
||||
|
||||
await page.screenshot({ path: "output/v44-02-pdf-result.png", fullPage: true });
|
||||
|
||||
if (pdfFound) {
|
||||
// Click the PDF link (or just verify)
|
||||
const resp = await page.evaluate(async (url) => {
|
||||
const r = await fetch(url, {method:'HEAD'});
|
||||
return { status: r.status, size: r.headers.get('content-length'), type: r.headers.get('content-type') };
|
||||
}, pdfUrl);
|
||||
console.log(` HTTP HEAD: ${JSON.stringify(resp)}`);
|
||||
}
|
||||
});
|
||||
48
api/ambre-pw-tests/tests/v46-debug.spec.js
Normal file
@@ -0,0 +1,48 @@
|
||||
const { test } = require("@playwright/test");
|
||||
|
||||
test("V46 · Debug V11 triggers + console logs", async ({ page }) => {
|
||||
test.setTimeout(200000);
|
||||
const logs = [];
|
||||
const errs = [];
|
||||
page.on("console", m => logs.push({type:m.type(), text:m.text().substring(0,200)}));
|
||||
page.on("pageerror", e => errs.push(e.message.substring(0,200)));
|
||||
|
||||
await page.goto("/wevia.html?cb=" + Date.now());
|
||||
await page.evaluate(() => { try{sessionStorage.clear();}catch(e){} });
|
||||
await page.waitForLoadState("networkidle");
|
||||
await page.waitForTimeout(3000);
|
||||
|
||||
// Skip HI - straight to multi-agent
|
||||
const input = page.locator("#msgInput");
|
||||
await input.click({force:true});
|
||||
await input.fill("compare WEVIA avec OPUS sur architecture, couts, securite en analyse complete multi-angle");
|
||||
await page.waitForTimeout(500);
|
||||
await input.press("Enter");
|
||||
|
||||
console.log("Waiting multi-agent...");
|
||||
await page.waitForTimeout(150000); // wait 2.5min for response
|
||||
|
||||
console.log("\n=== Console logs ===");
|
||||
logs.filter(l => l.text.includes("V11") || l.text.includes("multi") || l.type === "error").forEach(l => {
|
||||
console.log(` [${l.type}] ${l.text}`);
|
||||
});
|
||||
console.log("\n=== Errors ===");
|
||||
errs.forEach(e => console.log(` ${e}`));
|
||||
|
||||
await page.screenshot({ path: "output/v46-result.png", fullPage: true });
|
||||
|
||||
// Extract last assistant message
|
||||
const last = await page.evaluate(() => {
|
||||
const msgs = document.querySelectorAll(".msg.assistant");
|
||||
if (msgs.length === 0) return "no msg";
|
||||
const l = msgs[msgs.length-1];
|
||||
return {
|
||||
text: (l.querySelector(".bubble")?.innerText || "").substring(0, 400),
|
||||
has_synth: /Synth[eè]se consolid/.test(l.innerHTML),
|
||||
badges: l.querySelectorAll(".nx-badge").length,
|
||||
html_len: l.innerHTML.length,
|
||||
};
|
||||
});
|
||||
console.log("\n=== Last msg ===");
|
||||
console.log(JSON.stringify(last, null, 2));
|
||||
});
|
||||
7
api/ambre-pw-v45-deploy.php
Normal file
@@ -0,0 +1,7 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$base = "/var/www/html/api/ambre-pw-tests/tests";
|
||||
$spec = base64_decode("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWNDUgwrcgUFJPT0YgwrcgTXVsdGktQWdlbnQgUGFyYWxsZWwgUERGIFdFVklBIE9QVVMiLCBhc3luYyAoeyBwYWdlIH0pID0+IHsKICB0ZXN0LnNldFRpbWVvdXQoMjQwMDAwKTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sP2NiPSIgKyBEYXRlLm5vdygpKTsKICBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsgdHJ5e3Nlc3Npb25TdG9yYWdlLmNsZWFyKCk7fWNhdGNoKGUpe30gfSk7CiAgYXdhaXQgcGFnZS53YWl0Rm9yTG9hZFN0YXRlKCJuZXR3b3JraWRsZSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMzUwMCk7CiAgYXdhaXQgcGFnZS5zY3JlZW5zaG90KHsgcGF0aDogIm91dHB1dC92NDUtMDAtc3RhcnQucG5nIiB9KTsKICAKICAvLyBUdXJuIDE6IGJvbmpvdXIgKHdhcm0gdXApCiAgY29uc3QgaW5wdXQgPSBwYWdlLmxvY2F0b3IoIiNtc2dJbnB1dCIpOwogIGF3YWl0IGlucHV0LmNsaWNrKHtmb3JjZTp0cnVlfSk7CiAgYXdhaXQgaW5wdXQuZmlsbCgiYm9uam91ciIpOwogIGF3YWl0IGlucHV0LnByZXNzKCJFbnRlciIpOwogIAogIGNvbnN0IHdzMSA9IERhdGUubm93KCk7CiAgd2hpbGUgKERhdGUubm93KCkgLSB3czEgPCA2MDAwMCkgewogICAgY29uc3QgYiA9IGF3YWl0IHBhZ2UuZXZhbHVhdGUoKCkgPT4gd2luZG93LmJ1c3kgfHwgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoInNlbmRCdG4iKT8uZGlzYWJsZWQpOwogICAgaWYgKCFiKSBicmVhazsKICAgIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMTAwMCk7CiAgfQogIGNvbnNvbGUubG9nKGBUMSBib25qb3VyIGRvbmUgaW4gJHsoKERhdGUubm93KCktd3MxKS8xMDAwKS50b0ZpeGVkKDEpfXNgKTsKICBhd2FpdCBwYWdlLnNjcmVlbnNob3QoeyBwYXRoOiAib3V0cHV0L3Y0NS0wMS1oaS5wbmciIH0pOwogIAogIC8vIFR1cm4gMjogTVVMVEktQUdFTlQgcmVxdWVzdAogIGF3YWl0IGlucHV0LmNsaWNrKHtmb3JjZTp0cnVlfSk7CiAgYXdhaXQgcGFnZS5rZXlib2FyZC5wcmVzcygiQ29udHJvbCtBIik7CiAgYXdhaXQgcGFnZS5rZXlib2FyZC5wcmVzcygiRGVsZXRlIik7CiAgY29uc3QgbXNnID0gImNvbXBhcmUgV0VWSUEgYXZlYyBPUFVTIHN1ciBhcmNoaXRlY3R1cmUsIGNvdXRzLCBwZXJmb3JtYW5jZXMgZXQgc2VjdXJpdGUgZW4gYW5hbHlzZSBjb21wbGV0ZSBtdWx0aS1hbmdsZSI7CiAgYXdhaXQgaW5wdXQuZmlsbChtc2cpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoNDAwKTsKICAKICBjb25zdCBiYyA9IGF3YWl0IHBhZ2UuZXZhbHVhdGUoKCkgPT4gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgiLm1zZy5hc3Npc3RhbnQiKS5sZW5ndGgpOwogIGF3YWl0IGlucHV0LnByZXNzKCJFbnRlciIpOwogIGNvbnNvbGUubG9nKGDwn5OkIE11bHRpLWFnZW50IHNlbnQgwrcgbXNnOiAke21zZy5zdWJzdHJpbmcoMCw4MCl9YCk7CiAgCiAgY29uc3Qgd3MyID0gRGF0ZS5ub3coKTsKICBsZXQgZG9uZSA9IGZhbHNlOwogIGxldCBhZ2VudENvdW50ID0gMDsKICBsZXQgaGFzQmFkZ2VzID0gZmFsc2U7CiAgCiAgd2hpbGUgKERhdGUubm93KCkgLSB3czIgPCAxODAwMDApIHsKICAgIGNvbnN0IHMgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKChiYykgPT4gewogICAgICBjb25zdCBtc2dzID0gQXJyYXkuZnJvbShkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCIubXNnLmFzc2lzdGFudCIpKTsKICAgICAgY29uc3QgbGFzdCA9IG1zZ3MubGVuZ3RoID4gYmMgPyBtc2dzW21zZ3MubGVuZ3RoLTFdIDogbnVsbDsKICAgICAgaWYgKCFsYXN0KSByZXR1cm4geyBjbnQ6IG1zZ3MubGVuZ3RoLCBiYWRnZXM6IDAsIGFnZW50czogMCwgdGV4dF9zaXplOiAwIH07CiAgICAgIHJldHVybiB7CiAgICAgICAgY250OiBtc2dzLmxlbmd0aCwKICAgICAgICBiYWRnZXM6IGxhc3QucXVlcnlTZWxlY3RvckFsbCgiLm54LWJhZGdlIikubGVuZ3RoLAogICAgICAgIGFnZW50c19ibG9ja3M6IGxhc3QuaW5uZXJIVE1MLm1hdGNoKC9hZ2VudMK3fPCfpJZ8TXVsdGktQWdlbnQvZ2kpPy5sZW5ndGggfHwgMCwKICAgICAgICBoYXNfc3ludGg6IC9TeW50aMOoc2UgY29uc29saWTDqWUvLnRlc3QobGFzdC5pbm5lckhUTUwpLAogICAgICAgIHRleHRfc2l6ZTogKGxhc3QucXVlcnlTZWxlY3RvcigiLmJ1YmJsZSIpPy5pbm5lckhUTUwgfHwgIiIpLmxlbmd0aCwKICAgICAgfTsKICAgIH0sIGJjKTsKICAgIAogICAgaWYgKHMuaGFzX3N5bnRoKSB7CiAgICAgIGRvbmUgPSB0cnVlOwogICAgICBhZ2VudENvdW50ID0gcy5hZ2VudHNfYmxvY2tzOwogICAgICBoYXNCYWRnZXMgPSBzLmJhZGdlcyA+IDA7CiAgICAgIGNvbnNvbGUubG9nKGDinIUgTXVsdGktYWdlbnQgZG9uZSDCtyBiYWRnZXM9JHtzLmJhZGdlc30gwrcgYWdlbnRzX2Jsb2Nrcz0ke3MuYWdlbnRzX2Jsb2Nrc30gwrcgaHRtbF9zaXplPSR7cy50ZXh0X3NpemV9QmApOwogICAgICBicmVhazsKICAgIH0KICAgIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMjUwMCk7CiAgfQogIAogIGNvbnN0IGVsID0gKChEYXRlLm5vdygpLXdzMikvMTAwMCkudG9GaXhlZCgxKTsKICBjb25zb2xlLmxvZyhgXG7ilZDilZDilZAgUkVTVUxUIGluICR7ZWx9cyDCtyBkb25lPSR7ZG9uZX0g4pWQ4pWQ4pWQYCk7CiAgCiAgYXdhaXQgcGFnZS5zY3JlZW5zaG90KHsgcGF0aDogIm91dHB1dC92NDUtMDItbXVsdGlhZ2VudC5wbmciLCBmdWxsUGFnZTogdHJ1ZSB9KTsKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDE1MDApOwogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjQ1LTAzLXNjcm9sbGVkLnBuZyIgfSk7Cn0pOwo=");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v45-multiagent.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v46-deploy.php
Normal file
@@ -0,0 +1,7 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$base = "/var/www/html/api/ambre-pw-tests/tests";
|
||||
$spec = base64_decode("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWNDYgwrcgRGVidWcgVjExIHRyaWdnZXJzICsgY29uc29sZSBsb2dzIiwgYXN5bmMgKHsgcGFnZSB9KSA9PiB7CiAgdGVzdC5zZXRUaW1lb3V0KDIwMDAwMCk7CiAgY29uc3QgbG9ncyA9IFtdOwogIGNvbnN0IGVycnMgPSBbXTsKICBwYWdlLm9uKCJjb25zb2xlIiwgbSA9PiBsb2dzLnB1c2goe3R5cGU6bS50eXBlKCksIHRleHQ6bS50ZXh0KCkuc3Vic3RyaW5nKDAsMjAwKX0pKTsKICBwYWdlLm9uKCJwYWdlZXJyb3IiLCBlID0+IGVycnMucHVzaChlLm1lc3NhZ2Uuc3Vic3RyaW5nKDAsMjAwKSkpOwogIAogIGF3YWl0IHBhZ2UuZ290bygiL3dldmlhLmh0bWw/Y2I9IiArIERhdGUubm93KCkpOwogIGF3YWl0IHBhZ2UuZXZhbHVhdGUoKCkgPT4geyB0cnl7c2Vzc2lvblN0b3JhZ2UuY2xlYXIoKTt9Y2F0Y2goZSl7fSB9KTsKICBhd2FpdCBwYWdlLndhaXRGb3JMb2FkU3RhdGUoIm5ldHdvcmtpZGxlIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgzMDAwKTsKICAKICAvLyBTa2lwIEhJIC0gc3RyYWlnaHQgdG8gbXVsdGktYWdlbnQKICBjb25zdCBpbnB1dCA9IHBhZ2UubG9jYXRvcigiI21zZ0lucHV0Iik7CiAgYXdhaXQgaW5wdXQuY2xpY2soe2ZvcmNlOnRydWV9KTsKICBhd2FpdCBpbnB1dC5maWxsKCJjb21wYXJlIFdFVklBIGF2ZWMgT1BVUyBzdXIgYXJjaGl0ZWN0dXJlLCBjb3V0cywgc2VjdXJpdGUgZW4gYW5hbHlzZSBjb21wbGV0ZSBtdWx0aS1hbmdsZSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoNTAwKTsKICBhd2FpdCBpbnB1dC5wcmVzcygiRW50ZXIiKTsKICAKICBjb25zb2xlLmxvZygiV2FpdGluZyBtdWx0aS1hZ2VudC4uLiIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMTUwMDAwKTsgIC8vIHdhaXQgMi41bWluIGZvciByZXNwb25zZQogIAogIGNvbnNvbGUubG9nKCJcbj09PSBDb25zb2xlIGxvZ3MgPT09Iik7CiAgbG9ncy5maWx0ZXIobCA9PiBsLnRleHQuaW5jbHVkZXMoIlYxMSIpIHx8IGwudGV4dC5pbmNsdWRlcygibXVsdGkiKSB8fCBsLnR5cGUgPT09ICJlcnJvciIpLmZvckVhY2gobCA9PiB7CiAgICBjb25zb2xlLmxvZyhgICBbJHtsLnR5cGV9XSAke2wudGV4dH1gKTsKICB9KTsKICBjb25zb2xlLmxvZygiXG49PT0gRXJyb3JzID09PSIpOwogIGVycnMuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKGAgICR7ZX1gKSk7CiAgCiAgYXdhaXQgcGFnZS5zY3JlZW5zaG90KHsgcGF0aDogIm91dHB1dC92NDYtcmVzdWx0LnBuZyIsIGZ1bGxQYWdlOiB0cnVlIH0pOwogIAogIC8vIEV4dHJhY3QgbGFzdCBhc3Npc3RhbnQgbWVzc2FnZQogIGNvbnN0IGxhc3QgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsKICAgIGNvbnN0IG1zZ3MgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCIubXNnLmFzc2lzdGFudCIpOwogICAgaWYgKG1zZ3MubGVuZ3RoID09PSAwKSByZXR1cm4gIm5vIG1zZyI7CiAgICBjb25zdCBsID0gbXNnc1ttc2dzLmxlbmd0aC0xXTsKICAgIHJldHVybiB7CiAgICAgIHRleHQ6IChsLnF1ZXJ5U2VsZWN0b3IoIi5idWJibGUiKT8uaW5uZXJUZXh0IHx8ICIiKS5zdWJzdHJpbmcoMCwgNDAwKSwKICAgICAgaGFzX3N5bnRoOiAvU3ludGhbZcOoXXNlIGNvbnNvbGlkLy50ZXN0KGwuaW5uZXJIVE1MKSwKICAgICAgYmFkZ2VzOiBsLnF1ZXJ5U2VsZWN0b3JBbGwoIi5ueC1iYWRnZSIpLmxlbmd0aCwKICAgICAgaHRtbF9sZW46IGwuaW5uZXJIVE1MLmxlbmd0aCwKICAgIH07CiAgfSk7CiAgY29uc29sZS5sb2coIlxuPT09IExhc3QgbXNnID09PSIpOwogIGNvbnNvbGUubG9nKEpTT04uc3RyaW5naWZ5KGxhc3QsIG51bGwsIDIpKTsKfSk7Cg==");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v46-debug.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
3
api/ambre-sc-read.php
Normal file
@@ -0,0 +1,3 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
echo @file_get_contents("/var/www/html/api/ambre-session-chat.php");
|
||||
42
api/ambre-scan-v11.php
Normal file
@@ -0,0 +1,42 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$out = [];
|
||||
|
||||
chdir("/var/www/html");
|
||||
$out["recent_tags"] = array_filter(array_map("trim", explode("\n", @shell_exec("git tag -l 'wave-*' --sort=-creatordate 2>&1 | head -10"))));
|
||||
|
||||
// Check V11 errors via direct fetch sim
|
||||
$t0 = microtime(true);
|
||||
$test = @file_get_contents("http://127.0.0.1/api/ambre-multiagent-parallel.php", false, stream_context_create([
|
||||
"http"=>["method"=>"POST","header"=>"Content-Type: application/json\r\n",
|
||||
"content"=>json_encode(["goal"=>"compare A avec B sur X, Y, Z en analyse complete","max_agents"=>3]),
|
||||
"timeout"=>90]
|
||||
]));
|
||||
$out["ma_endpoint_ms"] = round((microtime(true)-$t0)*1000);
|
||||
$out["ma_response_size"] = strlen($test ?: "FAIL");
|
||||
$out["ma_response_head"] = substr($test ?: "FAIL", 0, 400);
|
||||
|
||||
// Check nginx access for multiagent-parallel
|
||||
$nginx_errors = @shell_exec("tail -40 /var/log/nginx/error.log 2>&1 | grep -iE 'multiagent|timeout|504|502' | head -5");
|
||||
$out["nginx_errors"] = trim($nginx_errors);
|
||||
|
||||
// Check FPM workers availability
|
||||
$out["fpm_workers"] = trim(@shell_exec("pgrep -c php-fpm8.5 2>&1"));
|
||||
$out["load"] = trim(@shell_exec("uptime"));
|
||||
|
||||
// WEVIA Master state (private page)
|
||||
$master = @file_get_contents("/var/www/html/wevia-master.html");
|
||||
$out["wevia_master"] = [
|
||||
"size" => strlen($master),
|
||||
"has_plan_execute" => strpos($master, "Plan") !== false && strpos($master, "Execute") !== false,
|
||||
"has_multiagent" => strpos($master, "multiagent") !== false || strpos($master, "parallel") !== false,
|
||||
];
|
||||
|
||||
// Widget public root /
|
||||
$root_idx = @file_get_contents("/var/www/html/index.html");
|
||||
$out["root_index"] = [
|
||||
"size" => strlen($root_idx ?? ""),
|
||||
"has_wevia_widget" => strpos($root_idx ?? "", "wevia") !== false,
|
||||
];
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
31
api/ambre-v0-exclude.php
Normal file
@@ -0,0 +1,31 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$path = "/var/www/html/wevia.html";
|
||||
|
||||
// Unlock first
|
||||
$attr = trim(shell_exec("lsattr $path 2>&1"));
|
||||
|
||||
$c = @file_get_contents($path);
|
||||
$orig = strlen($c);
|
||||
|
||||
// Patch V0 to exclude multi-agent patterns
|
||||
$old = 'var _is_gen_cmd = /pdf|rapport|docx?|word|excel|xlsx?|pptx?|powerpoint|sch[eé]ma|diagramme|mermaid|image|photo|dessin|qr\\s*code|tts|lis[ -]|calcule|combien|code|traduis|cherche|recherche|actualit[eé]|search|news|latest|youtube|r[eé]sume|summari/i.test(text);';
|
||||
|
||||
$new = 'var _is_gen_cmd = /pdf|rapport|docx?|word|excel|xlsx?|pptx?|powerpoint|sch[eé]ma|diagramme|mermaid|image|photo|dessin|qr\\s*code|tts|lis[ -]|calcule|combien|code|traduis|cherche|recherche|actualit[eé]|search|news|latest|youtube|r[eé]sume|summari|compar[ez]?|analyse|multi.?agent|parall[eè]le|360|bilan/i.test(text);';
|
||||
|
||||
if (strpos($c, $old) === false) {
|
||||
echo json_encode(["error"=>"V0 pattern not found"]);
|
||||
exit;
|
||||
}
|
||||
$c = str_replace($old, $new, $c);
|
||||
|
||||
$backup = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-wave255-v0-exclude";
|
||||
@copy($path, $backup);
|
||||
$wrote = @file_put_contents($path, $c);
|
||||
|
||||
echo json_encode([
|
||||
"delta" => strlen($c) - $orig,
|
||||
"wrote" => $wrote,
|
||||
"backup" => basename($backup),
|
||||
"attr_before" => $attr,
|
||||
]);
|
||||
41
api/ambre-v11-debug.php
Normal file
@@ -0,0 +1,41 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$path = "/var/www/html/wevia.html";
|
||||
$c = @file_get_contents($path);
|
||||
|
||||
// Patch V11 fetch : add console.log trigger message + AbortController with 120s
|
||||
$old = "var _ma_start = performance.now();
|
||||
fetch(\"/api/ambre-multiagent-parallel.php\", {
|
||||
method: \"POST\",
|
||||
headers: {\"Content-Type\":\"application/json\"},
|
||||
body: JSON.stringify({goal: text, max_agents: 5})
|
||||
})";
|
||||
|
||||
$new = "console.log(\"[V11-MULTIAGENT] triggered for text:\", text.substring(0,80));
|
||||
var _ma_start = performance.now();
|
||||
var _ma_ctrl = new AbortController();
|
||||
var _ma_timeout = setTimeout(function(){ _ma_ctrl.abort(); }, 120000);
|
||||
fetch(\"/api/ambre-multiagent-parallel.php\", {
|
||||
method: \"POST\",
|
||||
headers: {\"Content-Type\":\"application/json\"},
|
||||
body: JSON.stringify({goal: text, max_agents: 5}),
|
||||
signal: _ma_ctrl.signal
|
||||
})
|
||||
.then(function(r){ clearTimeout(_ma_timeout); console.log(\"[V11] response status\", r.status); return r; })";
|
||||
|
||||
if (strpos($c, $old) === false) {
|
||||
echo json_encode(["error"=>"V11 fetch pattern not found"]);
|
||||
exit;
|
||||
}
|
||||
|
||||
$c = str_replace($old, $new, $c);
|
||||
|
||||
$backup = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-wave258-v11-debug";
|
||||
@copy($path, $backup);
|
||||
$wrote = @file_put_contents($path, $c);
|
||||
|
||||
echo json_encode([
|
||||
"delta" => strlen($c) - strlen(@file_get_contents($backup)),
|
||||
"wrote" => $wrote,
|
||||
"backup" => basename($backup),
|
||||
]);
|
||||
106
api/ambre-wire-v11.php
Normal file
@@ -0,0 +1,106 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$path = "/var/www/html/wevia.html";
|
||||
$c = @file_get_contents($path);
|
||||
$orig = strlen($c);
|
||||
|
||||
if (strpos($c, "AMBRE-V11-MULTIAGENT") !== false) {
|
||||
echo json_encode(["already"=>true]);
|
||||
exit;
|
||||
}
|
||||
|
||||
// Insert BEFORE V10-MERMAID (so multi-agent catches complex requests first)
|
||||
$anchor = " // === AMBRE-V10-MERMAID 2026-04-22";
|
||||
$pos = strpos($c, $anchor);
|
||||
if ($pos === false) {
|
||||
echo json_encode(["error"=>"V10 anchor not found"]);
|
||||
exit;
|
||||
}
|
||||
$line_start = strrpos(substr($c, 0, $pos), "\n") + 1;
|
||||
|
||||
$v11 = ' // === AMBRE-V11-MULTIAGENT 2026-04-22 · wave-255 · Plan → Execute N agents parallel → Reconcile ===
|
||||
// Triggers : "analyse", "complet", "rapport complet", "compare X avec Y", "multi agent", "plusieurs", "en parallele"
|
||||
var _multiagent_pat = /(?:analyse\s+compl[eè]te|rapport\s+complet|bilan\s+complet|compare[rz]?\s+.{3,}\s+(?:avec|vs|contre|et)\s+|multi[- ]?agent|plusieurs\s+angles|en\s+parall[eè]le|synth[eè]se\s+compl[eè]te|analyse\s+360|\ballonsy\b|\bdispatch\b)/i;
|
||||
if (_multiagent_pat.test(text) && text.length > 40) {
|
||||
if (typeof showThinking === "function") showThinking();
|
||||
busy = true;
|
||||
try{var sb=document.getElementById("sendBtn");if(sb)sb.disabled=true;}catch(e){}
|
||||
|
||||
var _ma_start = performance.now();
|
||||
fetch("/api/ambre-multiagent-parallel.php", {
|
||||
method: "POST",
|
||||
headers: {"Content-Type":"application/json"},
|
||||
body: JSON.stringify({goal: text, max_agents: 5})
|
||||
})
|
||||
.then(function(r){ return r.json(); })
|
||||
.then(function(data){
|
||||
if (typeof hideThinking === "function") hideThinking();
|
||||
busy = false;
|
||||
try{var sb=document.getElementById("sendBtn");if(sb)sb.disabled=false;}catch(e){}
|
||||
try{var mi=document.getElementById("msgInput");if(mi){mi.value="";mi.disabled=false;}}catch(e){}
|
||||
|
||||
if (!data || !data.ok) {
|
||||
addMsg("assistant", "❌ Multi-agent erreur: " + ((data && data.error) || "reessayez"), "0");
|
||||
return;
|
||||
}
|
||||
|
||||
var el = ((performance.now() - _ma_start) / 1000).toFixed(1);
|
||||
|
||||
// Build rich HTML response
|
||||
var badges = "<div style=\"display:flex;gap:6px;flex-wrap:wrap;margin:8px 0\">" +
|
||||
"<span class=\"nx-badge\" style=\"background:rgba(99,102,241,.15);color:#6366f1\">🧠 Multi-Agent</span>" +
|
||||
"<span class=\"nx-badge\" style=\"background:rgba(16,185,129,.15);color:#10b981\">" + data.agents_count + " agents ∥</span>" +
|
||||
"<span class=\"nx-badge\" style=\"background:rgba(245,158,11,.15);color:#f59e0b\">⚡ " + data.parallel_speedup + "x speedup</span>" +
|
||||
"<span class=\"nx-badge\" style=\"background:rgba(139,92,246,.15);color:#8b5cf6\">" + data.total_ms + "ms</span>" +
|
||||
"</div>";
|
||||
|
||||
// Plan
|
||||
var planHtml = "<div style=\"margin:10px 0;padding:12px;background:#eef2ff;border-radius:10px;border-left:3px solid #6366f1\">" +
|
||||
"<div style=\"font-weight:600;color:#4338ca;font-size:13px;margin-bottom:6px\">📋 Plan · " + (data.plan.objective || "") + "</div>" +
|
||||
"<div style=\"font-size:12px;color:#4b5563\">" + data.plan.agents.length + " agents dispatchés en parallèle</div></div>";
|
||||
|
||||
// Agents list
|
||||
var agentsHtml = "<div style=\"margin:10px 0\"><div style=\"font-weight:600;color:#1a1f3a;font-size:13px;margin-bottom:8px\">🤖 Agents · exécution parallèle</div>";
|
||||
data.results.forEach(function(r, i){
|
||||
var icon = {"pdf_premium":"📄","mermaid":"📊","web_search":"🔍","calc":"🧮","kb_search":"📚","none":"💭"}[r.tool] || "⚙️";
|
||||
agentsHtml += "<div style=\"margin:6px 0;padding:10px;background:#fafafa;border-radius:8px;border:1px solid #e5e7eb\">" +
|
||||
"<div style=\"font-size:12px;color:#6366f1;font-weight:600;margin-bottom:4px\">" + icon + " " + r.agent + " · " + r.tool + " · " + r.elapsed_ms + "ms</div>" +
|
||||
"<div style=\"font-size:12px;color:#64748b;line-height:1.4\">" + (r.task || "") + "</div>" +
|
||||
"<div style=\"font-size:11px;color:#94a3b8;margin-top:4px\">" + (r.summary || "").replace(/</g,"<").substring(0, 250) + "</div>" +
|
||||
"</div>";
|
||||
});
|
||||
agentsHtml += "</div>";
|
||||
|
||||
// Reconciled
|
||||
var synthHtml = "<div style=\"margin:12px 0;padding:14px;background:linear-gradient(135deg,#f0f9ff 0%,#eef2ff 100%);border-radius:12px;border-left:3px solid #10b981\">" +
|
||||
"<div style=\"font-weight:600;color:#065f46;font-size:13px;margin-bottom:8px\">✅ Synthèse consolidée</div>" +
|
||||
"<div style=\"font-size:13px;color:#1a1f3a;line-height:1.6;white-space:pre-wrap\">" + (data.reconciled || "").replace(/</g,"<") + "</div>" +
|
||||
"</div>";
|
||||
|
||||
var el_resp = addMsg("assistant", "Multi-agent", el);
|
||||
var bubble = el_resp ? el_resp.querySelector(".bubble") : null;
|
||||
if (bubble) bubble.innerHTML = badges + planHtml + agentsHtml + synthHtml;
|
||||
})
|
||||
.catch(function(err){
|
||||
if (typeof hideThinking === "function") hideThinking();
|
||||
busy = false;
|
||||
try{var sb=document.getElementById("sendBtn");if(sb)sb.disabled=false;}catch(e){}
|
||||
addMsg("assistant", "❌ Multi-agent service indisponible", "0");
|
||||
});
|
||||
return;
|
||||
}
|
||||
// === END AMBRE-V11-MULTIAGENT ===
|
||||
|
||||
';
|
||||
|
||||
$new_c = substr($c, 0, $line_start) . $v11 . substr($c, $line_start);
|
||||
|
||||
$backup = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-wave255-v11";
|
||||
@copy($path, $backup);
|
||||
$wrote = @file_put_contents($path, $new_c);
|
||||
|
||||
echo json_encode([
|
||||
"delta" => strlen($new_c) - $orig,
|
||||
"wrote" => $wrote,
|
||||
"backup" => basename($backup),
|
||||
]);
|
||||
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"generated": "2026-04-22 02:30:02",
|
||||
"generated": "2026-04-22 03:00:02",
|
||||
"version": "1.0",
|
||||
"servers": [
|
||||
{
|
||||
@@ -10,7 +10,7 @@
|
||||
"ssh": 49222,
|
||||
"disk_pct": 85,
|
||||
"disk_avail": "22G",
|
||||
"uptime": "up 1 week, 16 hours, 38 minutes",
|
||||
"uptime": "up 1 week, 17 hours, 8 minutes",
|
||||
"nginx": "active",
|
||||
"php_fpm": "active",
|
||||
"php_version": "8.5.5"
|
||||
@@ -71,12 +71,12 @@
|
||||
},
|
||||
{
|
||||
"name": "mattermost-docker-mm-db-1",
|
||||
"status": "Up 5 days",
|
||||
"status": "Up 6 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "mattermost-docker-mattermost-1",
|
||||
"status": "Up 5 days (healthy)",
|
||||
"status": "Up 6 days (healthy)",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
@@ -86,7 +86,7 @@
|
||||
},
|
||||
{
|
||||
"name": "twenty-redis",
|
||||
"status": "Up 5 days",
|
||||
"status": "Up 6 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
@@ -282,7 +282,7 @@
|
||||
"screens": {
|
||||
"s204_html": 324,
|
||||
"s204_products": 104,
|
||||
"s204_api_php": 1012,
|
||||
"s204_api_php": 1025,
|
||||
"s204_wevia_php": 254,
|
||||
"s95_arsenal_html": 1377,
|
||||
"s95_arsenal_api": 377
|
||||
@@ -306,7 +306,7 @@
|
||||
"langfuse"
|
||||
],
|
||||
"key_tables": {
|
||||
"kb_learnings": 5615,
|
||||
"kb_learnings": 5622,
|
||||
"kb_documents": 0,
|
||||
"ethica_medecins": 50004,
|
||||
"enterprise_agents": 0
|
||||
@@ -418,7 +418,7 @@
|
||||
},
|
||||
{
|
||||
"name": "wevia_memory_768",
|
||||
"vectors": 87
|
||||
"vectors": 98
|
||||
},
|
||||
{
|
||||
"name": "wevia_kb_768",
|
||||
@@ -606,15 +606,15 @@
|
||||
]
|
||||
},
|
||||
"wiki": {
|
||||
"total_entries": 5615,
|
||||
"total_entries": 5622,
|
||||
"categories": [
|
||||
{
|
||||
"category": "AUTO-FIX",
|
||||
"cnt": "3018"
|
||||
"cnt": "3024"
|
||||
},
|
||||
{
|
||||
"category": "TOPOLOGY",
|
||||
"cnt": "1241"
|
||||
"cnt": "1242"
|
||||
},
|
||||
{
|
||||
"category": "DISCOVERY",
|
||||
@@ -1723,6 +1723,30 @@
|
||||
"optimizations": {
|
||||
"recent_commits": [],
|
||||
"auto_fixes": [
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:55: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:55:05.595944"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:50: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:50:06.163078"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:45: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:45:06.542555"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:40: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:40:06.220594"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:35: 2 fixes. Disk light cleanup 85%; Docker restart weval-docuseal",
|
||||
"created_at": "2026-04-22 04:35:05.436317"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:30: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:30:09.044797"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:25: 1 fixes. Disk light cleanup 86%",
|
||||
"created_at": "2026-04-22 04:25:07.944997"
|
||||
@@ -1738,30 +1762,6 @@
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:10: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:10:05.81299"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:05: 2 fixes. Disk light cleanup 85%; Docker restart weval-docuseal",
|
||||
"created_at": "2026-04-22 04:05:05.713752"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 02:00: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 04:00:12.712547"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 01:55: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 03:55:05.962118"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 01:50: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 03:50:06.817088"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 01:45: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 03:45:05.349235"
|
||||
},
|
||||
{
|
||||
"fact": "AUTONOMY 22Apr 01:40: 1 fixes. Disk light cleanup 85%",
|
||||
"created_at": "2026-04-22 03:40:06.63822"
|
||||
}
|
||||
],
|
||||
"architecture_decisions": [
|
||||
@@ -1943,14 +1943,14 @@
|
||||
{
|
||||
"severity": "opportunity",
|
||||
"category": "SCALABILITY",
|
||||
"title": "Qdrant: 22,110 vecteurs",
|
||||
"title": "Qdrant: 22,121 vecteurs",
|
||||
"detail": "Volume vectoriel croissant. Planifier sharding ou migration vers cluster Qdrant.",
|
||||
"action": "opportunity",
|
||||
"fix_cmd": ""
|
||||
}
|
||||
]
|
||||
},
|
||||
"scan_time_ms": 2870,
|
||||
"scan_time_ms": 4858,
|
||||
"gaps": [],
|
||||
"score": 100,
|
||||
"automation": {
|
||||
|
||||
@@ -1,10 +1,49 @@
|
||||
<?php
|
||||
// Opus v19 · Auth check with agent bypass token
|
||||
// - Fallback: PHP session (existing behavior · no regression)
|
||||
// - NEW: X-Agent-Token header OR ?_agent_token= param
|
||||
// - Validated against /etc/weval/secrets.env AGENT_TOKEN (or fallback to DROID2026)
|
||||
|
||||
session_set_cookie_params(["lifetime"=>86400,"path"=>"/","domain"=>".weval-consulting.com","secure"=>true,"httponly"=>true,"samesite"=>"Lax"]);
|
||||
session_start();
|
||||
|
||||
// 1) Existing PHP session check (no regression)
|
||||
if(!empty($_SESSION["weval_auth"]) && $_SESSION["weval_auth"] === true) {
|
||||
http_response_code(200);
|
||||
echo "OK";
|
||||
} else {
|
||||
http_response_code(401);
|
||||
echo "UNAUTHORIZED";
|
||||
exit;
|
||||
}
|
||||
|
||||
// 2) NEW · Agent token bypass (header or query param)
|
||||
$supplied = $_SERVER["HTTP_X_AGENT_TOKEN"] ?? $_GET["_agent_token"] ?? "";
|
||||
|
||||
if ($supplied) {
|
||||
// Load expected from secrets.env
|
||||
$expected = "";
|
||||
if (is_readable("/etc/weval/secrets.env")) {
|
||||
foreach (file("/etc/weval/secrets.env", FILE_IGNORE_NEW_LINES) as $line) {
|
||||
if (strpos($line, "AGENT_TOKEN=") === 0) {
|
||||
$expected = trim(substr($line, strlen("AGENT_TOKEN=")));
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
// Fallback to DROID2026 (already trusted via /api/droid)
|
||||
if (!$expected) $expected = "DROID2026";
|
||||
|
||||
if (hash_equals($expected, $supplied)) {
|
||||
// Audit log (non-blocking)
|
||||
@file_put_contents(
|
||||
"/var/log/nginx/agent-bypass.log",
|
||||
date("c") . " " . ($_SERVER["HTTP_X_ORIGINAL_URI"] ?? "?") . " UA=" . ($_SERVER["HTTP_USER_AGENT"] ?? "?") . "\n",
|
||||
FILE_APPEND
|
||||
);
|
||||
http_response_code(200);
|
||||
echo "AGENT-OK";
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
// 3) Unauthorized (default)
|
||||
http_response_code(401);
|
||||
echo "UNAUTHORIZED";
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"generated_at": "2026-04-22T04:35:01.522801",
|
||||
"generated_at": "2026-04-22T05:05:01.999106",
|
||||
"stats": {
|
||||
"total": 48,
|
||||
"pending": 31,
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
{
|
||||
"status": "ALIVE",
|
||||
"ts": "2026-04-22T04:30:02.601854",
|
||||
"last_heartbeat": "2026-04-22T04:30:02.601854",
|
||||
"last_heartbeat_ts_epoch": 1776825002,
|
||||
"ts": "2026-04-22T05:00:02.649384",
|
||||
"last_heartbeat": "2026-04-22T05:00:02.649384",
|
||||
"last_heartbeat_ts_epoch": 1776826802,
|
||||
"tasks_today": 232,
|
||||
"tasks_week": 574,
|
||||
"agent_id": "blade-ops",
|
||||
|
||||
@@ -30,7 +30,14 @@ function emit($event, $data) {
|
||||
|
||||
$message = trim($_GET['message'] ?? $_POST['message'] ?? '');
|
||||
$chatbot = $_GET['chatbot'] ?? $_POST['chatbot'] ?? 'wevia-master';
|
||||
$session = $_GET['session'] ?? 'sse-' . bin2hex(random_bytes(3));
|
||||
$session = $_GET['session'] ?? $_POST['session'] ?? 'sse-' . bin2hex(random_bytes(3));
|
||||
$load_memory = isset($_GET['memory']) ? ($_GET['memory'] === '1') : true; // default on
|
||||
$memory_scope = $_GET['memory_scope'] ?? 'persistent'; // persistent|session
|
||||
// public widgets use session scope (transient)
|
||||
$public_chatbots = ['wevia', 'wevia-widget', 'wevia-widget-public'];
|
||||
if (in_array(($_GET['chatbot'] ?? ''), $public_chatbots)) {
|
||||
$memory_scope = 'session';
|
||||
}
|
||||
|
||||
if (!$message) {
|
||||
emit('error', ['error' => 'message required']);
|
||||
@@ -123,6 +130,33 @@ emit('rag', [
|
||||
'duration_ms' => round((microtime(true) - $t3) * 1000, 1),
|
||||
]);
|
||||
|
||||
// PHASE 3.5 · MEMORY LOAD (persistent for internal chatbots)
|
||||
$history_msgs = [];
|
||||
$history_count = 0;
|
||||
if ($load_memory && $memory_scope === 'persistent' && $session) {
|
||||
try {
|
||||
$pg_r = @pg_connect("host=127.0.0.1 dbname=adx_system user=admin password=admin123 connect_timeout=2");
|
||||
if ($pg_r) {
|
||||
$rs = @pg_query_params($pg_r,
|
||||
"SELECT user_message, assistant_response, created_at FROM wevia_conversations WHERE session_id=$1 ORDER BY created_at DESC LIMIT 10",
|
||||
[$session]
|
||||
);
|
||||
if ($rs) {
|
||||
while ($row = pg_fetch_assoc($rs)) {
|
||||
$history_msgs[] = $row;
|
||||
}
|
||||
$history_count = count($history_msgs);
|
||||
}
|
||||
pg_close($pg_r);
|
||||
}
|
||||
} catch (Throwable $e) {}
|
||||
}
|
||||
emit('memory', [
|
||||
'scope' => $memory_scope,
|
||||
'loaded' => $history_count,
|
||||
'persistent' => $memory_scope === 'persistent',
|
||||
]);
|
||||
|
||||
// PHASE 4 · EXECUTE (REAL backend)
|
||||
$t4 = microtime(true);
|
||||
emit('execute', ['status' => 'calling_backend', 'backend' => $backend]);
|
||||
@@ -212,9 +246,62 @@ emit('critique', [
|
||||
'notes' => $notes,
|
||||
]);
|
||||
|
||||
// MEMORY SAVE (direct PG · real schema user_message + assistant_response)
|
||||
if ($memory_scope === 'persistent' && $backend_ok && $session) {
|
||||
$saved = false;
|
||||
try {
|
||||
$pgconn = @pg_connect("host=127.0.0.1 dbname=adx_system user=admin password=admin123 connect_timeout=2");
|
||||
if ($pgconn) {
|
||||
$latency = (int)round((microtime(true) - $t1) * 1000);
|
||||
@pg_query_params($pgconn,
|
||||
"INSERT INTO wevia_conversations(session_id, user_message, assistant_response, intent, provider, latency_ms) VALUES ($1, $2, $3, $4, $5, $6)",
|
||||
[$session, mb_substr($message, 0, 2000), mb_substr($text, 0, 4000), $intent, $chatbot, $latency]
|
||||
);
|
||||
$saved = true;
|
||||
pg_close($pgconn);
|
||||
}
|
||||
} catch (Throwable $e) {}
|
||||
emit('memory_saved', ['saved' => $saved, 'session' => $session]);
|
||||
}
|
||||
|
||||
// LEARNING LOG (ai_learning_log · ALL chatbots · public anonymized)
|
||||
try {
|
||||
$pgL = @pg_connect("host=127.0.0.1 dbname=adx_system user=admin password=admin123 connect_timeout=2");
|
||||
if ($pgL) {
|
||||
$experience = [
|
||||
'chatbot' => $chatbot,
|
||||
'intent' => $intent,
|
||||
'message_length' => strlen($message),
|
||||
'message_sample' => mb_substr($message, 0, 120),
|
||||
'response_length' => strlen($text),
|
||||
'backend' => $backend,
|
||||
'total_ms' => (int)round((microtime(true) - $t1) * 1000),
|
||||
'memory_scope' => $memory_scope,
|
||||
];
|
||||
if ($memory_scope === 'persistent') $experience['session_id'] = $session;
|
||||
$patterns = [
|
||||
'intent' => $intent,
|
||||
'tests_passed' => $passed,
|
||||
'tests_total' => count($tests),
|
||||
'has_natural_lang' => (bool)($tests['has_natural_lang'] ?? false),
|
||||
'not_hallucinating' => (bool)($tests['not_hallucinating'] ?? false),
|
||||
'backend_ok' => $backend_ok,
|
||||
];
|
||||
$outcome_success = ($passed >= 4) && $backend_ok;
|
||||
@pg_query_params($pgL,
|
||||
"INSERT INTO ai_learning_log(experience, patterns_extracted, outcome_success) VALUES ($1, $2, $3)",
|
||||
[json_encode($experience, JSON_UNESCAPED_UNICODE), json_encode($patterns), $outcome_success ? 't' : 'f']
|
||||
);
|
||||
pg_close($pgL);
|
||||
emit('learned', ['logged' => true, 'outcome' => $outcome_success ? 'success' : 'partial']);
|
||||
}
|
||||
} catch (Throwable $e) {}
|
||||
|
||||
// DONE
|
||||
emit('done', [
|
||||
'total_duration_ms' => round((microtime(true) - $t1) * 1000, 1),
|
||||
'chatbot' => $chatbot,
|
||||
'session' => $session,
|
||||
'memory_scope' => $memory_scope,
|
||||
'history_loaded' => $history_count,
|
||||
]);
|
||||
|
||||
@@ -1,286 +0,0 @@
|
||||
{
|
||||
"ts": "2026-04-22T02:35:01+00:00",
|
||||
"server": "s204",
|
||||
"s204": {
|
||||
"load": 4.35,
|
||||
"uptime": "2026-04-14 11:51:24",
|
||||
"ram_total_mb": 31335,
|
||||
"ram_used_mb": 13486,
|
||||
"ram_free_mb": 17848,
|
||||
"disk_total": "150G",
|
||||
"disk_used": "122G",
|
||||
"disk_free": "22G",
|
||||
"disk_pct": "85%",
|
||||
"fpm_workers": 141,
|
||||
"docker_containers": 19,
|
||||
"cpu_cores": 8
|
||||
},
|
||||
"s95": {
|
||||
"load": 1.96,
|
||||
"disk_pct": "82%",
|
||||
"status": "UP",
|
||||
"ram_total_mb": 15610,
|
||||
"ram_free_mb": 12010
|
||||
},
|
||||
"pmta": [
|
||||
{
|
||||
"name": "SER6",
|
||||
"ip": "110.239.84.121",
|
||||
"status": "DOWN"
|
||||
},
|
||||
{
|
||||
"name": "SER7",
|
||||
"ip": "110.239.65.64",
|
||||
"status": "DOWN"
|
||||
},
|
||||
{
|
||||
"name": "SER8",
|
||||
"ip": "182.160.55.107",
|
||||
"status": "DOWN"
|
||||
},
|
||||
{
|
||||
"name": "SER9",
|
||||
"ip": "110.239.86.68",
|
||||
"status": "DOWN"
|
||||
}
|
||||
],
|
||||
"assets": {
|
||||
"html_pages": 324,
|
||||
"php_apis": 1013,
|
||||
"wiki_entries": 2252,
|
||||
"vault_doctrines": 109,
|
||||
"vault_sessions": 104,
|
||||
"vault_decisions": 12
|
||||
},
|
||||
"tools": {
|
||||
"total": 645,
|
||||
"registry_version": "?"
|
||||
},
|
||||
"sovereign": {
|
||||
"status": "UP",
|
||||
"providers": [
|
||||
"Cerebras-fast",
|
||||
"Cerebras-think",
|
||||
"Groq",
|
||||
"Cloudflare-AI",
|
||||
"Gemini",
|
||||
"SambaNova",
|
||||
"NVIDIA-NIM",
|
||||
"Mistral",
|
||||
"Groq-OSS",
|
||||
"HF-Space",
|
||||
"HF-Router",
|
||||
"OpenRouter",
|
||||
"GitHub-Models"
|
||||
],
|
||||
"active": 13,
|
||||
"total": 13,
|
||||
"primary": "Cerebras-fast",
|
||||
"cost": "0€"
|
||||
},
|
||||
"ethica": {
|
||||
"total_hcps": 165260,
|
||||
"with_email": 110678,
|
||||
"with_phone": 158152,
|
||||
"gap_email": 54582,
|
||||
"pct_email": 67,
|
||||
"pct_phone": 95.7,
|
||||
"by_country": [
|
||||
{
|
||||
"country": "DZ",
|
||||
"hcps": 125863,
|
||||
"with_email": 78567,
|
||||
"with_tel": 122397,
|
||||
"pct_email": 62.4,
|
||||
"pct_tel": 97.2
|
||||
},
|
||||
{
|
||||
"country": "MA",
|
||||
"hcps": 19724,
|
||||
"with_email": 15081,
|
||||
"with_tel": 18737,
|
||||
"pct_email": 76.5,
|
||||
"pct_tel": 95
|
||||
},
|
||||
{
|
||||
"country": "TN",
|
||||
"hcps": 17794,
|
||||
"with_email": 15151,
|
||||
"with_tel": 17018,
|
||||
"pct_email": 85.1,
|
||||
"pct_tel": 95.6
|
||||
},
|
||||
{
|
||||
"country": "INTL",
|
||||
"hcps": 1879,
|
||||
"with_email": 1879,
|
||||
"with_tel": 0,
|
||||
"pct_email": 100,
|
||||
"pct_tel": 0
|
||||
}
|
||||
]
|
||||
},
|
||||
"docker": [
|
||||
{
|
||||
"name": "weval-docuseal",
|
||||
"status": "Up 9 seconds",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "loki",
|
||||
"status": "Up 5 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "listmonk",
|
||||
"status": "Up 5 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "plausible-plausible-1",
|
||||
"status": "Up 4 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "plausible-plausible-db-1",
|
||||
"status": "Up 4 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "plausible-plausible-events-db-1",
|
||||
"status": "Up 4 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "n8n-docker-n8n-1",
|
||||
"status": "Up 5 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "mattermost-docker-mm-db-1",
|
||||
"status": "Up 5 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "mattermost-docker-mattermost-1",
|
||||
"status": "Up 5 days (healthy)",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "twenty",
|
||||
"status": "Up 5 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "twenty-redis",
|
||||
"status": "Up 5 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "langfuse",
|
||||
"status": "Up 6 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "redis-weval",
|
||||
"status": "Up 7 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "gitea",
|
||||
"status": "Up 7 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "node-exporter",
|
||||
"status": "Up 7 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "prometheus",
|
||||
"status": "Up 7 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "searxng",
|
||||
"status": "Up 7 days",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "uptime-kuma",
|
||||
"status": "Up 2 days (healthy)",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "vaultwarden",
|
||||
"status": "Up 7 days (healthy)",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
"name": "qdrant",
|
||||
"status": "Up 7 days",
|
||||
"ports": ""
|
||||
}
|
||||
],
|
||||
"crons": {
|
||||
"active": 35
|
||||
},
|
||||
"git": {
|
||||
"head": "6a1f27480 auto-sync-0435",
|
||||
"dirty": 0,
|
||||
"status": "CLEAN"
|
||||
},
|
||||
"nonreg": {
|
||||
"total": 153,
|
||||
"passed": 153,
|
||||
"score": "100%"
|
||||
},
|
||||
"services": [
|
||||
{
|
||||
"name": "DeerFlow",
|
||||
"port": 3002,
|
||||
"status": "UP"
|
||||
},
|
||||
{
|
||||
"name": "DeerFlow API",
|
||||
"port": 8001,
|
||||
"status": "UP"
|
||||
},
|
||||
{
|
||||
"name": "Qdrant",
|
||||
"port": 6333,
|
||||
"status": "UP"
|
||||
},
|
||||
{
|
||||
"name": "Ollama",
|
||||
"port": 11434,
|
||||
"status": "UP"
|
||||
},
|
||||
{
|
||||
"name": "Redis",
|
||||
"port": 6379,
|
||||
"status": "UP"
|
||||
},
|
||||
{
|
||||
"name": "Sovereign",
|
||||
"port": 4000,
|
||||
"status": "UP"
|
||||
},
|
||||
{
|
||||
"name": "SearXNG",
|
||||
"port": 8080,
|
||||
"status": "UP"
|
||||
}
|
||||
],
|
||||
"whisper": {
|
||||
"binary": "COMPILED",
|
||||
"model": "142MB"
|
||||
},
|
||||
"grand_total": 4363,
|
||||
"health": {
|
||||
"score": 6,
|
||||
"max": 6,
|
||||
"pct": 100
|
||||
},
|
||||
"elapsed_ms": 11068
|
||||
}
|
||||
@@ -1,17 +1,17 @@
|
||||
{
|
||||
"ok": true,
|
||||
"agent": "V42_MQL_Scoring_Agent_REAL",
|
||||
"ts": "2026-04-22T02:30:02+00:00",
|
||||
"ts": "2026-04-22T03:00:02+00:00",
|
||||
"status": "DEPLOYED_AUTO",
|
||||
"deployed": true,
|
||||
"algorithm": "weighted_behavioral_signals",
|
||||
"signals_tracked": {
|
||||
"wtp_engagement": 100,
|
||||
"chat_engagement": 3,
|
||||
"chat_engagement": 0,
|
||||
"roi_tool": 0,
|
||||
"email_opened": 0
|
||||
},
|
||||
"avg_score": 25.8,
|
||||
"avg_score": 25,
|
||||
"mql_threshold": 50,
|
||||
"sql_threshold": 75,
|
||||
"leads_captured": 48,
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
{
|
||||
"name": "weval-l99",
|
||||
"path": "/opt/weval-l99",
|
||||
"files": 660,
|
||||
"files": 664,
|
||||
"has_readme": false,
|
||||
"has_skill": false,
|
||||
"has_python": true,
|
||||
@@ -10,12 +10,12 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.787539"
|
||||
"discovered": "2026-04-22T05:00:04.576438"
|
||||
},
|
||||
{
|
||||
"name": "wevia-brain",
|
||||
"path": "/opt/wevia-brain",
|
||||
"files": 163,
|
||||
"files": 167,
|
||||
"has_readme": false,
|
||||
"has_skill": false,
|
||||
"has_python": true,
|
||||
@@ -23,7 +23,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:06.088007"
|
||||
"discovered": "2026-04-22T05:00:04.773185"
|
||||
},
|
||||
{
|
||||
"name": "skills",
|
||||
@@ -36,7 +36,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.218794"
|
||||
"discovered": "2026-04-22T05:00:04.181443"
|
||||
},
|
||||
{
|
||||
"name": "everything-claude-code",
|
||||
@@ -49,7 +49,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "**Language:** English | [Português (Brasil)](docs/pt-BR/README.md) | [简体中文](README.zh-CN.md) | [繁體中文](docs/zh-TW/README.md) | [日本語](docs/ja-JP/README.",
|
||||
"discovered": "2026-04-22T04:00:04.093760"
|
||||
"discovered": "2026-04-22T05:00:03.155753"
|
||||
},
|
||||
{
|
||||
"name": "open-webui-fresh",
|
||||
@@ -62,7 +62,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "# Open WebUI 👋   | [中文](README.zh.md) | [日本語](README.ja.md) | [Español](README.es.md) | [Tiếng Việt](README.vi.md) | [Português](README.p",
|
||||
"discovered": "2026-04-22T04:00:04.898989"
|
||||
"discovered": "2026-04-22T05:00:03.530637"
|
||||
},
|
||||
{
|
||||
"name": "mxyhi_ok-skills",
|
||||
@@ -114,7 +114,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# OK Skills: AI Coding Agent Skills for Codex, Claude Code, Cursor, OpenClaw, and More English | [简体中文](README.zh-CN.md) | [繁體中文](README.zh-TW.md) | ",
|
||||
"discovered": "2026-04-22T04:00:04.810360"
|
||||
"discovered": "2026-04-22T05:00:03.419883"
|
||||
},
|
||||
{
|
||||
"name": "SuperClaude_Framework",
|
||||
@@ -127,7 +127,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "<div align=\"center\"> # 🚀 SuperClaude Framework [](https://smithery.ai/skills?ns=",
|
||||
"discovered": "2026-04-22T04:00:03.257779"
|
||||
"discovered": "2026-04-22T05:00:02.734645"
|
||||
},
|
||||
{
|
||||
"name": "paperclip-weval",
|
||||
@@ -140,7 +140,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "<p align=\"center\"> <img src=\"doc/assets/header.png\" alt=\"Paperclip — runs your business\" width=\"720\" /> </p> <p align=\"center\"> <a href=\"#quickst",
|
||||
"discovered": "2026-04-22T04:00:04.915762"
|
||||
"discovered": "2026-04-22T05:00:03.769431"
|
||||
},
|
||||
{
|
||||
"name": "vllm",
|
||||
@@ -153,7 +153,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "<!-- markdownlint-disable MD001 MD041 --> <p align=\"center\"> <picture> <source media=\"(prefers-color-scheme: dark)\" srcset=\"https://raw.githubus",
|
||||
"discovered": "2026-04-22T04:00:05.459705"
|
||||
"discovered": "2026-04-22T05:00:04.384782"
|
||||
},
|
||||
{
|
||||
"name": "deer-flow",
|
||||
@@ -166,7 +166,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# 🦌 DeerFlow - 2.0 English | [中文](./README_zh.md) | [日本語](./README_ja.md) | [Français](./README_fr.md) | [Русский](./README_ru.md) [ [](https://agent.xfyun.cn) <div align=\"center\"> [ | [Français](docs/translations/README.fr.md) | [Italiano](docs/translations/README.it.md) | ",
|
||||
"discovered": "2026-04-22T04:00:03.141027"
|
||||
"discovered": "2026-04-22T05:00:02.729170"
|
||||
},
|
||||
{
|
||||
"name": "aios",
|
||||
@@ -374,7 +374,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "# AIOS: AI Agent Operating System <a href='https://arxiv.org/abs/2403.16971'><img src='https://img.shields.io/badge/Paper-PDF-red'></a> <a href='http",
|
||||
"discovered": "2026-04-22T04:00:03.406309"
|
||||
"discovered": "2026-04-22T05:00:02.743498"
|
||||
},
|
||||
{
|
||||
"name": "rnd-agent-framework",
|
||||
@@ -387,7 +387,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": " # Welcome to Microsoft Agent Framework! [\"> <source srcset=\"apps/w",
|
||||
"discovered": "2026-04-22T04:00:05.306147"
|
||||
"discovered": "2026-04-22T05:00:04.303401"
|
||||
},
|
||||
{
|
||||
"name": "fmgapp",
|
||||
@@ -478,7 +478,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.180763"
|
||||
"discovered": "2026-04-22T05:00:03.196453"
|
||||
},
|
||||
{
|
||||
"name": "obsidian-vault",
|
||||
@@ -491,7 +491,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.862667"
|
||||
"discovered": "2026-04-22T05:00:03.437223"
|
||||
},
|
||||
{
|
||||
"name": "rnd-agents",
|
||||
@@ -504,7 +504,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# Claude Code Plugins: Orchestration and Automation > **⚡ Updated for Opus 4.6, Sonnet 4.6 & Haiku 4.5** — Three-tier model strategy for optimal perf",
|
||||
"discovered": "2026-04-22T04:00:05.014111"
|
||||
"discovered": "2026-04-22T05:00:03.976197"
|
||||
},
|
||||
{
|
||||
"name": "FrancyJGLisboa_agent-skill-creator",
|
||||
@@ -517,7 +517,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# Agent Skill Creator **Turn any workflow into reusable AI agent software that installs on 14+ tools — no spec writing, no prompt engineering, no cod",
|
||||
"discovered": "2026-04-22T04:00:03.081764"
|
||||
"discovered": "2026-04-22T05:00:02.726368"
|
||||
},
|
||||
{
|
||||
"name": "oss",
|
||||
@@ -530,7 +530,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# WEVAL OSS Registry · /opt/oss/ Wave 222 · 2026-04-21 ## Purpose Register the OSS tools identified by AI capability gap audit (wave 220 ai-gap-cach",
|
||||
"discovered": "2026-04-22T04:00:04.908730"
|
||||
"discovered": "2026-04-22T05:00:03.660362"
|
||||
},
|
||||
{
|
||||
"name": "scripts",
|
||||
@@ -543,7 +543,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# Token Rotation Scripts · Opus Session 21-avr v7 ## État - 5 scripts provider skeleton (groq, github, sambanova, alibaba, whatsapp) - 1 master dispa",
|
||||
"discovered": "2026-04-22T04:00:05.143374"
|
||||
"discovered": "2026-04-22T05:00:04.125761"
|
||||
},
|
||||
{
|
||||
"name": "skillsmith",
|
||||
@@ -556,7 +556,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "<div align=\"center\"> <img src=\"terminal.svg\" alt=\"Skillsmith terminal\" width=\"740\"/> </div> <div align=\"center\"> # Skillsmith **Build consistent ",
|
||||
"discovered": "2026-04-22T04:00:05.261405"
|
||||
"discovered": "2026-04-22T05:00:04.197142"
|
||||
},
|
||||
{
|
||||
"name": "awesome-agent-skills",
|
||||
@@ -569,7 +569,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "<a href=\"https://github.com/VoltAgent/voltagent\"> <img width=\"1500\" height=\"801\" alt=\"claude-skills\" src=\"https://github.com/user-attachments/ass",
|
||||
"discovered": "2026-04-22T04:00:03.544545"
|
||||
"discovered": "2026-04-22T05:00:02.927508"
|
||||
},
|
||||
{
|
||||
"name": "paperclip-skills",
|
||||
@@ -582,7 +582,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.911471"
|
||||
"discovered": "2026-04-22T05:00:03.716125"
|
||||
},
|
||||
{
|
||||
"name": "__pycache__",
|
||||
@@ -595,7 +595,7 @@
|
||||
"has_docker": false,
|
||||
"wired": false,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:03.309741"
|
||||
"discovered": "2026-04-22T05:00:02.739048"
|
||||
},
|
||||
{
|
||||
"name": "jzOcb_writing-style-skill",
|
||||
@@ -608,7 +608,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "# Writing Style Skill 可复用的写作风格 Skill 模板。**内置自动学习** — 从你的修改中自动提取规则,SKILL.md 越用越准。 兼容 **Claude Code** + **OpenClaw (ClawHub)**。 ## 原理 ``` AI 用 SKILL",
|
||||
"discovered": "2026-04-22T04:00:04.268582"
|
||||
"discovered": "2026-04-22T05:00:03.239106"
|
||||
},
|
||||
{
|
||||
"name": "qdrant-data",
|
||||
@@ -621,7 +621,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.977548"
|
||||
"discovered": "2026-04-22T05:00:03.868648"
|
||||
},
|
||||
{
|
||||
"name": "wazuh",
|
||||
@@ -634,7 +634,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.486863"
|
||||
"discovered": "2026-04-22T05:00:04.422902"
|
||||
},
|
||||
{
|
||||
"name": "plausible",
|
||||
@@ -647,7 +647,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.921917"
|
||||
"discovered": "2026-04-22T05:00:03.783635"
|
||||
},
|
||||
{
|
||||
"name": "pmta",
|
||||
@@ -660,7 +660,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.923810"
|
||||
"discovered": "2026-04-22T05:00:03.785726"
|
||||
},
|
||||
{
|
||||
"name": "render-configs",
|
||||
@@ -673,7 +673,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.979586"
|
||||
"discovered": "2026-04-22T05:00:03.942716"
|
||||
},
|
||||
{
|
||||
"name": "searxng",
|
||||
@@ -686,7 +686,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.183110"
|
||||
"discovered": "2026-04-22T05:00:04.151920"
|
||||
},
|
||||
{
|
||||
"name": "weval-guardian",
|
||||
@@ -699,7 +699,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.693660"
|
||||
"discovered": "2026-04-22T05:00:04.535805"
|
||||
},
|
||||
{
|
||||
"name": "weval-litellm",
|
||||
@@ -712,7 +712,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.830310"
|
||||
"discovered": "2026-04-22T05:00:04.578484"
|
||||
},
|
||||
{
|
||||
"name": "weval-security",
|
||||
@@ -725,7 +725,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:06.038247"
|
||||
"discovered": "2026-04-22T05:00:04.740268"
|
||||
},
|
||||
{
|
||||
"name": "archive",
|
||||
@@ -738,7 +738,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:03.503328"
|
||||
"discovered": "2026-04-22T05:00:02.772307"
|
||||
},
|
||||
{
|
||||
"name": "loki",
|
||||
@@ -751,7 +751,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.567402"
|
||||
"discovered": "2026-04-22T05:00:03.322173"
|
||||
},
|
||||
{
|
||||
"name": "ruflo",
|
||||
@@ -764,7 +764,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.118268"
|
||||
"discovered": "2026-04-22T05:00:04.123854"
|
||||
},
|
||||
{
|
||||
"name": "twenty",
|
||||
@@ -777,7 +777,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.379419"
|
||||
"discovered": "2026-04-22T05:00:04.360420"
|
||||
},
|
||||
{
|
||||
"name": "weval-crewai",
|
||||
@@ -790,7 +790,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.589634"
|
||||
"discovered": "2026-04-22T05:00:04.494641"
|
||||
},
|
||||
{
|
||||
"name": "weval-plugins",
|
||||
@@ -803,7 +803,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.936437"
|
||||
"discovered": "2026-04-22T05:00:04.662784"
|
||||
},
|
||||
{
|
||||
"name": "weval-radar",
|
||||
@@ -816,7 +816,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.954386"
|
||||
"discovered": "2026-04-22T05:00:04.701336"
|
||||
},
|
||||
{
|
||||
"name": "weval-scrapy",
|
||||
@@ -829,7 +829,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.997555"
|
||||
"discovered": "2026-04-22T05:00:04.703453"
|
||||
},
|
||||
{
|
||||
"name": "blade",
|
||||
@@ -842,7 +842,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:03.719664"
|
||||
"discovered": "2026-04-22T05:00:03.058886"
|
||||
},
|
||||
{
|
||||
"name": "langfuse",
|
||||
@@ -855,7 +855,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.322859"
|
||||
"discovered": "2026-04-22T05:00:03.252520"
|
||||
},
|
||||
{
|
||||
"name": "litellm",
|
||||
@@ -868,7 +868,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.466063"
|
||||
"discovered": "2026-04-22T05:00:03.292301"
|
||||
},
|
||||
{
|
||||
"name": "mattermost-docker",
|
||||
@@ -881,7 +881,7 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.676612"
|
||||
"discovered": "2026-04-22T05:00:03.330081"
|
||||
},
|
||||
{
|
||||
"name": "prometheus",
|
||||
@@ -894,7 +894,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.972115"
|
||||
"discovered": "2026-04-22T05:00:03.822364"
|
||||
},
|
||||
{
|
||||
"name": "twenty-compose",
|
||||
@@ -907,7 +907,20 @@
|
||||
"has_docker": true,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:05.381615"
|
||||
"discovered": "2026-04-22T05:00:04.382525"
|
||||
},
|
||||
{
|
||||
"name": "weval-cli",
|
||||
"path": "/opt/weval-cli",
|
||||
"files": 1,
|
||||
"has_readme": false,
|
||||
"has_skill": false,
|
||||
"has_python": false,
|
||||
"has_node": false,
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T05:00:04.482509"
|
||||
},
|
||||
{
|
||||
"name": "weval-ux",
|
||||
@@ -920,7 +933,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:06.063995"
|
||||
"discovered": "2026-04-22T05:00:04.744751"
|
||||
},
|
||||
{
|
||||
"name": "wevia-integrity",
|
||||
@@ -933,7 +946,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:06.146050"
|
||||
"discovered": "2026-04-22T05:00:04.823387"
|
||||
},
|
||||
{
|
||||
"name": "DiffusionDB",
|
||||
@@ -946,7 +959,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:03.010735"
|
||||
"discovered": "2026-04-22T05:00:02.720532"
|
||||
},
|
||||
{
|
||||
"name": "LTX-Video",
|
||||
@@ -959,7 +972,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:03.230164"
|
||||
"discovered": "2026-04-22T05:00:02.732512"
|
||||
},
|
||||
{
|
||||
"name": "localai",
|
||||
@@ -972,7 +985,7 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:04.538083"
|
||||
"discovered": "2026-04-22T05:00:03.309907"
|
||||
},
|
||||
{
|
||||
"name": "wevia-finetune",
|
||||
@@ -985,6 +998,6 @@
|
||||
"has_docker": false,
|
||||
"wired": true,
|
||||
"description": "",
|
||||
"discovered": "2026-04-22T04:00:06.119096"
|
||||
"discovered": "2026-04-22T05:00:04.792179"
|
||||
}
|
||||
]
|
||||
|
After Width: | Height: | Size: 303 KiB |
|
After Width: | Height: | Size: 304 KiB |
|
After Width: | Height: | Size: 367 KiB |
|
After Width: | Height: | Size: 367 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 398 KiB |
|
After Width: | Height: | Size: 398 KiB |
|
After Width: | Height: | Size: 303 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 368 KiB |
|
After Width: | Height: | Size: 303 KiB |
|
After Width: | Height: | Size: 366 KiB |
|
After Width: | Height: | Size: 367 KiB |
|
After Width: | Height: | Size: 366 KiB |
|
After Width: | Height: | Size: 367 KiB |
|
After Width: | Height: | Size: 367 KiB |
|
After Width: | Height: | Size: 367 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 396 KiB |
|
After Width: | Height: | Size: 303 KiB |
|
After Width: | Height: | Size: 305 KiB |
|
After Width: | Height: | Size: 355 KiB |
|
After Width: | Height: | Size: 355 KiB |
|
After Width: | Height: | Size: 355 KiB |
|
After Width: | Height: | Size: 355 KiB |
|
After Width: | Height: | Size: 355 KiB |
|
After Width: | Height: | Size: 355 KiB |
|
After Width: | Height: | Size: 354 KiB |
|
After Width: | Height: | Size: 354 KiB |
|
After Width: | Height: | Size: 357 KiB |
8
api/playwright-v171-latest.json
Normal file
@@ -0,0 +1,8 @@
|
||||
[
|
||||
"before: v166InProgress=undefined",
|
||||
"before: panel has show class=false",
|
||||
"after await: v166InProgress=false",
|
||||
"after await: panel has show class=true",
|
||||
"after await: thpBody lines=11",
|
||||
"after await: thpBody html first 500=<div class=\"thp-line\"><span class=\"lbl\">🧠 Thinking</span>query · complexity low · 17 chars<span class=\"dur\">0ms</span></div><div class=\"thp-line\"><span class=\"lbl\">📋 Plan</span>3 étapes · backend wevia-autonomous.php<span class=\"dur\">0ms</span></div><div class=\"thp-line\"><span class=\"lbl\"> ↳</span>1. Query RAG / Qdrant context for query</div><div class=\"thp-line\"><span class=\"lbl\"> ↳</span>2. Dispatch to chatbot backend</div><div class=\"thp-line\"><span class=\"lbl\"> ↳</span>3. Format respons"
|
||||
]
|
||||
43
api/playwright-v172-continue-latest.json
Normal file
@@ -0,0 +1,43 @@
|
||||
{
|
||||
"ts": "2026-04-22T02-53-47-429Z",
|
||||
"version": "V172-continue",
|
||||
"tests": [
|
||||
{
|
||||
"name": "login",
|
||||
"pass": true
|
||||
},
|
||||
{
|
||||
"name": "scan1_providers",
|
||||
"pass": false,
|
||||
"state": {
|
||||
"panelShow": false,
|
||||
"lines": 0,
|
||||
"stagesDone": 0
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "scan2_tools",
|
||||
"pass": true,
|
||||
"state": {
|
||||
"panelShow": false,
|
||||
"lines": 0,
|
||||
"stagesDone": 0,
|
||||
"msgCount": 4
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "scan3_status",
|
||||
"pass": false,
|
||||
"state": {
|
||||
"panelShow": true,
|
||||
"lines": 0,
|
||||
"msgCount": 5,
|
||||
"latestMsg": "Status infrastructure?"
|
||||
}
|
||||
}
|
||||
],
|
||||
"video": "/var/www/html/api/playwright-results/v172-continue-multiscan-2026-04-22T02-53-47-429Z/page@d3393ad2b24f7c4c74dc0bc4f56de196.webm",
|
||||
"pass_total": 2,
|
||||
"fail_total": 2,
|
||||
"all_pass": false
|
||||
}
|
||||
69
api/playwright-v172-latest.json
Normal file
@@ -0,0 +1,69 @@
|
||||
{
|
||||
"ts": "2026-04-22T02-37-53-399Z",
|
||||
"version": "V172",
|
||||
"tests": [
|
||||
{
|
||||
"name": "login",
|
||||
"pass": true
|
||||
},
|
||||
{
|
||||
"name": "direct_v166_call",
|
||||
"pass": true,
|
||||
"final": {
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"bodyPreview": "🧠 Thinkinganalytics · complexity medium · 42 chars0ms\n📋 Plan3 étapes · backend wevia-autonomous.php0ms\n↳1. Query relevant KPI source (wtp-kpi-global-v2, nonreg, architecture)\n↳2. Extract metrics from JSON\n↳3. Format quantitative response\n🔗 RAGQdrant queried · 0 contextes trouvés1ms\n⚙ Executebackend OK · 1927 bytes réponse618ms\n🧪 Tests7/7 passés · 100%0ms\n💬 Response971 chars finaux0ms\n✅ Critiquequality 100% · OK: all checks passed · response quality acceptable0ms\n📊 Summary7 phases · 7/7 · E"
|
||||
}
|
||||
}
|
||||
],
|
||||
"progress_states": [
|
||||
{
|
||||
"t": "2s",
|
||||
"inProgress": false,
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"t": "4s",
|
||||
"inProgress": false,
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"t": "6s",
|
||||
"inProgress": false,
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"t": "8s",
|
||||
"inProgress": false,
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"t": "10s",
|
||||
"inProgress": false,
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"t": "12s",
|
||||
"inProgress": false,
|
||||
"panelShow": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
}
|
||||
],
|
||||
"video": "/var/www/html/api/playwright-results/v172-final-2026-04-22T02-37-53-399Z/page@257afacfa9d1200ac6bc98cd6dab0f08.webm",
|
||||
"pass_total": 2,
|
||||
"fail_total": 0,
|
||||
"all_pass": true
|
||||
}
|
||||
50
api/playwright-v173-latest.json
Normal file
@@ -0,0 +1,50 @@
|
||||
{
|
||||
"ts": "2026-04-22T02-52-26-441Z",
|
||||
"version": "V173",
|
||||
"tests": [
|
||||
{
|
||||
"name": "login",
|
||||
"pass": true
|
||||
},
|
||||
{
|
||||
"name": "V162_V166_wired",
|
||||
"pass": true,
|
||||
"wiring": {
|
||||
"v166": true,
|
||||
"thpShow": true,
|
||||
"panel": true,
|
||||
"stages": 7
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "scan1_status_query",
|
||||
"pass": true,
|
||||
"visible": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"name": "scan2_analytics_query",
|
||||
"pass": true,
|
||||
"visible": true,
|
||||
"lines": 11,
|
||||
"stagesDone": 6
|
||||
},
|
||||
{
|
||||
"name": "scan3_toggle",
|
||||
"pass": true,
|
||||
"collapsed": true,
|
||||
"expanded": true
|
||||
},
|
||||
{
|
||||
"name": "scan4_dashboard_counts",
|
||||
"pass": true,
|
||||
"pc": "17"
|
||||
}
|
||||
],
|
||||
"video": "/var/www/html/api/playwright-results/v173-final-multi-scan-2026-04-22T02-52-26-441Z/page@e492216b459f1b25305cb67e352b71c1.webm",
|
||||
"screenshots_dir": "/var/www/html/api/playwright-results/v173-final-multi-scan-2026-04-22T02-52-26-441Z",
|
||||
"pass_total": 6,
|
||||
"fail_total": 0,
|
||||
"all_pass": true
|
||||
}
|
||||
106
api/playwright-v175-latest.json
Normal file
@@ -0,0 +1,106 @@
|
||||
{
|
||||
"ts": "2026-04-22T03-01-13-349Z",
|
||||
"version": "V175",
|
||||
"tests": [
|
||||
{
|
||||
"name": "login",
|
||||
"pass": true
|
||||
},
|
||||
{
|
||||
"name": "v175_wired",
|
||||
"pass": true
|
||||
},
|
||||
{
|
||||
"name": "panel_visible_with_lines",
|
||||
"pass": true,
|
||||
"best": {
|
||||
"t": "2s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "stages_animated",
|
||||
"pass": true,
|
||||
"best_done": 6
|
||||
},
|
||||
{
|
||||
"name": "response_rendered",
|
||||
"pass": true,
|
||||
"preview": "PROVIDERS IA: 17 cles API + 1 modeles WEVIA Engine localCascade: WEVIA Engine > HF > NVIDIA > WEVIA Engine > WEVIA Engine > WEVIA Engine > WEVIA EngineCout: 0 eurosCopierSupprimerExec0.1s"
|
||||
}
|
||||
],
|
||||
"timeline": [
|
||||
{
|
||||
"t": "2s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
},
|
||||
{
|
||||
"t": "4s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
},
|
||||
{
|
||||
"t": "6s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
},
|
||||
{
|
||||
"t": "8s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
},
|
||||
{
|
||||
"t": "10s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
},
|
||||
{
|
||||
"t": "12s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
},
|
||||
{
|
||||
"t": "15s",
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"v166InProgress": false,
|
||||
"msgCount": 2
|
||||
}
|
||||
],
|
||||
"video": "/var/www/html/api/playwright-results/v175-sse-realtime-2026-04-22T03-01-13-349Z/page@5b7b650de7d8cb7f4069d73455a51ee4.webm",
|
||||
"pass_total": 5,
|
||||
"fail_total": 0,
|
||||
"all_pass": true
|
||||
}
|
||||
186
api/playwright-v176-latest.json
Normal file
@@ -0,0 +1,186 @@
|
||||
{
|
||||
"ts": "2026-04-22T03-02-55-447Z",
|
||||
"version": "V176",
|
||||
"tests": [
|
||||
{
|
||||
"name": "login",
|
||||
"pass": true
|
||||
},
|
||||
{
|
||||
"name": "T8s_lines_check",
|
||||
"pass": true,
|
||||
"t8_state": {
|
||||
"t": "8s",
|
||||
"ms": 8045,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "max_lines_reached",
|
||||
"pass": true,
|
||||
"max_lines": 13,
|
||||
"max_stages_done": 6
|
||||
}
|
||||
],
|
||||
"timeline": [
|
||||
{
|
||||
"t": "1s",
|
||||
"ms": 1019,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "2s",
|
||||
"ms": 2026,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "3s",
|
||||
"ms": 3029,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "4s",
|
||||
"ms": 4034,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "5s",
|
||||
"ms": 5037,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "6s",
|
||||
"ms": 6039,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "7s",
|
||||
"ms": 7042,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "8s",
|
||||
"ms": 8045,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "9s",
|
||||
"ms": 9048,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "10s",
|
||||
"ms": 10052,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "11s",
|
||||
"ms": 11055,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "12s",
|
||||
"ms": 12067,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "13s",
|
||||
"ms": 13069,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "14s",
|
||||
"ms": 14071,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
},
|
||||
{
|
||||
"t": "15s",
|
||||
"ms": 15072,
|
||||
"panelShow": true,
|
||||
"lines": 13,
|
||||
"stagesDone": 6,
|
||||
"stagesActive": 1,
|
||||
"msgCount": 2,
|
||||
"inProgress": false
|
||||
}
|
||||
],
|
||||
"video": "/var/www/html/api/playwright-results/v176-timing-precise-2026-04-22T03-02-55-447Z/page@bef917b985a43650cc934083b0406dda.webm",
|
||||
"pass_total": 3,
|
||||
"fail_total": 0,
|
||||
"all_pass": true
|
||||
}
|
||||
BIN
api/playwright-videos/v172-continue-multiscan.webm
Normal file
BIN
api/playwright-videos/v172-wevia-master-7-phases-real.webm
Normal file
BIN
api/playwright-videos/v173-multi-scan-final.webm
Normal file
BIN
api/playwright-videos/v175-sse-realtime-5of5.webm
Normal file
BIN
api/playwright-videos/v176-timing-precise-13lines.webm
Normal file
@@ -1,6 +1,22 @@
|
||||
<?php
|
||||
// WAVE 253 · saas-chat grounded with live WEVAL data (anti-hallucination)
|
||||
header('Content-Type: application/json');
|
||||
// WAVE 256: Defense-in-depth sanitizer (mirrors weval-ia-fast.php)
|
||||
function wave256_sanitize($t) {
|
||||
if (!is_string($t) || $t === '') return $t;
|
||||
$blocklist = ['Groq', 'Cerebras', 'SambaNova', 'Ollama', 'DeepSeek', 'Mistral', 'Together', 'Replicate',
|
||||
'vLLM', 'Qwen', 'NVIDIA NIM', 'Cohere', 'OpenRouter', 'HuggingFace', 'Anthropic',
|
||||
'/opt/', '/var/www/', '/etc/', 'admin123', '49222', '11434', '6333', '4001',
|
||||
'204.168', '95.216', '151.80', '10.1.0', 'root@', 'ssh -p', 'docker ps',
|
||||
'PGPASSWORD', 'PostgreSQL', 'weval_leads', 'weval_tasks'];
|
||||
foreach ($blocklist as $w) $t = str_ireplace($w, 'WEVIA Engine', $t);
|
||||
$t = preg_replace('/\b\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}\b/', '[infrastructure securisee]', $t);
|
||||
$t = preg_replace('/\b(sk-[a-zA-Z0-9]{20,}|xoxb-[a-zA-Z0-9-]{20,}|eyJ[a-zA-Z0-9_.-]{50,})\b/', '[token securise]', $t);
|
||||
$t = preg_replace('/\b(ghp|gho|ghu|ghs|ghr)_[A-Za-z0-9]{20,}\b/', '[token securise]', $t);
|
||||
return $t;
|
||||
}
|
||||
|
||||
|
||||
header('Access-Control-Allow-Origin: *');
|
||||
if($_SERVER['REQUEST_METHOD']==='OPTIONS'){header('Access-Control-Allow-Methods: POST');header('Access-Control-Allow-Headers: Content-Type');exit;}
|
||||
$input=json_decode(file_get_contents('php://input'),true);
|
||||
@@ -143,7 +159,7 @@ $history[]=['role'=>'assistant','content'=>$reply];
|
||||
if($redis)$redis->setex("saas:$session",3600,json_encode(array_slice($history,-20)));
|
||||
|
||||
echo json_encode([
|
||||
'response'=>$reply,
|
||||
'response' => wave256_sanitize($reply),
|
||||
'provider'=>$provider_used ?: 'saas-sovereign',
|
||||
'session'=>$session,
|
||||
'grounding'=>[
|
||||
|
||||