nis2-agile/public/simulate-nis2.php
DevEnv nis2-agile 49c62ab811 [FIX] simulate: proc_open streaming SSE (pattern lg231) + NIS2_SSE flag
- public/simulate-nis2.php: riscritta con proc_open come lg231 test-runner.
  Lancia simulate-nis2.php come subprocess CLI con NIS2_SSE=1, streama
  ogni riga SSE al browser immediatamente senza buffering Apache/FPM.
  Stderr del subprocess → eventi SSE 'error' visibili nel terminale.

- simulate-nis2.php: aggiunto supporto NIS2_SSE=1 (env var).
  Quando NIS2_SSE=1, IS_CLI=false → output SSE anche da sottoprocesso.
  API_BASE usa sempre server prod in modalità subprocess.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-10 10:51:05 +01:00

103 lines
4.2 KiB
PHP

<?php
/**
* NIS2 Agile — Wrapper SSE per simulate-nis2.php (pattern proc_open, lg231-inspired)
*
* DocumentRoot è public/ → il file principale è fuori dalla web root.
* Usa proc_open per eseguire il simulatore come sottoprocesso CLI con NIS2_SSE=1,
* streamando ogni riga SSE al browser man mano che viene prodotta (no buffering).
*
* Vantaggi rispetto a require diretto:
* - PHP-FPM gestisce solo il proxy SSE (leggero, non può fare timeout del simulatore)
* - Output reale in tempo reale (nessun buffering Apache/FPM)
* - Errori del simulatore visibili come eventi SSE
*
* URL: https://nis2.agile.software/simulate-nis2.php?sim=all
*/
set_time_limit(0);
ignore_user_abort(true);
ini_set('memory_limit', '64M');
// ── SSE headers ──────────────────────────────────────────────────────────────
header('Content-Type: text/event-stream');
header('Cache-Control: no-cache');
header('X-Accel-Buffering: no');
header('Connection: keep-alive');
ob_implicit_flush(true);
while (ob_get_level()) ob_end_flush();
// ── Parametri ────────────────────────────────────────────────────────────────
$simParam = strtolower(trim($_GET['sim'] ?? 'all'));
$scriptPath = realpath(__DIR__ . '/../simulate-nis2.php');
if (!$scriptPath || !is_file($scriptPath)) {
echo 'data: ' . json_encode(['t' => 'error', 'm' => 'simulate-nis2.php non trovato']) . "\n\n";
echo 'data: ' . json_encode(['t' => 'done', 'stats' => ['pass' => 0, 'fail' => 1, 'skip' => 0, 'warn' => 0]]) . "\n\n";
flush();
exit;
}
// ── Ambiente subprocess ───────────────────────────────────────────────────────
// NIS2_SSE=1 → il simulatore usa output SSE anche quando php_sapi_name()='cli'
// NIS2_SIM=SIM06 → esegue solo lo scenario B2B (opzionale)
$env = [];
foreach ($_ENV ?: [] as $k => $v) {
if (is_string($v)) $env[$k] = $v;
}
$env['NIS2_SSE'] = '1';
if ($simParam === 'sim06') {
$env['NIS2_SIM'] = 'SIM06';
}
// ── Avvia subprocess (pattern lg231 runCommand) ───────────────────────────────
$descriptors = [0 => ['pipe', 'r'], 1 => ['pipe', 'w'], 2 => ['pipe', 'w']];
$cwd = dirname($scriptPath);
$cmd = PHP_BINARY . ' ' . escapeshellarg($scriptPath);
$proc = proc_open($cmd, $descriptors, $pipes, $cwd, $env);
if (!is_resource($proc)) {
echo 'data: ' . json_encode(['t' => 'error', 'm' => 'Impossibile avviare il simulatore']) . "\n\n";
echo 'data: ' . json_encode(['t' => 'done', 'stats' => ['pass' => 0, 'fail' => 1, 'skip' => 0, 'warn' => 0]]) . "\n\n";
flush();
exit;
}
fclose($pipes[0]);
stream_set_blocking($pipes[1], false);
stream_set_blocking($pipes[2], false);
// ── Streaming riga per riga ──────────────────────────────────────────────────
// Il simulatore emette direttamente eventi SSE (data: {...}\n\n).
// Il wrapper li passa al browser immediatamente senza buffering.
while (true) {
$chunk = fread($pipes[1], 8192);
if ($chunk !== false && $chunk !== '') {
echo $chunk;
flush();
}
// Stderr → errori PHP fatali visibili nel terminale SSE
$err = fread($pipes[2], 1024);
if ($err !== false && $err !== '') {
foreach (explode("\n", trim($err)) as $errLine) {
if (trim($errLine) !== '') {
echo 'data: ' . json_encode(['t' => 'error', 'm' => '[stderr] ' . trim($errLine)]) . "\n\n";
}
}
flush();
}
$status = proc_get_status($proc);
if (!$status['running']) {
// Leggi eventuale output residuo dopo la fine del processo
$tail = stream_get_contents($pipes[1]);
if ($tail) { echo $tail; flush(); }
fclose($pipes[1]);
fclose($pipes[2]);
proc_close($proc);
break;
}
usleep(50000); // 50ms — bilanciamento latenza/CPU (come lg231)
}