16b7e227cSYamadaMiz<?php 247595f9cSYamadaMiz 347595f9cSYamadaMizuse dokuwiki\Extension\ActionPlugin; 447595f9cSYamadaMizuse dokuwiki\Extension\EventHandler; 547595f9cSYamadaMizuse dokuwiki\Extension\Event; 647595f9cSYamadaMiz 7f9af2148SYamadaMiz/** 8b68e3724SYamadaMiz * DokuWiki Plugin Mizar Verifiable Docs (Action Component) 9f9af2148SYamadaMiz * 10f9af2148SYamadaMiz * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html 114f65af2cSYamadaMiz * @author Yamada, M. 12f9af2148SYamadaMiz */ 13f9af2148SYamadaMiz 14b68e3724SYamadaMizclass action_plugin_mizarverifiabledocs extends ActionPlugin 1547595f9cSYamadaMiz{ 1647595f9cSYamadaMiz /** 174754b0a7SYamadaMiz * Registers a callback function for a given event 1847595f9cSYamadaMiz * 1947595f9cSYamadaMiz * @param EventHandler $controller DokuWiki's event controller object 2047595f9cSYamadaMiz * @return void 2147595f9cSYamadaMiz */ 2247595f9cSYamadaMiz public function register(EventHandler $controller) 2347595f9cSYamadaMiz { 246b7e227cSYamadaMiz $controller->register_hook('AJAX_CALL_UNKNOWN', 'BEFORE', $this, 'handle_ajax_call'); 254f65af2cSYamadaMiz $controller->register_hook('TPL_CONTENT_DISPLAY', 'BEFORE', $this, 'handle_tpl_content_display'); 264f65af2cSYamadaMiz } 274f65af2cSYamadaMiz 284f65af2cSYamadaMiz /** 294f65af2cSYamadaMiz * Handles the TPL_CONTENT_DISPLAY event to insert "Hide All" button 304f65af2cSYamadaMiz * 314f65af2cSYamadaMiz * @param Event $event 324f65af2cSYamadaMiz * @param mixed $_param Unused parameter 334f65af2cSYamadaMiz * @return void 344f65af2cSYamadaMiz */ 354f65af2cSYamadaMiz public function handle_tpl_content_display(Event $event, $_param) 364f65af2cSYamadaMiz { 374f65af2cSYamadaMiz // データが文字列かどうか確認 384f65af2cSYamadaMiz if (!is_string($event->data)) { 394f65af2cSYamadaMiz error_log('handle_tpl_content_display: data is not a string! ' . print_r($event->data, true)); 404f65af2cSYamadaMiz return; 414f65af2cSYamadaMiz } 424f65af2cSYamadaMiz 434f65af2cSYamadaMiz $html = $event->data; 444f65af2cSYamadaMiz 454f65af2cSYamadaMiz // "mizarWrapper" クラスが存在するか確認 464f65af2cSYamadaMiz if (strpos($html, 'mizarWrapper') !== false) { 474f65af2cSYamadaMiz // 既にボタンが挿入されているか確認(複数回挿入しないため) 484f65af2cSYamadaMiz if (strpos($html, 'id="hideAllButton"') === false) { 494f65af2cSYamadaMiz $buttonHtml = '<div class="hideAllContainer">' 504f65af2cSYamadaMiz . '<button id="hideAllButton" class="hide-all-button">Hide All</button>' 514f65af2cSYamadaMiz . '<button id="showAllButton" class="hide-all-button" style="display:none;">Show All</button>' 52*fcd231cdSYamadaMiz . '<button id="resetAllButton" class="reset-all-button">Reset All</button>' // ★ 追加 534f65af2cSYamadaMiz . '</div>'; 544f65af2cSYamadaMiz 554f65af2cSYamadaMiz // 先頭にボタンを挿入 564f65af2cSYamadaMiz $html = $buttonHtml . $html; 574f65af2cSYamadaMiz $event->data = $html; 584f65af2cSYamadaMiz error_log('handle_tpl_content_display: "Hide All" ボタンを挿入しました。'); 594f65af2cSYamadaMiz } else { 604f65af2cSYamadaMiz // ボタンが既に存在する場合 614f65af2cSYamadaMiz error_log('handle_tpl_content_display: "Hide All" ボタンは既に存在します。'); 624f65af2cSYamadaMiz } 634f65af2cSYamadaMiz } 646b7e227cSYamadaMiz } 656b7e227cSYamadaMiz 6647595f9cSYamadaMiz /** 6747595f9cSYamadaMiz * Handles AJAX requests 6847595f9cSYamadaMiz * 6947595f9cSYamadaMiz * @param Event $event 7047595f9cSYamadaMiz * @param $param 7147595f9cSYamadaMiz */ 7247595f9cSYamadaMiz public function handle_ajax_call(Event $event, $param) 7347595f9cSYamadaMiz { 746b7e227cSYamadaMiz unset($param); // 未使用のパラメータを無効化 7547595f9cSYamadaMiz 7647595f9cSYamadaMiz switch ($event->data) { 7747595f9cSYamadaMiz case 'clear_temp_files': 7847595f9cSYamadaMiz $event->preventDefault(); 7947595f9cSYamadaMiz $event->stopPropagation(); 8042c7ffb2SYamadaMiz $this->clearTempFiles(); 8147595f9cSYamadaMiz break; 8247595f9cSYamadaMiz case 'source_sse': 836b7e227cSYamadaMiz $event->preventDefault(); 846b7e227cSYamadaMiz $event->stopPropagation(); 856b7e227cSYamadaMiz $this->handleSourceSSERequest(); 8647595f9cSYamadaMiz break; 8747595f9cSYamadaMiz case 'source_compile': 886b7e227cSYamadaMiz $event->preventDefault(); 896b7e227cSYamadaMiz $event->stopPropagation(); 906b7e227cSYamadaMiz $this->handleSourceCompileRequest(); 9147595f9cSYamadaMiz break; 9247595f9cSYamadaMiz case 'view_compile': 936b7e227cSYamadaMiz $event->preventDefault(); 946b7e227cSYamadaMiz $event->stopPropagation(); 956b7e227cSYamadaMiz $this->handleViewCompileRequest(); 9647595f9cSYamadaMiz break; 9747595f9cSYamadaMiz case 'view_sse': 986b7e227cSYamadaMiz $event->preventDefault(); 996b7e227cSYamadaMiz $event->stopPropagation(); 1006b7e227cSYamadaMiz $this->handleViewSSERequest(); 10147595f9cSYamadaMiz break; 1029fc5dc4bSYamadaMiz case 'create_combined_file': 1039fc5dc4bSYamadaMiz $event->preventDefault(); 1049fc5dc4bSYamadaMiz $event->stopPropagation(); 1059fc5dc4bSYamadaMiz $this->handle_create_combined_file(); 1069fc5dc4bSYamadaMiz break; 1076b7e227cSYamadaMiz } 1086b7e227cSYamadaMiz } 1096b7e227cSYamadaMiz 1106b7e227cSYamadaMiz // source用のコンパイルリクエスト処理 11147595f9cSYamadaMiz private function handleSourceCompileRequest() 11247595f9cSYamadaMiz { 1136b7e227cSYamadaMiz global $INPUT; 1146b7e227cSYamadaMiz $pageContent = $INPUT->post->str('content'); 1156b7e227cSYamadaMiz $mizarData = $this->extractMizarContent($pageContent); 1166b7e227cSYamadaMiz 11742c7ffb2SYamadaMiz // エラーチェックを追加 1186b7e227cSYamadaMiz if ($mizarData === null) { 1196b7e227cSYamadaMiz $this->sendAjaxResponse(false, 'Mizar content not found'); 1206b7e227cSYamadaMiz return; 12142c7ffb2SYamadaMiz } elseif (isset($mizarData['error'])) { 12242c7ffb2SYamadaMiz $this->sendAjaxResponse(false, $mizarData['error']); 12342c7ffb2SYamadaMiz return; 1246b7e227cSYamadaMiz } 1256b7e227cSYamadaMiz 1266b7e227cSYamadaMiz $filePath = $this->saveMizarContent($mizarData); 1276b7e227cSYamadaMiz 1286b7e227cSYamadaMiz session_start(); 1296b7e227cSYamadaMiz $_SESSION['source_filepath'] = $filePath; 1306b7e227cSYamadaMiz 1316b7e227cSYamadaMiz $this->sendAjaxResponse(true, 'Mizar content processed successfully'); 1326b7e227cSYamadaMiz } 1336b7e227cSYamadaMiz 1346b7e227cSYamadaMiz // source用のSSEリクエスト処理 13547595f9cSYamadaMiz private function handleSourceSSERequest() 13647595f9cSYamadaMiz { 1376b7e227cSYamadaMiz header('Content-Type: text/event-stream'); 1386b7e227cSYamadaMiz header('Cache-Control: no-cache'); 1396b7e227cSYamadaMiz 1406b7e227cSYamadaMiz session_start(); 1416b7e227cSYamadaMiz if (!isset($_SESSION['source_filepath'])) { 1426b7e227cSYamadaMiz echo "data: Mizar file path not found in session\n\n"; 1436b7e227cSYamadaMiz ob_flush(); 1446b7e227cSYamadaMiz flush(); 1456b7e227cSYamadaMiz return; 1466b7e227cSYamadaMiz } 1476b7e227cSYamadaMiz 1486b7e227cSYamadaMiz $filePath = $_SESSION['source_filepath']; 1496b7e227cSYamadaMiz $this->streamSourceOutput($filePath); 1506b7e227cSYamadaMiz 1516b7e227cSYamadaMiz echo "event: end\n"; 1526b7e227cSYamadaMiz echo "data: Compilation complete\n\n"; 1536b7e227cSYamadaMiz ob_flush(); 1546b7e227cSYamadaMiz flush(); 1556b7e227cSYamadaMiz } 1566b7e227cSYamadaMiz 15747595f9cSYamadaMiz // view用のコンパイルリクエスト処理 15847595f9cSYamadaMiz private function handleViewCompileRequest() 15947595f9cSYamadaMiz { 16047595f9cSYamadaMiz global $INPUT; 16147595f9cSYamadaMiz $content = $INPUT->post->str('content'); 16247595f9cSYamadaMiz 16347595f9cSYamadaMiz $filePath = $this->createTempFile($content); 16447595f9cSYamadaMiz 16547595f9cSYamadaMiz session_start(); 16647595f9cSYamadaMiz $_SESSION['view_filepath'] = $filePath; 16747595f9cSYamadaMiz 16847595f9cSYamadaMiz $this->sendAjaxResponse(true, 'Mizar content processed successfully'); 16947595f9cSYamadaMiz } 17047595f9cSYamadaMiz 17147595f9cSYamadaMiz // view用のSSEリクエスト処理 17247595f9cSYamadaMiz private function handleViewSSERequest() 17347595f9cSYamadaMiz { 17447595f9cSYamadaMiz header('Content-Type: text/event-stream'); 17547595f9cSYamadaMiz header('Cache-Control: no-cache'); 17647595f9cSYamadaMiz 17747595f9cSYamadaMiz session_start(); 17847595f9cSYamadaMiz if (!isset($_SESSION['view_filepath'])) { 17947595f9cSYamadaMiz echo "data: Mizar file path not found in session\n\n"; 18047595f9cSYamadaMiz ob_flush(); 18147595f9cSYamadaMiz flush(); 18247595f9cSYamadaMiz return; 18347595f9cSYamadaMiz } 18447595f9cSYamadaMiz 18547595f9cSYamadaMiz $filePath = $_SESSION['view_filepath']; 1869fc5dc4bSYamadaMiz $this->streamViewCompileOutput($filePath); 18747595f9cSYamadaMiz 18847595f9cSYamadaMiz echo "event: end\n"; 18947595f9cSYamadaMiz echo "data: Compilation complete\n\n"; 19047595f9cSYamadaMiz ob_flush(); 19147595f9cSYamadaMiz flush(); 19247595f9cSYamadaMiz } 19347595f9cSYamadaMiz 19447595f9cSYamadaMiz // Mizarコンテンツの抽出 19547595f9cSYamadaMiz private function extractMizarContent($pageContent) 19647595f9cSYamadaMiz { 1976b7e227cSYamadaMiz $pattern = '/<mizar\s+([^>]+)>(.*?)<\/mizar>/s'; 1986b7e227cSYamadaMiz preg_match_all($pattern, $pageContent, $matches, PREG_SET_ORDER); 1996b7e227cSYamadaMiz 2006b7e227cSYamadaMiz if (empty($matches)) { 20147595f9cSYamadaMiz return null; 2026b7e227cSYamadaMiz } 2036b7e227cSYamadaMiz 20442c7ffb2SYamadaMiz // 最初のファイル名を取得し、拡張子を除去 20547595f9cSYamadaMiz $fileName = trim($matches[0][1]); 20642c7ffb2SYamadaMiz $fileNameWithoutExt = preg_replace('/\.miz$/i', '', $fileName); 20742c7ffb2SYamadaMiz 20842c7ffb2SYamadaMiz // ファイル名のバリデーションを追加 20942c7ffb2SYamadaMiz if (!$this->isValidFileName($fileNameWithoutExt)) { 21042c7ffb2SYamadaMiz return ['error' => "Invalid characters in file name: '{$fileNameWithoutExt}'. Only letters, numbers, underscores (_), and apostrophes (') are allowed, up to 8 characters."]; 21142c7ffb2SYamadaMiz } 21242c7ffb2SYamadaMiz 2136b7e227cSYamadaMiz $combinedContent = ''; 2146b7e227cSYamadaMiz 2156b7e227cSYamadaMiz foreach ($matches as $match) { 21642c7ffb2SYamadaMiz $currentFileName = trim($match[1]); 21742c7ffb2SYamadaMiz $currentFileNameWithoutExt = preg_replace('/\.miz$/i', '', $currentFileName); 21842c7ffb2SYamadaMiz 21942c7ffb2SYamadaMiz if ($currentFileNameWithoutExt !== $fileNameWithoutExt) { 22042c7ffb2SYamadaMiz return ['error' => "File name mismatch in <mizar> tags: '{$fileNameWithoutExt}' and '{$currentFileNameWithoutExt}'"]; 2216b7e227cSYamadaMiz } 22242c7ffb2SYamadaMiz 22342c7ffb2SYamadaMiz // バリデーションを各ファイル名にも適用 22442c7ffb2SYamadaMiz if (!$this->isValidFileName($currentFileNameWithoutExt)) { 22542c7ffb2SYamadaMiz return ['error' => "Invalid characters in file name: '{$currentFileNameWithoutExt}'. Only letters, numbers, underscores (_), and apostrophes (') are allowed, up to 8 characters."]; 22642c7ffb2SYamadaMiz } 22742c7ffb2SYamadaMiz 2286b7e227cSYamadaMiz $combinedContent .= trim($match[2]) . "\n"; 2296b7e227cSYamadaMiz } 2306b7e227cSYamadaMiz 23142c7ffb2SYamadaMiz // ファイル名に拡張子を付加 23242c7ffb2SYamadaMiz $fullFileName = $fileNameWithoutExt . '.miz'; 23342c7ffb2SYamadaMiz 23442c7ffb2SYamadaMiz return ['fileName' => $fullFileName, 'content' => $combinedContent]; 23542c7ffb2SYamadaMiz } 23642c7ffb2SYamadaMiz 23742c7ffb2SYamadaMiz // ファイル名のバリデーション関数を追加 23842c7ffb2SYamadaMiz private function isValidFileName($fileName) 23942c7ffb2SYamadaMiz { 24042c7ffb2SYamadaMiz // ファイル名の長さをチェック(最大8文字) 24142c7ffb2SYamadaMiz if (strlen($fileName) > 8) { 24242c7ffb2SYamadaMiz return false; 24342c7ffb2SYamadaMiz } 24442c7ffb2SYamadaMiz 24542c7ffb2SYamadaMiz // 許可される文字のみを含むかチェック 24642c7ffb2SYamadaMiz if (!preg_match('/^[A-Za-z0-9_\']+$/', $fileName)) { 24742c7ffb2SYamadaMiz return false; 24842c7ffb2SYamadaMiz } 24942c7ffb2SYamadaMiz 25042c7ffb2SYamadaMiz return true; 2516b7e227cSYamadaMiz } 2526b7e227cSYamadaMiz 25347595f9cSYamadaMiz // Mizarコンテンツの保存 25447595f9cSYamadaMiz private function saveMizarContent($mizarData) 25547595f9cSYamadaMiz { 2566b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\'); 2576b7e227cSYamadaMiz $filePath = $workPath . "/TEXT/" . $mizarData['fileName']; 2586b7e227cSYamadaMiz file_put_contents($filePath, $mizarData['content']); 2596b7e227cSYamadaMiz return $filePath; 2606b7e227cSYamadaMiz } 2616b7e227cSYamadaMiz 26247595f9cSYamadaMiz // source用の出力をストリーム 26347595f9cSYamadaMiz private function streamSourceOutput($filePath) 26447595f9cSYamadaMiz { 2656b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\'); 2666b7e227cSYamadaMiz chdir($workPath); 2676b7e227cSYamadaMiz 2686b7e227cSYamadaMiz $command = "miz2prel " . escapeshellarg($filePath); 2696b7e227cSYamadaMiz $process = proc_open($command, array(1 => array("pipe", "w")), $pipes); 2706b7e227cSYamadaMiz 2716b7e227cSYamadaMiz if (is_resource($process)) { 2726b7e227cSYamadaMiz while ($line = fgets($pipes[1])) { 2736b7e227cSYamadaMiz echo "data: " . $line . "\n\n"; 2746b7e227cSYamadaMiz ob_flush(); 2756b7e227cSYamadaMiz flush(); 2766b7e227cSYamadaMiz } 2776b7e227cSYamadaMiz fclose($pipes[1]); 2789fc5dc4bSYamadaMiz 2799fc5dc4bSYamadaMiz // エラー処理の追加 2809fc5dc4bSYamadaMiz $errFilename = str_replace('.miz', '.err', $filePath); 2819fc5dc4bSYamadaMiz if ($this->handleCompilationErrors($errFilename, rtrim($this->getConf('mizar_share_dir'), '/\\') . '/mizar.msg')) { 2829fc5dc4bSYamadaMiz // エラーがあった場合は処理を終了 2839fc5dc4bSYamadaMiz proc_close($process); 2849fc5dc4bSYamadaMiz return; 2859fc5dc4bSYamadaMiz } 2869fc5dc4bSYamadaMiz 2876b7e227cSYamadaMiz proc_close($process); 2886b7e227cSYamadaMiz } 2896b7e227cSYamadaMiz } 2906b7e227cSYamadaMiz 29147595f9cSYamadaMiz // view用の一時ファイル作成 29247595f9cSYamadaMiz private function createTempFile($content) 29347595f9cSYamadaMiz { 2946b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/'; 29547595f9cSYamadaMiz $uniqueName = str_replace('.', '_', uniqid('tmp', true)); 2966b7e227cSYamadaMiz $tempFilename = $workPath . $uniqueName . ".miz"; 2976b7e227cSYamadaMiz file_put_contents($tempFilename, $content); 2986b7e227cSYamadaMiz return $tempFilename; 2996b7e227cSYamadaMiz } 3006b7e227cSYamadaMiz 30142c7ffb2SYamadaMiz // 一時ファイルの削除 30247595f9cSYamadaMiz private function clearTempFiles() 30347595f9cSYamadaMiz { 30447595f9cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/'; 30547595f9cSYamadaMiz $files = glob($workPath . '*'); // TEXTフォルダ内のすべてのファイルを取得 30647595f9cSYamadaMiz 30747595f9cSYamadaMiz $errors = []; 30847595f9cSYamadaMiz foreach ($files as $file) { 30947595f9cSYamadaMiz if (is_file($file)) { 31047595f9cSYamadaMiz // ファイルが使用中かどうか確認 31147595f9cSYamadaMiz if (!$this->is_file_locked($file)) { 31247595f9cSYamadaMiz $retries = 3; // 最大3回リトライ 31347595f9cSYamadaMiz while ($retries > 0) { 31447595f9cSYamadaMiz if (unlink($file)) { 31547595f9cSYamadaMiz break; // 削除成功 31647595f9cSYamadaMiz } 3175cbf3a53SYamadaMiz $errors[] = "Error deleting $file: " . error_get_last()['message']; 31847595f9cSYamadaMiz $retries--; 31947595f9cSYamadaMiz sleep(1); // 1秒待ってリトライ 32047595f9cSYamadaMiz } 32147595f9cSYamadaMiz if ($retries === 0) { 32247595f9cSYamadaMiz $errors[] = "Failed to delete: $file"; // 削除失敗 32347595f9cSYamadaMiz } 32447595f9cSYamadaMiz } else { 32547595f9cSYamadaMiz $errors[] = "File is locked: $file"; // ファイルがロックされている 32647595f9cSYamadaMiz } 32747595f9cSYamadaMiz } 32847595f9cSYamadaMiz } 32947595f9cSYamadaMiz 33047595f9cSYamadaMiz if (empty($errors)) { 33147595f9cSYamadaMiz $this->sendAjaxResponse(true, 'Temporary files cleared successfully'); 33247595f9cSYamadaMiz } else { 33347595f9cSYamadaMiz $this->sendAjaxResponse(false, 'Some files could not be deleted', $errors); 33447595f9cSYamadaMiz } 33547595f9cSYamadaMiz } 33647595f9cSYamadaMiz 33747595f9cSYamadaMiz // ファイルがロックされているかをチェックする関数 33847595f9cSYamadaMiz private function is_file_locked($file) 33947595f9cSYamadaMiz { 34047595f9cSYamadaMiz $fileHandle = @fopen($file, "r+"); 34147595f9cSYamadaMiz 34247595f9cSYamadaMiz if ($fileHandle === false) { 34347595f9cSYamadaMiz return true; // ファイルが開けない、つまりロックされているかアクセス権がない 34447595f9cSYamadaMiz } 34547595f9cSYamadaMiz 34647595f9cSYamadaMiz $locked = !flock($fileHandle, LOCK_EX | LOCK_NB); // ロックの取得を試みる(非ブロッキングモード) 34747595f9cSYamadaMiz 34847595f9cSYamadaMiz fclose($fileHandle); 34947595f9cSYamadaMiz return $locked; // ロックが取得できなければファイルはロックされている 35047595f9cSYamadaMiz } 35147595f9cSYamadaMiz 3529fc5dc4bSYamadaMiz // View用コンパイル出力のストリーム 3539fc5dc4bSYamadaMiz private function streamViewCompileOutput($filePath) 35447595f9cSYamadaMiz { 3556b7e227cSYamadaMiz $workPath = $this->getConf('mizar_work_dir'); 3566b7e227cSYamadaMiz $sharePath = rtrim($this->getConf('mizar_share_dir'), '/\\') . '/'; 3576b7e227cSYamadaMiz 3586b7e227cSYamadaMiz chdir($workPath); 3596b7e227cSYamadaMiz 3609fc5dc4bSYamadaMiz $errFilename = str_replace('.miz', '.err', $filePath); 3616b7e227cSYamadaMiz $command = "makeenv " . escapeshellarg($filePath); 3626b7e227cSYamadaMiz $process = proc_open($command, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $pipes); 3636b7e227cSYamadaMiz 3646b7e227cSYamadaMiz if (is_resource($process)) { 3656b7e227cSYamadaMiz while ($line = fgets($pipes[1])) { 3666b7e227cSYamadaMiz echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3676b7e227cSYamadaMiz ob_flush(); 3686b7e227cSYamadaMiz flush(); 3696b7e227cSYamadaMiz } 3706b7e227cSYamadaMiz fclose($pipes[1]); 3716b7e227cSYamadaMiz 3726b7e227cSYamadaMiz while ($line = fgets($pipes[2])) { 3736b7e227cSYamadaMiz echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3746b7e227cSYamadaMiz ob_flush(); 3756b7e227cSYamadaMiz flush(); 3766b7e227cSYamadaMiz } 3776b7e227cSYamadaMiz fclose($pipes[2]); 3786b7e227cSYamadaMiz proc_close($process); 3796b7e227cSYamadaMiz 3801174571dSYamadaMiz // makeenvのエラー処理 3819fc5dc4bSYamadaMiz if ($this->handleCompilationErrors($errFilename, $sharePath . '/mizar.msg')) { 3826b7e227cSYamadaMiz return; 3836b7e227cSYamadaMiz } 3846b7e227cSYamadaMiz 3856b7e227cSYamadaMiz // verifierの実行 3866b7e227cSYamadaMiz $exePath = rtrim($this->getConf('mizar_exe_dir'), '/\\') . '/'; 3876b7e227cSYamadaMiz $verifierPath = escapeshellarg($exePath . "verifier"); 3886b7e227cSYamadaMiz if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') { 3896b7e227cSYamadaMiz $verifierPath .= ".exe"; 3906b7e227cSYamadaMiz } 3916b7e227cSYamadaMiz $verifierCommand = $verifierPath . " -q -l " . escapeshellarg("TEXT/" . basename($filePath)); 3926b7e227cSYamadaMiz 3936b7e227cSYamadaMiz $verifierProcess = proc_open($verifierCommand, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $verifierPipes); 3946b7e227cSYamadaMiz 3956b7e227cSYamadaMiz if (is_resource($verifierProcess)) { 3966b7e227cSYamadaMiz while ($line = fgets($verifierPipes[1])) { 3976b7e227cSYamadaMiz echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3986b7e227cSYamadaMiz ob_flush(); 3996b7e227cSYamadaMiz flush(); 4006b7e227cSYamadaMiz } 4016b7e227cSYamadaMiz fclose($verifierPipes[1]); 4026b7e227cSYamadaMiz 4036b7e227cSYamadaMiz while ($line = fgets($verifierPipes[2])) { 4046b7e227cSYamadaMiz echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 4056b7e227cSYamadaMiz ob_flush(); 4066b7e227cSYamadaMiz flush(); 4076b7e227cSYamadaMiz } 4086b7e227cSYamadaMiz fclose($verifierPipes[2]); 4096b7e227cSYamadaMiz proc_close($verifierProcess); 4101174571dSYamadaMiz 4111174571dSYamadaMiz // verifierのエラー処理 4121174571dSYamadaMiz if ($this->handleCompilationErrors($errFilename, $sharePath . '/mizar.msg')) { 4131174571dSYamadaMiz return; 4141174571dSYamadaMiz } 4156b7e227cSYamadaMiz } else { 4166b7e227cSYamadaMiz echo "data: ERROR: Failed to execute verifier command.\n\n"; 4176b7e227cSYamadaMiz ob_flush(); 4186b7e227cSYamadaMiz flush(); 4196b7e227cSYamadaMiz } 4206b7e227cSYamadaMiz } else { 4216b7e227cSYamadaMiz echo "data: ERROR: Failed to execute makeenv command.\n\n"; 4226b7e227cSYamadaMiz ob_flush(); 4236b7e227cSYamadaMiz flush(); 4246b7e227cSYamadaMiz } 4256b7e227cSYamadaMiz } 4266b7e227cSYamadaMiz 42747595f9cSYamadaMiz private function getMizarErrorMessages($mizarMsgFile) 42847595f9cSYamadaMiz { 4296b7e227cSYamadaMiz $errorMessages = []; 4306b7e227cSYamadaMiz $lines = file($mizarMsgFile, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); 4316b7e227cSYamadaMiz 4326b7e227cSYamadaMiz $isReadingErrorMsg = false; 4336b7e227cSYamadaMiz $key = 0; 4346b7e227cSYamadaMiz 4356b7e227cSYamadaMiz foreach ($lines as $line) { 4366b7e227cSYamadaMiz if (preg_match('/# (\d+)/', $line, $matches)) { 4376b7e227cSYamadaMiz $isReadingErrorMsg = true; 4386b7e227cSYamadaMiz $key = intval($matches[1]); 4396b7e227cSYamadaMiz } elseif ($isReadingErrorMsg) { 4406b7e227cSYamadaMiz $errorMessages[$key] = $line; 4416b7e227cSYamadaMiz $isReadingErrorMsg = false; 4426b7e227cSYamadaMiz } 4436b7e227cSYamadaMiz } 4446b7e227cSYamadaMiz 4456b7e227cSYamadaMiz return $errorMessages; 4466b7e227cSYamadaMiz } 4476b7e227cSYamadaMiz 44847595f9cSYamadaMiz private function sendAjaxResponse($success, $message, $data = '') 44947595f9cSYamadaMiz { 4506b7e227cSYamadaMiz header('Content-Type: application/json'); 4516b7e227cSYamadaMiz echo json_encode(['success' => $success, 'message' => $message, 'data' => $data]); 4526b7e227cSYamadaMiz exit; 4536b7e227cSYamadaMiz } 4549fc5dc4bSYamadaMiz 4559fc5dc4bSYamadaMiz private function handleCompilationErrors($errFilename, $mizarMsgFilePath) 4569fc5dc4bSYamadaMiz { 4579fc5dc4bSYamadaMiz if (file_exists($errFilename)) { 4589fc5dc4bSYamadaMiz $errors = []; 4599fc5dc4bSYamadaMiz $errorLines = file($errFilename, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); 4609fc5dc4bSYamadaMiz foreach ($errorLines as $errorLine) { 4619fc5dc4bSYamadaMiz if (preg_match('/(\d+)\s+(\d+)\s+(\d+)/', $errorLine, $matches)) { 4629fc5dc4bSYamadaMiz $errorCode = intval($matches[3]); 4639fc5dc4bSYamadaMiz $errors[] = [ 4649fc5dc4bSYamadaMiz 'code' => $errorCode, 4659fc5dc4bSYamadaMiz 'line' => intval($matches[1]), 4669fc5dc4bSYamadaMiz 'column' => intval($matches[2]), 4679fc5dc4bSYamadaMiz 'message' => $this->getMizarErrorMessages($mizarMsgFilePath)[$errorCode] ?? 'Unknown error' 4689fc5dc4bSYamadaMiz ]; 4699fc5dc4bSYamadaMiz } 4709fc5dc4bSYamadaMiz } 4719fc5dc4bSYamadaMiz if (!empty($errors)) { 4729fc5dc4bSYamadaMiz echo "event: compileErrors\n"; 4739fc5dc4bSYamadaMiz echo "data: " . json_encode($errors) . "\n\n"; 4749fc5dc4bSYamadaMiz ob_flush(); 4759fc5dc4bSYamadaMiz flush(); 4769fc5dc4bSYamadaMiz return true; 4779fc5dc4bSYamadaMiz } 4789fc5dc4bSYamadaMiz } 4799fc5dc4bSYamadaMiz return false; 4809fc5dc4bSYamadaMiz } 4819fc5dc4bSYamadaMiz 4829fc5dc4bSYamadaMiz private function handle_create_combined_file() 4839fc5dc4bSYamadaMiz { 4849fc5dc4bSYamadaMiz global $INPUT; 4859fc5dc4bSYamadaMiz 4869fc5dc4bSYamadaMiz // 投稿されたコンテンツを取得 4879fc5dc4bSYamadaMiz $combinedContent = $INPUT->post->str('content'); 488a0444036SYamadaMiz $filename = $INPUT->post->str('filename', 'combined_file.miz'); // デフォルトのファイル名を指定 4899fc5dc4bSYamadaMiz 490a0444036SYamadaMiz // ファイルを保存せず、コンテンツを直接返す 491a0444036SYamadaMiz if (!empty($combinedContent)) { 492a0444036SYamadaMiz // ファイルの内容をレスポンスで返す(PHP側でファイルを作成しない) 493a0444036SYamadaMiz $this->sendAjaxResponse(true, 'File created successfully', [ 494a0444036SYamadaMiz 'filename' => $filename, 495a0444036SYamadaMiz 'content' => $combinedContent 496a0444036SYamadaMiz ]); 497a0444036SYamadaMiz error_log("File content sent: " . $filename); 4989fc5dc4bSYamadaMiz } else { 499a0444036SYamadaMiz $this->sendAjaxResponse(false, 'Content is empty, no file created'); 5009fc5dc4bSYamadaMiz } 5019fc5dc4bSYamadaMiz } 5026b7e227cSYamadaMiz} 503