Compare commits
66 Commits
wave-225-p
...
wave-228-g
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bfa20ebe57 | ||
|
|
6df6fd7f35 | ||
|
|
3eda96d9d4 | ||
|
|
306552cec6 | ||
|
|
073d617d08 | ||
|
|
27f9e80bc9 | ||
|
|
14976ae05a | ||
|
|
4dd03ea3fb | ||
|
|
41e8202461 | ||
|
|
d9016feadc | ||
|
|
4193cac577 | ||
|
|
bb34f9695f | ||
|
|
f75092aa3f | ||
|
|
2ff7e3a0ea | ||
|
|
cb993ae41c | ||
|
|
dae689cecd | ||
|
|
fa16e6554e | ||
|
|
a4d0c4d564 | ||
|
|
adf9eba31c | ||
|
|
c22f115b3e | ||
|
|
9c69db151f | ||
|
|
bc6d6cb2fb | ||
|
|
c4bf820a92 | ||
|
|
99195cf362 | ||
|
|
a6c4850b58 | ||
|
|
874a7c6dfa | ||
|
|
917e2441af | ||
|
|
decb3e2904 | ||
|
|
84a6a12f1f | ||
|
|
97c4a5e1b3 | ||
|
|
3e44d926de | ||
|
|
7737c976ed | ||
|
|
c5fa4e7480 | ||
|
|
99c7db040f | ||
|
|
ac38795373 | ||
|
|
a78b554733 | ||
|
|
a0257bff01 | ||
|
|
b438489484 | ||
|
|
282cba3eda | ||
|
|
b157e5e6da | ||
|
|
ee1ce9d791 | ||
|
|
4cdd2f56ba | ||
|
|
2a6e707f38 | ||
|
|
23ef40516a | ||
|
|
03c2699122 | ||
|
|
79af700e98 | ||
|
|
124b23e60f | ||
|
|
ad93447f00 | ||
|
|
0558cf03ed | ||
|
|
19cb060d60 | ||
|
|
b74675f037 | ||
|
|
632f6349e3 | ||
|
|
05ce22a54c | ||
|
|
ec6762838f | ||
|
|
dc58ec560f | ||
|
|
b0f9523064 | ||
|
|
ff64a67fbb | ||
|
|
5e9aa9d772 | ||
|
|
0eb4825f7d | ||
|
|
e3e6e3ac54 | ||
|
|
8e37e1c3f4 | ||
|
|
5ed6857e78 | ||
|
|
8d0f0ceee4 | ||
|
|
e94c263624 | ||
|
|
e824e9c03e | ||
|
|
2c9ff7c958 |
@@ -1,10 +1,10 @@
|
||||
{
|
||||
"agent": "V41_Disk_Monitor",
|
||||
"ts": "2026-04-21T23:30:02+02:00",
|
||||
"disk_pct": 83,
|
||||
"disk_free_gb": 26,
|
||||
"ts": "2026-04-22T02:00:02+02:00",
|
||||
"disk_pct": 84,
|
||||
"disk_free_gb": 25,
|
||||
"growth_per_day_gb": 1.5,
|
||||
"runway_days": 17,
|
||||
"runway_days": 16,
|
||||
"alert": "WARN_runway_under_30d",
|
||||
"action_auto_if_under_7d": "trigger_hetzner_volume_extension_api",
|
||||
"hetzner_volume_size_gb_recommended": 500,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V41_Risk_Escalation",
|
||||
"ts": "2026-04-21T23:15:02+02:00",
|
||||
"ts": "2026-04-22T02:00:03+02:00",
|
||||
"dg_alerts_active": 7,
|
||||
"wevia_life_stats_preview": "{
|
||||
"ok": true,
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
{
|
||||
"routes": 446,
|
||||
"skills": 835,
|
||||
"wiki": 2046,
|
||||
"pages": 317,
|
||||
"apis": 251,
|
||||
"wiki": 2066,
|
||||
"pages": 318,
|
||||
"apis": 252,
|
||||
"docker": 19,
|
||||
"proposals": [
|
||||
{
|
||||
@@ -27,5 +27,5 @@
|
||||
"effort": "S"
|
||||
}
|
||||
],
|
||||
"timestamp": "2026-04-21 16:00"
|
||||
"timestamp": "2026-04-21 22:00"
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"timestamp": "2026-04-21 12:00",
|
||||
"timestamp": "2026-04-22 00:00",
|
||||
"analysis": {
|
||||
"existing_skills": 835,
|
||||
"missing": 15,
|
||||
|
||||
@@ -1,12 +1,12 @@
|
||||
{
|
||||
"agent": "V41_Feature_Adoption_Tracker",
|
||||
"ts": "2026-04-21T23:00:02+02:00",
|
||||
"ts": "2026-04-22T02:00:02+02:00",
|
||||
"features_tracked": 15,
|
||||
"features_used_24h": 12,
|
||||
"adoption_pct": 80,
|
||||
"chat_queries_last_1k_log": 10,
|
||||
"wtp_views_last_1k_log": 130,
|
||||
"dg_views_last_1k_log": 3,
|
||||
"features_used_24h": 11,
|
||||
"adoption_pct": 73,
|
||||
"chat_queries_last_1k_log": 4,
|
||||
"wtp_views_last_1k_log": 41,
|
||||
"dg_views_last_1k_log": 0,
|
||||
"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-21T23:20:02+02:00",
|
||||
"ts": "2026-04-22T02:00:05+02:00",
|
||||
"paperclip_total": 48,
|
||||
"active_customer": 4,
|
||||
"warm_prospect": 5,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V41_MQL_Scoring",
|
||||
"ts": "2026-04-21T23:00:03+02:00",
|
||||
"ts": "2026-04-22T02:00:03+02:00",
|
||||
"leads_total": 48,
|
||||
"mql_current": 16,
|
||||
"sql_current": 6,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V60_Nudge_Owner_Actions",
|
||||
"ts": "2026-04-21T16:00:01+02:00",
|
||||
"ts": "2026-04-22T00:00:02+02:00",
|
||||
"cron": "every_8_hours",
|
||||
"actions_pending_owner": {
|
||||
"emails_drafts_V45_to_send": {
|
||||
@@ -10,10 +10,10 @@
|
||||
"action": "Yacine envoie via Gmail ymahboub@weval-consulting.com"
|
||||
},
|
||||
"ethica_renewal_Q1": {
|
||||
"days_to_Q1_end": -21,
|
||||
"days_to_Q1_end": -22,
|
||||
"amount_keur": 280,
|
||||
"urgency": "CRITICAL",
|
||||
"action": "Close contrat avec Kaouther Najar avant -21 jours"
|
||||
"action": "Close contrat avec Kaouther Najar avant -22 jours"
|
||||
},
|
||||
"sourcing_39_emails_linkedin": {
|
||||
"count": 39,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"agent": "V54_Risk_Monitor_Live",
|
||||
"ts": "2026-04-21T23:00:03+02:00",
|
||||
"ts": "2026-04-22T02:00:04+02:00",
|
||||
"critical_risks": {
|
||||
"RW01_pipeline_vide": {
|
||||
"pipeline_keur": 0,
|
||||
@@ -22,7 +22,7 @@
|
||||
},
|
||||
"RW12_burnout": {
|
||||
"agents_cron_active": 15,
|
||||
"load_5min": "5.02",
|
||||
"load_5min": "11.29",
|
||||
"automation_coverage_pct": 70,
|
||||
"residual_risk_pct": 60,
|
||||
"trend": "V52_goldratt_options_active"
|
||||
|
||||
@@ -1,13 +1,13 @@
|
||||
{
|
||||
"timestamp": "2026-04-21 22:00",
|
||||
"timestamp": "2026-04-22 02:00",
|
||||
"sections": {
|
||||
"servers": {
|
||||
"S204": {
|
||||
"docker": 19,
|
||||
"disk": "82%",
|
||||
"ram": "12Gi/30Gi",
|
||||
"load": "9.85",
|
||||
"uptime": "up 1 week, 10 hours, 8 minutes"
|
||||
"docker": 20,
|
||||
"disk": "84%",
|
||||
"ram": "13Gi/30Gi",
|
||||
"load": "6.51",
|
||||
"uptime": "up 1 week, 14 hours, 8 minutes"
|
||||
}
|
||||
},
|
||||
"docker": {
|
||||
@@ -95,7 +95,7 @@
|
||||
},
|
||||
{
|
||||
"name": "uptime-kuma",
|
||||
"status": "Up 44 hours (healthy)",
|
||||
"status": "Up 2 days (healthy)",
|
||||
"ports": ""
|
||||
},
|
||||
{
|
||||
@@ -111,7 +111,7 @@
|
||||
]
|
||||
},
|
||||
"apis": {
|
||||
"count": 272,
|
||||
"count": 273,
|
||||
"files": [
|
||||
"wevia-stream-sovereign.php",
|
||||
"wevia-pending-loader.php",
|
||||
@@ -321,6 +321,7 @@
|
||||
"wevia-dream.php",
|
||||
"wevia-public-status.php",
|
||||
"wevia-sovereign-proxy.php",
|
||||
"wevia-intent-autowire.php",
|
||||
"wevia-dev-pipeline.php",
|
||||
"wevia-batch.php",
|
||||
"wevia-lean-toc.php",
|
||||
@@ -388,7 +389,7 @@
|
||||
]
|
||||
},
|
||||
"routes": {
|
||||
"lines": 3681,
|
||||
"lines": 3718,
|
||||
"count": 446
|
||||
},
|
||||
"skills": {
|
||||
@@ -480,16 +481,16 @@
|
||||
]
|
||||
},
|
||||
"pages": {
|
||||
"count": 317
|
||||
"count": 319
|
||||
},
|
||||
"opt_tools": {
|
||||
"count": 91
|
||||
"count": 95
|
||||
},
|
||||
"dataset": {
|
||||
"pairs": 5751
|
||||
},
|
||||
"wiki": {
|
||||
"entries": 2066
|
||||
"entries": 2123
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
{
|
||||
"generated_at": "2026-04-21T18:00:03.108093",
|
||||
"generated_at": "2026-04-22T00:00:03.159161",
|
||||
"agent_version": "V69_enhanced",
|
||||
"pages_scanned": 9,
|
||||
"fixed_elements_checked": 19,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"timestamp": "2026-04-21T16:00:05+00:00",
|
||||
"compute_ms": 2343,
|
||||
"timestamp": "2026-04-21T22:00:06+00:00",
|
||||
"compute_ms": 4023,
|
||||
"metrics": {
|
||||
"agents": 0,
|
||||
"agents_hierarchy": 0,
|
||||
@@ -13,31 +13,31 @@
|
||||
"oss_tools": 765,
|
||||
"oss_skills": 734,
|
||||
"oss_tests": 765,
|
||||
"docker": 19,
|
||||
"docker": 20,
|
||||
"ollama_models": 7,
|
||||
"git_repos": 38,
|
||||
"providers": [
|
||||
{
|
||||
"name": "Cerebras",
|
||||
"latency_ms": 609,
|
||||
"latency_ms": 968,
|
||||
"status": "up"
|
||||
},
|
||||
{
|
||||
"name": "Groq",
|
||||
"latency_ms": 499,
|
||||
"latency_ms": 1001,
|
||||
"status": "up"
|
||||
}
|
||||
]
|
||||
},
|
||||
"scores": {
|
||||
"combined": 75,
|
||||
"infra": 56,
|
||||
"infra": 57,
|
||||
"ecosystem": 100,
|
||||
"agents": 0,
|
||||
"skills": 100,
|
||||
"nonreg": 100,
|
||||
"oss": 100,
|
||||
"docker": 95,
|
||||
"docker": 100,
|
||||
"providers": 72,
|
||||
"hierarchy": 0,
|
||||
"instructions": 100
|
||||
@@ -45,7 +45,7 @@
|
||||
"leaderboard": [
|
||||
{
|
||||
"name": "WEVAL_Ecosystem",
|
||||
"score": 80.6,
|
||||
"score": 80.7,
|
||||
"skills": 839,
|
||||
"agents": 0
|
||||
},
|
||||
@@ -61,7 +61,7 @@
|
||||
},
|
||||
{
|
||||
"name": "WEVAL_MiroFish",
|
||||
"score": 95,
|
||||
"score": 100,
|
||||
"type": "sovereign"
|
||||
},
|
||||
{
|
||||
|
||||
@@ -3,9 +3,9 @@
|
||||
"total_gaps": 8,
|
||||
"gaps": {
|
||||
"pdf_report": {
|
||||
"current_score": 55,
|
||||
"gap": 35,
|
||||
"priority": "critical",
|
||||
"current_score": 63,
|
||||
"gap": 27,
|
||||
"priority": "high",
|
||||
"candidates": [
|
||||
{
|
||||
"name": "reportlab",
|
||||
@@ -28,7 +28,8 @@
|
||||
"full_name": "Kozea/WeasyPrint",
|
||||
"stars": 7500,
|
||||
"description": "HTML to PDF with CSS \u00b7 rich layouts",
|
||||
"installed": false
|
||||
"installed": true,
|
||||
"installed_at": "2026-04-21T23:43:33.502172"
|
||||
},
|
||||
{
|
||||
"name": "gotenberg",
|
||||
@@ -45,12 +46,12 @@
|
||||
"installed": false
|
||||
}
|
||||
],
|
||||
"previous_score": 47,
|
||||
"bump_reason": "2 OSS installed: reportlab, pypdf2 (+8)"
|
||||
"previous_score": 55,
|
||||
"bump_reason": "WeasyPrint installed +8"
|
||||
},
|
||||
"proposal": {
|
||||
"current_score": 51,
|
||||
"gap": 39,
|
||||
"current_score": 55,
|
||||
"gap": 35,
|
||||
"priority": "critical",
|
||||
"candidates": [
|
||||
{
|
||||
@@ -58,7 +59,8 @@
|
||||
"full_name": "docusealco/docuseal",
|
||||
"stars": 7800,
|
||||
"description": "Open source DocuSign alternative \u00b7 electronic signatures + proposals",
|
||||
"installed": false
|
||||
"installed": true,
|
||||
"installed_at": "2026-04-21T23:52:44.498853"
|
||||
},
|
||||
{
|
||||
"name": "pdfme",
|
||||
@@ -77,8 +79,8 @@
|
||||
"shared_with": "pdf_report"
|
||||
}
|
||||
],
|
||||
"previous_score": 47,
|
||||
"bump_reason": "1 OSS installed (shared): reportlab (+4)"
|
||||
"previous_score": 51,
|
||||
"bump_reason": "docuseal deployed +4"
|
||||
},
|
||||
"code": {
|
||||
"current_score": 67,
|
||||
@@ -187,10 +189,20 @@
|
||||
"bump_reason": "2 OSS installed: funnlp, pandas-ai (+8)"
|
||||
},
|
||||
"pharma": {
|
||||
"current_score": 62,
|
||||
"gap": 28,
|
||||
"current_score": 66,
|
||||
"gap": 24,
|
||||
"priority": "medium",
|
||||
"candidates": []
|
||||
"candidates": [
|
||||
{
|
||||
"name": "biopython",
|
||||
"full_name": "biopython/biopython",
|
||||
"stars": 1700,
|
||||
"installed": true,
|
||||
"installed_at": "2026-04-21T23:52:44.498824"
|
||||
}
|
||||
],
|
||||
"previous_score": 62,
|
||||
"bump_reason": "biopython installed +4"
|
||||
},
|
||||
"strategy": {
|
||||
"current_score": 65,
|
||||
@@ -274,12 +286,12 @@
|
||||
"installed": false
|
||||
}
|
||||
],
|
||||
"last_refresh": "2026-04-21T23:23:06.309385",
|
||||
"last_refresh": "2026-04-21T23:52:44.498855",
|
||||
"refreshed_by": "opus-wave-223-audit-refresh",
|
||||
"oss_installed_count": 4,
|
||||
"oss_installed_count": 10,
|
||||
"oss_registry_disk_mb": 828,
|
||||
"last_audit_rescan": "2026-04-21T23:26:52.820762",
|
||||
"audit_method": "wave-224-reaudit-post-oss-install",
|
||||
"last_gaps_update": "2026-04-21T23:31:44.984815",
|
||||
"gaps_update_wave": 225
|
||||
"gaps_update_wave": 227
|
||||
}
|
||||
52
api/ambre-6sigma-scan.php
Normal file
@@ -0,0 +1,52 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$out = [];
|
||||
|
||||
// 1. Check PHP-FPM status
|
||||
$out["fpm_status"] = @shell_exec("systemctl is-active php8.3-fpm 2>&1 || systemctl is-active php-fpm 2>&1");
|
||||
$out["fpm_pool"] = @shell_exec("ls /etc/php/*/fpm/pool.d/*.conf 2>&1 | head -3");
|
||||
|
||||
// max_children from pool config
|
||||
$fpm_conf = @shell_exec("grep -h 'pm.max_children\\|pm.start_servers\\|pm.max_spare_servers' /etc/php/*/fpm/pool.d/*.conf 2>&1 | head -20");
|
||||
$out["fpm_pool_config"] = trim($fpm_conf);
|
||||
|
||||
// 2. Current load
|
||||
$out["load_avg"] = trim(@shell_exec("uptime") ?: "");
|
||||
$out["fpm_processes"] = intval(@shell_exec("ps aux | grep -c php-fpm8") ?: 0);
|
||||
|
||||
// 3. Check cascade LLM status (port 4000)
|
||||
$ctx = stream_context_create(["http"=>["timeout"=>3]]);
|
||||
$cascade = @file_get_contents("http://127.0.0.1:4000/health", false, $ctx);
|
||||
$out["cascade_health"] = $cascade ? "OK" : "DOWN/UNREACHABLE";
|
||||
|
||||
// 4. Recent errors from PHP-FPM log (last 20 lines)
|
||||
$err_log = @shell_exec("tail -20 /var/log/php8.3-fpm.log 2>/dev/null || tail -20 /var/log/php-fpm.log 2>/dev/null");
|
||||
$out["fpm_errors"] = substr($err_log ?? "", 0, 1500);
|
||||
|
||||
// 5. Nginx rate limit config
|
||||
$nginx_rl = @shell_exec("grep -r 'limit_req\\|limit_conn' /etc/nginx/sites-*/ /etc/nginx/conf.d/ 2>/dev/null | head -5");
|
||||
$out["nginx_rate_limit"] = trim($nginx_rl);
|
||||
|
||||
// 6. Cloudflare rate limit? Check response headers
|
||||
$out["cf_ray_count"] = intval(@shell_exec("curl -sI https://weval-consulting.com/ 2>&1 | grep -c 'cf-ray'") ?: 0);
|
||||
|
||||
// 7. Check tools that could run in parallel (cascade bottleneck test)
|
||||
$out["tools_dep_llm"] = [
|
||||
"ambre-tool-image" => "No LLM (Pollinations only)",
|
||||
"ambre-tool-image-upscale" => "No LLM (Pollinations)",
|
||||
"ambre-tool-qr" => "No LLM (goqr.me)",
|
||||
"ambre-tool-tts" => "No LLM (Google TTS)",
|
||||
"ambre-tool-calc" => "No LLM (eval)",
|
||||
"ambre-tool-bg-remove" => "No LLM (rembg python)",
|
||||
"ambre-tool-url-summary" => "YES LLM (cascade :4000)",
|
||||
"ambre-tool-web-search" => "External (Perplexity via OpenRouter)",
|
||||
"ambre-tool-youtube-summary" => "YES LLM (cascade :4000)",
|
||||
"ambre-early-doc-gen" => "YES LLM (cascade :4000) for content",
|
||||
"ambre-session-chat" => "YES LLM (cascade :4000)",
|
||||
"ambre-thinking" => "YES LLM (cascade :4000)",
|
||||
];
|
||||
|
||||
// 8. Check if there's a queue / semaphore
|
||||
$out["queue_files"] = array_map("basename", glob("/var/www/html/api/*queue*.php") ?: []);
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
24
api/ambre-compare-gold.php
Normal file
@@ -0,0 +1,24 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
$gold = "/opt/wevads/vault/wevia.html.GOLD-20260421-230109-pre-safe-write";
|
||||
$current = "/var/www/html/wevia.html";
|
||||
|
||||
$g_content = @file_get_contents($gold);
|
||||
$c_content = @file_get_contents($current);
|
||||
|
||||
echo "GOLD size: " . strlen($g_content) . "\n";
|
||||
echo "Current size: " . strlen($c_content) . "\n\n";
|
||||
|
||||
// Parse both with node to see which has error
|
||||
file_put_contents("/tmp/gold.js", "void function(){" . extract_main_script($g_content) . "}();");
|
||||
file_put_contents("/tmp/current.js", "void function(){" . extract_main_script($c_content) . "}();");
|
||||
|
||||
echo "=== GOLD node --check ===\n";
|
||||
echo @shell_exec("node --check /tmp/gold.js 2>&1 | head -10");
|
||||
echo "\n=== Current node --check ===\n";
|
||||
echo @shell_exec("node --check /tmp/current.js 2>&1 | head -10");
|
||||
|
||||
function extract_main_script($html) {
|
||||
preg_match('/<script>(.*?)<\/script>/s', $html, $m);
|
||||
return $m[1] ?? "";
|
||||
}
|
||||
37
api/ambre-deps-find.php
Normal file
@@ -0,0 +1,37 @@
|
||||
<?php
|
||||
/* V144: cache 1h for ambre-deps-find - previous version took 30+s for find / */
|
||||
header("Content-Type: application/json");
|
||||
|
||||
$cache_file = "/tmp/ambre-deps-cache.json";
|
||||
$cache_ttl = 3600; /* 1 hour */
|
||||
|
||||
if (file_exists($cache_file) && (time() - filemtime($cache_file)) < $cache_ttl) {
|
||||
/* V144 cache HIT: respond instantly from cache */
|
||||
$cached = @file_get_contents($cache_file);
|
||||
if ($cached) {
|
||||
header("X-V144-Cache: HIT");
|
||||
echo $cached;
|
||||
exit;
|
||||
}
|
||||
}
|
||||
|
||||
/* V144 cache MISS: scan limited paths only (not full /) */
|
||||
$out = [];
|
||||
$out["rembg_find"] = trim(@shell_exec("find /usr/local/bin /usr/bin /opt/venv -name rembg -type f -executable 2>/dev/null | head -3") ?: "");
|
||||
$out["python_path"] = trim(@shell_exec("python3 -c \"import sys; print(sys.executable)\"") ?: "");
|
||||
|
||||
/* Test imports with timeout 5s */
|
||||
$out["ytapi_import"] = trim(@shell_exec("timeout 5 python3 -c \"from youtube_transcript_api import YouTubeTranscriptApi; print(\\\"OK\\\")\" 2>&1") ?: "");
|
||||
$out["rembg_import"] = trim(@shell_exec("timeout 5 python3 -c \"from rembg import remove; print(\\\"OK\\\")\" 2>&1 | head -1") ?: "");
|
||||
|
||||
$out["_v144_cached_at"] = date("c");
|
||||
$out["_v144_cache_ttl_sec"] = $cache_ttl;
|
||||
|
||||
$response = json_encode($out, JSON_PRETTY_PRINT);
|
||||
|
||||
/* Save cache */
|
||||
@file_put_contents($cache_file, $response);
|
||||
@chmod($cache_file, 0644);
|
||||
|
||||
header("X-V144-Cache: MISS");
|
||||
echo $response;
|
||||
97
api/ambre-fix-and-v9.php
Normal file
@@ -0,0 +1,97 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$path = "/var/www/html/wevia.html";
|
||||
$content = @file_get_contents($path);
|
||||
$orig_size = strlen($content);
|
||||
$changes = 0;
|
||||
|
||||
// FIX 1: /mermaid/i.test → indexOf (removes runtime regex parse issue)
|
||||
$old1 = "if(e&&e.message&&/mermaid/i.test(e.message)) return true;";
|
||||
$new1 = "if(e&&e.message&&String(e.message).toLowerCase().indexOf('mermaid')>=0) return true;";
|
||||
if (strpos($content, $old1) !== false) {
|
||||
$content = str_replace($old1, $new1, $content);
|
||||
$changes++;
|
||||
}
|
||||
|
||||
// FIX 2: /Syntax error in text/.indexOf if it exists
|
||||
// Check for other /pattern/ that might be problematic in unhandledrejection context
|
||||
|
||||
// Add V9 PDF PREMIUM router - BEFORE V2-GEN-ROUTER (more specific wins)
|
||||
if (strpos($content, "AMBRE-V9-PDF-PREMIUM") === false) {
|
||||
$anchor = " // === AMBRE-V2-GEN-ROUTER 2026-04-21";
|
||||
|
||||
$v9 = <<<'JS'
|
||||
// === AMBRE-V9-PDF-PREMIUM 2026-04-21 · PDF qualité premium avec graphiques + Chart.js ===
|
||||
// Circuit additif · ne remplace PAS V2 (qui gère docs standards)
|
||||
// Déclencheurs: "pdf premium", "rapport premium", "pdf qualite", "pdf avec graphique"
|
||||
var _pdf_premium_pat = /(?:pdf|rapport)\s+(?:premium|qualit[eé]|pro|professionnel|avec\s+graphique|hd|chart)|(?:cr[eé]e[zr]?|g[eé]n[eè]re[zr]?|fais|fait|produi[st])\s+(?:un\s+)?(?:rapport|pdf)\s+(?:premium|pro|complet|avec\s+graphique|hd|qualit[eé])/i;
|
||||
if (_pdf_premium_pat.test(text)) {
|
||||
var _topic = text.replace(/^(?:cr[eé]e[zr]?|g[eé]n[eè]re[zr]?|fais|fait|produi[st])\s+(?:moi\s+)?(?:un\s+)?(?:rapport|pdf)\s+(?:premium|pro|complet|qualit[eé]|hd|avec\s+graphique)?\s*(?:sur|pour|de|du|:|à\s+propos\s+de)?\s*/i, '').trim();
|
||||
if (_topic.length < 5) _topic = text;
|
||||
fetch('/api/ambre-tool-pdf-premium.php', {
|
||||
method: 'POST',
|
||||
headers: {'Content-Type':'application/json'},
|
||||
body: JSON.stringify({topic: _topic})
|
||||
})
|
||||
.then(function(r){ return r.text().then(function(t){ try{return JSON.parse(t);}catch(e){return null;} }); })
|
||||
.then(function(data){
|
||||
hideThinking();
|
||||
var elapsed = ((performance.now()-startTime)/1000).toFixed(1);
|
||||
var resp;
|
||||
if (data && data.success) {
|
||||
resp = '📊 **PDF Premium généré** : ' + data.title + '\n\n' +
|
||||
'- **' + data.sections + ' sections** avec contenu détaillé\n' +
|
||||
'- **' + data.kpis + ' KPI** visualisés\n' +
|
||||
'- **Graphique interactif** (' + (data.has_chart ? 'Chart.js intégré' : 'aucun') + ')\n' +
|
||||
'- ~**' + data.pages + ' pages** · ' + data.size_kb + ' KB\n\n' +
|
||||
'📥 [**Télécharger PDF**](' + data.url + ')\n' +
|
||||
'🖼 [Prévisualiser HTML](' + data.html_preview + ')';
|
||||
} else {
|
||||
resp = '❌ Génération PDF Premium échouée. ' + (data && data.error ? data.error : 'Réessayez.');
|
||||
}
|
||||
chatHistory.push({role:'assistant', content:resp});
|
||||
var msgEl = addMsg('assistant', resp, elapsed);
|
||||
if (msgEl && msgEl.querySelector('.msg-inner')) {
|
||||
var b = document.createElement('div'); b.className = 'nx-badges';
|
||||
b.innerHTML = '<span class="nx-badge" style="background:rgba(124,58,237,0.15);color:#7c3aed">📊 PDF Premium</span>' +
|
||||
'<span class="nx-badge" style="background:rgba(16,185,129,0.15);color:#10b981">📈 Chart.js</span>';
|
||||
msgEl.querySelector('.msg-inner').appendChild(b);
|
||||
}
|
||||
// Open artifact panel with HTML preview
|
||||
if (data && data.html_preview && typeof openPreview === 'function') {
|
||||
try { openPreview(data.html_preview, 'pdf'); } catch(e){}
|
||||
}
|
||||
busy=false; try{var s=document.getElementById("sendBtn");if(s)s.disabled=false;}catch(e){}
|
||||
try{var m=document.getElementById("msgInput");if(m){m.value="";m.disabled=false;}}catch(e){}
|
||||
})
|
||||
.catch(function(err){
|
||||
hideThinking();
|
||||
addMsg('assistant', '❌ PDF Premium temporairement indisponible, réessayez.', '0');
|
||||
busy=false;
|
||||
try{var s=document.getElementById("sendBtn");if(s)s.disabled=false;}catch(e){}
|
||||
});
|
||||
return;
|
||||
}
|
||||
// === END AMBRE-V9-PDF-PREMIUM ===
|
||||
|
||||
JS;
|
||||
|
||||
if (strpos($content, $anchor) !== false) {
|
||||
$content = str_replace($anchor, $v9 . $anchor, $content);
|
||||
$changes++;
|
||||
}
|
||||
}
|
||||
|
||||
if ($changes === 0) { echo json_encode(["error"=>"no changes", "orig"=>$orig_size]); exit; }
|
||||
|
||||
$backup = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-v9-pdf-premium";
|
||||
@copy($path, $backup);
|
||||
$wrote = @file_put_contents($path, $content);
|
||||
echo json_encode([
|
||||
"orig" => $orig_size,
|
||||
"new" => strlen($content),
|
||||
"delta" => strlen($content) - $orig_size,
|
||||
"changes" => $changes,
|
||||
"wrote" => $wrote,
|
||||
"backup" => basename($backup),
|
||||
]);
|
||||
34
api/ambre-fix-regex-final.php
Normal file
@@ -0,0 +1,34 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$path = "/var/www/html/wevia.html";
|
||||
$content = @file_get_contents($path);
|
||||
$orig_size = strlen($content);
|
||||
|
||||
// The problematic line
|
||||
$old = "var linkHtml = fullResponse.replace(new RegExp(finalFileUrl, \x27g\x27), \x27<a href=\"\x27+finalFileUrl+\x27\"";
|
||||
|
||||
$has_old = strpos($content, $old);
|
||||
if ($has_old === false) {
|
||||
echo json_encode(["error"=>"pattern not found", "size"=>$orig_size]);
|
||||
exit;
|
||||
}
|
||||
|
||||
// Escape function: replace all regex-special chars with \\char
|
||||
// The fix: use a simple string.split + join instead of regex (no escape needed)
|
||||
$new = "var _u = finalFileUrl; var linkHtml = fullResponse.split(_u).join(\x27<a href=\"\x27+_u+\x27\"";
|
||||
|
||||
$new_content = str_replace($old, $new, $content);
|
||||
$new_size = strlen($new_content);
|
||||
|
||||
// Backup + write
|
||||
$backup = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-regex-split-fix";
|
||||
@copy($path, $backup);
|
||||
$wrote = @file_put_contents($path, $new_content);
|
||||
|
||||
echo json_encode([
|
||||
"orig_size" => $orig_size,
|
||||
"new_size" => $new_size,
|
||||
"delta" => $new_size - $orig_size,
|
||||
"wrote" => $wrote,
|
||||
"backup" => basename($backup),
|
||||
]);
|
||||
13
api/ambre-gold-list.php
Normal file
@@ -0,0 +1,13 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$golds = glob("/opt/wevads/vault/wevia.html.GOLD-2026*");
|
||||
usort($golds, function($a,$b){return filemtime($b)-filemtime($a);});
|
||||
$out = [];
|
||||
foreach (array_slice($golds, 0, 20) as $g) {
|
||||
$out[] = [
|
||||
"name" => basename($g),
|
||||
"size" => filesize($g),
|
||||
"mtime" => date("c", filemtime($g)),
|
||||
];
|
||||
}
|
||||
echo json_encode($out, JSON_PRETTY_PRINT);
|
||||
13
api/ambre-golds.php
Normal file
@@ -0,0 +1,13 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$golds = glob("/opt/wevads/vault/wevia.html.GOLD-*");
|
||||
usort($golds, function($a,$b){return filemtime($b)-filemtime($a);});
|
||||
$out = [];
|
||||
foreach (array_slice($golds, 0, 20) as $g) {
|
||||
$out[] = [
|
||||
"file" => basename($g),
|
||||
"bytes" => filesize($g),
|
||||
"mtime" => date("Y-m-d H:i:s", filemtime($g)),
|
||||
];
|
||||
}
|
||||
echo json_encode($out, JSON_PRETTY_PRINT);
|
||||
12
api/ambre-install-deps.php
Normal file
@@ -0,0 +1,12 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
$out = "";
|
||||
$out .= "=== Installing youtube_transcript_api ===\n";
|
||||
$out .= @shell_exec("pip install --break-system-packages youtube-transcript-api 2>&1 | tail -5");
|
||||
$out .= "\n=== Installing rembg (CPU only) ===\n";
|
||||
$out .= @shell_exec("pip install --break-system-packages 'rembg[cpu]' 2>&1 | tail -10");
|
||||
$out .= "\n=== Verify ===\n";
|
||||
$out .= "tesseract: " . trim(@shell_exec("which tesseract")) . "\n";
|
||||
$out .= "rembg: " . trim(@shell_exec("which rembg")) . "\n";
|
||||
$out .= "yt_api: " . trim(@shell_exec("python3 -c 'import youtube_transcript_api; print(\"OK\", youtube_transcript_api.__version__)' 2>&1")) . "\n";
|
||||
echo $out;
|
||||
28
api/ambre-js-lint.php
Normal file
@@ -0,0 +1,28 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
// Extract the big script from wevia.html and try to parse it
|
||||
$wevia = @file_get_contents("/var/www/html/wevia.html");
|
||||
// Find the main script starting around line 718
|
||||
$pos = 0;
|
||||
$scripts = [];
|
||||
while (($start = strpos($wevia, "<script>", $pos)) !== false) {
|
||||
$end = strpos($wevia, "</script>", $start);
|
||||
if ($end === false) break;
|
||||
$content = substr($wevia, $start + 8, $end - $start - 8);
|
||||
if (strlen($content) > 5000) { // only big scripts
|
||||
$line_start = substr_count(substr($wevia, 0, $start), "\n") + 1;
|
||||
$scripts[] = ["start_line" => $line_start, "size" => strlen($content), "content" => $content];
|
||||
}
|
||||
$pos = $end + 9;
|
||||
}
|
||||
|
||||
echo "Big scripts found: " . count($scripts) . "\n";
|
||||
foreach ($scripts as $i => $s) {
|
||||
$tmp = "/tmp/wevia-script-$i.js";
|
||||
file_put_contents($tmp, $s["content"]);
|
||||
echo "Script $i: line $s[start_line], $s[size]B → $tmp\n";
|
||||
|
||||
// Parse with node to find syntax errors
|
||||
$parse_result = @shell_exec("node --check $tmp 2>&1");
|
||||
echo " Parse: " . substr($parse_result, 0, 500) . "\n\n";
|
||||
}
|
||||
29
api/ambre-js-parse.php
Normal file
@@ -0,0 +1,29 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
$wevia = @file_get_contents("/var/www/html/wevia.html");
|
||||
$pos = 0;
|
||||
$big = null; $start_abs = 0;
|
||||
while (($m = strpos($wevia, "<script>", $pos)) !== false) {
|
||||
$end = strpos($wevia, "</script>", $m);
|
||||
if ($end === false) break;
|
||||
$content = substr($wevia, $m + 8, $end - $m - 8);
|
||||
if (strlen($content) > 20000) {
|
||||
$big = $content;
|
||||
$start_abs = substr_count(substr($wevia, 0, $m + 8), "\n") + 1;
|
||||
break;
|
||||
}
|
||||
$pos = $end + 9;
|
||||
}
|
||||
|
||||
$tmp = "/tmp/wevia-big.js";
|
||||
file_put_contents($tmp, $big);
|
||||
echo "Size: " . strlen($big) . "B\n";
|
||||
echo "Start abs line in HTML: $start_abs\n\n";
|
||||
|
||||
// Try to parse with node
|
||||
$parse = @shell_exec("node --check $tmp 2>&1");
|
||||
echo "=== node --check ===\n$parse\n\n";
|
||||
|
||||
// Also try to execute just to see if RUNTIME regex error appears
|
||||
$exec = @shell_exec("timeout 3 node -e \"const fs=require('fs'); const src=fs.readFileSync('$tmp','utf8'); try { new Function(src); console.log('new Function OK'); } catch(e) { console.log('new Function ERROR:', e.message); console.log(e.stack ? e.stack.split(String.fromCharCode(10))[0] : ''); }\" 2>&1");
|
||||
echo "=== new Function test ===\n$exec\n";
|
||||
27
api/ambre-keys-scan.php
Normal file
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$out = [];
|
||||
|
||||
// All API keys from secrets.env
|
||||
$secrets = @file_get_contents("/etc/weval/secrets.env");
|
||||
if ($secrets) {
|
||||
preg_match_all("/^(\w+)=(\S+)/m", $secrets, $m);
|
||||
$keys_present = [];
|
||||
foreach ($m[1] as $i => $name) {
|
||||
if (strpos($name, "KEY") !== false || strpos($name, "TOKEN") !== false) {
|
||||
$val = $m[2][$i];
|
||||
$keys_present[] = ["name"=>$name, "len"=>strlen($val)];
|
||||
}
|
||||
}
|
||||
$out["keys"] = $keys_present;
|
||||
}
|
||||
|
||||
// Check skill-image-gen.php content
|
||||
$f = "/var/www/html/api/skill-image-gen.php";
|
||||
if (file_exists($f)) $out["skill_image_gen_preview"] = substr(@file_get_contents($f), 0, 1500);
|
||||
|
||||
// Check wevia-deepseek-web.php content
|
||||
$f2 = "/var/www/html/api/wevia-deepseek-web.php";
|
||||
if (file_exists($f2)) $out["deepseek_web_preview"] = substr(@file_get_contents($f2), 0, 800);
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
5
api/ambre-kill25.php
Normal file
@@ -0,0 +1,5 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
@shell_exec("pkill -f playwright 2>&1");
|
||||
echo "killed\n";
|
||||
echo @shell_exec("pgrep -af playwright | head -5");
|
||||
28
api/ambre-line-dump.php
Normal file
@@ -0,0 +1,28 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$wevia = @file_get_contents("/var/www/html/wevia.html");
|
||||
$pos = 0;
|
||||
$big = null;
|
||||
while (($m = strpos($wevia, "<script>", $pos)) !== false) {
|
||||
$end = strpos($wevia, "</script>", $m);
|
||||
if ($end === false) break;
|
||||
$content = substr($wevia, $m + 8, $end - $m - 8);
|
||||
if (strlen($content) > 20000) { $big = $content; break; }
|
||||
$pos = $end + 9;
|
||||
}
|
||||
if (!$big) { echo json_encode(["error"=>"no big script"]); exit; }
|
||||
|
||||
$lines = explode("\n", $big);
|
||||
// Browser reports line 920 col 105 within that script
|
||||
$out = [
|
||||
"total_lines" => count($lines),
|
||||
"line_918" => $lines[917] ?? "",
|
||||
"line_919" => $lines[918] ?? "",
|
||||
"line_920" => $lines[919] ?? "",
|
||||
"line_921" => $lines[920] ?? "",
|
||||
"line_922" => $lines[921] ?? "",
|
||||
"line_920_length" => strlen($lines[919] ?? ""),
|
||||
"col_95_115_of_920" => substr($lines[919] ?? "", 94, 21),
|
||||
];
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
83
api/ambre-llm-semaphore.php
Normal file
@@ -0,0 +1,83 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-llm-semaphore.php · 6σ Lean server-side throttle for LLM cascade :4000
|
||||
* Prevents > 5 simultaneous LLM calls to protect cascade from burst overload.
|
||||
* Used transparently by tools that call LLM.
|
||||
*/
|
||||
|
||||
class AmbreLLMSemaphore {
|
||||
const DIR = "/var/tmp/ambre-llm-sem";
|
||||
const MAX_CONCURRENT = 5;
|
||||
const MAX_WAIT_MS = 20000; // 20s max wait in queue
|
||||
const STALE_LOCK_SEC = 60; // kill locks older than 60s
|
||||
|
||||
public static function init() {
|
||||
if (!is_dir(self::DIR)) @mkdir(self::DIR, 0777, true);
|
||||
}
|
||||
|
||||
/** Clean stale locks (older than 60s) */
|
||||
public static function cleanup() {
|
||||
self::init();
|
||||
$now = time();
|
||||
foreach (glob(self::DIR . "/*.lock") as $f) {
|
||||
if (($now - @filemtime($f)) > self::STALE_LOCK_SEC) @unlink($f);
|
||||
}
|
||||
}
|
||||
|
||||
/** Count active locks */
|
||||
public static function count_active() {
|
||||
self::cleanup();
|
||||
return count(glob(self::DIR . "/*.lock"));
|
||||
}
|
||||
|
||||
/** Acquire a slot (blocks up to MAX_WAIT_MS) */
|
||||
public static function acquire() {
|
||||
self::init();
|
||||
$id = bin2hex(random_bytes(6)) . "-" . getmypid();
|
||||
$start = microtime(true);
|
||||
|
||||
while (true) {
|
||||
if (self::count_active() < self::MAX_CONCURRENT) {
|
||||
@file_put_contents(self::DIR . "/$id.lock", date("c"));
|
||||
return $id;
|
||||
}
|
||||
// Wait 200ms then retry
|
||||
if ((microtime(true) - $start) * 1000 > self::MAX_WAIT_MS) {
|
||||
return null; // timeout - caller should handle
|
||||
}
|
||||
usleep(200000);
|
||||
}
|
||||
}
|
||||
|
||||
/** Release a slot */
|
||||
public static function release($id) {
|
||||
if (!$id) return;
|
||||
@unlink(self::DIR . "/$id.lock");
|
||||
}
|
||||
|
||||
/** Wrap a callable with semaphore protection */
|
||||
public static function guarded($callable) {
|
||||
$id = self::acquire();
|
||||
try {
|
||||
$result = $callable($id);
|
||||
} finally {
|
||||
self::release($id);
|
||||
}
|
||||
return $result;
|
||||
}
|
||||
|
||||
/** Stats for monitoring */
|
||||
public static function stats() {
|
||||
return [
|
||||
"active" => self::count_active(),
|
||||
"max" => self::MAX_CONCURRENT,
|
||||
"dir" => self::DIR,
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
// Expose as endpoint for stats
|
||||
if (basename($_SERVER["SCRIPT_NAME"]) === "ambre-llm-semaphore.php") {
|
||||
header("Content-Type: application/json");
|
||||
echo json_encode(AmbreLLMSemaphore::stats());
|
||||
}
|
||||
6
api/ambre-load-quick.php
Normal file
@@ -0,0 +1,6 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
echo json_encode([
|
||||
"load" => trim(shell_exec("uptime")),
|
||||
"fpm_procs" => intval(shell_exec("pgrep -c php-fpm8")),
|
||||
]);
|
||||
25
api/ambre-or-models.php
Normal file
@@ -0,0 +1,25 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$secrets = @file_get_contents("/etc/weval/secrets.env");
|
||||
preg_match("/^OPENROUTER_KEY=(\S+)/m", $secrets ?? "", $m);
|
||||
$or_key = $m[1] ?? "";
|
||||
|
||||
$ch = curl_init("https://openrouter.ai/api/v1/models");
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_RETURNTRANSFER => true,
|
||||
CURLOPT_TIMEOUT => 15,
|
||||
CURLOPT_HTTPHEADER => ["Authorization: Bearer $or_key"],
|
||||
]);
|
||||
$raw = curl_exec($ch);
|
||||
curl_close($ch);
|
||||
|
||||
$d = json_decode($raw, true);
|
||||
$online = [];
|
||||
$sonar = [];
|
||||
foreach ($d["data"] ?? [] as $m) {
|
||||
$id = $m["id"] ?? "";
|
||||
if (strpos($id, "online") !== false || strpos($id, "sonar") !== false || strpos($id, "perplexity") !== false) {
|
||||
$online[] = $id;
|
||||
}
|
||||
}
|
||||
echo json_encode(["online_models" => $online, "total_models" => count($d["data"] ?? [])], JSON_PRETTY_PRINT);
|
||||
31
api/ambre-parse-all.php
Normal file
@@ -0,0 +1,31 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
|
||||
$wevia = @file_get_contents("/var/www/html/wevia.html");
|
||||
|
||||
// Extract all <script> contents and try each with node
|
||||
$pos = 0; $n = 0;
|
||||
while (($m = strpos($wevia, "<script", $pos)) !== false) {
|
||||
$tag_end = strpos($wevia, ">", $m);
|
||||
if ($tag_end === false) break;
|
||||
$tag = substr($wevia, $m, $tag_end - $m + 1);
|
||||
// Skip if has src=
|
||||
if (strpos($tag, "src=") !== false) { $pos = $tag_end + 1; continue; }
|
||||
$end = strpos($wevia, "</script>", $tag_end);
|
||||
if ($end === false) break;
|
||||
$content = substr($wevia, $tag_end + 1, $end - $tag_end - 1);
|
||||
if (strlen($content) < 50) { $pos = $end + 9; continue; }
|
||||
|
||||
$n++;
|
||||
$abs_line = substr_count(substr($wevia, 0, $tag_end + 1), "\n") + 1;
|
||||
$tmp = "/tmp/wscript-$n.js";
|
||||
file_put_contents($tmp, $content);
|
||||
$parse = @shell_exec("node --check $tmp 2>&1");
|
||||
if (trim($parse)) {
|
||||
echo "=== Script #$n (starts abs line $abs_line, " . strlen($content) . "B) ===\n";
|
||||
echo "$parse\n\n";
|
||||
} else {
|
||||
echo "Script #$n (line $abs_line, " . strlen($content) . "B): OK\n";
|
||||
}
|
||||
$pos = $end + 9;
|
||||
}
|
||||
54
api/ambre-pdf-memory.php
Normal file
@@ -0,0 +1,54 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$out = [];
|
||||
|
||||
// 1. Search wikis about "pdf premium" or "pdf graphique"
|
||||
$wiki_hits = [];
|
||||
foreach (glob("/opt/obsidian-vault/**/*.md") as $f) {
|
||||
$content = @file_get_contents($f);
|
||||
if (!$content) continue;
|
||||
if (stripos($content, "pdf premium") !== false ||
|
||||
stripos($content, "weasyprint") !== false ||
|
||||
stripos($content, "pdf graphique") !== false ||
|
||||
stripos($content, "reportlab") !== false ||
|
||||
stripos($content, "pdf chart") !== false ||
|
||||
stripos($content, "pdfmake") !== false) {
|
||||
$wiki_hits[] = [
|
||||
"file" => str_replace("/opt/obsidian-vault/", "", $f),
|
||||
"size" => filesize($f),
|
||||
];
|
||||
}
|
||||
}
|
||||
$out["wiki_hits_pdf"] = array_slice($wiki_hits, 0, 15);
|
||||
|
||||
// 2. Existing PDF endpoints
|
||||
$out["pdf_endpoints"] = [];
|
||||
foreach (glob("/var/www/html/api/*pdf*.php") as $f) {
|
||||
$out["pdf_endpoints"][] = [
|
||||
"name" => basename($f),
|
||||
"size" => filesize($f),
|
||||
"mtime" => date("Y-m-d H:i", filemtime($f)),
|
||||
];
|
||||
}
|
||||
|
||||
// 3. PDF generation binaries available
|
||||
$out["pdf_tools"] = [
|
||||
"wkhtmltopdf" => trim(@shell_exec("which wkhtmltopdf") ?: "NO"),
|
||||
"weasyprint" => trim(@shell_exec("which weasyprint") ?: "NO"),
|
||||
"pandoc" => trim(@shell_exec("which pandoc") ?: "NO"),
|
||||
"chromium" => trim(@shell_exec("which chromium chromium-browser google-chrome 2>&1 | head -1") ?: "NO"),
|
||||
"reportlab" => trim(@shell_exec("python3 -c 'import reportlab; print(reportlab.__version__)' 2>&1") ?: "NO"),
|
||||
"matplotlib" => trim(@shell_exec("python3 -c 'import matplotlib; print(matplotlib.__version__)' 2>&1") ?: "NO"),
|
||||
"plotly" => trim(@shell_exec("python3 -c 'import plotly; print(plotly.__version__)' 2>&1") ?: "NO"),
|
||||
];
|
||||
|
||||
// 4. Earlier wiki sessions about PDF
|
||||
$recent_sessions = [];
|
||||
foreach (glob("/opt/obsidian-vault/sessions/*.md") as $f) {
|
||||
$base = basename($f);
|
||||
$recent_sessions[] = ["name"=>$base, "mtime"=>date("Y-m-d", filemtime($f))];
|
||||
}
|
||||
usort($recent_sessions, function($a,$b){return strcmp($b["mtime"], $a["mtime"]);});
|
||||
$out["recent_sessions"] = array_slice($recent_sessions, 0, 10);
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
4
api/ambre-pw-check.php
Normal file
@@ -0,0 +1,4 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$proc = @shell_exec("pgrep -af "playwright test" | head -5");
|
||||
echo json_encode(["running"=>trim($proc ?: "none")]);
|
||||
@@ -1,73 +1,6 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$base = "/var/www/html/api/ambre-pw-tests";
|
||||
$spec = <<<'JS'
|
||||
const { test, expect } = require("@playwright/test");
|
||||
|
||||
test("debug router click flow", async ({ page }) => {
|
||||
test.setTimeout(120000);
|
||||
|
||||
// Capture console logs
|
||||
const logs = [];
|
||||
page.on("console", msg => {
|
||||
logs.push(`[${msg.type()}] ${msg.text().substring(0, 300)}`);
|
||||
});
|
||||
|
||||
// Inject diagnostic script
|
||||
await page.addInitScript(() => {
|
||||
window.__ambre_fetch_calls = [];
|
||||
const origFetch = window.fetch;
|
||||
window.fetch = function(...args) {
|
||||
window.__ambre_fetch_calls.push({url: args[0], body: args[1] && args[1].body ? args[1].body.toString().substring(0, 200) : null});
|
||||
return origFetch.apply(this, args);
|
||||
};
|
||||
});
|
||||
|
||||
await page.goto("/wevia.html");
|
||||
await page.waitForLoadState("networkidle");
|
||||
await page.waitForTimeout(1500);
|
||||
|
||||
// Send PDF
|
||||
const input = page.locator("#msgInput");
|
||||
await input.fill("Genere un PDF sur: test debug");
|
||||
await input.press("Enter");
|
||||
await page.waitForTimeout(8000);
|
||||
|
||||
// Dump state
|
||||
const state = await page.evaluate(() => ({
|
||||
busy: typeof busy !== "undefined" ? busy : "undefined",
|
||||
pendingFile: typeof pendingFile !== "undefined" ? pendingFile : "undefined",
|
||||
fetch_calls: window.__ambre_fetch_calls || [],
|
||||
msg_input_disabled: document.getElementById("msgInput").disabled,
|
||||
msg_input_value: document.getElementById("msgInput").value,
|
||||
assistant_count: document.querySelectorAll(".msg.assistant").length,
|
||||
user_count: document.querySelectorAll(".msg.user").length,
|
||||
has_router: !!window._ambre_gen_pat,
|
||||
}));
|
||||
|
||||
console.log("=== STATE AFTER PDF SEND ===");
|
||||
console.log(JSON.stringify(state, null, 2));
|
||||
|
||||
// Now try 2nd message
|
||||
await input.fill("Genere un document Word sur: test2");
|
||||
await input.press("Enter");
|
||||
await page.waitForTimeout(8000);
|
||||
|
||||
const state2 = await page.evaluate(() => ({
|
||||
busy: typeof busy !== "undefined" ? busy : "undefined",
|
||||
fetch_calls_count: (window.__ambre_fetch_calls || []).length,
|
||||
last_fetch: (window.__ambre_fetch_calls || []).slice(-3),
|
||||
assistant_count: document.querySelectorAll(".msg.assistant").length,
|
||||
user_count: document.querySelectorAll(".msg.user").length,
|
||||
}));
|
||||
console.log("=== STATE AFTER WORD SEND ===");
|
||||
console.log(JSON.stringify(state2, null, 2));
|
||||
|
||||
// Write logs to file
|
||||
require("fs").writeFileSync("output/debug-console.log", logs.join("\n"));
|
||||
});
|
||||
JS;
|
||||
file_put_contents("$base/tests/debug-flow.spec.js", $spec);
|
||||
// Remove v5 to not rerun
|
||||
@unlink("$base/tests/chat-capabilities-v5.spec.js");
|
||||
echo json_encode(["ok"=>true, "size"=>filesize("$base/tests/debug-flow.spec.js")]);
|
||||
$base = "/var/www/html/api/ambre-pw-tests/tests";
|
||||
@unlink("$base/conversation-v12.spec.js");
|
||||
$written = @file_put_contents("$base/debug-trace.spec.js", base64_decode("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMTItZGVidWcgwrcgdHJhY2Ugd2hhdCBoYXBwZW5zIG9uIHNpbXBsZSBtZXNzYWdlIiwgYXN5bmMgKHsgcGFnZSB9KSA9PiB7CiAgdGVzdC5zZXRUaW1lb3V0KDYwMDAwKTsKICAKICBjb25zdCBsb2dzID0gW107CiAgY29uc3QgZXJyb3JzID0gW107CiAgY29uc3QgbmV0d29ya3MgPSBbXTsKICAKICBwYWdlLm9uKCJjb25zb2xlIiwgbSA9PiBsb2dzLnB1c2goYFske20udHlwZSgpfV0gJHttLnRleHQoKS5zdWJzdHJpbmcoMCwzMDApfWApKTsKICBwYWdlLm9uKCJwYWdlZXJyb3IiLCBlID0+IGVycm9ycy5wdXNoKGUubWVzc2FnZS5zdWJzdHJpbmcoMCwzMDApKSk7CiAgcGFnZS5vbigicmVxdWVzdCIsIHIgPT4gewogICAgaWYgKHIudXJsKCkuaW5jbHVkZXMoImFtYnJlIikgfHwgci51cmwoKS5pbmNsdWRlcygic292ZXJlaWduIikgfHwgci51cmwoKS5pbmNsdWRlcygibWFzdGVyIikpIHsKICAgICAgbmV0d29ya3MucHVzaChg4oaSICR7ci5tZXRob2QoKX0gJHtyLnVybCgpLnN1YnN0cmluZygwLDEyMCl9YCk7CiAgICB9CiAgfSk7CiAgcGFnZS5vbigicmVzcG9uc2UiLCByID0+IHsKICAgIGlmIChyLnVybCgpLmluY2x1ZGVzKCJhbWJyZSIpIHx8IHIudXJsKCkuaW5jbHVkZXMoInNvdmVyZWlnbiIpIHx8IHIudXJsKCkuaW5jbHVkZXMoIm1hc3RlciIpKSB7CiAgICAgIG5ldHdvcmtzLnB1c2goYOKGkCAke3Iuc3RhdHVzKCl9ICR7ci51cmwoKS5zdWJzdHJpbmcoMCwxMjApfWApOwogICAgfQogIH0pOwogIAogIGF3YWl0IHBhZ2UuZ290bygiL3dldmlhLmh0bWwiKTsKICBhd2FpdCBwYWdlLndhaXRGb3JMb2FkU3RhdGUoIm5ldHdvcmtpZGxlIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgyMDAwKTsKICAKICAvLyBMb2cgZ2xvYmFsIHN0YXRlCiAgY29uc3Qgc3RhdGUxID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiAoewogICAgaGFzVjVNZW1vcnlWMjogZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50Lm91dGVySFRNTC5pbmNsdWRlcygiQU1CUkUtVjUtTUVNT1JZLXYyIiksCiAgICBoYXNTZW5kTXNnOiB0eXBlb2Ygd2luZG93LnNlbmQgPT09ICJmdW5jdGlvbiIsCiAgICBidXN5OiB0eXBlb2YgYnVzeSAhPT0gInVuZGVmaW5lZCIgPyBidXN5IDogInVuZGVmaW5lZCIsCiAgfSkpOwogIGNvbnNvbGUubG9nKCJTVEFURSBBVCBMT0FEOiIsIEpTT04uc3RyaW5naWZ5KHN0YXRlMSkpOwogIAogIC8vIFNlbmQgYSBtZXNzYWdlCiAgY29uc3QgaW5wdXQgPSBwYWdlLmxvY2F0b3IoIiNtc2dJbnB1dCIpOwogIGF3YWl0IGlucHV0LmZpbGwoImJvbmpvdXIgamUgc3VpcyBZYWNpbmUgY29tbWVudCBjYSB2YSBhdWpvdXJkIGh1aSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoNTAwKTsKICBhd2FpdCBpbnB1dC5wcmVzcygiRW50ZXIiKTsKICBjb25zb2xlLmxvZygiTUVTU0FHRSBTRU5UIik7CiAgCiAgLy8gV2FpdCBhbmQgY2FwdHVyZSB3aGF0IGhhcHBlbnMKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDE1MDAwKTsKICAKICBjb25zdCBzdGF0ZTIgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+ICh7CiAgICBhc3Npc3RhbnRfY291bnQ6IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3JBbGwoIi5tc2cuYXNzaXN0YW50IikubGVuZ3RoLAogICAgbGFzdF9hc3Npc3RhbnQ6IEFycmF5LmZyb20oZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgiLm1zZy5hc3Npc3RhbnQiKSkuc2xpY2UoLTEpWzBdPy5pbm5lclRleHQ/LnN1YnN0cmluZygwLDMwMCksCiAgICBidXN5OiB0eXBlb2YgYnVzeSAhPT0gInVuZGVmaW5lZCIgPyBidXN5IDogInVuZGVmaW5lZCIsCiAgICBzZXNzaW9uX2lkOiB3aW5kb3cuX2FtYnJlX3Nlc3Npb25faWQsCiAgfSkpOwogIGNvbnNvbGUubG9nKCJTVEFURSAxNXMgbGF0ZXI6IiwgSlNPTi5zdHJpbmdpZnkoc3RhdGUyKSk7CiAgCiAgY29uc29sZS5sb2coIlxuPT09IE5FVFdPUksgPT09Iik7CiAgbmV0d29ya3MuZm9yRWFjaChuID0+IGNvbnNvbGUubG9nKG4pKTsKICBjb25zb2xlLmxvZygiXG49PT0gQ09OU09MRSBMT0dTID09PSIpOwogIGxvZ3Muc2xpY2UoMCwgMzApLmZvckVhY2gobCA9PiBjb25zb2xlLmxvZyhsKSk7CiAgY29uc29sZS5sb2coIlxuPT09IFBBR0UgRVJST1JTID09PSIpOwogIGVycm9ycy5mb3JFYWNoKGUgPT4gY29uc29sZS5sb2coZSkpOwp9KTsK"));
|
||||
echo json_encode(["written"=>$written]);
|
||||
|
||||
6
api/ambre-pw-debug2.php
Normal file
@@ -0,0 +1,6 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$base = "/var/www/html/api/ambre-pw-tests/tests";
|
||||
@unlink("$base/debug-trace.spec.js");
|
||||
$written = @file_put_contents("$base/debug2.spec.js", base64_decode("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMTItZGVidWcyIMK3IGNhcHR1cmUgdGhlIGV4YWN0IGVycm9yIHNvdXJjZSIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCg2MDAwMCk7CiAgCiAgY29uc3QgZXJyb3JzID0gW107CiAgY29uc3Qgd2FybmluZ3MgPSBbXTsKICBjb25zdCBzY3JpcHRFcnJvcnMgPSBbXTsKICAKICBwYWdlLm9uKCJjb25zb2xlIiwgbSA9PiB7CiAgICBjb25zdCB0ID0gbS50ZXh0KCk7CiAgICBpZiAobS50eXBlKCkgPT09ICJ3YXJuaW5nIikgd2FybmluZ3MucHVzaCh0LnN1YnN0cmluZygwLDQwMCkpOwogICAgaWYgKG0udHlwZSgpID09PSAiZXJyb3IiKSBlcnJvcnMucHVzaCh0LnN1YnN0cmluZygwLDQwMCkpOwogIH0pOwogIHBhZ2Uub24oInBhZ2VlcnJvciIsIGUgPT4gewogICAgc2NyaXB0RXJyb3JzLnB1c2goYCR7ZS5uYW1lfTogJHtlLm1lc3NhZ2V9XG4keyhlLnN0YWNrfHwnJykuc3Vic3RyaW5nKDAsNTAwKX1gKTsKICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yTG9hZFN0YXRlKCJuZXR3b3JraWRsZSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMzAwMCk7CiAgCiAgY29uc29sZS5sb2coIj09PSBTQ1JJUFQgRVJST1JTID09PSIpOwogIHNjcmlwdEVycm9ycy5mb3JFYWNoKGUgPT4gY29uc29sZS5sb2coZSkpOwogIGNvbnNvbGUubG9nKCJcbj09PSBXQVJOSU5HUyA9PT0iKTsKICB3YXJuaW5ncy5zbGljZSgwLDEwKS5mb3JFYWNoKHcgPT4gY29uc29sZS5sb2codykpOwogIGNvbnNvbGUubG9nKCJcbj09PSBDT05TT0xFIEVSUk9SUyA9PT0iKTsKICBlcnJvcnMuc2xpY2UoMCwxMCkuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKGUpKTsKICAKICAvLyBOb3cgdHJ5IHRvIHNlbmQgYSBtZXNzYWdlCiAgY29uc3QgaW5wdXQgPSBwYWdlLmxvY2F0b3IoIiNtc2dJbnB1dCIpOwogIGF3YWl0IGlucHV0LmZpbGwoInNhbHV0IFlhY2luZSBpY2kgY29tbWVudCBjYSB2YSIpOwogIGF3YWl0IGlucHV0LnByZXNzKCJFbnRlciIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMTIwMDApOwogIAogIGNvbnNvbGUubG9nKCJcbj09PSBBRlRFUiBTRU5EIEVSUk9SUyA9PT0iKTsKICBzY3JpcHRFcnJvcnMuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKGUuc3Vic3RyaW5nKDAsNDAwKSkpOwogIGNvbnNvbGUubG9nKCJcbj09PSBBRlRFUiBTRU5EIFdBUk5JTkdTID09PSIpOwogIHdhcm5pbmdzLmZvckVhY2godyA9PiBjb25zb2xlLmxvZyh3LnN1YnN0cmluZygwLDQwMCkpKTsKICAKICBjb25zdCBsYXN0ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7CiAgICBjb25zdCBtc2dzID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgiLm1zZy5hc3Npc3RhbnQiKTsKICAgIHJldHVybiBtc2dzLmxlbmd0aCA+IDAgPyBtc2dzW21zZ3MubGVuZ3RoLTFdLmlubmVyVGV4dC5zdWJzdHJpbmcoMCwyNTApIDogIm5vIG1zZyI7CiAgfSk7CiAgY29uc29sZS5sb2coIlxuTGFzdCBhc3Npc3RhbnQgbXNnOiIsIGxhc3QpOwp9KTsK"));
|
||||
echo json_encode(["written"=>$written]);
|
||||
@@ -1,10 +1,6 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
// Kill running playwright processes
|
||||
$killed = @shell_exec("pkill -f playwright 2>&1");
|
||||
sleep(1);
|
||||
$still = @shell_exec("ps aux | grep playwright | grep -v grep | wc -l");
|
||||
echo json_encode([
|
||||
"killed_output" => trim($killed ?: ""),
|
||||
"processes_remaining" => trim($still ?: ""),
|
||||
], JSON_PRETTY_PRINT);
|
||||
header("Content-Type: text/plain");
|
||||
@shell_exec("pkill -f playwright 2>&1");
|
||||
@shell_exec("pkill -f v17-6sigma 2>&1");
|
||||
echo "killed\n";
|
||||
echo @shell_exec("pgrep -af playwright");
|
||||
|
||||
18
api/ambre-pw-readlog-latest.php
Normal file
@@ -0,0 +1,18 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
// Get the debug2 log specifically
|
||||
$log = "/tmp/ambre-pw-run-20260421-215101.log";
|
||||
if (file_exists($log)) {
|
||||
echo "=== $log ===\n";
|
||||
echo "Size: " . filesize($log) . "B\n\n";
|
||||
echo @file_get_contents($log);
|
||||
} else {
|
||||
// Get latest log
|
||||
$logs = glob("/tmp/ambre-pw-run-*.log");
|
||||
usort($logs, function($a,$b){return filemtime($b)-filemtime($a);});
|
||||
if ($logs) {
|
||||
echo "=== " . basename($logs[0]) . " ===\n";
|
||||
echo "Size: " . filesize($logs[0]) . "B\n\n";
|
||||
echo @file_get_contents($logs[0]);
|
||||
}
|
||||
}
|
||||
@@ -1,4 +0,0 @@
|
||||
{
|
||||
"status": "passed",
|
||||
"failedTests": []
|
||||
}
|
||||
|
Before Width: | Height: | Size: 117 KiB |
|
Before Width: | Height: | Size: 505 KiB |
@@ -1,367 +0,0 @@
|
||||
{
|
||||
"config": {
|
||||
"configFile": "/var/www/html/api/ambre-pw-tests/playwright.config.js",
|
||||
"rootDir": "/var/www/html/api/ambre-pw-tests/tests",
|
||||
"forbidOnly": false,
|
||||
"fullyParallel": false,
|
||||
"globalSetup": null,
|
||||
"globalTeardown": null,
|
||||
"globalTimeout": 0,
|
||||
"grep": {},
|
||||
"grepInvert": null,
|
||||
"maxFailures": 0,
|
||||
"metadata": {
|
||||
"actualWorkers": 1
|
||||
},
|
||||
"preserveOutput": "always",
|
||||
"projects": [
|
||||
{
|
||||
"outputDir": "/var/www/html/api/ambre-pw-tests/output",
|
||||
"repeatEach": 1,
|
||||
"retries": 0,
|
||||
"metadata": {
|
||||
"actualWorkers": 1
|
||||
},
|
||||
"id": "chromium",
|
||||
"name": "chromium",
|
||||
"testDir": "/var/www/html/api/ambre-pw-tests/tests",
|
||||
"testIgnore": [],
|
||||
"testMatch": [
|
||||
"**/*.@(spec|test).?(c|m)[jt]s?(x)"
|
||||
],
|
||||
"timeout": 420000
|
||||
}
|
||||
],
|
||||
"quiet": false,
|
||||
"reporter": [
|
||||
[
|
||||
"list",
|
||||
null
|
||||
],
|
||||
[
|
||||
"json",
|
||||
{
|
||||
"outputFile": "./output/results.json"
|
||||
}
|
||||
]
|
||||
],
|
||||
"reportSlowTests": {
|
||||
"max": 5,
|
||||
"threshold": 300000
|
||||
},
|
||||
"shard": null,
|
||||
"tags": [],
|
||||
"updateSnapshots": "missing",
|
||||
"updateSourceMethod": "patch",
|
||||
"version": "1.59.1",
|
||||
"workers": 1,
|
||||
"webServer": null
|
||||
},
|
||||
"suites": [
|
||||
{
|
||||
"title": "capabilities-v11.spec.js",
|
||||
"file": "capabilities-v11.spec.js",
|
||||
"column": 0,
|
||||
"line": 0,
|
||||
"specs": [
|
||||
{
|
||||
"title": "V11 Claude-pattern streaming · full video experience",
|
||||
"ok": true,
|
||||
"tags": [],
|
||||
"tests": [
|
||||
{
|
||||
"timeout": 360000,
|
||||
"annotations": [],
|
||||
"expectedStatus": "passed",
|
||||
"projectId": "chromium",
|
||||
"projectName": "chromium",
|
||||
"results": [
|
||||
{
|
||||
"workerIndex": 0,
|
||||
"parallelIndex": 0,
|
||||
"status": "passed",
|
||||
"duration": 114343,
|
||||
"errors": [],
|
||||
"stdout": [
|
||||
{
|
||||
"text": "📸 initial\n"
|
||||
},
|
||||
{
|
||||
"text": "\n═══ [01/4] PDF ═══\n"
|
||||
},
|
||||
{
|
||||
"text": " 📤 sent: Genere un PDF sur: strategie WEVIA 2026\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 thinking snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 plan snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 final snap (full page)\n"
|
||||
},
|
||||
{
|
||||
"text": " phases detected: null\n"
|
||||
},
|
||||
{
|
||||
"text": "\n═══ [02/4] Mermaid ═══\n"
|
||||
},
|
||||
{
|
||||
"text": " 📤 sent: Genere un schema mermaid pour: workflow ventes\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 thinking snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 plan snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 final snap (full page)\n"
|
||||
},
|
||||
{
|
||||
"text": " phases detected: null\n"
|
||||
},
|
||||
{
|
||||
"text": "\n═══ [03/4] Code ═══\n"
|
||||
},
|
||||
{
|
||||
"text": " 📤 sent: Ecris le code python pour: fibonacci recursif\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 thinking snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 plan snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 final snap (full page)\n"
|
||||
},
|
||||
{
|
||||
"text": " phases detected: null\n"
|
||||
},
|
||||
{
|
||||
"text": "\n═══ [04/4] Image ═══\n"
|
||||
},
|
||||
{
|
||||
"text": " 📤 sent: Genere une image: oasis palmiers\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 thinking snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 plan snap\n"
|
||||
},
|
||||
{
|
||||
"text": " 📸 final snap (full page)\n"
|
||||
},
|
||||
{
|
||||
"text": " phases detected: null\n"
|
||||
},
|
||||
{
|
||||
"text": "\n✅ V11 done\n"
|
||||
}
|
||||
],
|
||||
"stderr": [],
|
||||
"retry": 0,
|
||||
"startTime": "2026-04-21T21:22:50.627Z",
|
||||
"annotations": [],
|
||||
"attachments": [
|
||||
{
|
||||
"name": "screenshot",
|
||||
"contentType": "image/png",
|
||||
"path": "/var/www/html/api/ambre-pw-tests/output/capabilities-v11-V11-Claud-61813-ing-·-full-video-experience-chromium/test-finished-1.png"
|
||||
}
|
||||
]
|
||||
}
|
||||
],
|
||||
"status": "expected"
|
||||
}
|
||||
],
|
||||
"id": "891ac0df5a7278e0a023-26e2434937214a7fb59a",
|
||||
"file": "capabilities-v11.spec.js",
|
||||
"line": 3,
|
||||
"column": 1
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
"title": "claude-pattern-v11.spec.js",
|
||||
"file": "claude-pattern-v11.spec.js",
|
||||
"column": 0,
|
||||
"line": 0,
|
||||
"specs": [
|
||||
{
|
||||
"title": "V11 · Claude Pattern Dashboard · SSE streaming long · full video",
|
||||
"ok": true,
|
||||
"tags": [],
|
||||
"tests": [
|
||||
{
|
||||
"timeout": 600000,
|
||||
"annotations": [],
|
||||
"expectedStatus": "passed",
|
||||
"projectId": "chromium",
|
||||
"projectName": "chromium",
|
||||
"results": [
|
||||
{
|
||||
"workerIndex": 0,
|
||||
"parallelIndex": 0,
|
||||
"status": "passed",
|
||||
"duration": 82198,
|
||||
"errors": [],
|
||||
"stdout": [
|
||||
{
|
||||
"text": "📸 Landing page captured\n"
|
||||
},
|
||||
{
|
||||
"text": "📸 Query filled\n"
|
||||
},
|
||||
{
|
||||
"text": "🚀 Launched SSE stream\n"
|
||||
},
|
||||
{
|
||||
"text": " t=2.0s · doneDots=0/7 · chips=0 · bodyLen=165\n"
|
||||
},
|
||||
{
|
||||
"text": " t=4.4s · doneDots=0/7 · chips=0 · bodyLen=165\n"
|
||||
},
|
||||
{
|
||||
"text": " t=6.4s · doneDots=0/7 · chips=0 · bodyLen=165\n"
|
||||
},
|
||||
{
|
||||
"text": " t=8.4s · doneDots=0/7 · chips=0 · bodyLen=165\n"
|
||||
},
|
||||
{
|
||||
"text": " t=10.9s · doneDots=0/7 · chips=0 · bodyLen=283\n"
|
||||
},
|
||||
{
|
||||
"text": " t=13.0s · doneDots=0/7 · chips=0 · bodyLen=591\n"
|
||||
},
|
||||
{
|
||||
"text": " t=15.0s · doneDots=0/7 · chips=0 · bodyLen=918\n"
|
||||
},
|
||||
{
|
||||
"text": " t=20.2s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=22.3s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=24.3s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=29.2s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=31.2s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=33.2s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=35.5s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=37.5s · doneDots=5/7 · chips=8 · bodyLen=1540\n"
|
||||
},
|
||||
{
|
||||
"text": " t=39.6s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=42.1s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=44.1s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=46.2s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=49.7s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=51.7s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=53.7s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=56.2s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=58.2s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=60.3s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=62.8s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=64.8s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=66.8s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=69.2s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": " t=71.2s · doneDots=5/7 · chips=8 · bodyLen=1539\n"
|
||||
},
|
||||
{
|
||||
"text": "📸 Complete dashboard captured\n"
|
||||
},
|
||||
{
|
||||
"text": "\n═══════════════ V11 BILAN ═══════════════\n"
|
||||
},
|
||||
{
|
||||
"text": "✅ 5/7 phases streamées via SSE\n"
|
||||
},
|
||||
{
|
||||
"text": "✅ 8 badges success\n"
|
||||
},
|
||||
{
|
||||
"text": "✅ Confidence: null\n"
|
||||
},
|
||||
{
|
||||
"text": "✅ Result length: 0 chars\n"
|
||||
},
|
||||
{
|
||||
"text": "Result preview: \n"
|
||||
}
|
||||
],
|
||||
"stderr": [],
|
||||
"retry": 0,
|
||||
"startTime": "2026-04-21T21:24:45.068Z",
|
||||
"annotations": [],
|
||||
"attachments": [
|
||||
{
|
||||
"name": "screenshot",
|
||||
"contentType": "image/png",
|
||||
"path": "/var/www/html/api/ambre-pw-tests/output/claude-pattern-v11-V11-·-C-bd800-streaming-long-·-full-video-chromium/test-finished-1.png"
|
||||
}
|
||||
]
|
||||
}
|
||||
],
|
||||
"status": "expected"
|
||||
}
|
||||
],
|
||||
"id": "995711e79a56a7f87945-b11bf97faa12763b16b5",
|
||||
"file": "claude-pattern-v11.spec.js",
|
||||
"line": 3,
|
||||
"column": 1
|
||||
}
|
||||
]
|
||||
}
|
||||
],
|
||||
"errors": [],
|
||||
"stats": {
|
||||
"startTime": "2026-04-21T21:22:50.024Z",
|
||||
"duration": 197328.535,
|
||||
"expected": 2,
|
||||
"skipped": 0,
|
||||
"unexpected": 0,
|
||||
"flaky": 0
|
||||
}
|
||||
}
|
||||
|
Before Width: | Height: | Size: 568 KiB |
|
Before Width: | Height: | Size: 575 KiB |
|
Before Width: | Height: | Size: 625 KiB |
|
Before Width: | Height: | Size: 501 KiB |
|
Before Width: | Height: | Size: 505 KiB |
|
Before Width: | Height: | Size: 505 KiB |
|
Before Width: | Height: | Size: 505 KiB |
|
Before Width: | Height: | Size: 505 KiB |
|
Before Width: | Height: | Size: 505 KiB |
|
Before Width: | Height: | Size: 625 KiB |
|
Before Width: | Height: | Size: 616 KiB |
|
Before Width: | Height: | Size: 501 KiB |
|
Before Width: | Height: | Size: 511 KiB |
|
Before Width: | Height: | Size: 117 KiB |
|
Before Width: | Height: | Size: 505 KiB |
|
Before Width: | Height: | Size: 586 KiB |
|
Before Width: | Height: | Size: 117 KiB |
@@ -1,96 +0,0 @@
|
||||
const { test } = require("@playwright/test");
|
||||
|
||||
test("V11 · Claude Pattern Dashboard · SSE streaming long · full video", async ({ page }) => {
|
||||
test.setTimeout(600000);
|
||||
|
||||
await page.goto("/wevia-claude-pattern.html");
|
||||
await page.waitForLoadState("networkidle");
|
||||
await page.waitForTimeout(2500);
|
||||
await page.screenshot({ path: "output/v11-00-landing.png" });
|
||||
console.log("📸 Landing page captured");
|
||||
|
||||
// Fill query
|
||||
const query = "Comment WEVIA peut orchestrer une strategie pharma multicanal MENA avec agents autonomes et RAG Qdrant ?";
|
||||
await page.locator("input.input").fill(query);
|
||||
await page.waitForTimeout(500);
|
||||
await page.screenshot({ path: "output/v11-01-query-filled.png" });
|
||||
console.log("📸 Query filled");
|
||||
|
||||
// Click launch
|
||||
await page.locator("button.btn").click();
|
||||
console.log("🚀 Launched SSE stream");
|
||||
|
||||
// Capture phases in real-time
|
||||
const phases = ["thinking", "plan", "rag", "execute", "test", "result", "critique"];
|
||||
const captured = {};
|
||||
|
||||
// Wait for each phase progressively
|
||||
const startTime = Date.now();
|
||||
for (let i = 0; i < 30; i++) { // max 30s × 2s wait
|
||||
await page.waitForTimeout(2000);
|
||||
const state = await page.evaluate(() => {
|
||||
const chips = document.querySelectorAll(".chip-success");
|
||||
const doneDots = document.querySelectorAll(".phase-dot.done");
|
||||
const h3s = Array.from(document.querySelectorAll("h3")).map(h=>h.innerText);
|
||||
const result_card = document.querySelector(".card[style*='borderColor']");
|
||||
return {
|
||||
doneDots: doneDots.length,
|
||||
successChips: chips.length,
|
||||
headers: h3s,
|
||||
hasResult: !!result_card,
|
||||
bodyLen: document.body.innerText.length,
|
||||
};
|
||||
});
|
||||
const elapsed = ((Date.now() - startTime) / 1000).toFixed(1);
|
||||
console.log(` t=${elapsed}s · doneDots=${state.doneDots}/7 · chips=${state.successChips} · bodyLen=${state.bodyLen}`);
|
||||
|
||||
// Capture progressive screenshot
|
||||
if (i % 3 === 0) {
|
||||
await page.screenshot({ path: `output/v11-02-progress-${i}.png` });
|
||||
}
|
||||
|
||||
if (state.doneDots >= 7 || state.hasResult) {
|
||||
console.log(` ✅ All 7 phases done at t=${elapsed}s`);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Wait a final moment for critique to render
|
||||
await page.waitForTimeout(2500);
|
||||
|
||||
// Full page final screenshot
|
||||
await page.screenshot({ path: "output/v11-03-complete.png", fullPage: true });
|
||||
console.log("📸 Complete dashboard captured");
|
||||
|
||||
// Scroll to top and capture header area
|
||||
await page.evaluate(() => window.scrollTo(0, 0));
|
||||
await page.waitForTimeout(800);
|
||||
await page.screenshot({ path: "output/v11-04-top.png" });
|
||||
|
||||
// Capture the critique card specifically
|
||||
const critiqueCard = page.locator(".card").filter({ hasText: "Self-critique" });
|
||||
if (await critiqueCard.count() > 0) {
|
||||
await critiqueCard.first().scrollIntoViewIfNeeded();
|
||||
await page.waitForTimeout(800);
|
||||
await page.screenshot({ path: "output/v11-05-critique.png" });
|
||||
console.log("📸 Critique card captured");
|
||||
}
|
||||
|
||||
// Final metrics
|
||||
const metrics = await page.evaluate(() => ({
|
||||
phases_done: document.querySelectorAll(".phase-dot.done").length,
|
||||
total_chips: document.querySelectorAll(".chip-success").length,
|
||||
result_text: (document.querySelector(".card[style*='rgba(16,185,129']") || {}).innerText || "",
|
||||
confidence: (() => {
|
||||
const t = Array.from(document.querySelectorAll(".text-3xl")).find(e => e.innerText.includes("%"));
|
||||
return t ? t.innerText : null;
|
||||
})(),
|
||||
}));
|
||||
|
||||
console.log("\n═══════════════ V11 BILAN ═══════════════");
|
||||
console.log(`✅ ${metrics.phases_done}/7 phases streamées via SSE`);
|
||||
console.log(`✅ ${metrics.total_chips} badges success`);
|
||||
console.log(`✅ Confidence: ${metrics.confidence}`);
|
||||
console.log(`✅ Result length: ${metrics.result_text.length} chars`);
|
||||
console.log(`Result preview: ${metrics.result_text.substring(0, 200)}`);
|
||||
});
|
||||
56
api/ambre-pw-tests/tests/v28-fetch-log.spec.js
Normal file
@@ -0,0 +1,56 @@
|
||||
const { test } = require("@playwright/test");
|
||||
|
||||
test("V28 · wrap fetch and send HI", async ({ page }) => {
|
||||
test.setTimeout(60000);
|
||||
|
||||
const netlog = [];
|
||||
const errs = [];
|
||||
page.on("pageerror", e => errs.push(e.message));
|
||||
page.on("response", res => {
|
||||
if (res.url().includes("/api/")) {
|
||||
netlog.push({ url: res.url().split("?")[0], status: res.status() });
|
||||
}
|
||||
});
|
||||
|
||||
await page.goto("/wevia.html");
|
||||
await page.waitForTimeout(2500);
|
||||
|
||||
// Monkey-patch fetch to see all calls
|
||||
await page.evaluate(() => {
|
||||
window._fetchLog = [];
|
||||
const orig = window.fetch;
|
||||
window.fetch = function(u, opts) {
|
||||
const url = typeof u === "string" ? u : u.url;
|
||||
window._fetchLog.push({ url: url.split("?")[0], method: (opts && opts.method) || "GET" });
|
||||
return orig.apply(this, arguments);
|
||||
};
|
||||
});
|
||||
|
||||
// Send "HI"
|
||||
await page.fill("#msgInput", "HI");
|
||||
await page.waitForTimeout(300);
|
||||
await page.press("#msgInput", "Enter");
|
||||
|
||||
await page.waitForTimeout(15000);
|
||||
|
||||
const fetchLog = await page.evaluate(() => window._fetchLog);
|
||||
console.log("\n=== fetch calls from JS ===");
|
||||
console.log(JSON.stringify(fetchLog, null, 2));
|
||||
|
||||
console.log("\n=== Network log (via Playwright) ===");
|
||||
console.log(JSON.stringify(netlog, null, 2));
|
||||
|
||||
console.log("\n=== Page errors ===");
|
||||
errs.forEach(e => console.log(" ", e.substring(0, 200)));
|
||||
|
||||
// DOM state
|
||||
const domState = await page.evaluate(() => {
|
||||
const a = document.querySelectorAll(".msg.assistant .bubble");
|
||||
return {
|
||||
count: a.length,
|
||||
messages: Array.from(a).map(el => el.innerText.substring(0, 200)),
|
||||
};
|
||||
});
|
||||
console.log("\n=== DOM messages ===");
|
||||
console.log(JSON.stringify(domState, null, 2));
|
||||
});
|
||||
10
api/ambre-pw-v12-deploy.php
Normal file
7
api/ambre-pw-v13-deploy.php
Normal file
23
api/ambre-pw-v13-files.php
Normal file
@@ -0,0 +1,23 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$base = "/var/www/html/api/ambre-pw-tests/output";
|
||||
$out = ["v13_screenshots"=>[], "v13_video"=>null];
|
||||
|
||||
foreach (glob("$base/v13-*.png") as $p) {
|
||||
$out["v13_screenshots"][] = [
|
||||
"name" => basename($p),
|
||||
"kb" => round(filesize($p)/1024, 1),
|
||||
"url" => "https://weval-consulting.com/api/ambre-pw-tests/output/" . basename($p),
|
||||
];
|
||||
}
|
||||
usort($out["v13_screenshots"], function($a,$b){return strcmp($a["name"],$b["name"]);});
|
||||
|
||||
foreach (glob("$base/long-conversation-v13*/*.webm") as $w) {
|
||||
$rel = str_replace($base."/", "", $w);
|
||||
$out["v13_video"] = [
|
||||
"size_mb" => round(filesize($w)/1048576, 2),
|
||||
"url" => "https://weval-consulting.com/api/ambre-pw-tests/output/" . $rel,
|
||||
];
|
||||
}
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
7
api/ambre-pw-v14-deploy.php
Normal file
7
api/ambre-pw-v15-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMTUgZml4IHZlcmlmaWNhdGlvbiDCtyBRUiArIEhEICsgVFRTIHNpbmdsZSBhdHRlbXB0cyIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCgxODAwMDApOwogIAogIHBhZ2Uub24oInBhZ2VlcnJvciIsIGUgPT4gY29uc29sZS5sb2coIltwYWdlZXJyb3JdIiwgZS5tZXNzYWdlLnN1YnN0cmluZygwLCAyMDApKSk7CiAgcGFnZS5vbigiY29uc29sZSIsIG1zZyA9PiB7CiAgICBpZiAobXNnLnR5cGUoKSA9PT0gImVycm9yIikgY29uc29sZS5sb2coIltjb25zb2xlLmVycm9yXSIsIG1zZy50ZXh0KCkuc3Vic3RyaW5nKDAsIDIwMCkpOwogIH0pOwogIAogIGF3YWl0IHBhZ2UuZ290bygiL3dldmlhLmh0bWwiKTsKICBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsgdHJ5IHsgc2Vzc2lvblN0b3JhZ2UuY2xlYXIoKTsgfSBjYXRjaChlKXt9IH0pOwogIGF3YWl0IHBhZ2Uud2FpdEZvckxvYWRTdGF0ZSgibmV0d29ya2lkbGUiKTsKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDMwMDApOwogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjE1LTAwLnBuZyIgfSk7CiAgY29uc29sZS5sb2coIvCfk7ggbGFuZGluZyIpOwogIAogIGNvbnN0IHRlc3RzID0gWwogICAgeyBsYWJlbDogInFyIiwgbXNnOiAiUVIgY29kZSBwb3VyIGh0dHBzOi8vd2V2YWwtY29uc3VsdGluZy5jb20iLCBuZWVkbGU6IC93ZXZpYS1xci18XC5wbmcvaSB9LAogICAgeyBsYWJlbDogImhkIiwgbXNnOiAiSW1hZ2UgSEQgZGU6IGJlYXV0aWZ1bCBzdW5zZXQgb3ZlciBtb3VudGFpbiIsIG5lZWRsZTogL3dldmlhLWhkLXxIRC9pIH0sCiAgICB7IGxhYmVsOiAidHRzIiwgbXNnOiAibGlzIDogYm9uam91ciBjb21tZW50IGFsbGV6IHZvdXMgYXVqb3VyZCBodWkgbWVyY2kiLCBuZWVkbGU6IC9cLm1wM3xBdWRpby9pIH0sCiAgXTsKICAKICBmb3IgKGxldCBpID0gMDsgaSA8IHRlc3RzLmxlbmd0aDsgaSsrKSB7CiAgICBjb25zdCB0ID0gdGVzdHNbaV07CiAgICBjb25zb2xlLmxvZyhgXG5bJHtpKzF9LyR7dGVzdHMubGVuZ3RofV0gJHt0LmxhYmVsfWApOwogICAgCiAgICBjb25zdCBpbnB1dCA9IHBhZ2UubG9jYXRvcigiI21zZ0lucHV0Iik7CiAgICBhd2FpdCBpbnB1dC5jbGljayh7IGZvcmNlOiB0cnVlIH0pOwogICAgYXdhaXQgcGFnZS5rZXlib2FyZC5wcmVzcygiQ29udHJvbCtBIik7CiAgICBhd2FpdCBwYWdlLmtleWJvYXJkLnByZXNzKCJEZWxldGUiKTsKICAgIGF3YWl0IGlucHV0LmZpbGwodC5tc2cpOwogICAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCg0MDApOwogICAgYXdhaXQgaW5wdXQucHJlc3MoIkVudGVyIik7CiAgICAKICAgIC8vIFdhaXQgdXAgdG8gNjBzIHdpdGggcG9sbGluZwogICAgY29uc3Qgd2FpdFN0YXJ0ID0gRGF0ZS5ub3coKTsKICAgIGxldCBmb3VuZCA9IGZhbHNlOwogICAgbGV0IGxhc3RCb2R5ID0gIiI7CiAgICB3aGlsZSAoRGF0ZS5ub3coKSAtIHdhaXRTdGFydCA8IDYwMDAwKSB7CiAgICAgIGNvbnN0IGJvZHkgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IGRvY3VtZW50LmJvZHkuaW5uZXJUZXh0KTsKICAgICAgbGFzdEJvZHkgPSBib2R5OwogICAgICBpZiAodC5uZWVkbGUudGVzdChib2R5KSkgeyBmb3VuZCA9IHRydWU7IGJyZWFrOyB9CiAgICAgIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMjAwMCk7CiAgICB9CiAgICBjb25zdCBlbGFwc2VkID0gKChEYXRlLm5vdygpIC0gd2FpdFN0YXJ0KSAvIDEwMDApLnRvRml4ZWQoMSk7CiAgICBjb25zb2xlLmxvZyhmb3VuZCA/IGAgIOKchSAke2VsYXBzZWR9c2AgOiBgICDimqDvuI8gJHtlbGFwc2VkfXNgKTsKICAgIAogICAgLy8gTG9nIHdoYXQncyBpbiB0aGUgcmVzcG9uc2UgdmlzaWJsZQogICAgaWYgKCFmb3VuZCkgewogICAgICBjb25zdCBtc2dzID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7CiAgICAgICAgcmV0dXJuIEFycmF5LmZyb20oZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgnLm1zZy5hc3Npc3RhbnQgLmJ1YmJsZScpKS5zbGljZSgtMykubWFwKGIgPT4gYi5pbm5lclRleHQuc3Vic3RyaW5nKDAsIDIwMCkpOwogICAgICB9KTsKICAgICAgY29uc29sZS5sb2coYCAgbGFzdCAzIGFzc2lzdGFudCBtc2dzOiAke0pTT04uc3RyaW5naWZ5KG1zZ3MpfWApOwogICAgfQogICAgCiAgICBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsgY29uc3QgbSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCJtZXNzYWdlcyIpOyBpZiAobSkgbS5zY3JvbGxUb3AgPSBtLnNjcm9sbEhlaWdodDsgfSk7CiAgICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDIwMDApOwogICAgYXdhaXQgcGFnZS5zY3JlZW5zaG90KHsgcGF0aDogYG91dHB1dC92MTUtJHtTdHJpbmcoaSsxKS5wYWRTdGFydCgyLCIwIil9LSR7dC5sYWJlbH0ucG5nYCB9KTsKICAgIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMTAwMCk7CiAgfQogIAogIGNvbnNvbGUubG9nKCJcblYxNSBkb25lIik7Cn0pOwo=");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v15-fix.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v16-deploy.php
Normal file
27
api/ambre-pw-v16-files.php
Normal file
@@ -0,0 +1,27 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$base = "/var/www/html/api/ambre-pw-tests/output";
|
||||
$out = ["v16_screenshots"=>[], "v16_video"=>null, "v16_results"=>null];
|
||||
|
||||
foreach (glob("$base/v16-*.png") as $p) {
|
||||
$out["v16_screenshots"][] = [
|
||||
"name" => basename($p),
|
||||
"kb" => round(filesize($p)/1024, 1),
|
||||
"url" => "https://weval-consulting.com/api/ambre-pw-tests/output/" . basename($p),
|
||||
];
|
||||
}
|
||||
usort($out["v16_screenshots"], function($a,$b){return strcmp($a["name"],$b["name"]);});
|
||||
|
||||
foreach (glob("$base/v16-full-showcase*/*.webm") as $w) {
|
||||
$rel = str_replace($base."/", "", $w);
|
||||
$out["v16_video"] = [
|
||||
"size_mb" => round(filesize($w)/1048576, 2),
|
||||
"url" => "https://weval-consulting.com/api/ambre-pw-tests/output/" . $rel,
|
||||
];
|
||||
}
|
||||
|
||||
if (file_exists("$base/v16-results.json")) {
|
||||
$out["v16_results"] = @json_decode(file_get_contents("$base/v16-results.json"), true);
|
||||
}
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
7
api/ambre-pw-v17-deploy.php
Normal file
7
api/ambre-pw-v18-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMTggwrcgc21va2UgdmVyaWZ5IFY4LVJPQlVTVCBnbG9iYWwgKyAxIG1lc3NhZ2UiLCBhc3luYyAoeyBwYWdlIH0pID0+IHsKICB0ZXN0LnNldFRpbWVvdXQoMTIwMDAwKTsKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7IHRyeSB7IHNlc3Npb25TdG9yYWdlLmNsZWFyKCk7IH0gY2F0Y2goZSl7fSB9KTsKICBhd2FpdCBwYWdlLndhaXRGb3JMb2FkU3RhdGUoIm5ldHdvcmtpZGxlIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgyNTAwKTsKICAKICBjb25zdCByb2J1c3QgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+ICh7CiAgICBmZXRjaF9mbjogdHlwZW9mIHdpbmRvdy5fX2FtYnJlRmV0Y2gsCiAgICBjaXJjdWl0X2ZuOiB0eXBlb2Ygd2luZG93Ll9fYW1icmVDaXJjdWl0U3RhdGUsCiAgfSkpOwogIGNvbnNvbGUubG9nKCJWOC1ST0JVU1Qgc3RhdHVzOiIsIEpTT04uc3RyaW5naWZ5KHJvYnVzdCkpOwogIAogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjE4LTAwLWxhbmRpbmcucG5nIiB9KTsKICAKICBjb25zdCBpbnB1dCA9IHBhZ2UubG9jYXRvcigiI21zZ0lucHV0Iik7CiAgYXdhaXQgaW5wdXQuZmlsbCgiYm9uam91ciB0ZXN0IHNtb2tlIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgzMDApOwogIGF3YWl0IGlucHV0LnByZXNzKCJFbnRlciIpOwogIAogIGNvbnN0IHN0YXJ0ID0gRGF0ZS5ub3coKTsKICBsZXQgYXNzaXN0YW50UmVwbHkgPSAiIjsKICB3aGlsZSAoRGF0ZS5ub3coKSAtIHN0YXJ0IDwgNDUwMDApIHsKICAgIGNvbnN0IHN0YXRlID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7CiAgICAgIGNvbnN0IGEgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCcubXNnLmFzc2lzdGFudCAuYnViYmxlJyk7CiAgICAgIHJldHVybiBhLmxlbmd0aCA+IDAgPyBhW2EubGVuZ3RoIC0gMV0uaW5uZXJUZXh0IDogIiI7CiAgICB9KTsKICAgIGlmIChzdGF0ZSAmJiAhc3RhdGUuc3RhcnRzV2l0aCgiQm9uam91ciAhIENvbW1lbnQiKSkgewogICAgICBhc3Npc3RhbnRSZXBseSA9IHN0YXRlOwogICAgICBicmVhazsKICAgIH0KICAgIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMTUwMCk7CiAgfQogIGNvbnN0IGVsYXBzZWQgPSAoKERhdGUubm93KCkgLSBzdGFydCkgLyAxMDAwKS50b0ZpeGVkKDEpOwogIGNvbnNvbGUubG9nKCJSZXBseSBpbiAiICsgZWxhcHNlZCArICJzOiIsIGFzc2lzdGFudFJlcGx5LnN1YnN0cmluZygwLCAyNTApKTsKICAKICBhd2FpdCBwYWdlLnNjcmVlbnNob3QoeyBwYXRoOiAib3V0cHV0L3YxOC0wMS1yZXBseS5wbmciIH0pOwogIAogIGNvbnN0IGNpcmN1aXQgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHdpbmRvdy5fX2FtYnJlQ2lyY3VpdFN0YXRlID8gd2luZG93Ll9fYW1icmVDaXJjdWl0U3RhdGUoKSA6IG51bGwpOwogIGNvbnNvbGUubG9nKCJDaXJjdWl0OiIsIEpTT04uc3RyaW5naWZ5KGNpcmN1aXQpKTsKfSk7Cg==");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v18-smoke.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v19-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMTkgwrcgZGVidWcgX19hbWJyZUZldGNoIGRpcmVjdGx5IiwgYXN5bmMgKHsgcGFnZSB9KSA9PiB7CiAgdGVzdC5zZXRUaW1lb3V0KDYwMDAwKTsKICAKICBjb25zdCBjb25zb2xlTWVzc2FnZXMgPSBbXTsKICBwYWdlLm9uKCJjb25zb2xlIiwgbXNnID0+IHsKICAgIGNvbnNvbGVNZXNzYWdlcy5wdXNoKGBbJHttc2cudHlwZSgpfV0gJHttc2cudGV4dCgpLnN1YnN0cmluZygwLCAyMDApfWApOwogIH0pOwogIHBhZ2Uub24oInBhZ2VlcnJvciIsIGUgPT4gY29uc29sZS5sb2coIltwYWdlZXJyb3JdIiwgZS5tZXNzYWdlLnN1YnN0cmluZygwLCAyMDApKSk7CiAgCiAgYXdhaXQgcGFnZS5nb3RvKCIvd2V2aWEuaHRtbCIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvckxvYWRTdGF0ZSgibmV0d29ya2lkbGUiKTsKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDIwMDApOwogIAogIC8vIFRlc3QgX19hbWJyZUZldGNoIGRpcmVjdGx5CiAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZShhc3luYyAoKSA9PiB7CiAgICB0cnkgewogICAgICBjb25zdCByID0gYXdhaXQgd2luZG93Ll9fYW1icmVGZXRjaCgnL2FwaS9hbWJyZS10b29sLWNhbGMucGhwJywgewogICAgICAgIG1ldGhvZDogJ1BPU1QnLAogICAgICAgIGhlYWRlcnM6IHsnQ29udGVudC1UeXBlJzonYXBwbGljYXRpb24vanNvbid9LAogICAgICAgIGJvZHk6IEpTT04uc3RyaW5naWZ5KHtleHByZXNzaW9uOiAnMisyJ30pCiAgICAgIH0pOwogICAgICBjb25zdCB0ZXh0ID0gYXdhaXQgci50ZXh0KCk7CiAgICAgIHJldHVybiB7IHN1Y2Nlc3M6IHRydWUsIHN0YXR1czogci5zdGF0dXMsIG9rOiByLm9rLCB0ZXh0OiB0ZXh0LnN1YnN0cmluZygwLCAzMDApIH07CiAgICB9IGNhdGNoIChlKSB7CiAgICAgIHJldHVybiB7IHN1Y2Nlc3M6IGZhbHNlLCBlcnJvcjogZS5tZXNzYWdlLCBzdGFjazogKGUuc3RhY2sgfHwgIiIpLnN1YnN0cmluZygwLCA1MDApIH07CiAgICB9CiAgfSk7CiAgCiAgY29uc29sZS5sb2coIlxuPT09IF9fYW1icmVGZXRjaCBkaXJlY3QgdGVzdCA9PT0iKTsKICBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShyZXN1bHQsIG51bGwsIDIpKTsKICAKICBjb25zb2xlLmxvZygiXG49PT0gQ29uc29sZSBtZXNzYWdlcyA9PT0iKTsKICBjb25zb2xlTWVzc2FnZXMuZm9yRWFjaChtID0+IGNvbnNvbGUubG9nKG0pKTsKICAKICAvLyBBbHNvIHRlc3QgbmF0aXZlIGZldGNoIGZvciBjb21wYXJpc29uCiAgY29uc3QgbmF0aXZlID0gYXdhaXQgcGFnZS5ldmFsdWF0ZShhc3luYyAoKSA9PiB7CiAgICB0cnkgewogICAgICBjb25zdCByID0gYXdhaXQgZmV0Y2goJy9hcGkvYW1icmUtdG9vbC1jYWxjLnBocCcsIHsKICAgICAgICBtZXRob2Q6ICdQT1NUJywKICAgICAgICBoZWFkZXJzOiB7J0NvbnRlbnQtVHlwZSc6J2FwcGxpY2F0aW9uL2pzb24nfSwKICAgICAgICBib2R5OiBKU09OLnN0cmluZ2lmeSh7ZXhwcmVzc2lvbjogJzMrMyd9KQogICAgICB9KTsKICAgICAgY29uc3QgdGV4dCA9IGF3YWl0IHIudGV4dCgpOwogICAgICByZXR1cm4geyBvazogci5vaywgc3RhdHVzOiByLnN0YXR1cywgdGV4dDogdGV4dC5zdWJzdHJpbmcoMCwgMjAwKSB9OwogICAgfSBjYXRjaCAoZSkgewogICAgICByZXR1cm4geyBlcnJvcjogZS5tZXNzYWdlIH07CiAgICB9CiAgfSk7CiAgY29uc29sZS5sb2coIlxuPT09IG5hdGl2ZSBmZXRjaCA9PT0iKTsKICBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShuYXRpdmUsIG51bGwsIDIpKTsKfSk7Cg==");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v19-debug.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v20-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjAgwrcgZmluZCByZWdleCBzeW50YXggZXJyb3IgbGluZSIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCgzMDAwMCk7CiAgCiAgY29uc3QgZXJyb3JzID0gW107CiAgcGFnZS5vbigicGFnZWVycm9yIiwgZSA9PiB7CiAgICBlcnJvcnMucHVzaCh7IG1zZzogZS5tZXNzYWdlLCBzdGFjazogKGUuc3RhY2sgfHwgIiIpLnN1YnN0cmluZygwLCA4MDApIH0pOwogIH0pOwogIHBhZ2Uub24oImNvbnNvbGUiLCBtc2cgPT4gewogICAgaWYgKG1zZy50eXBlKCkgPT09ICJlcnJvciIgfHwgbXNnLnR5cGUoKSA9PT0gIndhcm5pbmciKSB7CiAgICAgIGVycm9ycy5wdXNoKHsgdHlwZTogbXNnLnR5cGUoKSwgdGV4dDogbXNnLnRleHQoKS5zdWJzdHJpbmcoMCwgNDAwKSwgbG9jOiBtc2cubG9jYXRpb24oKSB9KTsKICAgIH0KICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yTG9hZFN0YXRlKCJuZXR3b3JraWRsZSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMzAwMCk7CiAgCiAgY29uc29sZS5sb2coIj09PSBFcnJvcnMgPT09Iik7CiAgZXJyb3JzLmZvckVhY2goZSA9PiBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShlLCBudWxsLCAyKSkpOwp9KTsK");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v20-regex.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v21-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjEgwrcgY2FwdHVyZSBleGFjdCBlcnJvciBzdGFjayIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCgzMDAwMCk7CiAgY29uc3QgZXJyb3JzID0gW107CiAgcGFnZS5vbigicGFnZWVycm9yIiwgZSA9PiB7CiAgICBlcnJvcnMucHVzaCh7IHR5cGU6ICdwYWdlZXJyb3InLCBtc2c6IGUubWVzc2FnZSwgc3RhY2s6IGUuc3RhY2sgfSk7CiAgfSk7CiAgcGFnZS5vbigiY29uc29sZSIsIG1zZyA9PiB7CiAgICBpZiAobXNnLnR5cGUoKSAhPT0gImxvZyIpIHsKICAgICAgZXJyb3JzLnB1c2goeyB0eXBlOiBtc2cudHlwZSgpLCB0ZXh0OiBtc2cudGV4dCgpLCBsb2M6IG1zZy5sb2NhdGlvbigpLCBhcmdzOiBtc2cuYXJncygpLmxlbmd0aCB9KTsKICAgIH0KICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgzNTAwKTsKICAKICAvLyBEZXRhaWxlZCBlcnJvciBpbmZvCiAgZm9yIChjb25zdCBlIG9mIGVycm9ycykgewogICAgY29uc29sZS5sb2coIlxuLS0tIik7CiAgICBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShlLCBudWxsLCAyKSk7CiAgfQogIAogIC8vIEFsc28gZHVtcCB0aGUgY2hhcnMgYXJvdW5kIGxpbmUgOTIwIGNvbCAxMDUgdmlhIHBhZ2UuZXZhbHVhdGUKICBjb25zdCBzbmlwcGV0ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZShhc3luYyAoKSA9PiB7CiAgICB0cnkgewogICAgICBjb25zdCByID0gYXdhaXQgZmV0Y2goJy93ZXZpYS5odG1sJyk7CiAgICAgIGNvbnN0IHRleHQgPSBhd2FpdCByLnRleHQoKTsKICAgICAgY29uc3QgbGluZXMgPSB0ZXh0LnNwbGl0KCdcbicpOwogICAgICAvLyBMaW5lIDkyMCAoMC1pbmRleGVkIDkxOSkKICAgICAgY29uc3QgbGluZSA9IGxpbmVzWzkxOV0gfHwgIiI7CiAgICAgIHJldHVybiB7CiAgICAgICAgbGluZV85MjA6IGxpbmUsCiAgICAgICAgbGluZV85MTk6IGxpbmVzWzkxOF0gfHwgIiIsCiAgICAgICAgbGluZV85MjE6IGxpbmVzWzkyMF0gfHwgIiIsCiAgICAgICAgY29sXzEwMF8xMTU6IGxpbmUuc3Vic3RyaW5nKDk5LCAxMTUpLAogICAgICB9OwogICAgfSBjYXRjaCAoZSkgeyByZXR1cm4geyBlcnI6IGUubWVzc2FnZSB9OyB9CiAgfSk7CiAgY29uc29sZS5sb2coIlxuPT09IExpbmUgOTIwIHNuaXBwZXQgPT09Iik7CiAgY29uc29sZS5sb2coSlNPTi5zdHJpbmdpZnkoc25pcHBldCwgbnVsbCwgMikpOwp9KTsK");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v21-find.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v22-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjIgwrcgdmVyaWZ5IHJlZ2V4IGZpeCArIHNlc3Npb24tY2hhdCB3b3JrcyIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCg5MDAwMCk7CiAgY29uc3QgZXJyb3JzID0gW107CiAgcGFnZS5vbigicGFnZWVycm9yIiwgZSA9PiBlcnJvcnMucHVzaCgiUEU6ICIgKyBlLm1lc3NhZ2Uuc3Vic3RyaW5nKDAsMjAwKSkpOwogIHBhZ2Uub24oImNvbnNvbGUiLCBtc2cgPT4gewogICAgaWYgKG1zZy50eXBlKCkgPT09ICJlcnJvciIgfHwgbXNnLnR5cGUoKSA9PT0gIndhcm5pbmciKSB7CiAgICAgIGVycm9ycy5wdXNoKG1zZy50eXBlKCkgKyAiOiAiICsgbXNnLnRleHQoKS5zdWJzdHJpbmcoMCwyMDApKTsKICAgIH0KICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yTG9hZFN0YXRlKCJuZXR3b3JraWRsZSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMzAwMCk7CiAgCiAgY29uc29sZS5sb2coIj09PSBFcnJvcnMgb24gbG9hZCA9PT0iKTsKICBlcnJvcnMuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKCIgIiwgZSkpOwogIAogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjIyLTAwLWxvYWQucG5nIiB9KTsKICAKICAvLyBWZXJpZnkgX19hbWJyZUZldGNoCiAgY29uc3QgaGFzUm9idXN0ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB0eXBlb2Ygd2luZG93Ll9fYW1icmVGZXRjaCA9PT0gImZ1bmN0aW9uIik7CiAgY29uc29sZS5sb2coIlY4LVJPQlVTVCBsb2FkZWQ6IiwgaGFzUm9idXN0KTsKICAKICAvLyBTZW5kIGEgc2ltcGxlIG1lc3NhZ2UgKFY1IHNlc3Npb24tY2hhdCBwYXRoLCBub3QgTExNLWhlYXZ5KQogIGNvbnN0IGlucHV0ID0gcGFnZS5sb2NhdG9yKCIjbXNnSW5wdXQiKTsKICBhd2FpdCBpbnB1dC5jbGljayh7IGZvcmNlOiB0cnVlIH0pOwogIGF3YWl0IGlucHV0LmZpbGwoImJvbmpvdXIiKTsKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDUwMCk7CiAgYXdhaXQgaW5wdXQucHJlc3MoIkVudGVyIik7CiAgCiAgY29uc3Qgd2FpdFN0YXJ0ID0gRGF0ZS5ub3coKTsKICBsZXQgZm91bmQgPSBmYWxzZTsKICBsZXQgbGFzdFJlcGx5ID0gIiI7CiAgd2hpbGUgKERhdGUubm93KCkgLSB3YWl0U3RhcnQgPCA0NTAwMCkgewogICAgY29uc3QgcmVwbHkgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsKICAgICAgY29uc3QgYXNzdCA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3JBbGwoIi5tc2cuYXNzaXN0YW50IC5idWJibGUiKTsKICAgICAgcmV0dXJuIGFzc3QubGVuZ3RoID4gMSA/IGFzc3RbYXNzdC5sZW5ndGgtMV0uaW5uZXJUZXh0IDogIiI7CiAgICB9KTsKICAgIGlmIChyZXBseSAmJiByZXBseS5sZW5ndGggPiAyMCAmJiAhcmVwbHkuaW5jbHVkZXMoIkNvbW1lbnQgcHVpcy1qZSB2b3VzIGFpZGVyIikpIHsKICAgICAgbGFzdFJlcGx5ID0gcmVwbHk7CiAgICAgIGZvdW5kID0gdHJ1ZTsKICAgICAgYnJlYWs7CiAgICB9CiAgICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDE1MDApOwogIH0KICBjb25zdCBlbGFwc2VkID0gKChEYXRlLm5vdygpLXdhaXRTdGFydCkvMTAwMCkudG9GaXhlZCgxKTsKICBjb25zb2xlLmxvZyhgXG5SZXBseSBpbiAke2VsYXBzZWR9czpgLCBsYXN0UmVwbHkuc3Vic3RyaW5nKDAsMzAwKSk7CiAgY29uc29sZS5sb2coIkZvdW5kOiIsIGZvdW5kKTsKICAKICBhd2FpdCBwYWdlLnNjcmVlbnNob3QoeyBwYXRoOiAib3V0cHV0L3YyMi0wMS1yZXBseS5wbmciIH0pOwogIAogIC8vIEFsc28gdHJ5IGNhbGMKICBhd2FpdCBpbnB1dC5jbGljayh7IGZvcmNlOiB0cnVlIH0pOwogIGF3YWl0IHBhZ2Uua2V5Ym9hcmQucHJlc3MoIkNvbnRyb2wrQSIpOwogIGF3YWl0IHBhZ2Uua2V5Ym9hcmQucHJlc3MoIkRlbGV0ZSIpOwogIGF3YWl0IGlucHV0LmZpbGwoImNhbGN1bGUgNDIgKiAzIik7CiAgYXdhaXQgaW5wdXQucHJlc3MoIkVudGVyIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCg1MDAwKTsKICAKICBjb25zdCBjYWxjUmVwbHkgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsKICAgIGNvbnN0IGFzc3QgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCIubXNnLmFzc2lzdGFudCAuYnViYmxlIik7CiAgICByZXR1cm4gYXNzdC5sZW5ndGggPiAwID8gYXNzdFthc3N0Lmxlbmd0aC0xXS5pbm5lclRleHQgOiAiIjsKICB9KTsKICBjb25zb2xlLmxvZygiXG5DYWxjIHJlcGx5OiIsIGNhbGNSZXBseS5zdWJzdHJpbmcoMCwyMDApKTsKICAKICBhd2FpdCBwYWdlLnNjcmVlbnNob3QoeyBwYXRoOiAib3V0cHV0L3YyMi0wMi1jYWxjLnBuZyIgfSk7Cn0pOwo=");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v22-verify.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v22.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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjIgwrcgdmVyaWZ5IHJlZ2V4IGZpeCArIHNtb2tlIG1lc3NhZ2UiLCBhc3luYyAoeyBwYWdlIH0pID0+IHsKICB0ZXN0LnNldFRpbWVvdXQoNjAwMDApOwogIGNvbnN0IGVycm9ycyA9IFtdOwogIHBhZ2Uub24oInBhZ2VlcnJvciIsIGUgPT4gZXJyb3JzLnB1c2goZS5tZXNzYWdlKSk7CiAgcGFnZS5vbigiY29uc29sZSIsIG1zZyA9PiB7CiAgICBpZiAobXNnLnR5cGUoKSA9PT0gIndhcm5pbmciIHx8IG1zZy50eXBlKCkgPT09ICJlcnJvciIpIHsKICAgICAgZXJyb3JzLnB1c2gobXNnLnR5cGUoKSArICI6ICIgKyBtc2cudGV4dCgpLnN1YnN0cmluZygwLCAyMDApKTsKICAgIH0KICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgzNTAwKTsKICAKICBjb25zb2xlLmxvZygiPT09IGVycm9ycyA9PT0iKTsKICBlcnJvcnMuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKCIgICIgKyBlKSk7CiAgCiAgY29uc3Qgcm9idXN0ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB0eXBlb2Ygd2luZG93Ll9fYW1icmVGZXRjaCk7CiAgY29uc29sZS5sb2coIl9fYW1icmVGZXRjaDoiLCByb2J1c3QpOwogIAogIC8vIFNlbmQgYSBjYWxjIG1lc3NhZ2UKICBjb25zdCBpbnB1dCA9IHBhZ2UubG9jYXRvcigiI21zZ0lucHV0Iik7CiAgYXdhaXQgaW5wdXQuZmlsbCgiY2FsY3VsZSAyKzIiKTsKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDUwMCk7CiAgYXdhaXQgaW5wdXQucHJlc3MoIkVudGVyIik7CiAgCiAgY29uc3Qgc3RhcnQgPSBEYXRlLm5vdygpOwogIGxldCByZXBseSA9ICIiOwogIHdoaWxlIChEYXRlLm5vdygpIC0gc3RhcnQgPCAyNTAwMCkgewogICAgY29uc3Qgc3RhdGUgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsKICAgICAgY29uc3QgYSA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3JBbGwoJy5tc2cuYXNzaXN0YW50IC5idWJibGUnKTsKICAgICAgcmV0dXJuIGEubGVuZ3RoID4gMCA/IGFbYS5sZW5ndGggLSAxXS5pbm5lclRleHQgOiAiIjsKICAgIH0pOwogICAgaWYgKHN0YXRlICYmICFzdGF0ZS5zdGFydHNXaXRoKCJCb25qb3VyICEgQ29tbWVudCIpKSB7CiAgICAgIHJlcGx5ID0gc3RhdGU7CiAgICAgIGlmICgvPS4qNHxyZXN1bHR8XFxkKy8udGVzdChzdGF0ZSkpIGJyZWFrOwogICAgfQogICAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgxNTAwKTsKICB9CiAgY29uc3QgZWxhcHNlZCA9ICgoRGF0ZS5ub3coKSAtIHN0YXJ0KSAvIDEwMDApLnRvRml4ZWQoMSk7CiAgY29uc29sZS5sb2coInJlcGx5IGluICIgKyBlbGFwc2VkICsgInM6ICIgKyByZXBseS5zdWJzdHJpbmcoMCwgMjAwKSk7CiAgCiAgYXdhaXQgcGFnZS5zY3JlZW5zaG90KHsgcGF0aDogIm91dHB1dC92MjItY2FsYy5wbmciIH0pOwp9KTsK");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v22-smoke.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v23-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjMgwrcgYWZ0ZXIgR09MRCByZXN0b3JlIMK3IHZlcmlmeSBlcnJvciBnb25lICsgc2Vzc2lvbiB3b3JrcyIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCg5MDAwMCk7CiAgY29uc3QgZXJyb3JzID0gW107CiAgcGFnZS5vbigicGFnZWVycm9yIiwgZSA9PiBlcnJvcnMucHVzaCgiUEU6ICIgKyBlLm1lc3NhZ2Uuc3Vic3RyaW5nKDAsMjAwKSkpOwogIHBhZ2Uub24oImNvbnNvbGUiLCBtc2cgPT4gewogICAgaWYgKG1zZy50eXBlKCkgPT09ICJlcnJvciIgfHwgbXNnLnR5cGUoKSA9PT0gIndhcm5pbmciKSB7CiAgICAgIGVycm9ycy5wdXNoKG1zZy50eXBlKCkgKyAiOiAiICsgbXNnLnRleHQoKS5zdWJzdHJpbmcoMCwyMDApKTsKICAgIH0KICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yTG9hZFN0YXRlKCJuZXR3b3JraWRsZSIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMzAwMCk7CiAgCiAgY29uc29sZS5sb2coIj09PSBFcnJvcnMgYWZ0ZXIgcmVzdG9yZSA9PT0iKTsKICBlcnJvcnMuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKCIgIiwgZSkpOwogIAogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjIzLTAwLWxvYWQucG5nIiB9KTsKICAKICAvLyBUZXN0IGNhbGMgKFY2KQogIGNvbnN0IGlucHV0ID0gcGFnZS5sb2NhdG9yKCIjbXNnSW5wdXQiKTsKICBhd2FpdCBpbnB1dC5jbGljayh7IGZvcmNlOiB0cnVlIH0pOwogIGF3YWl0IGlucHV0LmZpbGwoImNhbGN1bGUgNDIgKiAzIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCg0MDApOwogIGF3YWl0IGlucHV0LnByZXNzKCJFbnRlciIpOwogIAogIGNvbnN0IHdhaXRTdGFydCA9IERhdGUubm93KCk7CiAgbGV0IGZvdW5kID0gZmFsc2U7IGxldCByZXBseSA9ICIiOwogIHdoaWxlIChEYXRlLm5vdygpIC0gd2FpdFN0YXJ0IDwgMjUwMDApIHsKICAgIHJlcGx5ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7CiAgICAgIGNvbnN0IGEgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKCIubXNnLmFzc2lzdGFudCAuYnViYmxlIik7CiAgICAgIHJldHVybiBhLmxlbmd0aCA+IDEgPyBhW2EubGVuZ3RoLTFdLmlubmVyVGV4dCA6ICIiOwogICAgfSk7CiAgICBpZiAocmVwbHkuaW5jbHVkZXMoIjEyNiIpIHx8IHJlcGx5LmluY2x1ZGVzKCLwn6euIikpIHsgZm91bmQgPSB0cnVlOyBicmVhazsgfQogICAgaWYgKHJlcGx5LmluY2x1ZGVzKCJlcnJldXIiKSkgYnJlYWs7CiAgICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDE1MDApOwogIH0KICBjb25zdCBlbCA9ICgoRGF0ZS5ub3coKS13YWl0U3RhcnQpLzEwMDApLnRvRml4ZWQoMSk7CiAgY29uc29sZS5sb2coYFxuQ2FsYyAke2ZvdW5kPyLinIUiOiLinYwifSBpbiAke2VsfXM6YCwgcmVwbHkuc3Vic3RyaW5nKDAsIDI1MCkpOwogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjIzLTAxLWNhbGMucG5nIiB9KTsKICAKICAvLyBUZXN0IFFSIChWNykKICBhd2FpdCBpbnB1dC5jbGljayh7IGZvcmNlOiB0cnVlIH0pOwogIGF3YWl0IHBhZ2Uua2V5Ym9hcmQucHJlc3MoIkNvbnRyb2wrQSIpOwogIGF3YWl0IHBhZ2Uua2V5Ym9hcmQucHJlc3MoIkRlbGV0ZSIpOwogIGF3YWl0IGlucHV0LmZpbGwoIlFSIGNvZGUgcG91ciBXRVZBTCIpOwogIGF3YWl0IGlucHV0LnByZXNzKCJFbnRlciIpOwogIAogIGNvbnN0IHdhaXRTdGFydDIgPSBEYXRlLm5vdygpOwogIGxldCBmb3VuZDIgPSBmYWxzZTsgbGV0IHJlcGx5MiA9ICIiOwogIHdoaWxlIChEYXRlLm5vdygpIC0gd2FpdFN0YXJ0MiA8IDI1MDAwKSB7CiAgICByZXBseTIgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHsKICAgICAgY29uc3QgYSA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3JBbGwoIi5tc2cuYXNzaXN0YW50IC5idWJibGUiKTsKICAgICAgcmV0dXJuIGEubGVuZ3RoID4gMCA/IGFbYS5sZW5ndGgtMV0uaW5uZXJUZXh0IDogIiI7CiAgICB9KTsKICAgIGlmIChyZXBseTIuaW5jbHVkZXMoIndldmlhLXFyIikgfHwgcmVwbHkyLmluY2x1ZGVzKCLwn5OxIikpIHsgZm91bmQyID0gdHJ1ZTsgYnJlYWs7IH0KICAgIGlmIChyZXBseTIuaW5jbHVkZXMoImVycmV1ciIpKSBicmVhazsKICAgIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoMTUwMCk7CiAgfQogIGNvbnN0IGVsMiA9ICgoRGF0ZS5ub3coKS13YWl0U3RhcnQyKS8xMDAwKS50b0ZpeGVkKDEpOwogIGNvbnNvbGUubG9nKGBRUiAke2ZvdW5kMj8i4pyFIjoi4p2MIn0gaW4gJHtlbDJ9czpgLCByZXBseTIuc3Vic3RyaW5nKDAsIDI1MCkpOwogIGF3YWl0IHBhZ2Uuc2NyZWVuc2hvdCh7IHBhdGg6ICJvdXRwdXQvdjIzLTAyLXFyLnBuZyIgfSk7Cn0pOwo=");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v23-post-restore.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v24-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjQgwrcgY2FwdHVyZSBzdGFjayB0cmFjZSArIGRldGVjdCByZWdleCBsaW5lIiwgYXN5bmMgKHsgcGFnZSB9KSA9PiB7CiAgdGVzdC5zZXRUaW1lb3V0KDYwMDAwKTsKICAKICBjb25zdCBtZXNzYWdlcyA9IFtdOwogIHBhZ2Uub24oInBhZ2VlcnJvciIsIGUgPT4gewogICAgbWVzc2FnZXMucHVzaCh7IHR5cGU6ICJwYWdlZXJyb3IiLCBtc2c6IGUubWVzc2FnZSwgc3RhY2s6IChlLnN0YWNrIHx8ICJubyBzdGFjayIpLnN1YnN0cmluZygwLCAyMDAwKSB9KTsKICB9KTsKICBwYWdlLm9uKCJjb25zb2xlIiwgbXNnID0+IHsKICAgIGlmIChtc2cudHlwZSgpICE9PSAibG9nIiAmJiBtc2cudHlwZSgpICE9PSAiZGVidWciKSB7CiAgICAgIGNvbnN0IGxvYyA9IG1zZy5sb2NhdGlvbigpOwogICAgICBtZXNzYWdlcy5wdXNoKHsKICAgICAgICB0eXBlOiBtc2cudHlwZSgpLAogICAgICAgIHRleHQ6IG1zZy50ZXh0KCkuc3Vic3RyaW5nKDAsIDMwMCksCiAgICAgICAgdXJsOiBsb2MudXJsLAogICAgICAgIGxpbmU6IGxvYy5saW5lTnVtYmVyLAogICAgICAgIGNvbDogbG9jLmNvbHVtbk51bWJlciwKICAgICAgfSk7CiAgICB9CiAgfSk7CiAgCiAgYXdhaXQgcGFnZS5nb3RvKCIvd2V2aWEuaHRtbCIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoNTAwMCk7CiAgCiAgY29uc29sZS5sb2coYFxuPT09ICR7bWVzc2FnZXMubGVuZ3RofSBtZXNzYWdlcyBjYXB0dXJlZCA9PT1gKTsKICBmb3IgKGNvbnN0IG0gb2YgbWVzc2FnZXMpIHsKICAgIGNvbnNvbGUubG9nKCJcbi0tLSIpOwogICAgY29uc29sZS5sb2coSlNPTi5zdHJpbmdpZnkobSwgbnVsbCwgMikpOwogIH0KICAKICAvLyBBbHNvIGV2YWwgaW5saW5lIGFuZCB3YXRjaCBmb3IgcnVudGltZSBlcnJvcnMKICBjb25zdCBldmFsUmVzdWx0ID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7CiAgICBjb25zdCByZXN1bHRzID0geyBmdW5jczoge30sIGVycm9yczogW10gfTsKICAgIHRyeSB7CiAgICAgIHJlc3VsdHMuZnVuY3Muc2VuZE1zZyA9IHR5cGVvZiB3aW5kb3cuc2VuZE1zZzsKICAgICAgcmVzdWx0cy5mdW5jcy5hZGRNc2cgPSB0eXBlb2Ygd2luZG93LmFkZE1zZzsKICAgICAgcmVzdWx0cy5mdW5jcy5zaG93VGhpbmtpbmcgPSB0eXBlb2Ygd2luZG93LnNob3dUaGlua2luZzsKICAgICAgcmVzdWx0cy5mdW5jcy5oaWRlVGhpbmtpbmcgPSB0eXBlb2Ygd2luZG93LmhpZGVUaGlua2luZzsKICAgICAgcmVzdWx0cy5mdW5jcy5hbWJyZUZldGNoID0gdHlwZW9mIHdpbmRvdy5fX2FtYnJlRmV0Y2g7CiAgICB9IGNhdGNoKGUpIHsgcmVzdWx0cy5lcnJvcnMucHVzaCgiZnVuY3M6ICIgKyBlLm1lc3NhZ2UpOyB9CiAgICByZXR1cm4gcmVzdWx0czsKICB9KTsKICBjb25zb2xlLmxvZygiXG49PT0gR2xvYmFsIGZ1bmN0aW9ucyA9PT0iKTsKICBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShldmFsUmVzdWx0LCBudWxsLCAyKSk7Cn0pOwo=");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v24-stacktrace.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v25-deploy.php
Normal file
7
api/ambre-pw-v26-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjYgwrcgQ0RQIGZpbmQgZXhhY3QgcmVnZXggZXJyb3IiLCBhc3luYyAoeyBicm93c2VyIH0pID0+IHsKICB0ZXN0LnNldFRpbWVvdXQoNjAwMDApOwogIGNvbnN0IGNvbnRleHQgPSBhd2FpdCBicm93c2VyLm5ld0NvbnRleHQoKTsKICBjb25zdCBwYWdlID0gYXdhaXQgY29udGV4dC5uZXdQYWdlKCk7CiAgCiAgLy8gVXNlIENEUCB0byBnZXQgZGV0YWlsZWQgZXJyb3JzCiAgY29uc3QgY2xpZW50ID0gYXdhaXQgY29udGV4dC5uZXdDRFBTZXNzaW9uKHBhZ2UpOwogIGF3YWl0IGNsaWVudC5zZW5kKCJSdW50aW1lLmVuYWJsZSIpOwogIGF3YWl0IGNsaWVudC5zZW5kKCJEZWJ1Z2dlci5lbmFibGUiKTsKICAKICBjb25zdCBleGNlcHRpb25zID0gW107CiAgY2xpZW50Lm9uKCJSdW50aW1lLmV4Y2VwdGlvblRocm93biIsIChwYXJhbXMpID0+IHsKICAgIGV4Y2VwdGlvbnMucHVzaChwYXJhbXMuZXhjZXB0aW9uRGV0YWlscyk7CiAgfSk7CiAgCiAgYXdhaXQgcGFnZS5nb3RvKCIvd2V2aWEuaHRtbCIpOwogIGF3YWl0IHBhZ2Uud2FpdEZvclRpbWVvdXQoNDAwMCk7CiAgCiAgY29uc29sZS5sb2coIj09PSBDRFAgZXhjZXB0aW9ucyBjb3VudDoiLCBleGNlcHRpb25zLmxlbmd0aCk7CiAgZm9yIChjb25zdCBlIG9mIGV4Y2VwdGlvbnMpIHsKICAgIGNvbnNvbGUubG9nKCJcbi0tLSBFeGNlcHRpb24gLS0tIik7CiAgICBjb25zb2xlLmxvZygidGV4dDoiLCBlLnRleHQpOwogICAgY29uc29sZS5sb2coInVybDoiLCBlLnVybCk7CiAgICBjb25zb2xlLmxvZygibGluZToiLCBlLmxpbmVOdW1iZXIpOwogICAgY29uc29sZS5sb2coImNvbDoiLCBlLmNvbHVtbk51bWJlcik7CiAgICBpZiAoZS5leGNlcHRpb24pIHsKICAgICAgY29uc29sZS5sb2coImRlc2NyaXB0aW9uOiIsIGUuZXhjZXB0aW9uLmRlc2NyaXB0aW9uID8gZS5leGNlcHRpb24uZGVzY3JpcHRpb24uc3Vic3RyaW5nKDAsIDE1MDApIDogIiIpOwogICAgfQogICAgaWYgKGUuc3RhY2tUcmFjZSkgewogICAgICBjb25zb2xlLmxvZygic3RhY2s6IiwgSlNPTi5zdHJpbmdpZnkoZS5zdGFja1RyYWNlLmNhbGxGcmFtZXMsIG51bGwsIDIpLnN1YnN0cmluZygwLCAxNTAwKSk7CiAgICB9CiAgfQp9KTsK");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v26-cdp.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v27-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjcgwrcgdGVzdCBlYWNoIHNjcmlwdCB3aXRoIG5ldyBGdW5jdGlvbiBpbiBicm93c2VyIiwgYXN5bmMgKHsgcGFnZSB9KSA9PiB7CiAgdGVzdC5zZXRUaW1lb3V0KDYwMDAwKTsKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgyNTAwKTsKICAKICAvLyBGZXRjaCB0aGUgSFRNTCBzb3VyY2UgYW5kIHBhcnNlIGVhY2ggaW5saW5lIDxzY3JpcHQ+IHZpYSBuZXcgRnVuY3Rpb24KICBjb25zdCByZXN1bHQgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKGFzeW5jICgpID0+IHsKICAgIGNvbnN0IHIgPSBhd2FpdCBmZXRjaCgnL3dldmlhLmh0bWwnKTsKICAgIGNvbnN0IGh0bWwgPSBhd2FpdCByLnRleHQoKTsKICAgIAogICAgLy8gRXh0cmFjdCBpbmxpbmUgc2NyaXB0cyAobm8gc3JjIGF0dHJpYnV0ZSkKICAgIGNvbnN0IHNjcmlwdHMgPSBbXTsKICAgIGNvbnN0IHJlID0gLzxzY3JpcHQoPyFbXj5dKlxzc3JjPSlbXj5dKj4oW1xzXFNdKj8pPFwvc2NyaXB0Pi9naTsKICAgIGxldCBtOwogICAgbGV0IGlkeCA9IDA7CiAgICB3aGlsZSAoKG0gPSByZS5leGVjKGh0bWwpKSAhPT0gbnVsbCkgewogICAgICBpZHgrKzsKICAgICAgY29uc3QgY29udGVudCA9IG1bMV07CiAgICAgIGlmIChjb250ZW50Lmxlbmd0aCA8IDIwKSBjb250aW51ZTsKICAgICAgY29uc3Qgc3RhcnRMaW5lID0gaHRtbC5zdWJzdHJpbmcoMCwgbS5pbmRleCkuc3BsaXQoJ1xuJykubGVuZ3RoOwogICAgICAKICAgICAgLy8gVHJ5IHRvIHBhcnNlIHZpYSBuZXcgRnVuY3Rpb24KICAgICAgbGV0IHBhcnNlRXJyID0gbnVsbDsKICAgICAgdHJ5IHsKICAgICAgICBuZXcgRnVuY3Rpb24oY29udGVudCk7CiAgICAgIH0gY2F0Y2ggKGUpIHsKICAgICAgICBwYXJzZUVyciA9IHsgbXNnOiBlLm1lc3NhZ2UsIHN0YWNrOiAoZS5zdGFjayB8fCAnJykuc3Vic3RyaW5nKDAsIDgwMCkgfTsKICAgICAgfQogICAgICAKICAgICAgc2NyaXB0cy5wdXNoKHsKICAgICAgICBpZHgsCiAgICAgICAgc3RhcnRMaW5lLAogICAgICAgIHNpemU6IGNvbnRlbnQubGVuZ3RoLAogICAgICAgIHBhcnNlRXJyLAogICAgICB9KTsKICAgIH0KICAgIHJldHVybiBzY3JpcHRzOwogIH0pOwogIAogIGNvbnNvbGUubG9nKEpTT04uc3RyaW5naWZ5KHJlc3VsdCwgbnVsbCwgMikpOwp9KTsK");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v27-parse.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
7
api/ambre-pw-v28-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("Y29uc3QgeyB0ZXN0IH0gPSByZXF1aXJlKCJAcGxheXdyaWdodC90ZXN0Iik7Cgp0ZXN0KCJWMjggwrcgd3JhcCBmZXRjaCBhbmQgc2VuZCBISSIsIGFzeW5jICh7IHBhZ2UgfSkgPT4gewogIHRlc3Quc2V0VGltZW91dCg2MDAwMCk7CiAgCiAgY29uc3QgbmV0bG9nID0gW107CiAgY29uc3QgZXJycyA9IFtdOwogIHBhZ2Uub24oInBhZ2VlcnJvciIsIGUgPT4gZXJycy5wdXNoKGUubWVzc2FnZSkpOwogIHBhZ2Uub24oInJlc3BvbnNlIiwgcmVzID0+IHsKICAgIGlmIChyZXMudXJsKCkuaW5jbHVkZXMoIi9hcGkvIikpIHsKICAgICAgbmV0bG9nLnB1c2goeyB1cmw6IHJlcy51cmwoKS5zcGxpdCgiPyIpWzBdLCBzdGF0dXM6IHJlcy5zdGF0dXMoKSB9KTsKICAgIH0KICB9KTsKICAKICBhd2FpdCBwYWdlLmdvdG8oIi93ZXZpYS5odG1sIik7CiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgyNTAwKTsKICAKICAvLyBNb25rZXktcGF0Y2ggZmV0Y2ggdG8gc2VlIGFsbCBjYWxscwogIGF3YWl0IHBhZ2UuZXZhbHVhdGUoKCkgPT4gewogICAgd2luZG93Ll9mZXRjaExvZyA9IFtdOwogICAgY29uc3Qgb3JpZyA9IHdpbmRvdy5mZXRjaDsKICAgIHdpbmRvdy5mZXRjaCA9IGZ1bmN0aW9uKHUsIG9wdHMpIHsKICAgICAgY29uc3QgdXJsID0gdHlwZW9mIHUgPT09ICJzdHJpbmciID8gdSA6IHUudXJsOwogICAgICB3aW5kb3cuX2ZldGNoTG9nLnB1c2goeyB1cmw6IHVybC5zcGxpdCgiPyIpWzBdLCBtZXRob2Q6IChvcHRzICYmIG9wdHMubWV0aG9kKSB8fCAiR0VUIiB9KTsKICAgICAgcmV0dXJuIG9yaWcuYXBwbHkodGhpcywgYXJndW1lbnRzKTsKICAgIH07CiAgfSk7CiAgCiAgLy8gU2VuZCAiSEkiCiAgYXdhaXQgcGFnZS5maWxsKCIjbXNnSW5wdXQiLCAiSEkiKTsKICBhd2FpdCBwYWdlLndhaXRGb3JUaW1lb3V0KDMwMCk7CiAgYXdhaXQgcGFnZS5wcmVzcygiI21zZ0lucHV0IiwgIkVudGVyIik7CiAgCiAgYXdhaXQgcGFnZS53YWl0Rm9yVGltZW91dCgxNTAwMCk7CiAgCiAgY29uc3QgZmV0Y2hMb2cgPSBhd2FpdCBwYWdlLmV2YWx1YXRlKCgpID0+IHdpbmRvdy5fZmV0Y2hMb2cpOwogIGNvbnNvbGUubG9nKCJcbj09PSBmZXRjaCBjYWxscyBmcm9tIEpTID09PSIpOwogIGNvbnNvbGUubG9nKEpTT04uc3RyaW5naWZ5KGZldGNoTG9nLCBudWxsLCAyKSk7CiAgCiAgY29uc29sZS5sb2coIlxuPT09IE5ldHdvcmsgbG9nICh2aWEgUGxheXdyaWdodCkgPT09Iik7CiAgY29uc29sZS5sb2coSlNPTi5zdHJpbmdpZnkobmV0bG9nLCBudWxsLCAyKSk7CiAgCiAgY29uc29sZS5sb2coIlxuPT09IFBhZ2UgZXJyb3JzID09PSIpOwogIGVycnMuZm9yRWFjaChlID0+IGNvbnNvbGUubG9nKCIgIiwgZS5zdWJzdHJpbmcoMCwgMjAwKSkpOwogIAogIC8vIERPTSBzdGF0ZQogIGNvbnN0IGRvbVN0YXRlID0gYXdhaXQgcGFnZS5ldmFsdWF0ZSgoKSA9PiB7CiAgICBjb25zdCBhID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgiLm1zZy5hc3Npc3RhbnQgLmJ1YmJsZSIpOwogICAgcmV0dXJuIHsKICAgICAgY291bnQ6IGEubGVuZ3RoLAogICAgICBtZXNzYWdlczogQXJyYXkuZnJvbShhKS5tYXAoZWwgPT4gZWwuaW5uZXJUZXh0LnN1YnN0cmluZygwLCAyMDApKSwKICAgIH07CiAgfSk7CiAgY29uc29sZS5sb2coIlxuPT09IERPTSBtZXNzYWdlcyA9PT0iKTsKICBjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeShkb21TdGF0ZSwgbnVsbCwgMikpOwp9KTsK");
|
||||
foreach (glob("$base/*.spec.js") as $old) @unlink($old);
|
||||
$written = @file_put_contents("$base/v28-fetch-log.spec.js", $spec);
|
||||
echo json_encode(["written" => $written]);
|
||||
3
api/ambre-read-chat.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");
|
||||
6
api/ambre-read-doctrines.php
Normal file
@@ -0,0 +1,6 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
echo "=== doctrine 102 wave226 PDF ===\n";
|
||||
echo @file_get_contents("/opt/obsidian-vault/doctrines/102-wave226-godmode-mega-weasyprint-pandasai-llm.md");
|
||||
echo "\n\n=== doctrine 101 wave225 reportlab ===\n";
|
||||
echo @file_get_contents("/opt/obsidian-vault/doctrines/101-wave225-reportlab-pypdf2-wire-intents-gap-bumps.md");
|
||||
36
api/ambre-regex-direct-fix.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
|
||||
$file = "/var/www/html/wevia.html";
|
||||
$content = file_get_contents($file);
|
||||
$len_before = strlen($content);
|
||||
|
||||
// The broken regex on line 1503 uses [.*+?^${}()|[\]\\/] which has [] inside [] chars -- INVALID
|
||||
// Replace with safer version using non-class-based escape
|
||||
|
||||
$broken = 'var _escUrl = finalFileUrl.replace(/[.*+?^${}()|[\]\\\\\/]/g, \'\\\\$&\');';
|
||||
$fixed = 'var _escUrl = finalFileUrl.split(/([^A-Za-z0-9])/).map(function(p,i){return i%2?"\\\\"+p:p;}).join("");';
|
||||
|
||||
if (strpos($content, $broken) !== false) {
|
||||
$new_content = str_replace($broken, $fixed, $content);
|
||||
// GOLD backup
|
||||
$gold = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-regex-fix";
|
||||
@copy($file, $gold);
|
||||
// Write
|
||||
@chattr_remove(); // ignore - chattr might not be set
|
||||
file_put_contents($file, $new_content);
|
||||
echo "FIXED! delta=" . (strlen($new_content) - $len_before) . " gold=$gold\n";
|
||||
} else {
|
||||
echo "Pattern not found in file\n";
|
||||
// Show what we have at line 1503
|
||||
$lines = explode("\n", $content);
|
||||
echo "Line 1503: " . ($lines[1502] ?? "N/A") . "\n";
|
||||
// Extract exact bytes
|
||||
$line = $lines[1502];
|
||||
$pos_var = strpos($line, "var _escUrl");
|
||||
if ($pos_var !== false) {
|
||||
echo "Bytes 60-200: " . bin2hex(substr($line, 60, 50)) . "\n";
|
||||
}
|
||||
}
|
||||
|
||||
function chattr_remove(){ @shell_exec("chattr -i /var/www/html/wevia.html 2>&1"); }
|
||||
11
api/ambre-rembg-check.php
Normal file
@@ -0,0 +1,11 @@
|
||||
<?php
|
||||
header("Content-Type: text/plain");
|
||||
echo "=== Try rembg direct ===\n";
|
||||
$out = @shell_exec("/var/www/.local/bin/rembg --version 2>&1");
|
||||
echo $out;
|
||||
echo "\n=== Python -m rembg ===\n";
|
||||
$out2 = @shell_exec("python3 -m rembg.cli --help 2>&1 | head -20");
|
||||
echo $out2;
|
||||
echo "\n=== import check ===\n";
|
||||
$out3 = @shell_exec("python3 -c 'from rembg import remove; from PIL import Image; print(\"import OK\")' 2>&1");
|
||||
echo $out3;
|
||||
16
api/ambre-restore-gold.php
Normal file
@@ -0,0 +1,16 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$gold = "/opt/wevads/vault/wevia.html.GOLD-20260421-230109-pre-safe-write";
|
||||
$dest = "/var/www/html/wevia.html";
|
||||
if (!file_exists($gold)) { echo json_encode(["error"=>"gold missing"]); exit; }
|
||||
// Current backup first
|
||||
$now_backup = "/opt/wevads/vault/wevia.html.GOLD-" . date("Ymd-His") . "-v8-broken";
|
||||
@copy($dest, $now_backup);
|
||||
// Restore
|
||||
$bytes = @file_put_contents($dest, file_get_contents($gold));
|
||||
echo json_encode([
|
||||
"restored" => $bytes,
|
||||
"gold" => basename($gold),
|
||||
"backup_broken" => basename($now_backup),
|
||||
"new_size" => filesize($dest),
|
||||
]);
|
||||
46
api/ambre-script-dig.php
Normal file
@@ -0,0 +1,46 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$wevia = @file_get_contents("/var/www/html/wevia.html");
|
||||
|
||||
// Find big script starting around line 718 (first <script> without src= and big content)
|
||||
$pos = 0;
|
||||
$big = null;
|
||||
$start_abs = 0;
|
||||
while (($m = strpos($wevia, "<script>", $pos)) !== false) {
|
||||
$end = strpos($wevia, "</script>", $m);
|
||||
if ($end === false) break;
|
||||
$content = substr($wevia, $m + 8, $end - $m - 8);
|
||||
if (strlen($content) > 20000) {
|
||||
$big = $content;
|
||||
$start_abs = substr_count(substr($wevia, 0, $m + 8), "\n") + 1;
|
||||
break;
|
||||
}
|
||||
$pos = $end + 9;
|
||||
}
|
||||
|
||||
$lines = explode("\n", $big);
|
||||
// Browser "line 920" likely 1-indexed within script body
|
||||
$out = ["script_starts_at_abs_line" => $start_abs, "total_lines" => count($lines)];
|
||||
|
||||
// lines 915 - 925 of script
|
||||
$context = [];
|
||||
for ($i = 914; $i <= 925 && $i < count($lines); $i++) {
|
||||
$L = $lines[$i];
|
||||
$context[] = [
|
||||
"script_line" => $i + 1,
|
||||
"abs_line" => $start_abs + $i,
|
||||
"length" => strlen($L),
|
||||
"content" => substr($L, 0, 300),
|
||||
"col_100_120" => strlen($L) >= 100 ? substr($L, 99, 21) : "(short)",
|
||||
];
|
||||
}
|
||||
$out["context"] = $context;
|
||||
|
||||
// Find all regex literals in lines 915-925 (they span multiple lines perhaps)
|
||||
// Also check the script_line 920 col 105 exactly
|
||||
if (isset($lines[919])) {
|
||||
$out["line_920_full"] = $lines[919];
|
||||
$out["line_920_char_at_104"] = isset($lines[919][104]) ? $lines[919][104] . " (ord " . ord($lines[919][104]) . ")" : "(out of range)";
|
||||
}
|
||||
|
||||
echo json_encode($out, JSON_PRETTY_PRINT|JSON_UNESCAPED_SLASHES);
|
||||
26
api/ambre-scripts-list.php
Normal file
@@ -0,0 +1,26 @@
|
||||
<?php
|
||||
header("Content-Type: application/json");
|
||||
$wevia = @file_get_contents("/var/www/html/wevia.html");
|
||||
|
||||
$scripts = [];
|
||||
$pos = 0;
|
||||
while (($m = strpos($wevia, "<script", $pos)) !== false) {
|
||||
$tag_end = strpos($wevia, ">", $m);
|
||||
if ($tag_end === false) break;
|
||||
$tag = substr($wevia, $m, $tag_end - $m + 1);
|
||||
$content_start = $tag_end + 1;
|
||||
$end = strpos($wevia, "</script>", $content_start);
|
||||
if ($end === false) break;
|
||||
$content = substr($wevia, $content_start, $end - $content_start);
|
||||
$start_line = substr_count(substr($wevia, 0, $m), "\n") + 1;
|
||||
$nlines = substr_count($content, "\n") + 1;
|
||||
$scripts[] = [
|
||||
"tag_start_line" => $start_line,
|
||||
"tag" => substr($tag, 0, 80),
|
||||
"content_size" => strlen($content),
|
||||
"content_lines" => $nlines,
|
||||
];
|
||||
$pos = $end + 9;
|
||||
}
|
||||
|
||||
echo json_encode($scripts, JSON_PRETTY_PRINT);
|
||||
130
api/ambre-session-chat.php
Normal file
@@ -0,0 +1,130 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-session-chat.php v3 · onboarding + empathy + identity + LLM semaphore (6σ)
|
||||
* POST {message, session_id}
|
||||
* - Semaphore-protected LLM call (max 5 concurrent)
|
||||
* - Automatic identity extraction
|
||||
* - Memory persistence
|
||||
*/
|
||||
require_once __DIR__ . "/ambre-session-memory.php";
|
||||
require_once __DIR__ . "/ambre-llm-semaphore.php";
|
||||
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
|
||||
$raw = file_get_contents("php://input");
|
||||
$in = json_decode($raw, true) ?: $_POST;
|
||||
$msg = trim($in["message"] ?? "");
|
||||
$sid = trim($in["session_id"] ?? "");
|
||||
|
||||
if (!$msg) { echo json_encode(["error"=>"message required"]); exit; }
|
||||
if (!$sid) $sid = "anon-" . substr(md5(($_SERVER["REMOTE_ADDR"] ?? "x") . time()), 0, 10);
|
||||
|
||||
$history = AmbreSessionMemory::context_messages($sid, 12);
|
||||
$turns_before = count($history);
|
||||
|
||||
$identity = null;
|
||||
$history_full = AmbreSessionMemory::load($sid);
|
||||
foreach ($history_full as $m) {
|
||||
if ($m["role"] === "meta" && strpos($m["content"], "identity:") === 0) {
|
||||
$identity = trim(substr($m["content"], 9));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
$extracted_name = null;
|
||||
$extracted_org = null;
|
||||
if (preg_match('/(?:je\s+m[\'\s]?appelle|mon\s+nom\s+est|je\s+suis|c[\'\s]?est\s+moi|i[\'\s]?m|my\s+name\s+is)\s+([A-ZÀ-Üa-zà-ü][A-ZÀ-Üa-zà-ü\-\s]{1,40}?)(?:\s+(?:de|from|at|chez|pour|travaille|,|\.|!|$))/iu', $msg, $m)) {
|
||||
$extracted_name = trim($m[1]);
|
||||
}
|
||||
if (preg_match('/(?:de|from|chez|at|travaille\s+(?:chez|pour|à))\s+([A-ZÀ-Üa-zà-ü][A-ZÀ-Üa-zà-ü0-9\-\s]{2,40}?)(?:\s*[\.,!]|\s+et|\s*$)/iu', $msg, $m)) {
|
||||
$extracted_org = trim($m[1]);
|
||||
}
|
||||
|
||||
if (($extracted_name || $extracted_org) && !$identity) {
|
||||
$id_parts = [];
|
||||
if ($extracted_name) $id_parts[] = "nom=$extracted_name";
|
||||
if ($extracted_org) $id_parts[] = "org=$extracted_org";
|
||||
$identity = implode(" · ", $id_parts);
|
||||
AmbreSessionMemory::append($sid, "meta", "identity: $identity");
|
||||
}
|
||||
|
||||
if ($turns_before === 0 && !$identity) {
|
||||
if ($extracted_name || $extracted_org) {
|
||||
$greeting_sys = "L'utilisateur vient de se présenter. Salue-le chaleureusement en utilisant son nom si connu et son entreprise si connue. Demande-lui comment tu peux l'aider. 2-3 phrases max. Reste en français.";
|
||||
} else {
|
||||
$greeting_sys = "Premier échange. Réponds brièvement à sa question, PUIS demande avec élégance son prénom et son entreprise. Style chaleureux, 3-4 phrases.";
|
||||
}
|
||||
$messages = [["role"=>"system","content"=>"Tu es WEVIA de WEVAL Consulting. $greeting_sys"]];
|
||||
$messages[] = ["role"=>"user","content"=>$msg];
|
||||
} else {
|
||||
$sys_parts = [
|
||||
"Tu es WEVIA, l'IA de WEVAL Consulting.",
|
||||
"Tu mémorises les échanges et t'adaptes au ton et contexte.",
|
||||
];
|
||||
if ($identity) {
|
||||
$sys_parts[] = "Identité: $identity. Utilise le nom naturellement.";
|
||||
}
|
||||
$sys_parts[] = "Réponds en français, concis, empathique si émotion détectée.";
|
||||
$sys_parts[] = "Si retour sur sujet antérieur, reconnais. Si changement, adapte-toi.";
|
||||
|
||||
$messages = [["role"=>"system","content"=>implode(" ", $sys_parts)]];
|
||||
foreach ($history as $h) {
|
||||
if ($h["role"] !== "meta") $messages[] = $h;
|
||||
}
|
||||
$messages[] = ["role"=>"user","content"=>$msg];
|
||||
}
|
||||
|
||||
// === LLM call WITH SEMAPHORE (6σ throttle) ===
|
||||
$t0 = microtime(true);
|
||||
$wait_ms_before = 0;
|
||||
$sem_id = AmbreLLMSemaphore::acquire();
|
||||
$wait_ms_before = round((microtime(true)-$t0)*1000);
|
||||
|
||||
if (!$sem_id) {
|
||||
// Queue full after 20s wait - fail gracefully
|
||||
echo json_encode([
|
||||
"response" => "Le service est saturé, réessayez dans quelques secondes.",
|
||||
"provider" => "ambre-session-v3",
|
||||
"intent" => "semaphore_timeout",
|
||||
"session_id" => $sid,
|
||||
"wait_ms" => $wait_ms_before,
|
||||
], JSON_UNESCAPED_UNICODE);
|
||||
exit;
|
||||
}
|
||||
|
||||
try {
|
||||
$llm_start = microtime(true);
|
||||
$raw_llm = @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"=>$messages,"max_tokens"=>1200,"temperature"=>0.5]),
|
||||
"timeout"=>30,
|
||||
],
|
||||
]));
|
||||
$llm_elapsed = round((microtime(true)-$llm_start)*1000);
|
||||
} finally {
|
||||
AmbreLLMSemaphore::release($sem_id);
|
||||
}
|
||||
|
||||
$d = @json_decode($raw_llm, true);
|
||||
$reply = $d["choices"][0]["message"]["content"] ?? "";
|
||||
if (!$reply) $reply = "Désolé, je n'ai pas pu traiter la demande. Peux-tu reformuler ?";
|
||||
|
||||
AmbreSessionMemory::append($sid, "user", $msg);
|
||||
AmbreSessionMemory::append($sid, "assistant", $reply);
|
||||
|
||||
$summary = AmbreSessionMemory::summary($sid);
|
||||
|
||||
echo json_encode([
|
||||
"response" => $reply,
|
||||
"provider" => "ambre-session-v3",
|
||||
"intent" => $turns_before === 0 ? "onboarding" : "contextual_reply",
|
||||
"session_id" => $sid,
|
||||
"turns_in_memory" => $summary["turns"],
|
||||
"history_used" => count($history),
|
||||
"identity" => $identity,
|
||||
"wait_ms" => $wait_ms_before,
|
||||
"llm_ms" => $llm_elapsed ?? 0,
|
||||
"elapsed_ms" => round((microtime(true)-$t0)*1000),
|
||||
], JSON_UNESCAPED_UNICODE);
|
||||
88
api/ambre-session-memory.php
Normal file
@@ -0,0 +1,88 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-session-memory.php · AMBRE v1 · per-session memory store
|
||||
* Stores/retrieves last N messages per session_id in /var/tmp/wevia-sessions/
|
||||
*/
|
||||
|
||||
class AmbreSessionMemory {
|
||||
const DIR = "/var/tmp/wevia-sessions";
|
||||
const MAX_TURNS = 20;
|
||||
const TTL_HOURS = 24;
|
||||
|
||||
public static function init() {
|
||||
if (!is_dir(self::DIR)) @mkdir(self::DIR, 0777, true);
|
||||
}
|
||||
|
||||
public static function path($sid) {
|
||||
self::init();
|
||||
$safe = preg_replace("/[^a-zA-Z0-9_-]/", "", $sid);
|
||||
if (!$safe) $safe = "default";
|
||||
return self::DIR . "/" . $safe . ".json";
|
||||
}
|
||||
|
||||
public static function load($sid) {
|
||||
$p = self::path($sid);
|
||||
if (!file_exists($p)) return [];
|
||||
$content = @file_get_contents($p);
|
||||
if (!$content) return [];
|
||||
$data = @json_decode($content, true);
|
||||
if (!is_array($data)) return [];
|
||||
// TTL cleanup
|
||||
$now = time();
|
||||
$data = array_filter($data, function($m) use ($now) {
|
||||
return isset($m["ts"]) && ($now - $m["ts"]) < (self::TTL_HOURS * 3600);
|
||||
});
|
||||
return array_values($data);
|
||||
}
|
||||
|
||||
public static function append($sid, $role, $content) {
|
||||
if (!$sid || !$role || !$content) return;
|
||||
$msgs = self::load($sid);
|
||||
$msgs[] = [
|
||||
"role" => $role,
|
||||
"content" => substr((string)$content, 0, 4000),
|
||||
"ts" => time(),
|
||||
];
|
||||
// Keep only last N
|
||||
if (count($msgs) > self::MAX_TURNS) {
|
||||
$msgs = array_slice($msgs, -self::MAX_TURNS);
|
||||
}
|
||||
@file_put_contents(self::path($sid), json_encode($msgs, JSON_UNESCAPED_UNICODE), LOCK_EX);
|
||||
}
|
||||
|
||||
public static function context_messages($sid, $max = 10) {
|
||||
$msgs = self::load($sid);
|
||||
$last = array_slice($msgs, -$max);
|
||||
$out = [];
|
||||
foreach ($last as $m) {
|
||||
$out[] = ["role" => $m["role"], "content" => $m["content"]];
|
||||
}
|
||||
return $out;
|
||||
}
|
||||
|
||||
public static function summary($sid) {
|
||||
$msgs = self::load($sid);
|
||||
return [
|
||||
"session" => $sid,
|
||||
"turns" => count($msgs),
|
||||
"first_ts" => !empty($msgs) ? date("c", $msgs[0]["ts"]) : null,
|
||||
"last_ts" => !empty($msgs) ? date("c", end($msgs)["ts"]) : null,
|
||||
];
|
||||
}
|
||||
|
||||
public static function clear($sid) {
|
||||
$p = self::path($sid);
|
||||
if (file_exists($p)) @unlink($p);
|
||||
}
|
||||
}
|
||||
|
||||
// Direct API usage
|
||||
if (basename($_SERVER["SCRIPT_NAME"]) === "ambre-session-memory.php") {
|
||||
header("Content-Type: application/json");
|
||||
$sid = $_GET["sid"] ?? "";
|
||||
$action = $_GET["action"] ?? "summary";
|
||||
if ($action === "summary") echo json_encode(AmbreSessionMemory::summary($sid));
|
||||
elseif ($action === "load") echo json_encode(AmbreSessionMemory::load($sid));
|
||||
elseif ($action === "clear") { AmbreSessionMemory::clear($sid); echo json_encode(["cleared"=>true]); }
|
||||
else echo json_encode(["error"=>"unknown action"]);
|
||||
}
|
||||
59
api/ambre-tool-bg-remove.php
Normal file
@@ -0,0 +1,59 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$url = trim($in["url"] ?? $in["image_url"] ?? "");
|
||||
if (!$url) { echo json_encode(["error"=>"image url required"]); exit; }
|
||||
|
||||
$t0 = microtime(true);
|
||||
|
||||
$ctx = stream_context_create(["http"=>["timeout"=>20]]);
|
||||
$orig = @file_get_contents($url, false, $ctx);
|
||||
if (!$orig) { echo json_encode(["error"=>"source fetch failed"]); exit; }
|
||||
|
||||
// Ensure cache dir exists and is writable by www-data
|
||||
$cache_dir = "/tmp/u2net_cache";
|
||||
if (!is_dir($cache_dir)) @mkdir($cache_dir, 0777, true);
|
||||
|
||||
$tmp_in = tempnam(sys_get_temp_dir(), "bgin_") . ".png";
|
||||
$tmp_out = tempnam(sys_get_temp_dir(), "bgout_") . ".png";
|
||||
file_put_contents($tmp_in, $orig);
|
||||
|
||||
$py = <<<PYTHON
|
||||
import os
|
||||
os.environ['U2NET_HOME'] = '/tmp/u2net_cache'
|
||||
os.environ['XDG_CACHE_HOME'] = '/tmp/xdg_cache'
|
||||
from rembg import remove
|
||||
with open('$tmp_in', 'rb') as f:
|
||||
input_bytes = f.read()
|
||||
output = remove(input_bytes)
|
||||
with open('$tmp_out', 'wb') as f:
|
||||
f.write(output)
|
||||
print('OK')
|
||||
PYTHON;
|
||||
|
||||
$py_file = tempnam(sys_get_temp_dir(), "bgpy_") . ".py";
|
||||
file_put_contents($py_file, $py);
|
||||
// Run with extended timeout for first-time model download (~170MB)
|
||||
$run_out = @shell_exec("timeout 300 python3 $py_file 2>&1");
|
||||
@unlink($py_file);
|
||||
|
||||
$result = @file_get_contents($tmp_out);
|
||||
@unlink($tmp_in); @unlink($tmp_out);
|
||||
|
||||
if (!$result || strlen($result) < 500) {
|
||||
echo json_encode(["error"=>"rembg processing failed", "py_out"=>substr($run_out ?? "", 0, 500)]);
|
||||
exit;
|
||||
}
|
||||
|
||||
$dir = "/var/www/html/generated";
|
||||
if (!is_dir($dir)) @mkdir($dir, 0755, true);
|
||||
$filename = "wevia-bgremove-" . date("Ymd-His") . "-" . bin2hex(random_bytes(3)) . ".png";
|
||||
file_put_contents("$dir/$filename", $result);
|
||||
|
||||
echo json_encode([
|
||||
"success"=>true, "original"=>$url,
|
||||
"url"=>"https://weval-consulting.com/generated/$filename",
|
||||
"size_kb"=>round(strlen($result)/1024, 1),
|
||||
"elapsed_ms"=>round((microtime(true)-$t0)*1000),
|
||||
"provider"=>"WEVIA BG Remove",
|
||||
]);
|
||||
22
api/ambre-tool-calc.php
Normal file
@@ -0,0 +1,22 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$expr = trim($in["expression"] ?? $in["q"] ?? "");
|
||||
if (!$expr) { echo json_encode(["error"=>"expression required"]); exit; }
|
||||
|
||||
// Sanitize: allow only numbers, operators, parens, decimal
|
||||
$safe = preg_replace('/[^0-9+\-*\/().\s,]/', '', $expr);
|
||||
$safe = str_replace(",", ".", $safe);
|
||||
if (!$safe) { echo json_encode(["error"=>"invalid expression"]); exit; }
|
||||
|
||||
try {
|
||||
$result = @eval("return ($safe);");
|
||||
echo json_encode([
|
||||
"expression" => $expr,
|
||||
"sanitized" => $safe,
|
||||
"result" => $result,
|
||||
"provider" => "WEVIA Calc",
|
||||
]);
|
||||
} catch (Throwable $e) {
|
||||
echo json_encode(["error"=>"eval failed", "expression"=>$expr]);
|
||||
}
|
||||
31
api/ambre-tool-image-upscale.php
Normal file
@@ -0,0 +1,31 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$prompt = trim($in["prompt"] ?? $in["q"] ?? "");
|
||||
if (!$prompt) { echo json_encode(["error"=>"prompt required"]); exit; }
|
||||
|
||||
$t0 = microtime(true);
|
||||
$clean = preg_replace('/[^\p{L}\p{N}\s,.\-]/u', '', $prompt);
|
||||
$clean = substr(trim($clean), 0, 300);
|
||||
$seed = rand(1, 99999);
|
||||
|
||||
// HD size 2048x2048 for hi-res
|
||||
$url = "https://image.pollinations.ai/prompt/" . urlencode($clean) . "?width=2048&height=2048&seed={$seed}&nologo=true&enhance=true&model=flux";
|
||||
$ctx = stream_context_create(["http"=>["timeout"=>60]]);
|
||||
$img = @file_get_contents($url, false, $ctx);
|
||||
|
||||
if (!$img || strlen($img) < 1000) { echo json_encode(["error"=>"upscale failed"]); exit; }
|
||||
|
||||
$dir = "/var/www/html/generated";
|
||||
if (!is_dir($dir)) @mkdir($dir, 0755, true);
|
||||
$slug = substr(preg_replace('/[^a-z0-9]+/', '-', strtolower($clean)), 0, 40);
|
||||
$filename = "wevia-hd-{$slug}-" . date("Ymd-His") . "-" . bin2hex(random_bytes(3)) . ".png";
|
||||
file_put_contents("$dir/$filename", $img);
|
||||
|
||||
echo json_encode([
|
||||
"success"=>true, "prompt"=>$clean, "size"=>"2048x2048 HD",
|
||||
"url"=>"https://weval-consulting.com/generated/$filename",
|
||||
"size_kb"=>round(strlen($img)/1024, 1),
|
||||
"elapsed_ms"=>round((microtime(true)-$t0)*1000),
|
||||
"provider"=>"WEVIA Image HD",
|
||||
]);
|
||||
55
api/ambre-tool-image.php
Normal file
@@ -0,0 +1,55 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-tool-image.php · Real image generation via Pollinations.ai (free, no auth)
|
||||
* Returns downloadable PNG from prompt
|
||||
*/
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
|
||||
$raw = file_get_contents("php://input");
|
||||
$in = json_decode($raw, true) ?: $_POST ?: $_GET;
|
||||
$prompt = trim($in["prompt"] ?? $in["q"] ?? "");
|
||||
|
||||
if (!$prompt) { echo json_encode(["error"=>"prompt required"]); exit; }
|
||||
|
||||
// Clean + translate prompt (keep as-is - Pollinations handles multilingue)
|
||||
$clean = preg_replace('/[^\p{L}\p{N}\s,.\-]/u', '', $prompt);
|
||||
$clean = substr(trim($clean), 0, 300);
|
||||
|
||||
// Pollinations API
|
||||
$seed = rand(1, 99999);
|
||||
$encoded = urlencode($clean);
|
||||
$pollinations_url = "https://image.pollinations.ai/prompt/$encoded?width=1024&height=1024&seed=$seed&nologo=true&enhance=true";
|
||||
|
||||
// Fetch image (with 30s timeout)
|
||||
$ctx = stream_context_create([
|
||||
"http" => ["timeout"=>30, "header"=>"User-Agent: WEVIA/1.0\r\n"],
|
||||
"https" => ["timeout"=>30, "header"=>"User-Agent: WEVIA/1.0\r\n"],
|
||||
]);
|
||||
$t0 = microtime(true);
|
||||
$img_data = @file_get_contents($pollinations_url, false, $ctx);
|
||||
$elapsed = round((microtime(true)-$t0)*1000);
|
||||
|
||||
if (!$img_data || strlen($img_data) < 1000) {
|
||||
echo json_encode(["error"=>"image generation failed", "prompt"=>$clean, "elapsed"=>$elapsed]);
|
||||
exit;
|
||||
}
|
||||
|
||||
// Save to /generated/
|
||||
$dir = "/var/www/html/generated";
|
||||
if (!is_dir($dir)) @mkdir($dir, 0755, true);
|
||||
$slug = preg_replace('/[^a-z0-9]+/', '-', strtolower($clean));
|
||||
$slug = substr(trim($slug, "-"), 0, 50);
|
||||
$ts = date("Ymd-His");
|
||||
$rand = bin2hex(random_bytes(3));
|
||||
$filename = "wevia-img-{$slug}-{$ts}-{$rand}.png";
|
||||
$path = "$dir/$filename";
|
||||
file_put_contents($path, $img_data);
|
||||
|
||||
echo json_encode([
|
||||
"success" => true,
|
||||
"prompt" => $clean,
|
||||
"url" => "https://weval-consulting.com/generated/$filename",
|
||||
"size_kb" => round(strlen($img_data)/1024, 1),
|
||||
"elapsed_ms" => $elapsed,
|
||||
"provider" => "WEVIA Image Engine",
|
||||
]);
|
||||
36
api/ambre-tool-ocr.php
Normal file
@@ -0,0 +1,36 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$url = trim($in["url"] ?? $in["image_url"] ?? "");
|
||||
if (!$url) { echo json_encode(["error"=>"image url required"]); exit; }
|
||||
|
||||
$t0 = microtime(true);
|
||||
|
||||
// Download image temporarily
|
||||
$tmp = tempnam(sys_get_temp_dir(), "wevia_ocr_") . ".png";
|
||||
$ctx = stream_context_create(["http"=>["timeout"=>20]]);
|
||||
$img = @file_get_contents($url, false, $ctx);
|
||||
if (!$img || strlen($img) < 100) { echo json_encode(["error"=>"image fetch failed"]); exit; }
|
||||
file_put_contents($tmp, $img);
|
||||
|
||||
// Try tesseract OCR
|
||||
$tess = @shell_exec("which tesseract 2>/dev/null");
|
||||
$text = "";
|
||||
if (trim($tess)) {
|
||||
$cmd = "tesseract " . escapeshellarg($tmp) . " - -l fra+eng 2>/dev/null";
|
||||
$text = @shell_exec($cmd);
|
||||
}
|
||||
@unlink($tmp);
|
||||
|
||||
if (!trim($text)) {
|
||||
echo json_encode(["error"=>"OCR extraction failed, text too short or tesseract missing", "image_url"=>$url]);
|
||||
exit;
|
||||
}
|
||||
|
||||
echo json_encode([
|
||||
"success"=>true, "image_url"=>$url,
|
||||
"text"=>trim($text),
|
||||
"char_count"=>strlen(trim($text)),
|
||||
"elapsed_ms"=>round((microtime(true)-$t0)*1000),
|
||||
"provider"=>"WEVIA OCR",
|
||||
]);
|
||||
283
api/ambre-tool-pdf-premium.php
Normal file
@@ -0,0 +1,283 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-tool-pdf-premium.php · Premium PDF with charts, zero template, LLM-driven
|
||||
* Circuit additif : NE TOUCHE PAS early-doc-gen (préservé).
|
||||
*
|
||||
* Flow:
|
||||
* 1. LLM génère structure JSON {title, sections[{heading,content,chart?}], conclusion}
|
||||
* 2. Inject JSON → HTML premium template avec Chart.js
|
||||
* 3. Chromium headless render → PDF full quality
|
||||
* 4. Return URL + artifact HTML preview
|
||||
*/
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$topic = trim($in["topic"] ?? $in["message"] ?? $in["q"] ?? "");
|
||||
$style = trim($in["style"] ?? "executive");
|
||||
if (!$topic) { echo json_encode(["error"=>"topic required"]); exit; }
|
||||
|
||||
$t0 = microtime(true);
|
||||
|
||||
// === 1. LLM JSON generation ===
|
||||
$sys = "Tu génères des rapports professionnels. Sortie JSON STRICT uniquement, aucun texte hors JSON:\n" .
|
||||
"{\n" .
|
||||
" \"title\": \"...\",\n" .
|
||||
" \"subtitle\": \"...\",\n" .
|
||||
" \"date\": \"$(date +'%d %B %Y')\",\n" .
|
||||
" \"author\": \"WEVIA Report Engine\",\n" .
|
||||
" \"executive_summary\": \"2-3 phrases fortes\",\n" .
|
||||
" \"sections\": [\n" .
|
||||
" {\"heading\": \"...\", \"content\": \"paragraphe 150-300 mots\", \"bullets\": [\"...\",\"...\"]},\n" .
|
||||
" (4-6 sections)\n" .
|
||||
" ],\n" .
|
||||
" \"kpis\": [{\"label\": \"...\", \"value\": \"...\", \"trend\": \"+X%\"}],\n" .
|
||||
" \"chart_data\": {\"labels\": [...], \"values\": [...], \"type\": \"bar|line|pie\", \"title\": \"...\"},\n" .
|
||||
" \"conclusion\": \"1 paragraphe de synthèse + recommandations\"\n" .
|
||||
"}";
|
||||
|
||||
$body = json_encode([
|
||||
"model" => "fast",
|
||||
"messages" => [
|
||||
["role"=>"system", "content"=>$sys],
|
||||
["role"=>"user", "content"=>"Rapport sur: $topic"],
|
||||
],
|
||||
"max_tokens" => 2500,
|
||||
"temperature" => 0.3,
|
||||
]);
|
||||
|
||||
$ch = curl_init("http://127.0.0.1:4000/v1/chat/completions");
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_RETURNTRANSFER => true,
|
||||
CURLOPT_POST => true,
|
||||
CURLOPT_TIMEOUT => 60,
|
||||
CURLOPT_HTTPHEADER => ["Content-Type: application/json"],
|
||||
CURLOPT_POSTFIELDS => $body,
|
||||
]);
|
||||
$raw = curl_exec($ch);
|
||||
curl_close($ch);
|
||||
$llm_elapsed = round((microtime(true)-$t0)*1000);
|
||||
|
||||
$d = @json_decode($raw, true);
|
||||
$content = $d["choices"][0]["message"]["content"] ?? "";
|
||||
$content = preg_replace('/```(?:json)?\s*|```/', '', $content);
|
||||
$data = @json_decode(trim($content), true);
|
||||
|
||||
if (!$data || !isset($data["title"])) {
|
||||
// Fallback simple structure
|
||||
$data = [
|
||||
"title" => ucfirst($topic),
|
||||
"subtitle" => "Rapport stratégique",
|
||||
"date" => date("d/m/Y"),
|
||||
"author" => "WEVIA Report Engine",
|
||||
"executive_summary" => "Analyse synthétique de $topic.",
|
||||
"sections" => [
|
||||
["heading" => "Contexte", "content" => "Analyse du contexte de $topic dans l'environnement actuel.", "bullets" => []],
|
||||
["heading" => "Enjeux", "content" => "Les enjeux clés à considérer.", "bullets" => []],
|
||||
["heading" => "Recommandations", "content" => "Actions recommandées.", "bullets" => []],
|
||||
],
|
||||
"kpis" => [
|
||||
["label" => "Impact estimé", "value" => "High", "trend" => ""],
|
||||
],
|
||||
"chart_data" => ["labels" => ["Q1","Q2","Q3","Q4"], "values" => [25,35,45,60], "type" => "bar", "title" => "Évolution projetée"],
|
||||
"conclusion" => "Synthèse et prochaines étapes.",
|
||||
];
|
||||
}
|
||||
|
||||
// === 2. Build HTML premium ===
|
||||
$chart_json = json_encode($data["chart_data"] ?? ["labels"=>[],"values"=>[],"type"=>"bar","title"=>""]);
|
||||
|
||||
$kpis_html = "";
|
||||
foreach ($data["kpis"] ?? [] as $k) {
|
||||
$v = htmlspecialchars($k["value"] ?? "");
|
||||
$l = htmlspecialchars($k["label"] ?? "");
|
||||
$t = htmlspecialchars($k["trend"] ?? "");
|
||||
$trend_color = (strpos($t, "+") === 0) ? "#10b981" : ((strpos($t, "-") === 0) ? "#ef4444" : "#6366f1");
|
||||
$kpis_html .= "<div class=kpi><div class=kv>$v</div><div class=kl>$l</div><div class=kt style='color:$trend_color'>$t</div></div>";
|
||||
}
|
||||
|
||||
$sections_html = "";
|
||||
foreach ($data["sections"] ?? [] as $i => $s) {
|
||||
$h = htmlspecialchars($s["heading"] ?? "");
|
||||
$c = htmlspecialchars($s["content"] ?? "");
|
||||
$bullets = "";
|
||||
foreach ($s["bullets"] ?? [] as $b) $bullets .= "<li>" . htmlspecialchars($b) . "</li>";
|
||||
$bullets = $bullets ? "<ul>$bullets</ul>" : "";
|
||||
$sections_html .= "<section class=sec><div class=snum>" . str_pad($i+1, 2, "0", STR_PAD_LEFT) . "</div><h2>$h</h2><p>$c</p>$bullets</section>";
|
||||
}
|
||||
|
||||
$title = htmlspecialchars($data["title"] ?? "Rapport");
|
||||
$subtitle = htmlspecialchars($data["subtitle"] ?? "");
|
||||
$date_str = htmlspecialchars($data["date"] ?? date("d/m/Y"));
|
||||
$author = htmlspecialchars($data["author"] ?? "WEVIA");
|
||||
$exec_summary = htmlspecialchars($data["executive_summary"] ?? "");
|
||||
$conclusion = htmlspecialchars($data["conclusion"] ?? "");
|
||||
|
||||
$html = <<<HTML
|
||||
<!DOCTYPE html>
|
||||
<html lang="fr">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>$title</title>
|
||||
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
|
||||
<style>
|
||||
@page { size: A4; margin: 15mm; }
|
||||
* { box-sizing: border-box; margin: 0; padding: 0; }
|
||||
body { font-family: 'Helvetica Neue', Arial, sans-serif; color: #1a1f3a; line-height: 1.6; font-size: 11pt; }
|
||||
.cover { page-break-after: always; height: 270mm; display: flex; flex-direction: column; justify-content: space-between; padding: 20mm 10mm; background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; border-radius: 12px; }
|
||||
.cover .brand { font-size: 14pt; letter-spacing: 3px; opacity: 0.8; }
|
||||
.cover h1 { font-size: 36pt; font-weight: 800; line-height: 1.1; margin: 40mm 0 10mm; }
|
||||
.cover .sub { font-size: 16pt; opacity: 0.9; font-weight: 300; }
|
||||
.cover .meta { font-size: 11pt; opacity: 0.7; margin-top: 30mm; }
|
||||
.cover .meta div { margin-bottom: 4mm; }
|
||||
.exec-summary { background: #f8f9ff; border-left: 4px solid #667eea; padding: 12mm 10mm; margin: 10mm 0; border-radius: 6px; font-size: 13pt; font-style: italic; }
|
||||
.kpi-row { display: flex; gap: 8mm; margin: 10mm 0; }
|
||||
.kpi { flex: 1; background: white; border: 1px solid #e5e7eb; border-radius: 10px; padding: 8mm 6mm; text-align: center; }
|
||||
.kpi .kv { font-size: 24pt; font-weight: 800; color: #667eea; }
|
||||
.kpi .kl { font-size: 10pt; color: #6b7280; margin-top: 3mm; }
|
||||
.kpi .kt { font-size: 10pt; font-weight: 600; margin-top: 2mm; }
|
||||
.sec { margin: 10mm 0; padding: 8mm; background: white; border-radius: 8px; border: 1px solid #f0f0f5; page-break-inside: avoid; }
|
||||
.sec .snum { display: inline-block; background: #667eea; color: white; padding: 2mm 4mm; border-radius: 20px; font-weight: 700; font-size: 10pt; margin-bottom: 4mm; }
|
||||
.sec h2 { font-size: 16pt; color: #1a1f3a; margin-bottom: 4mm; }
|
||||
.sec p { text-align: justify; margin-bottom: 4mm; color: #374151; }
|
||||
.sec ul { list-style: none; padding-left: 0; }
|
||||
.sec ul li { padding: 2mm 0 2mm 8mm; position: relative; color: #4b5563; }
|
||||
.sec ul li::before { content: '▸'; position: absolute; left: 2mm; color: #667eea; font-weight: 700; }
|
||||
.chart-wrap { background: white; padding: 10mm; border-radius: 10px; border: 1px solid #e5e7eb; margin: 10mm 0; page-break-inside: avoid; }
|
||||
.chart-wrap h3 { font-size: 14pt; color: #1a1f3a; margin-bottom: 5mm; }
|
||||
.chart-wrap canvas { max-height: 100mm; }
|
||||
.conclusion { margin-top: 12mm; padding: 10mm; background: linear-gradient(135deg, rgba(102,126,234,0.08), rgba(118,75,162,0.08)); border-radius: 10px; border: 1px solid rgba(102,126,234,0.2); font-size: 12pt; }
|
||||
.conclusion h3 { color: #667eea; margin-bottom: 4mm; }
|
||||
.footer { text-align: center; color: #9ca3af; font-size: 9pt; margin-top: 15mm; border-top: 1px solid #e5e7eb; padding-top: 5mm; }
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="cover">
|
||||
<div class="brand">WEVIA · WEVAL CONSULTING</div>
|
||||
<div>
|
||||
<h1>$title</h1>
|
||||
<div class="sub">$subtitle</div>
|
||||
</div>
|
||||
<div class="meta">
|
||||
<div><strong>Date</strong> · $date_str</div>
|
||||
<div><strong>Auteur</strong> · $author</div>
|
||||
<div><strong>Type</strong> · Rapport exécutif</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="exec-summary">
|
||||
<strong>Résumé exécutif · </strong> $exec_summary
|
||||
</div>
|
||||
|
||||
<div class="kpi-row">$kpis_html</div>
|
||||
|
||||
<div class="chart-wrap">
|
||||
<h3 id="chart-title">Visualisation des données</h3>
|
||||
<canvas id="mainChart"></canvas>
|
||||
</div>
|
||||
|
||||
$sections_html
|
||||
|
||||
<div class="conclusion">
|
||||
<h3>Conclusion & Perspectives</h3>
|
||||
<p>$conclusion</p>
|
||||
</div>
|
||||
|
||||
<div class="footer">
|
||||
Rapport généré par WEVIA · WEVAL Consulting · $date_str · Confidentiel
|
||||
</div>
|
||||
|
||||
<script>
|
||||
(function(){
|
||||
var d = $chart_json;
|
||||
if (!d || !d.values || !d.values.length) return;
|
||||
try {
|
||||
var el = document.getElementById('mainChart');
|
||||
if (!el) return;
|
||||
document.getElementById('chart-title').innerText = d.title || 'Données';
|
||||
var colors = ['#667eea','#764ba2','#10b981','#f59e0b','#ef4444','#3b82f6','#8b5cf6','#ec4899'];
|
||||
new Chart(el, {
|
||||
type: d.type || 'bar',
|
||||
data: {
|
||||
labels: d.labels,
|
||||
datasets: [{
|
||||
label: d.title || '',
|
||||
data: d.values,
|
||||
backgroundColor: (d.type === 'pie' || d.type === 'doughnut') ? colors.slice(0, d.values.length) : 'rgba(102,126,234,0.7)',
|
||||
borderColor: '#667eea',
|
||||
borderWidth: 2,
|
||||
tension: 0.3,
|
||||
}],
|
||||
},
|
||||
options: {
|
||||
responsive: true,
|
||||
animation: false,
|
||||
plugins: { legend: { display: d.type === 'pie' || d.type === 'doughnut' } },
|
||||
scales: (d.type === 'pie' || d.type === 'doughnut') ? {} : {
|
||||
y: { beginAtZero: true, grid: { color: '#f0f0f5' } },
|
||||
x: { grid: { display: false } },
|
||||
},
|
||||
},
|
||||
});
|
||||
} catch(e) { console.error('Chart render fail:', e); }
|
||||
})();
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
HTML;
|
||||
|
||||
// === 3. Save HTML + render with Chromium ===
|
||||
$dir = "/var/www/html/generated";
|
||||
if (!is_dir($dir)) @mkdir($dir, 0755, true);
|
||||
$slug = substr(preg_replace('/[^a-z0-9]+/', '-', strtolower($topic)), 0, 40);
|
||||
$slug = trim($slug, "-");
|
||||
$ts = date("Ymd-His");
|
||||
$rand = bin2hex(random_bytes(3));
|
||||
$html_file = "$dir/wevia-pdf-html-$slug-$ts-$rand.html";
|
||||
$pdf_file = "$dir/wevia-pdf-premium-$slug-$ts-$rand.pdf";
|
||||
|
||||
file_put_contents($html_file, $html);
|
||||
|
||||
// Use chromium headless to render (supports Chart.js !)
|
||||
$chromium_cmd = "timeout 60 /usr/bin/chromium-browser " .
|
||||
"--headless --disable-gpu --no-sandbox --disable-dev-shm-usage " .
|
||||
"--virtual-time-budget=8000 " . // wait 8s for Chart.js
|
||||
"--run-all-compositor-stages-before-draw " .
|
||||
"--print-to-pdf=" . escapeshellarg($pdf_file) . " " .
|
||||
"--print-to-pdf-no-header " .
|
||||
"file://" . escapeshellarg($html_file) . " 2>&1";
|
||||
|
||||
$chrome_out = @shell_exec($chromium_cmd);
|
||||
|
||||
if (!file_exists($pdf_file) || filesize($pdf_file) < 1000) {
|
||||
// Fallback: wkhtmltopdf (no JS but still stylish)
|
||||
$wk_cmd = "timeout 30 /usr/bin/wkhtmltopdf --quiet --enable-local-file-access " .
|
||||
"--page-size A4 --margin-top 15 --margin-bottom 15 " .
|
||||
escapeshellarg($html_file) . " " . escapeshellarg($pdf_file) . " 2>&1";
|
||||
$wk_out = @shell_exec($wk_cmd);
|
||||
}
|
||||
|
||||
if (!file_exists($pdf_file) || filesize($pdf_file) < 1000) {
|
||||
echo json_encode([
|
||||
"error" => "pdf render failed",
|
||||
"chrome_out" => substr($chrome_out ?? "", 0, 500),
|
||||
"html_url" => "https://weval-consulting.com/generated/" . basename($html_file),
|
||||
]);
|
||||
exit;
|
||||
}
|
||||
|
||||
echo json_encode([
|
||||
"success" => true,
|
||||
"title" => $data["title"],
|
||||
"topic" => $topic,
|
||||
"url" => "https://weval-consulting.com/generated/" . basename($pdf_file),
|
||||
"html_preview" => "https://weval-consulting.com/generated/" . basename($html_file),
|
||||
"pages" => max(1, intval(filesize($pdf_file) / 15000)),
|
||||
"size_kb" => round(filesize($pdf_file)/1024, 1),
|
||||
"sections" => count($data["sections"] ?? []),
|
||||
"kpis" => count($data["kpis"] ?? []),
|
||||
"has_chart" => !empty($data["chart_data"]["values"]),
|
||||
"llm_ms" => $llm_elapsed,
|
||||
"total_ms" => round((microtime(true)-$t0)*1000),
|
||||
"provider" => "WEVIA PDF Premium",
|
||||
"engine" => file_exists($pdf_file) ? "Chromium" : "wkhtmltopdf",
|
||||
]);
|
||||
29
api/ambre-tool-qr.php
Normal file
@@ -0,0 +1,29 @@
|
||||
<?php
|
||||
/**
|
||||
* ambre-tool-qr.php · QR code generator (free via goqr.me API)
|
||||
*/
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$text = trim($in["text"] ?? $in["q"] ?? "");
|
||||
$size = intval($in["size"] ?? 512);
|
||||
if (!$text) { echo json_encode(["error"=>"text required"]); exit; }
|
||||
|
||||
$t0 = microtime(true);
|
||||
$url = "https://api.qrserver.com/v1/create-qr-code/?size={$size}x{$size}&data=" . urlencode($text);
|
||||
$ctx = stream_context_create(["http"=>["timeout"=>15]]);
|
||||
$img = @file_get_contents($url, false, $ctx);
|
||||
if (!$img || strlen($img) < 100) { echo json_encode(["error"=>"qr gen failed"]); exit; }
|
||||
|
||||
$dir = "/var/www/html/generated";
|
||||
if (!is_dir($dir)) @mkdir($dir, 0755, true);
|
||||
$slug = substr(preg_replace('/[^a-z0-9]/i', '-', $text), 0, 30);
|
||||
$filename = "wevia-qr-{$slug}-" . date("Ymd-His") . "-" . bin2hex(random_bytes(3)) . ".png";
|
||||
file_put_contents("$dir/$filename", $img);
|
||||
|
||||
echo json_encode([
|
||||
"success"=>true, "text"=>$text, "size"=>$size,
|
||||
"url"=>"https://weval-consulting.com/generated/$filename",
|
||||
"size_kb"=>round(strlen($img)/1024, 1),
|
||||
"elapsed_ms"=>round((microtime(true)-$t0)*1000),
|
||||
"provider"=>"WEVIA QR",
|
||||
]);
|
||||
32
api/ambre-tool-tts.php
Normal file
@@ -0,0 +1,32 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$text = trim($in["text"] ?? "");
|
||||
$lang = $in["lang"] ?? "fr";
|
||||
if (!$text) { echo json_encode(["error"=>"text required"]); exit; }
|
||||
if (strlen($text) > 500) $text = substr($text, 0, 500);
|
||||
|
||||
$t0 = microtime(true);
|
||||
|
||||
// Use free Google TTS (no key, proxy via translate)
|
||||
$tts_url = "https://translate.google.com/translate_tts?ie=UTF-8&client=tw-ob&tl={$lang}&q=" . urlencode($text);
|
||||
|
||||
$ctx = stream_context_create([
|
||||
"http" => ["timeout"=>15, "header"=>"User-Agent: Mozilla/5.0 WEVIA/1.0\r\n"],
|
||||
]);
|
||||
$audio = @file_get_contents($tts_url, false, $ctx);
|
||||
if (!$audio || strlen($audio) < 500) { echo json_encode(["error"=>"tts generation failed"]); exit; }
|
||||
|
||||
$dir = "/var/www/html/generated";
|
||||
if (!is_dir($dir)) @mkdir($dir, 0755, true);
|
||||
$slug = substr(preg_replace('/[^a-z0-9]/i', '-', substr($text, 0, 30)), 0, 20);
|
||||
$filename = "wevia-tts-{$slug}-" . date("Ymd-His") . "-" . bin2hex(random_bytes(3)) . ".mp3";
|
||||
file_put_contents("$dir/$filename", $audio);
|
||||
|
||||
echo json_encode([
|
||||
"success"=>true, "text"=>$text, "lang"=>$lang,
|
||||
"url"=>"https://weval-consulting.com/generated/$filename",
|
||||
"size_kb"=>round(strlen($audio)/1024, 1),
|
||||
"elapsed_ms"=>round((microtime(true)-$t0)*1000),
|
||||
"provider"=>"WEVIA Voice",
|
||||
]);
|
||||
38
api/ambre-tool-url-summary.php
Normal file
@@ -0,0 +1,38 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$url = trim($in["url"] ?? "");
|
||||
if (!$url) { echo json_encode(["error"=>"url required"]); exit; }
|
||||
if (!preg_match('/^https?:\/\//', $url)) $url = "https://$url";
|
||||
|
||||
$t0 = microtime(true);
|
||||
$ctx = stream_context_create(["http"=>["timeout"=>20,"header"=>"User-Agent: Mozilla/5.0 WEVIA/1.0\r\n"]]);
|
||||
$html = @file_get_contents($url, false, $ctx);
|
||||
if (!$html) { echo json_encode(["error"=>"fetch failed", "url"=>$url]); exit; }
|
||||
|
||||
// Strip to text
|
||||
$text = preg_replace('/<script[^>]*>.*?<\/script>/is', '', $html);
|
||||
$text = preg_replace('/<style[^>]*>.*?<\/style>/is', '', $text);
|
||||
$text = html_entity_decode(strip_tags($text), ENT_QUOTES, "UTF-8");
|
||||
$text = preg_replace('/\s+/', ' ', $text);
|
||||
$text = substr(trim($text), 0, 8000);
|
||||
|
||||
// Summarize via LLM
|
||||
$sys = "Résume l'article en 5-8 points clés bullet, en français, clair et structuré.";
|
||||
$llm_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"=>$sys],
|
||||
["role"=>"user","content"=>"URL: $url\n\nContenu:\n" . $text],
|
||||
],"max_tokens"=>700,"temperature"=>0.3]),"timeout"=>30]
|
||||
]));
|
||||
$summary = @json_decode($llm_raw, true)["choices"][0]["message"]["content"] ?? "Erreur de résumé.";
|
||||
|
||||
echo json_encode([
|
||||
"url" => $url,
|
||||
"summary" => trim($summary),
|
||||
"source_length" => strlen($text),
|
||||
"elapsed_ms" => round((microtime(true)-$t0)*1000),
|
||||
"provider" => "WEVIA URL Summarizer",
|
||||
]);
|
||||
51
api/ambre-tool-web-search.php
Normal file
@@ -0,0 +1,51 @@
|
||||
<?php
|
||||
header("Content-Type: application/json; charset=utf-8");
|
||||
$in = json_decode(file_get_contents("php://input"), true) ?: $_POST ?: $_GET;
|
||||
$q = trim($in["query"] ?? $in["q"] ?? "");
|
||||
if (!$q) { echo json_encode(["error"=>"query required"]); exit; }
|
||||
|
||||
$t0 = microtime(true);
|
||||
$secrets = @file_get_contents("/etc/weval/secrets.env");
|
||||
preg_match("/^OPENROUTER_KEY=(\S+)/m", $secrets ?? "", $m);
|
||||
$or_key = $m[1] ?? "";
|
||||
|
||||
$ch = curl_init("https://openrouter.ai/api/v1/chat/completions");
|
||||
curl_setopt_array($ch, [
|
||||
CURLOPT_RETURNTRANSFER => true,
|
||||
CURLOPT_POST => true,
|
||||
CURLOPT_TIMEOUT => 30,
|
||||
CURLOPT_HTTPHEADER => [
|
||||
"Content-Type: application/json",
|
||||
"Authorization: Bearer $or_key",
|
||||
"HTTP-Referer: https://weval-consulting.com",
|
||||
"X-Title: WEVIA",
|
||||
],
|
||||
CURLOPT_POSTFIELDS => json_encode([
|
||||
"model" => "perplexity/sonar",
|
||||
"messages" => [
|
||||
["role"=>"system","content"=>"Tu es un moteur de recherche. Réponds factuellement en français, cite les sources URLs inline entre crochets."],
|
||||
["role"=>"user","content"=>$q],
|
||||
],
|
||||
"max_tokens" => 700,
|
||||
"temperature" => 0.2,
|
||||
]),
|
||||
]);
|
||||
$raw = curl_exec($ch);
|
||||
$code = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
||||
curl_close($ch);
|
||||
|
||||
$d = json_decode($raw, true);
|
||||
$answer = $d["choices"][0]["message"]["content"] ?? "";
|
||||
|
||||
preg_match_all('/https?:\/\/[^\s\)\]\"]+/', $answer, $urls_m);
|
||||
$urls = array_slice(array_unique($urls_m[0] ?? []), 0, 5);
|
||||
|
||||
echo json_encode([
|
||||
"query" => $q,
|
||||
"answer" => trim($answer),
|
||||
"sources" => $urls,
|
||||
"http" => $code,
|
||||
"error" => $d["error"]["message"] ?? null,
|
||||
"elapsed_ms" => round((microtime(true)-$t0)*1000),
|
||||
"provider" => "WEVIA Search",
|
||||
]);
|
||||