12 lines
13 KiB
PHP
12 lines
13 KiB
PHP
<?php
|
|
header("Content-Type: application/json");
|
|
$base = "/var/www/html/api/ambre-pw-tests/tests";
|
|
$content = base64_decode("const { test, expect } = require("@playwright/test");
const fs = require("fs");
const path = require("path");

const CAPABILITIES = [
  { name: "PDF",      msg: "Genere un PDF sur: strategie enterprise 2026",         regex: /https:\/\/\S+?\.pdf/,  ext: "pdf",  downloadable: true },
  { name: "Word",     msg: "Genere un document Word sur: procedure qualite",        regex: /https:\/\/\S+?\.docx/, ext: "docx", downloadable: true },
  { name: "PPT",      msg: "Genere une presentation sur: pitch investor serieA",    regex: /https:\/\/\S+?\.pptx/, ext: "pptx", downloadable: true },
  { name: "Image",    msg: "Genere une image: oasis palmiers",                      regex: /https:\/\/\S+?\.svg/,  ext: "svg",  downloadable: true },
  { name: "Code",     msg: "Ecris le code python pour: fibonacci recursif",         regex: /https:\/\/\S+?\.py/,   ext: "py",   downloadable: true },
  { name: "Mermaid",  msg: "Genere un schema mermaid pour: workflow commandes",     regex: /graph TD[\s\S]{10,500}/, ext: null, downloadable: false },
  { name: "Traduire", msg: "Traduis en anglais: merci beaucoup pour votre aide",    regex: /English:\s*\S[\s\S]{3,200}/, ext: null, downloadable: false },
];

test("V9 real-file verification · download + screenshot proof", async ({ page, context }) => {
  test.setTimeout(600000);
  
  await page.goto("/wevia.html");
  await page.waitForLoadState("networkidle");
  await page.waitForTimeout(2500);
  await page.screenshot({ path: "output/v9-00-chat-initial.png" });
  
  const results = [];
  
  for (let i = 0; i < CAPABILITIES.length; i++) {
    const cap = CAPABILITIES[i];
    const num = String(i + 1).padStart(2, "0");
    console.log("\n═══ [" + num + "/7] " + cap.name + " ═══");
    
    try {
      // 1. Send message
      const input = page.locator("#msgInput");
      await input.click({ force: true });
      await page.keyboard.press("Control+A");
      await page.keyboard.press("Delete");
      await input.fill(cap.msg);
      await page.waitForTimeout(400);
      await input.press("Enter");
      console.log("  📤 sent: " + cap.msg);
      
      // 2. Wait for response matching regex
      const waitStart = Date.now();
      let matchedUrl = null;
      let matchedContent = null;
      while (Date.now() - waitStart < 45000) {
        const bodyText = await page.evaluate(() => document.body.innerText);
        const m = bodyText.match(cap.regex);
        if (m) { 
          matchedContent = m[0];
          if (cap.downloadable) matchedUrl = m[0];
          break;
        }
        await page.waitForTimeout(1500);
      }
      const elapsed = ((Date.now() - waitStart) / 1000).toFixed(1);
      
      if (!matchedContent) {
        console.log("  ❌ no match after " + elapsed + "s");
        await page.screenshot({ path: "output/v9-" + num + "-" + cap.name + "-FAIL.png" });
        results.push({ name: cap.name, ok: false, reason: "no_response_match" });
        continue;
      }
      console.log("  ✅ response matched in " + elapsed + "s");
      
      // Scroll + chat screenshot
      await page.evaluate(() => { const m = document.getElementById("messages"); if (m) m.scrollTop = m.scrollHeight; });
      await page.waitForTimeout(1500);
      await page.screenshot({ path: "output/v9-" + num + "-" + cap.name + "-chat.png" });
      
      // 3. For downloadable: open the file URL in a new tab and screenshot IT
      if (cap.downloadable && matchedUrl) {
        console.log("  📥 file URL: " + matchedUrl);
        
        // Check file via HEAD request (fetch in browser)
        const httpInfo = await page.evaluate(async (u) => {
          try {
            const r = await fetch(u, { method: "HEAD" });
            return { ok: r.ok, status: r.status, type: r.headers.get("content-type"), size: r.headers.get("content-length") };
          } catch (e) { return { err: e.message }; }
        }, matchedUrl);
        console.log("  HTTP: " + JSON.stringify(httpInfo));
        
        if (httpInfo.ok) {
          // Open the file in a new page and screenshot
          const newPage = await context.newPage();
          try {
            // PDFs open in viewer, others may download; for SVG/HTML we can render
            if (cap.ext === "pdf" || cap.ext === "svg") {
              await newPage.goto(matchedUrl, { waitUntil: "load", timeout: 30000 });
              await newPage.waitForTimeout(3000);
              await newPage.screenshot({ path: "output/v9-" + num + "-" + cap.name + "-FILE.png", fullPage: false });
              console.log("  📸 FILE rendered screenshot: v9-" + num + "-" + cap.name + "-FILE.png");
            } else if (cap.ext === "py") {
              // Display code as text
              await newPage.goto(matchedUrl, { waitUntil: "load", timeout: 30000 });
              await newPage.waitForTimeout(2000);
              await newPage.screenshot({ path: "output/v9-" + num + "-" + cap.name + "-FILE.png", fullPage: false });
              console.log("  📸 CODE screenshot: v9-" + num + "-" + cap.name + "-FILE.png");
            } else if (cap.ext === "docx" || cap.ext === "pptx") {
              // Office docs: download binary + save + note we cannot render in browser
              const r = await context.request.get(matchedUrl);
              const buffer = await r.body();
              const savePath = path.join("output", "v9-" + num + "-" + cap.name + ".") + cap.ext;
              fs.writeFileSync(savePath, buffer);
              console.log("  💾 binary saved: " + savePath + " (" + buffer.length + "B)");
              
              // Render an HTML page that shows the download link + metadata as "visual proof"
              const proofHtml = `<!DOCTYPE html><html><head><style>
                body{font-family:system-ui;padding:40px;background:linear-gradient(135deg,#667eea 0%,#764ba2 100%);min-height:100vh;color:#fff}
                .card{background:#fff;color:#333;padding:40px;border-radius:16px;box-shadow:0 20px 60px rgba(0,0,0,0.3);max-width:800px;margin:auto}
                .title{font-size:32px;margin-bottom:8px;color:#1a1a2e}
                .status{display:inline-block;background:#10b981;color:#fff;padding:6px 16px;border-radius:20px;font-size:14px;margin-bottom:20px}
                .row{padding:12px 0;border-bottom:1px solid #eee;display:flex;justify-content:space-between}
                .label{color:#666}
                .value{font-weight:600;color:#1a1a2e}
                a{color:#667eea;text-decoration:none;font-weight:600}
              </style></head><body>
              <div class="card">
                <div class="status">✅ FILE VERIFIED</div>
                <div class="title">${cap.name} · ${cap.ext.toUpperCase()} File</div>
                <div class="row"><span class="label">URL:</span><span class="value" style="word-break:break-all"><a href="${matchedUrl}">${matchedUrl}</a></span></div>
                <div class="row"><span class="label">HTTP Status:</span><span class="value">${httpInfo.status} OK</span></div>
                <div class="row"><span class="label">Content-Type:</span><span class="value">${httpInfo.type}</span></div>
                <div class="row"><span class="label">File size:</span><span class="value">${(buffer.length/1024).toFixed(1)} KB</span></div>
                <div class="row"><span class="label">Downloaded locally:</span><span class="value">v9-${num}-${cap.name}.${cap.ext}</span></div>
                <p style="margin-top:30px;color:#666;font-size:13px">Binary Office document · rendered proof card · click URL to download in browser</p>
              </div></body></html>`;
              await newPage.setContent(proofHtml);
              await newPage.waitForTimeout(1000);
              await newPage.screenshot({ path: "output/v9-" + num + "-" + cap.name + "-FILE.png", fullPage: false });
              console.log("  📸 proof card screenshot: v9-" + num + "-" + cap.name + "-FILE.png");
            }
            
            await newPage.close();
            results.push({ name: cap.name, ok: true, url: matchedUrl, size: httpInfo.size, type: httpInfo.type });
          } catch (e) {
            console.log("  ⚠️ file render err: " + e.message.substring(0, 100));
            await newPage.close().catch(() => {});
            results.push({ name: cap.name, ok: false, reason: "render_err" });
          }
        } else {
          console.log("  ❌ file not accessible");
          results.push({ name: cap.name, ok: false, reason: "http_fail" });
        }
      } else {
        // Non-downloadable (Mermaid, Traduire): content match is the proof
        console.log("  📝 inline content: " + matchedContent.substring(0, 80).replace(/\n/g, " "));
        results.push({ name: cap.name, ok: true, inline: matchedContent.substring(0, 200) });
      }
      
      await page.waitForTimeout(1500);
    } catch (e) {
      console.log("  ❌ error: " + e.message.substring(0, 120));
      results.push({ name: cap.name, ok: false, reason: "exception" });
    }
  }
  
  // Final summary
  const passCount = results.filter(r => r.ok).length;
  console.log("\n═══════════════ V9 BILAN ═══════════════");
  console.log(passCount + "/7 capabilities PASS with REAL FILE VERIFICATION");
  results.forEach(r => {
    const mark = r.ok ? "✅" : "❌";
    const detail = r.url ? (r.url + " · " + (r.size || "?") + "B · " + (r.type || "?")) : (r.inline || r.reason || "");
    console.log("  " + mark + " " + r.name + " · " + detail.substring(0, 140));
  });
  
  fs.writeFileSync("output/v9-results.json", JSON.stringify(results, null, 2));
});
");
|
|
$written = @file_put_contents("$base/capabilities-v9.spec.js", $content);
|
|
// Remove old specs
|
|
@unlink("$base/capabilities-v8.spec.js");
|
|
echo json_encode([
|
|
"written" => $written,
|
|
"specs_after" => array_map("basename", glob("$base/*.spec.js")),
|
|
]);
|