16b7e227cSYamadaMiz<?php 247595f9cSYamadaMiz 347595f9cSYamadaMizuse dokuwiki\Extension\ActionPlugin; 447595f9cSYamadaMizuse dokuwiki\Extension\EventHandler; 547595f9cSYamadaMizuse dokuwiki\Extension\Event; 647595f9cSYamadaMiz 747595f9cSYamadaMizclass action_plugin_mizarproofchecker extends ActionPlugin 847595f9cSYamadaMiz{ 947595f9cSYamadaMiz /** 1047595f9cSYamadaMiz * Registers a callback function for a given event 1147595f9cSYamadaMiz * 1247595f9cSYamadaMiz * @param EventHandler $controller DokuWiki's event controller object 1347595f9cSYamadaMiz * @return void 1447595f9cSYamadaMiz */ 1547595f9cSYamadaMiz public function register(EventHandler $controller) 1647595f9cSYamadaMiz { 176b7e227cSYamadaMiz $controller->register_hook('AJAX_CALL_UNKNOWN', 'BEFORE', $this, 'handle_ajax_call'); 186b7e227cSYamadaMiz } 196b7e227cSYamadaMiz 2047595f9cSYamadaMiz /** 2147595f9cSYamadaMiz * Handles AJAX requests 2247595f9cSYamadaMiz * 2347595f9cSYamadaMiz * @param Event $event 2447595f9cSYamadaMiz * @param $param 2547595f9cSYamadaMiz */ 2647595f9cSYamadaMiz public function handle_ajax_call(Event $event, $param) 2747595f9cSYamadaMiz { 286b7e227cSYamadaMiz unset($param); // 未使用のパラメータを無効化 2947595f9cSYamadaMiz 3047595f9cSYamadaMiz switch ($event->data) { 3147595f9cSYamadaMiz case 'clear_temp_files': 3247595f9cSYamadaMiz $event->preventDefault(); 3347595f9cSYamadaMiz $event->stopPropagation(); 3447595f9cSYamadaMiz $this->clearTempFiles(); // 追加: 一時ファイル削除の呼び出し 3547595f9cSYamadaMiz break; 3647595f9cSYamadaMiz case 'source_sse': 376b7e227cSYamadaMiz $event->preventDefault(); 386b7e227cSYamadaMiz $event->stopPropagation(); 396b7e227cSYamadaMiz $this->handleSourceSSERequest(); 4047595f9cSYamadaMiz break; 4147595f9cSYamadaMiz case 'source_compile': 426b7e227cSYamadaMiz $event->preventDefault(); 436b7e227cSYamadaMiz $event->stopPropagation(); 446b7e227cSYamadaMiz $this->handleSourceCompileRequest(); 4547595f9cSYamadaMiz break; 4647595f9cSYamadaMiz case 'view_compile': 476b7e227cSYamadaMiz $event->preventDefault(); 486b7e227cSYamadaMiz $event->stopPropagation(); 496b7e227cSYamadaMiz $this->handleViewCompileRequest(); 5047595f9cSYamadaMiz break; 5147595f9cSYamadaMiz case 'view_sse': 526b7e227cSYamadaMiz $event->preventDefault(); 536b7e227cSYamadaMiz $event->stopPropagation(); 546b7e227cSYamadaMiz $this->handleViewSSERequest(); 5547595f9cSYamadaMiz break; 566b7e227cSYamadaMiz } 576b7e227cSYamadaMiz } 586b7e227cSYamadaMiz 596b7e227cSYamadaMiz // source用のコンパイルリクエスト処理 6047595f9cSYamadaMiz private function handleSourceCompileRequest() 6147595f9cSYamadaMiz { 626b7e227cSYamadaMiz global $INPUT; 636b7e227cSYamadaMiz $pageContent = $INPUT->post->str('content'); 646b7e227cSYamadaMiz $mizarData = $this->extractMizarContent($pageContent); 656b7e227cSYamadaMiz 666b7e227cSYamadaMiz if ($mizarData === null) { 676b7e227cSYamadaMiz $this->sendAjaxResponse(false, 'Mizar content not found'); 686b7e227cSYamadaMiz return; 696b7e227cSYamadaMiz } 706b7e227cSYamadaMiz 716b7e227cSYamadaMiz $filePath = $this->saveMizarContent($mizarData); 726b7e227cSYamadaMiz 736b7e227cSYamadaMiz session_start(); 746b7e227cSYamadaMiz $_SESSION['source_filepath'] = $filePath; 756b7e227cSYamadaMiz 766b7e227cSYamadaMiz $this->sendAjaxResponse(true, 'Mizar content processed successfully'); 776b7e227cSYamadaMiz } 786b7e227cSYamadaMiz 796b7e227cSYamadaMiz // source用のSSEリクエスト処理 8047595f9cSYamadaMiz private function handleSourceSSERequest() 8147595f9cSYamadaMiz { 826b7e227cSYamadaMiz header('Content-Type: text/event-stream'); 836b7e227cSYamadaMiz header('Cache-Control: no-cache'); 846b7e227cSYamadaMiz 856b7e227cSYamadaMiz session_start(); 866b7e227cSYamadaMiz if (!isset($_SESSION['source_filepath'])) { 876b7e227cSYamadaMiz echo "data: Mizar file path not found in session\n\n"; 886b7e227cSYamadaMiz ob_flush(); 896b7e227cSYamadaMiz flush(); 906b7e227cSYamadaMiz return; 916b7e227cSYamadaMiz } 926b7e227cSYamadaMiz 936b7e227cSYamadaMiz $filePath = $_SESSION['source_filepath']; 946b7e227cSYamadaMiz $this->streamSourceOutput($filePath); 956b7e227cSYamadaMiz 966b7e227cSYamadaMiz echo "event: end\n"; 976b7e227cSYamadaMiz echo "data: Compilation complete\n\n"; 986b7e227cSYamadaMiz ob_flush(); 996b7e227cSYamadaMiz flush(); 1006b7e227cSYamadaMiz } 1016b7e227cSYamadaMiz 10247595f9cSYamadaMiz // view用のコンパイルリクエスト処理 10347595f9cSYamadaMiz private function handleViewCompileRequest() 10447595f9cSYamadaMiz { 10547595f9cSYamadaMiz global $INPUT; 10647595f9cSYamadaMiz $content = $INPUT->post->str('content'); 10747595f9cSYamadaMiz 10847595f9cSYamadaMiz $filePath = $this->createTempFile($content); 10947595f9cSYamadaMiz 11047595f9cSYamadaMiz session_start(); 11147595f9cSYamadaMiz $_SESSION['view_filepath'] = $filePath; 11247595f9cSYamadaMiz 11347595f9cSYamadaMiz $this->sendAjaxResponse(true, 'Mizar content processed successfully'); 11447595f9cSYamadaMiz } 11547595f9cSYamadaMiz 11647595f9cSYamadaMiz // view用のSSEリクエスト処理 11747595f9cSYamadaMiz private function handleViewSSERequest() 11847595f9cSYamadaMiz { 11947595f9cSYamadaMiz header('Content-Type: text/event-stream'); 12047595f9cSYamadaMiz header('Cache-Control: no-cache'); 12147595f9cSYamadaMiz 12247595f9cSYamadaMiz session_start(); 12347595f9cSYamadaMiz if (!isset($_SESSION['view_filepath'])) { 12447595f9cSYamadaMiz echo "data: Mizar file path not found in session\n\n"; 12547595f9cSYamadaMiz ob_flush(); 12647595f9cSYamadaMiz flush(); 12747595f9cSYamadaMiz return; 12847595f9cSYamadaMiz } 12947595f9cSYamadaMiz 13047595f9cSYamadaMiz $filePath = $_SESSION['view_filepath']; 13147595f9cSYamadaMiz $this->streamCompileOutput($filePath); 13247595f9cSYamadaMiz 13347595f9cSYamadaMiz echo "event: end\n"; 13447595f9cSYamadaMiz echo "data: Compilation complete\n\n"; 13547595f9cSYamadaMiz ob_flush(); 13647595f9cSYamadaMiz flush(); 13747595f9cSYamadaMiz } 13847595f9cSYamadaMiz 13947595f9cSYamadaMiz // Mizarコンテンツの抽出 14047595f9cSYamadaMiz private function extractMizarContent($pageContent) 14147595f9cSYamadaMiz { 1426b7e227cSYamadaMiz $pattern = '/<mizar\s+([^>]+)>(.*?)<\/mizar>/s'; 1436b7e227cSYamadaMiz preg_match_all($pattern, $pageContent, $matches, PREG_SET_ORDER); 1446b7e227cSYamadaMiz 1456b7e227cSYamadaMiz if (empty($matches)) { 14647595f9cSYamadaMiz return null; 1476b7e227cSYamadaMiz } 1486b7e227cSYamadaMiz 14947595f9cSYamadaMiz $fileName = trim($matches[0][1]); 1506b7e227cSYamadaMiz $combinedContent = ''; 1516b7e227cSYamadaMiz 1526b7e227cSYamadaMiz foreach ($matches as $match) { 1536b7e227cSYamadaMiz if (trim($match[1]) !== $fileName) { 1546b7e227cSYamadaMiz return ['error' => 'File name mismatch in <mizar> tags']; 1556b7e227cSYamadaMiz } 1566b7e227cSYamadaMiz $combinedContent .= trim($match[2]) . "\n"; 1576b7e227cSYamadaMiz } 1586b7e227cSYamadaMiz 1596b7e227cSYamadaMiz return ['fileName' => $fileName, 'content' => $combinedContent]; 1606b7e227cSYamadaMiz } 1616b7e227cSYamadaMiz 16247595f9cSYamadaMiz // Mizarコンテンツの保存 16347595f9cSYamadaMiz private function saveMizarContent($mizarData) 16447595f9cSYamadaMiz { 1656b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\'); 1666b7e227cSYamadaMiz $filePath = $workPath . "/TEXT/" . $mizarData['fileName']; 1676b7e227cSYamadaMiz file_put_contents($filePath, $mizarData['content']); 1686b7e227cSYamadaMiz return $filePath; 1696b7e227cSYamadaMiz } 1706b7e227cSYamadaMiz 17147595f9cSYamadaMiz // source用の出力をストリーム 17247595f9cSYamadaMiz private function streamSourceOutput($filePath) 17347595f9cSYamadaMiz { 1746b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\'); 1756b7e227cSYamadaMiz chdir($workPath); 1766b7e227cSYamadaMiz 1776b7e227cSYamadaMiz $command = "miz2prel " . escapeshellarg($filePath); 1786b7e227cSYamadaMiz $process = proc_open($command, array(1 => array("pipe", "w")), $pipes); 1796b7e227cSYamadaMiz 1806b7e227cSYamadaMiz if (is_resource($process)) { 1816b7e227cSYamadaMiz while ($line = fgets($pipes[1])) { 1826b7e227cSYamadaMiz echo "data: " . $line . "\n\n"; 1836b7e227cSYamadaMiz ob_flush(); 1846b7e227cSYamadaMiz flush(); 1856b7e227cSYamadaMiz } 1866b7e227cSYamadaMiz fclose($pipes[1]); 1876b7e227cSYamadaMiz proc_close($process); 1886b7e227cSYamadaMiz } 1896b7e227cSYamadaMiz } 1906b7e227cSYamadaMiz 19147595f9cSYamadaMiz // view用の一時ファイル作成 19247595f9cSYamadaMiz private function createTempFile($content) 19347595f9cSYamadaMiz { 1946b7e227cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/'; 19547595f9cSYamadaMiz $uniqueName = str_replace('.', '_', uniqid('tmp', true)); 1966b7e227cSYamadaMiz $tempFilename = $workPath . $uniqueName . ".miz"; 1976b7e227cSYamadaMiz file_put_contents($tempFilename, $content); 1986b7e227cSYamadaMiz return $tempFilename; 1996b7e227cSYamadaMiz } 2006b7e227cSYamadaMiz 20147595f9cSYamadaMiz // Clear all temporary files in the TEXT folder 20247595f9cSYamadaMiz private function clearTempFiles() 20347595f9cSYamadaMiz { 20447595f9cSYamadaMiz $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/'; 20547595f9cSYamadaMiz $files = glob($workPath . '*'); // TEXTフォルダ内のすべてのファイルを取得 20647595f9cSYamadaMiz 20747595f9cSYamadaMiz $errors = []; 20847595f9cSYamadaMiz foreach ($files as $file) { 20947595f9cSYamadaMiz if (is_file($file)) { 21047595f9cSYamadaMiz // ファイルが使用中かどうか確認 21147595f9cSYamadaMiz if (!$this->is_file_locked($file)) { 21247595f9cSYamadaMiz $retries = 3; // 最大3回リトライ 21347595f9cSYamadaMiz while ($retries > 0) { 21447595f9cSYamadaMiz if (unlink($file)) { 21547595f9cSYamadaMiz break; // 削除成功 21647595f9cSYamadaMiz } 217*5cbf3a53SYamadaMiz $errors[] = "Error deleting $file: " . error_get_last()['message']; 21847595f9cSYamadaMiz $retries--; 21947595f9cSYamadaMiz sleep(1); // 1秒待ってリトライ 22047595f9cSYamadaMiz } 22147595f9cSYamadaMiz if ($retries === 0) { 22247595f9cSYamadaMiz $errors[] = "Failed to delete: $file"; // 削除失敗 22347595f9cSYamadaMiz } 22447595f9cSYamadaMiz } else { 22547595f9cSYamadaMiz $errors[] = "File is locked: $file"; // ファイルがロックされている 22647595f9cSYamadaMiz } 22747595f9cSYamadaMiz } 22847595f9cSYamadaMiz } 22947595f9cSYamadaMiz 23047595f9cSYamadaMiz if (empty($errors)) { 23147595f9cSYamadaMiz $this->sendAjaxResponse(true, 'Temporary files cleared successfully'); 23247595f9cSYamadaMiz } else { 23347595f9cSYamadaMiz $this->sendAjaxResponse(false, 'Some files could not be deleted', $errors); 23447595f9cSYamadaMiz } 23547595f9cSYamadaMiz } 23647595f9cSYamadaMiz 23747595f9cSYamadaMiz // ファイルがロックされているかをチェックする関数 23847595f9cSYamadaMiz private function is_file_locked($file) 23947595f9cSYamadaMiz { 24047595f9cSYamadaMiz $fileHandle = @fopen($file, "r+"); 24147595f9cSYamadaMiz 24247595f9cSYamadaMiz if ($fileHandle === false) { 24347595f9cSYamadaMiz return true; // ファイルが開けない、つまりロックされているかアクセス権がない 24447595f9cSYamadaMiz } 24547595f9cSYamadaMiz 24647595f9cSYamadaMiz $locked = !flock($fileHandle, LOCK_EX | LOCK_NB); // ロックの取得を試みる(非ブロッキングモード) 24747595f9cSYamadaMiz 24847595f9cSYamadaMiz fclose($fileHandle); 24947595f9cSYamadaMiz return $locked; // ロックが取得できなければファイルはロックされている 25047595f9cSYamadaMiz } 25147595f9cSYamadaMiz 25247595f9cSYamadaMiz // コンパイル出力のストリーム 25347595f9cSYamadaMiz private function streamCompileOutput($filePath) 25447595f9cSYamadaMiz { 2556b7e227cSYamadaMiz $workPath = $this->getConf('mizar_work_dir'); 2566b7e227cSYamadaMiz $sharePath = rtrim($this->getConf('mizar_share_dir'), '/\\') . '/'; 2576b7e227cSYamadaMiz 2586b7e227cSYamadaMiz chdir($workPath); 2596b7e227cSYamadaMiz 2606b7e227cSYamadaMiz $tempErrFilename = str_replace('.miz', '.err', $filePath); 2616b7e227cSYamadaMiz $command = "makeenv " . escapeshellarg($filePath); 2626b7e227cSYamadaMiz $process = proc_open($command, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $pipes); 2636b7e227cSYamadaMiz 2646b7e227cSYamadaMiz if (is_resource($process)) { 2656b7e227cSYamadaMiz while ($line = fgets($pipes[1])) { 2666b7e227cSYamadaMiz echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 2676b7e227cSYamadaMiz ob_flush(); 2686b7e227cSYamadaMiz flush(); 2696b7e227cSYamadaMiz } 2706b7e227cSYamadaMiz fclose($pipes[1]); 2716b7e227cSYamadaMiz 2726b7e227cSYamadaMiz while ($line = fgets($pipes[2])) { 2736b7e227cSYamadaMiz echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 2746b7e227cSYamadaMiz ob_flush(); 2756b7e227cSYamadaMiz flush(); 2766b7e227cSYamadaMiz } 2776b7e227cSYamadaMiz fclose($pipes[2]); 2786b7e227cSYamadaMiz proc_close($process); 2796b7e227cSYamadaMiz 28047595f9cSYamadaMiz // エラー処理 2816b7e227cSYamadaMiz if (file_exists($tempErrFilename)) { 2826b7e227cSYamadaMiz $errors = []; 2836b7e227cSYamadaMiz $errorLines = file($tempErrFilename, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); 2846b7e227cSYamadaMiz foreach ($errorLines as $errorLine) { 2856b7e227cSYamadaMiz if (preg_match('/(\d+)\s+(\d+)\s+(\d+)/', $errorLine, $matches)) { 2866b7e227cSYamadaMiz $errors[] = [ 2876b7e227cSYamadaMiz 'code' => intval($matches[3]), 2886b7e227cSYamadaMiz 'line' => intval($matches[1]), 2896b7e227cSYamadaMiz 'column' => intval($matches[2]), 2906b7e227cSYamadaMiz 'message' => $this->getMizarErrorMessages($sharePath . '/mizar.msg')[intval($matches[3])] ?? 'Unknown error' 2916b7e227cSYamadaMiz ]; 2926b7e227cSYamadaMiz } 2936b7e227cSYamadaMiz } 2946b7e227cSYamadaMiz if (!empty($errors)) { 2956b7e227cSYamadaMiz echo "event: compileErrors\n"; 2966b7e227cSYamadaMiz echo "data: " . json_encode($errors) . "\n\n"; 2976b7e227cSYamadaMiz ob_flush(); 2986b7e227cSYamadaMiz flush(); 2996b7e227cSYamadaMiz return; 3006b7e227cSYamadaMiz } 3016b7e227cSYamadaMiz } 3026b7e227cSYamadaMiz 3036b7e227cSYamadaMiz // verifierの実行 3046b7e227cSYamadaMiz $exePath = rtrim($this->getConf('mizar_exe_dir'), '/\\') . '/'; 3056b7e227cSYamadaMiz $verifierPath = escapeshellarg($exePath . "verifier"); 3066b7e227cSYamadaMiz if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') { 3076b7e227cSYamadaMiz $verifierPath .= ".exe"; 3086b7e227cSYamadaMiz } 3096b7e227cSYamadaMiz $verifierCommand = $verifierPath . " -q -l " . escapeshellarg("TEXT/" . basename($filePath)); 3106b7e227cSYamadaMiz 3116b7e227cSYamadaMiz $verifierProcess = proc_open($verifierCommand, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $verifierPipes); 3126b7e227cSYamadaMiz 3136b7e227cSYamadaMiz if (is_resource($verifierProcess)) { 3146b7e227cSYamadaMiz while ($line = fgets($verifierPipes[1])) { 3156b7e227cSYamadaMiz echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3166b7e227cSYamadaMiz ob_flush(); 3176b7e227cSYamadaMiz flush(); 3186b7e227cSYamadaMiz } 3196b7e227cSYamadaMiz fclose($verifierPipes[1]); 3206b7e227cSYamadaMiz 3216b7e227cSYamadaMiz while ($line = fgets($verifierPipes[2])) { 3226b7e227cSYamadaMiz echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n"; 3236b7e227cSYamadaMiz ob_flush(); 3246b7e227cSYamadaMiz flush(); 3256b7e227cSYamadaMiz } 3266b7e227cSYamadaMiz fclose($verifierPipes[2]); 3276b7e227cSYamadaMiz proc_close($verifierProcess); 3286b7e227cSYamadaMiz } else { 3296b7e227cSYamadaMiz echo "data: ERROR: Failed to execute verifier command.\n\n"; 3306b7e227cSYamadaMiz ob_flush(); 3316b7e227cSYamadaMiz flush(); 3326b7e227cSYamadaMiz } 3336b7e227cSYamadaMiz } else { 3346b7e227cSYamadaMiz echo "data: ERROR: Failed to execute makeenv command.\n\n"; 3356b7e227cSYamadaMiz ob_flush(); 3366b7e227cSYamadaMiz flush(); 3376b7e227cSYamadaMiz } 3386b7e227cSYamadaMiz } 3396b7e227cSYamadaMiz 34047595f9cSYamadaMiz private function getMizarErrorMessages($mizarMsgFile) 34147595f9cSYamadaMiz { 3426b7e227cSYamadaMiz $errorMessages = []; 3436b7e227cSYamadaMiz $lines = file($mizarMsgFile, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); 3446b7e227cSYamadaMiz 3456b7e227cSYamadaMiz $isReadingErrorMsg = false; 3466b7e227cSYamadaMiz $key = 0; 3476b7e227cSYamadaMiz 3486b7e227cSYamadaMiz foreach ($lines as $line) { 3496b7e227cSYamadaMiz if (preg_match('/# (\d+)/', $line, $matches)) { 3506b7e227cSYamadaMiz $isReadingErrorMsg = true; 3516b7e227cSYamadaMiz $key = intval($matches[1]); 3526b7e227cSYamadaMiz } elseif ($isReadingErrorMsg) { 3536b7e227cSYamadaMiz $errorMessages[$key] = $line; 3546b7e227cSYamadaMiz $isReadingErrorMsg = false; 3556b7e227cSYamadaMiz } 3566b7e227cSYamadaMiz } 3576b7e227cSYamadaMiz 3586b7e227cSYamadaMiz return $errorMessages; 3596b7e227cSYamadaMiz } 3606b7e227cSYamadaMiz 36147595f9cSYamadaMiz private function sendAjaxResponse($success, $message, $data = '') 36247595f9cSYamadaMiz { 3636b7e227cSYamadaMiz header('Content-Type: application/json'); 3646b7e227cSYamadaMiz echo json_encode(['success' => $success, 'message' => $message, 'data' => $data]); 3656b7e227cSYamadaMiz exit; 3666b7e227cSYamadaMiz } 3676b7e227cSYamadaMiz}