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 11*4f65af2cSYamadaMiz * @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'); 25*4f65af2cSYamadaMiz $controller->register_hook('TPL_CONTENT_DISPLAY', 'BEFORE', $this, 'handle_tpl_content_display'); 26*4f65af2cSYamadaMiz } 27*4f65af2cSYamadaMiz 28*4f65af2cSYamadaMiz /** 29*4f65af2cSYamadaMiz * Handles the TPL_CONTENT_DISPLAY event to insert "Hide All" button 30*4f65af2cSYamadaMiz * 31*4f65af2cSYamadaMiz * @param Event $event 32*4f65af2cSYamadaMiz * @param mixed $_param Unused parameter 33*4f65af2cSYamadaMiz * @return void 34*4f65af2cSYamadaMiz */ 35*4f65af2cSYamadaMiz public function handle_tpl_content_display(Event $event, $_param) 36*4f65af2cSYamadaMiz { 37*4f65af2cSYamadaMiz // データが文字列かどうか確認 38*4f65af2cSYamadaMiz if (!is_string($event->data)) { 39*4f65af2cSYamadaMiz error_log('handle_tpl_content_display: data is not a string! ' . print_r($event->data, true)); 40*4f65af2cSYamadaMiz return; 41*4f65af2cSYamadaMiz } 42*4f65af2cSYamadaMiz 43*4f65af2cSYamadaMiz $html = $event->data; 44*4f65af2cSYamadaMiz 45*4f65af2cSYamadaMiz // "mizarWrapper" クラスが存在するか確認 46*4f65af2cSYamadaMiz if (strpos($html, 'mizarWrapper') !== false) { 47*4f65af2cSYamadaMiz // 既にボタンが挿入されているか確認(複数回挿入しないため) 48*4f65af2cSYamadaMiz if (strpos($html, 'id="hideAllButton"') === false) { 49*4f65af2cSYamadaMiz $buttonHtml = '<div class="hideAllContainer">' 50*4f65af2cSYamadaMiz . '<button id="hideAllButton" class="hide-all-button">Hide All</button>' 51*4f65af2cSYamadaMiz . '<button id="showAllButton" class="hide-all-button" style="display:none;">Show All</button>' 52*4f65af2cSYamadaMiz . '</div>'; 53*4f65af2cSYamadaMiz 54*4f65af2cSYamadaMiz // 先頭にボタンを挿入 55*4f65af2cSYamadaMiz $html = $buttonHtml . $html; 56*4f65af2cSYamadaMiz $event->data = $html; 57*4f65af2cSYamadaMiz 58*4f65af2cSYamadaMiz // デバッグ用ログ 59*4f65af2cSYamadaMiz error_log('handle_tpl_content_display: "Hide All" ボタンを挿入しました。'); 60*4f65af2cSYamadaMiz } else { 61*4f65af2cSYamadaMiz // ボタンが既に存在する場合 62*4f65af2cSYamadaMiz error_log('handle_tpl_content_display: "Hide All" ボタンは既に存在します。'); 63*4f65af2cSYamadaMiz } 64*4f65af2cSYamadaMiz } 656b7e227cSYamadaMiz } 666b7e227cSYamadaMiz 6747595f9cSYamadaMiz /** 6847595f9cSYamadaMiz * Handles AJAX requests 6947595f9cSYamadaMiz * 7047595f9cSYamadaMiz * @param Event $event 7147595f9cSYamadaMiz * @param $param 7247595f9cSYamadaMiz */ 7347595f9cSYamadaMiz public function handle_ajax_call(Event $event, $param) 7447595f9cSYamadaMiz { 756b7e227cSYamadaMiz unset($param); // 未使用のパラメータを無効化 7647595f9cSYamadaMiz 7747595f9cSYamadaMiz switch ($event->data) { 7847595f9cSYamadaMiz case 'clear_temp_files': 7947595f9cSYamadaMiz $event->preventDefault(); 8047595f9cSYamadaMiz $event->stopPropagation(); 8142c7ffb2SYamadaMiz $this->clearTempFiles(); 8247595f9cSYamadaMiz break; 8347595f9cSYamadaMiz case 'source_sse': 846b7e227cSYamadaMiz $event->preventDefault(); 856b7e227cSYamadaMiz $event->stopPropagation(); 866b7e227cSYamadaMiz $this->handleSourceSSERequest(); 8747595f9cSYamadaMiz break; 8847595f9cSYamadaMiz case 'source_compile': 896b7e227cSYamadaMiz $event->preventDefault(); 906b7e227cSYamadaMiz $event->stopPropagation(); 916b7e227cSYamadaMiz $this->handleSourceCompileRequest(); 9247595f9cSYamadaMiz break; 9347595f9cSYamadaMiz case 'view_compile': 946b7e227cSYamadaMiz $event->preventDefault(); 956b7e227cSYamadaMiz $event->stopPropagation(); 966b7e227cSYamadaMiz $this->handleViewCompileRequest(); 9747595f9cSYamadaMiz break; 9847595f9cSYamadaMiz case 'view_sse': 996b7e227cSYamadaMiz $event->preventDefault(); 1006b7e227cSYamadaMiz $event->stopPropagation(); 1016b7e227cSYamadaMiz $this->handleViewSSERequest(); 10247595f9cSYamadaMiz break; 1039fc5dc4bSYamadaMiz case 'create_combined_file': 1049fc5dc4bSYamadaMiz $event->preventDefault(); 1059fc5dc4bSYamadaMiz $event->stopPropagation(); 1069fc5dc4bSYamadaMiz $this->handle_create_combined_file(); 1079fc5dc4bSYamadaMiz break; 1086b7e227cSYamadaMiz } 1096b7e227cSYamadaMiz } 1106b7e227cSYamadaMiz 1116b7e227cSYamadaMiz // source用のコンパイルリクエスト処理 11247595f9cSYamadaMiz private function handleSourceCompileRequest() 11347595f9cSYamadaMiz { 1146b7e227cSYamadaMiz global $INPUT; 1156b7e227cSYamadaMiz $pageContent = $INPUT->post->str('content'); 1166b7e227cSYamadaMiz $mizarData = $this->extractMizarContent($pageContent); 1176b7e227cSYamadaMiz 11842c7ffb2SYamadaMiz // エラーチェックを追加 1196b7e227cSYamadaMiz if ($mizarData === null) { 1206b7e227cSYamadaMiz $this->sendAjaxResponse(false, 'Mizar content not found'); 1216b7e227cSYamadaMiz return; 12242c7ffb2SYamadaMiz } elseif (isset($mizarData['error'])) { 12342c7ffb2SYamadaMiz $this->sendAjaxResponse(false, $mizarData['error']); 12442c7ffb2SYamadaMiz return; 1256b7e227cSYamadaMiz } 1266b7e227cSYamadaMiz 1276b7e227cSYamadaMiz $filePath = $this->saveMizarContent($mizarData); 1286b7e227cSYamadaMiz 1296b7e227cSYamadaMiz session_start(); 1306b7e227cSYamadaMiz $_SESSION['source_filepath'] = $filePath; 1316b7e227cSYamadaMiz 1326b7e227cSYamadaMiz $this->sendAjaxResponse(true, 'Mizar content processed successfully'); 1336b7e227cSYamadaMiz } 1346b7e227cSYamadaMiz 1356b7e227cSYamadaMiz // source用のSSEリクエスト処理 13647595f9cSYamadaMiz private function handleSourceSSERequest() 13747595f9cSYamadaMiz { 1386b7e227cSYamadaMiz header('Content-Type: text/event-stream'); 1396b7e227cSYamadaMiz header('Cache-Control: no-cache'); 1406b7e227cSYamadaMiz 1416b7e227cSYamadaMiz session_start(); 1426b7e227cSYamadaMiz if (!isset($_SESSION['source_filepath'])) { 1436b7e227cSYamadaMiz echo "data: Mizar file path not found in session\n\n"; 1446b7e227cSYamadaMiz ob_flush(); 1456b7e227cSYamadaMiz flush(); 1466b7e227cSYamadaMiz return; 1476b7e227cSYamadaMiz } 1486b7e227cSYamadaMiz 1496b7e227cSYamadaMiz $filePath = $_SESSION['source_filepath']; 1506b7e227cSYamadaMiz $this->streamSourceOutput($filePath); 1516b7e227cSYamadaMiz 1526b7e227cSYamadaMiz echo "event: end\n"; 1536b7e227cSYamadaMiz echo "data: Compilation complete\n\n"; 1546b7e227cSYamadaMiz ob_flush(); 1556b7e227cSYamadaMiz flush(); 1566b7e227cSYamadaMiz } 1576b7e227cSYamadaMiz 15847595f9cSYamadaMiz // view用のコンパイルリクエスト処理 15947595f9cSYamadaMiz private function handleViewCompileRequest() 16047595f9cSYamadaMiz { 16147595f9cSYamadaMiz global $INPUT; 16247595f9cSYamadaMiz $content = $INPUT->post->str('content'); 16347595f9cSYamadaMiz 16447595f9cSYamadaMiz $filePath = $this->createTempFile($content); 16547595f9cSYamadaMiz 16647595f9cSYamadaMiz session_start(); 16747595f9cSYamadaMiz $_SESSION['view_filepath'] = $filePath; 16847595f9cSYamadaMiz 16947595f9cSYamadaMiz $this->sendAjaxResponse(true, 'Mizar content processed successfully'); 17047595f9cSYamadaMiz } 17147595f9cSYamadaMiz 17247595f9cSYamadaMiz // view用のSSEリクエスト処理 17347595f9cSYamadaMiz private function handleViewSSERequest() 17447595f9cSYamadaMiz { 17547595f9cSYamadaMiz header('Content-Type: text/event-stream'); 17647595f9cSYamadaMiz header('Cache-Control: no-cache'); 17747595f9cSYamadaMiz 17847595f9cSYamadaMiz session_start(); 17947595f9cSYamadaMiz if (!isset($_SESSION['view_filepath'])) { 18047595f9cSYamadaMiz echo "data: Mizar file path not found in session\n\n"; 18147595f9cSYamadaMiz ob_flush(); 18247595f9cSYamadaMiz flush(); 18347595f9cSYamadaMiz return; 18447595f9cSYamadaMiz } 18547595f9cSYamadaMiz 18647595f9cSYamadaMiz $filePath = $_SESSION['view_filepath']; 1879fc5dc4bSYamadaMiz $this->streamViewCompileOutput($filePath); 18847595f9cSYamadaMiz 18947595f9cSYamadaMiz echo "event: end\n"; 19047595f9cSYamadaMiz echo "data: Compilation complete\n\n"; 19147595f9cSYamadaMiz ob_flush(); 19247595f9cSYamadaMiz flush(); 19347595f9cSYamadaMiz } 19447595f9cSYamadaMiz 19547595f9cSYamadaMiz // Mizarコンテンツの抽出 19647595f9cSYamadaMiz private function extractMizarContent($pageContent) 19747595f9cSYamadaMiz { 1986b7e227cSYamadaMiz $pattern = '/<mizar\s+([^>]+)>(.*?)<\/mizar>/s'; 1996b7e227cSYamadaMiz preg_match_all($pattern, $pageContent, $matches, PREG_SET_ORDER); 2006b7e227cSYamadaMiz 2016b7e227cSYamadaMiz if (empty($matches)) { 20247595f9cSYamadaMiz return null; 2036b7e227cSYamadaMiz } 2046b7e227cSYamadaMiz 20542c7ffb2SYamadaMiz // 最初のファイル名を取得し、拡張子を除去 20647595f9cSYamadaMiz $fileName = trim($matches[0][1]); 20742c7ffb2SYamadaMiz $fileNameWithoutExt = preg_replace('/\.miz$/i', '', $fileName); 20842c7ffb2SYamadaMiz 20942c7ffb2SYamadaMiz // ファイル名のバリデーションを追加 21042c7ffb2SYamadaMiz if (!$this->isValidFileName($fileNameWithoutExt)) { 21142c7ffb2SYamadaMiz return ['error' => "Invalid characters in file name: '{$fileNameWithoutExt}'. Only letters, numbers, underscores (_), and apostrophes (') are allowed, up to 8 characters."]; 21242c7ffb2SYamadaMiz } 21342c7ffb2SYamadaMiz 2146b7e227cSYamadaMiz $combinedContent = ''; 2156b7e227cSYamadaMiz 2166b7e227cSYamadaMiz foreach ($matches as $match) { 21742c7ffb2SYamadaMiz $currentFileName = trim($match[1]); 21842c7ffb2SYamadaMiz $currentFileNameWithoutExt = preg_replace('/\.miz$/i', '', $currentFileName); 21942c7ffb2SYamadaMiz 22042c7ffb2SYamadaMiz if ($currentFileNameWithoutExt !== $fileNameWithoutExt) { 22142c7ffb2SYamadaMiz return ['error' => "File name mismatch in <mizar> tags: '{$fileNameWithoutExt}' and '{$currentFileNameWithoutExt}'"]; 2226b7e227cSYamadaMiz } 22342c7ffb2SYamadaMiz 22442c7ffb2SYamadaMiz // バリデーションを各ファイル名にも適用 22542c7ffb2SYamadaMiz if (!$this->isValidFileName($currentFileNameWithoutExt)) { 22642c7ffb2SYamadaMiz return ['error' => "Invalid characters in file name: '{$currentFileNameWithoutExt}'. Only letters, numbers, underscores (_), and apostrophes (') are allowed, up to 8 characters."]; 22742c7ffb2SYamadaMiz } 22842c7ffb2SYamadaMiz 2296b7e227cSYamadaMiz $combinedContent .= trim($match[2]) . "\n"; 2306b7e227cSYamadaMiz } 2316b7e227cSYamadaMiz 23242c7ffb2SYamadaMiz // ファイル名に拡張子を付加 23342c7ffb2SYamadaMiz $fullFileName = $fileNameWithoutExt . '.miz'; 23442c7ffb2SYamadaMiz 23542c7ffb2SYamadaMiz return ['fileName' => $fullFileName, 'content' => $combinedContent]; 23642c7ffb2SYamadaMiz } 23742c7ffb2SYamadaMiz 23842c7ffb2SYamadaMiz // ファイル名のバリデーション関数を追加 23942c7ffb2SYamadaMiz private function isValidFileName($fileName) 24042c7ffb2SYamadaMiz { 24142c7ffb2SYamadaMiz // ファイル名の長さをチェック(最大8文字) 24242c7ffb2SYamadaMiz if (strlen($fileName) > 8) { 24342c7ffb2SYamadaMiz return false; 24442c7ffb2SYamadaMiz } 24542c7ffb2SYamadaMiz 24642c7ffb2SYamadaMiz // 許可される文字のみを含むかチェック 24742c7ffb2SYamadaMiz if (!preg_match('/^[A-Za-z0-9_\']+$/', $fileName)) { 24842c7ffb2SYamadaMiz return false; 24942c7ffb2SYamadaMiz } 25042c7ffb2SYamadaMiz 25142c7ffb2SYamadaMiz return true; 2526b7e227cSYamadaMiz } 2536b7e227cSYamadaMiz 25447595f9cSYamadaMiz // Mizarコンテンツの保存 25547595f9cSYamadaMiz private function saveMizarContent($mizarData) 25647595f9cSYamadaMiz { 2576b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\'); 2586b7e227cSYamadaMiz $filePath = $workPath . "/TEXT/" . $mizarData['fileName']; 2596b7e227cSYamadaMiz file_put_contents($filePath, $mizarData['content']); 2606b7e227cSYamadaMiz return $filePath; 2616b7e227cSYamadaMiz } 2626b7e227cSYamadaMiz 26347595f9cSYamadaMiz // source用の出力をストリーム 26447595f9cSYamadaMiz private function streamSourceOutput($filePath) 26547595f9cSYamadaMiz { 2666b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\'); 2676b7e227cSYamadaMiz chdir($workPath); 2686b7e227cSYamadaMiz 2696b7e227cSYamadaMiz $command = "miz2prel " . escapeshellarg($filePath); 2706b7e227cSYamadaMiz $process = proc_open($command, array(1 => array("pipe", "w")), $pipes); 2716b7e227cSYamadaMiz 2726b7e227cSYamadaMiz if (is_resource($process)) { 2736b7e227cSYamadaMiz while ($line = fgets($pipes[1])) { 2746b7e227cSYamadaMiz echo "data: " . $line . "\n\n"; 2756b7e227cSYamadaMiz ob_flush(); 2766b7e227cSYamadaMiz flush(); 2776b7e227cSYamadaMiz } 2786b7e227cSYamadaMiz fclose($pipes[1]); 2799fc5dc4bSYamadaMiz 2809fc5dc4bSYamadaMiz // エラー処理の追加 2819fc5dc4bSYamadaMiz $errFilename = str_replace('.miz', '.err', $filePath); 2829fc5dc4bSYamadaMiz if ($this->handleCompilationErrors($errFilename, rtrim($this->getConf('mizar_share_dir'), '/\\') . '/mizar.msg')) { 2839fc5dc4bSYamadaMiz // エラーがあった場合は処理を終了 2849fc5dc4bSYamadaMiz proc_close($process); 2859fc5dc4bSYamadaMiz return; 2869fc5dc4bSYamadaMiz } 2879fc5dc4bSYamadaMiz 2886b7e227cSYamadaMiz proc_close($process); 2896b7e227cSYamadaMiz } 2906b7e227cSYamadaMiz } 2916b7e227cSYamadaMiz 29247595f9cSYamadaMiz // view用の一時ファイル作成 29347595f9cSYamadaMiz private function createTempFile($content) 29447595f9cSYamadaMiz { 2956b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/'; 29647595f9cSYamadaMiz $uniqueName = str_replace('.', '_', uniqid('tmp', true)); 2976b7e227cSYamadaMiz $tempFilename = $workPath . $uniqueName . ".miz"; 2986b7e227cSYamadaMiz file_put_contents($tempFilename, $content); 2996b7e227cSYamadaMiz return $tempFilename; 3006b7e227cSYamadaMiz } 3016b7e227cSYamadaMiz 30242c7ffb2SYamadaMiz // 一時ファイルの削除 30347595f9cSYamadaMiz private function clearTempFiles() 30447595f9cSYamadaMiz { 30547595f9cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/'; 30647595f9cSYamadaMiz $files = glob($workPath . '*'); // TEXTフォルダ内のすべてのファイルを取得 30747595f9cSYamadaMiz 30847595f9cSYamadaMiz $errors = []; 30947595f9cSYamadaMiz foreach ($files as $file) { 31047595f9cSYamadaMiz if (is_file($file)) { 31147595f9cSYamadaMiz // ファイルが使用中かどうか確認 31247595f9cSYamadaMiz if (!$this->is_file_locked($file)) { 31347595f9cSYamadaMiz $retries = 3; // 最大3回リトライ 31447595f9cSYamadaMiz while ($retries > 0) { 31547595f9cSYamadaMiz if (unlink($file)) { 31647595f9cSYamadaMiz break; // 削除成功 31747595f9cSYamadaMiz } 3185cbf3a53SYamadaMiz $errors[] = "Error deleting $file: " . error_get_last()['message']; 31947595f9cSYamadaMiz $retries--; 32047595f9cSYamadaMiz sleep(1); // 1秒待ってリトライ 32147595f9cSYamadaMiz } 32247595f9cSYamadaMiz if ($retries === 0) { 32347595f9cSYamadaMiz $errors[] = "Failed to delete: $file"; // 削除失敗 32447595f9cSYamadaMiz } 32547595f9cSYamadaMiz } else { 32647595f9cSYamadaMiz $errors[] = "File is locked: $file"; // ファイルがロックされている 32747595f9cSYamadaMiz } 32847595f9cSYamadaMiz } 32947595f9cSYamadaMiz } 33047595f9cSYamadaMiz 33147595f9cSYamadaMiz if (empty($errors)) { 33247595f9cSYamadaMiz $this->sendAjaxResponse(true, 'Temporary files cleared successfully'); 33347595f9cSYamadaMiz } else { 33447595f9cSYamadaMiz $this->sendAjaxResponse(false, 'Some files could not be deleted', $errors); 33547595f9cSYamadaMiz } 33647595f9cSYamadaMiz } 33747595f9cSYamadaMiz 33847595f9cSYamadaMiz // ファイルがロックされているかをチェックする関数 33947595f9cSYamadaMiz private function is_file_locked($file) 34047595f9cSYamadaMiz { 34147595f9cSYamadaMiz $fileHandle = @fopen($file, "r+"); 34247595f9cSYamadaMiz 34347595f9cSYamadaMiz if ($fileHandle === false) { 34447595f9cSYamadaMiz return true; // ファイルが開けない、つまりロックされているかアクセス権がない 34547595f9cSYamadaMiz } 34647595f9cSYamadaMiz 34747595f9cSYamadaMiz $locked = !flock($fileHandle, LOCK_EX | LOCK_NB); // ロックの取得を試みる(非ブロッキングモード) 34847595f9cSYamadaMiz 34947595f9cSYamadaMiz fclose($fileHandle); 35047595f9cSYamadaMiz return $locked; // ロックが取得できなければファイルはロックされている 35147595f9cSYamadaMiz } 35247595f9cSYamadaMiz 3539fc5dc4bSYamadaMiz // View用コンパイル出力のストリーム 3549fc5dc4bSYamadaMiz private function streamViewCompileOutput($filePath) 35547595f9cSYamadaMiz { 3566b7e227cSYamadaMiz $workPath = $this->getConf('mizar_work_dir'); 3576b7e227cSYamadaMiz $sharePath = rtrim($this->getConf('mizar_share_dir'), '/\\') . '/'; 3586b7e227cSYamadaMiz 3596b7e227cSYamadaMiz chdir($workPath); 3606b7e227cSYamadaMiz 3619fc5dc4bSYamadaMiz $errFilename = str_replace('.miz', '.err', $filePath); 3626b7e227cSYamadaMiz $command = "makeenv " . escapeshellarg($filePath); 3636b7e227cSYamadaMiz $process = proc_open($command, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $pipes); 3646b7e227cSYamadaMiz 3656b7e227cSYamadaMiz if (is_resource($process)) { 3666b7e227cSYamadaMiz while ($line = fgets($pipes[1])) { 3676b7e227cSYamadaMiz echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3686b7e227cSYamadaMiz ob_flush(); 3696b7e227cSYamadaMiz flush(); 3706b7e227cSYamadaMiz } 3716b7e227cSYamadaMiz fclose($pipes[1]); 3726b7e227cSYamadaMiz 3736b7e227cSYamadaMiz while ($line = fgets($pipes[2])) { 3746b7e227cSYamadaMiz echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3756b7e227cSYamadaMiz ob_flush(); 3766b7e227cSYamadaMiz flush(); 3776b7e227cSYamadaMiz } 3786b7e227cSYamadaMiz fclose($pipes[2]); 3796b7e227cSYamadaMiz proc_close($process); 3806b7e227cSYamadaMiz 3811174571dSYamadaMiz // makeenvのエラー処理 3829fc5dc4bSYamadaMiz if ($this->handleCompilationErrors($errFilename, $sharePath . '/mizar.msg')) { 3836b7e227cSYamadaMiz return; 3846b7e227cSYamadaMiz } 3856b7e227cSYamadaMiz 3866b7e227cSYamadaMiz // verifierの実行 3876b7e227cSYamadaMiz $exePath = rtrim($this->getConf('mizar_exe_dir'), '/\\') . '/'; 3886b7e227cSYamadaMiz $verifierPath = escapeshellarg($exePath . "verifier"); 3896b7e227cSYamadaMiz if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') { 3906b7e227cSYamadaMiz $verifierPath .= ".exe"; 3916b7e227cSYamadaMiz } 3926b7e227cSYamadaMiz $verifierCommand = $verifierPath . " -q -l " . escapeshellarg("TEXT/" . basename($filePath)); 3936b7e227cSYamadaMiz 3946b7e227cSYamadaMiz $verifierProcess = proc_open($verifierCommand, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $verifierPipes); 3956b7e227cSYamadaMiz 3966b7e227cSYamadaMiz if (is_resource($verifierProcess)) { 3976b7e227cSYamadaMiz while ($line = fgets($verifierPipes[1])) { 3986b7e227cSYamadaMiz echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3996b7e227cSYamadaMiz ob_flush(); 4006b7e227cSYamadaMiz flush(); 4016b7e227cSYamadaMiz } 4026b7e227cSYamadaMiz fclose($verifierPipes[1]); 4036b7e227cSYamadaMiz 4046b7e227cSYamadaMiz while ($line = fgets($verifierPipes[2])) { 4056b7e227cSYamadaMiz echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 4066b7e227cSYamadaMiz ob_flush(); 4076b7e227cSYamadaMiz flush(); 4086b7e227cSYamadaMiz } 4096b7e227cSYamadaMiz fclose($verifierPipes[2]); 4106b7e227cSYamadaMiz proc_close($verifierProcess); 4111174571dSYamadaMiz 4121174571dSYamadaMiz // verifierのエラー処理 4131174571dSYamadaMiz if ($this->handleCompilationErrors($errFilename, $sharePath . '/mizar.msg')) { 4141174571dSYamadaMiz return; 4151174571dSYamadaMiz } 4166b7e227cSYamadaMiz } else { 4176b7e227cSYamadaMiz echo "data: ERROR: Failed to execute verifier command.\n\n"; 4186b7e227cSYamadaMiz ob_flush(); 4196b7e227cSYamadaMiz flush(); 4206b7e227cSYamadaMiz } 4216b7e227cSYamadaMiz } else { 4226b7e227cSYamadaMiz echo "data: ERROR: Failed to execute makeenv command.\n\n"; 4236b7e227cSYamadaMiz ob_flush(); 4246b7e227cSYamadaMiz flush(); 4256b7e227cSYamadaMiz } 4266b7e227cSYamadaMiz } 4276b7e227cSYamadaMiz 42847595f9cSYamadaMiz private function getMizarErrorMessages($mizarMsgFile) 42947595f9cSYamadaMiz { 4306b7e227cSYamadaMiz $errorMessages = []; 4316b7e227cSYamadaMiz $lines = file($mizarMsgFile, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); 4326b7e227cSYamadaMiz 4336b7e227cSYamadaMiz $isReadingErrorMsg = false; 4346b7e227cSYamadaMiz $key = 0; 4356b7e227cSYamadaMiz 4366b7e227cSYamadaMiz foreach ($lines as $line) { 4376b7e227cSYamadaMiz if (preg_match('/# (\d+)/', $line, $matches)) { 4386b7e227cSYamadaMiz $isReadingErrorMsg = true; 4396b7e227cSYamadaMiz $key = intval($matches[1]); 4406b7e227cSYamadaMiz } elseif ($isReadingErrorMsg) { 4416b7e227cSYamadaMiz $errorMessages[$key] = $line; 4426b7e227cSYamadaMiz $isReadingErrorMsg = false; 4436b7e227cSYamadaMiz } 4446b7e227cSYamadaMiz } 4456b7e227cSYamadaMiz 4466b7e227cSYamadaMiz return $errorMessages; 4476b7e227cSYamadaMiz } 4486b7e227cSYamadaMiz 44947595f9cSYamadaMiz private function sendAjaxResponse($success, $message, $data = '') 45047595f9cSYamadaMiz { 4516b7e227cSYamadaMiz header('Content-Type: application/json'); 4526b7e227cSYamadaMiz echo json_encode(['success' => $success, 'message' => $message, 'data' => $data]); 4536b7e227cSYamadaMiz exit; 4546b7e227cSYamadaMiz } 4559fc5dc4bSYamadaMiz 4569fc5dc4bSYamadaMiz private function handleCompilationErrors($errFilename, $mizarMsgFilePath) 4579fc5dc4bSYamadaMiz { 4589fc5dc4bSYamadaMiz if (file_exists($errFilename)) { 4599fc5dc4bSYamadaMiz $errors = []; 4609fc5dc4bSYamadaMiz $errorLines = file($errFilename, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); 4619fc5dc4bSYamadaMiz foreach ($errorLines as $errorLine) { 4629fc5dc4bSYamadaMiz if (preg_match('/(\d+)\s+(\d+)\s+(\d+)/', $errorLine, $matches)) { 4639fc5dc4bSYamadaMiz $errorCode = intval($matches[3]); 4649fc5dc4bSYamadaMiz $errors[] = [ 4659fc5dc4bSYamadaMiz 'code' => $errorCode, 4669fc5dc4bSYamadaMiz 'line' => intval($matches[1]), 4679fc5dc4bSYamadaMiz 'column' => intval($matches[2]), 4689fc5dc4bSYamadaMiz 'message' => $this->getMizarErrorMessages($mizarMsgFilePath)[$errorCode] ?? 'Unknown error' 4699fc5dc4bSYamadaMiz ]; 4709fc5dc4bSYamadaMiz } 4719fc5dc4bSYamadaMiz } 4729fc5dc4bSYamadaMiz if (!empty($errors)) { 4739fc5dc4bSYamadaMiz echo "event: compileErrors\n"; 4749fc5dc4bSYamadaMiz echo "data: " . json_encode($errors) . "\n\n"; 4759fc5dc4bSYamadaMiz ob_flush(); 4769fc5dc4bSYamadaMiz flush(); 4779fc5dc4bSYamadaMiz return true; 4789fc5dc4bSYamadaMiz } 4799fc5dc4bSYamadaMiz } 4809fc5dc4bSYamadaMiz return false; 4819fc5dc4bSYamadaMiz } 4829fc5dc4bSYamadaMiz 4839fc5dc4bSYamadaMiz private function handle_create_combined_file() 4849fc5dc4bSYamadaMiz { 4859fc5dc4bSYamadaMiz global $INPUT; 4869fc5dc4bSYamadaMiz 4879fc5dc4bSYamadaMiz // 投稿されたコンテンツを取得 4889fc5dc4bSYamadaMiz $combinedContent = $INPUT->post->str('content'); 489a0444036SYamadaMiz $filename = $INPUT->post->str('filename', 'combined_file.miz'); // デフォルトのファイル名を指定 4909fc5dc4bSYamadaMiz 491a0444036SYamadaMiz // ファイルを保存せず、コンテンツを直接返す 492a0444036SYamadaMiz if (!empty($combinedContent)) { 493a0444036SYamadaMiz // ファイルの内容をレスポンスで返す(PHP側でファイルを作成しない) 494a0444036SYamadaMiz $this->sendAjaxResponse(true, 'File created successfully', [ 495a0444036SYamadaMiz 'filename' => $filename, 496a0444036SYamadaMiz 'content' => $combinedContent 497a0444036SYamadaMiz ]); 498a0444036SYamadaMiz error_log("File content sent: " . $filename); 4999fc5dc4bSYamadaMiz } else { 500a0444036SYamadaMiz $this->sendAjaxResponse(false, 'Content is empty, no file created'); 5019fc5dc4bSYamadaMiz } 5029fc5dc4bSYamadaMiz } 5036b7e227cSYamadaMiz} 504