xref: /plugin/mizarverifiabledocs/action.php (revision b68e37241baa51dc1a83300f58ba2507a52b9325) !
16b7e227cSYamadaMiz<?php
247595f9cSYamadaMiz
347595f9cSYamadaMizuse dokuwiki\Extension\ActionPlugin;
447595f9cSYamadaMizuse dokuwiki\Extension\EventHandler;
547595f9cSYamadaMizuse dokuwiki\Extension\Event;
647595f9cSYamadaMiz
7f9af2148SYamadaMiz/**
8*b68e3724SYamadaMiz * DokuWiki Plugin Mizar Verifiable Docs (Action Component)
9f9af2148SYamadaMiz *
10f9af2148SYamadaMiz * @license GPL 2 http://www.gnu.org/licenses/gpl-2.0.html
11f9af2148SYamadaMiz * @author  Yamada, M. <yamadam@mizar.work>
12f9af2148SYamadaMiz */
13f9af2148SYamadaMiz
14f9af2148SYamadaMiz
15*b68e3724SYamadaMizclass action_plugin_mizarverifiabledocs extends ActionPlugin
1647595f9cSYamadaMiz{
1747595f9cSYamadaMiz    /**
184754b0a7SYamadaMiz     * Registers a callback function for a given event
1947595f9cSYamadaMiz     *
2047595f9cSYamadaMiz     * @param EventHandler $controller DokuWiki's event controller object
2147595f9cSYamadaMiz     * @return void
2247595f9cSYamadaMiz     */
2347595f9cSYamadaMiz    public function register(EventHandler $controller)
2447595f9cSYamadaMiz    {
256b7e227cSYamadaMiz        $controller->register_hook('AJAX_CALL_UNKNOWN', 'BEFORE', $this, 'handle_ajax_call');
266b7e227cSYamadaMiz    }
276b7e227cSYamadaMiz
2847595f9cSYamadaMiz    /**
2947595f9cSYamadaMiz     * Handles AJAX requests
3047595f9cSYamadaMiz     *
3147595f9cSYamadaMiz     * @param Event $event
3247595f9cSYamadaMiz     * @param $param
3347595f9cSYamadaMiz     */
3447595f9cSYamadaMiz    public function handle_ajax_call(Event $event, $param)
3547595f9cSYamadaMiz    {
366b7e227cSYamadaMiz        unset($param); // 未使用のパラメータを無効化
3747595f9cSYamadaMiz
3847595f9cSYamadaMiz        switch ($event->data) {
3947595f9cSYamadaMiz            case 'clear_temp_files':
4047595f9cSYamadaMiz                $event->preventDefault();
4147595f9cSYamadaMiz                $event->stopPropagation();
4247595f9cSYamadaMiz                $this->clearTempFiles();  // 追加: 一時ファイル削除の呼び出し
4347595f9cSYamadaMiz                break;
4447595f9cSYamadaMiz            case 'source_sse':
456b7e227cSYamadaMiz                $event->preventDefault();
466b7e227cSYamadaMiz                $event->stopPropagation();
476b7e227cSYamadaMiz                $this->handleSourceSSERequest();
4847595f9cSYamadaMiz                break;
4947595f9cSYamadaMiz            case 'source_compile':
506b7e227cSYamadaMiz                $event->preventDefault();
516b7e227cSYamadaMiz                $event->stopPropagation();
526b7e227cSYamadaMiz                $this->handleSourceCompileRequest();
5347595f9cSYamadaMiz                break;
5447595f9cSYamadaMiz            case 'view_compile':
556b7e227cSYamadaMiz                $event->preventDefault();
566b7e227cSYamadaMiz                $event->stopPropagation();
576b7e227cSYamadaMiz                $this->handleViewCompileRequest();
5847595f9cSYamadaMiz                break;
5947595f9cSYamadaMiz            case 'view_sse':
606b7e227cSYamadaMiz                $event->preventDefault();
616b7e227cSYamadaMiz                $event->stopPropagation();
626b7e227cSYamadaMiz                $this->handleViewSSERequest();
6347595f9cSYamadaMiz                break;
649fc5dc4bSYamadaMiz            case 'create_combined_file':
659fc5dc4bSYamadaMiz                $event->preventDefault();
669fc5dc4bSYamadaMiz                $event->stopPropagation();
679fc5dc4bSYamadaMiz                $this->handle_create_combined_file();
689fc5dc4bSYamadaMiz                break;
696b7e227cSYamadaMiz        }
706b7e227cSYamadaMiz    }
716b7e227cSYamadaMiz
726b7e227cSYamadaMiz    // source用のコンパイルリクエスト処理
7347595f9cSYamadaMiz    private function handleSourceCompileRequest()
7447595f9cSYamadaMiz    {
756b7e227cSYamadaMiz        global $INPUT;
766b7e227cSYamadaMiz        $pageContent = $INPUT->post->str('content');
776b7e227cSYamadaMiz        $mizarData = $this->extractMizarContent($pageContent);
786b7e227cSYamadaMiz
796b7e227cSYamadaMiz        if ($mizarData === null) {
806b7e227cSYamadaMiz            $this->sendAjaxResponse(false, 'Mizar content not found');
816b7e227cSYamadaMiz            return;
826b7e227cSYamadaMiz        }
836b7e227cSYamadaMiz
846b7e227cSYamadaMiz        $filePath = $this->saveMizarContent($mizarData);
856b7e227cSYamadaMiz
866b7e227cSYamadaMiz        session_start();
876b7e227cSYamadaMiz        $_SESSION['source_filepath'] = $filePath;
886b7e227cSYamadaMiz
896b7e227cSYamadaMiz        $this->sendAjaxResponse(true, 'Mizar content processed successfully');
906b7e227cSYamadaMiz    }
916b7e227cSYamadaMiz
926b7e227cSYamadaMiz    // source用のSSEリクエスト処理
9347595f9cSYamadaMiz    private function handleSourceSSERequest()
9447595f9cSYamadaMiz    {
956b7e227cSYamadaMiz        header('Content-Type: text/event-stream');
966b7e227cSYamadaMiz        header('Cache-Control: no-cache');
976b7e227cSYamadaMiz
986b7e227cSYamadaMiz        session_start();
996b7e227cSYamadaMiz        if (!isset($_SESSION['source_filepath'])) {
1006b7e227cSYamadaMiz            echo "data: Mizar file path not found in session\n\n";
1016b7e227cSYamadaMiz            ob_flush();
1026b7e227cSYamadaMiz            flush();
1036b7e227cSYamadaMiz            return;
1046b7e227cSYamadaMiz        }
1056b7e227cSYamadaMiz
1066b7e227cSYamadaMiz        $filePath = $_SESSION['source_filepath'];
1076b7e227cSYamadaMiz        $this->streamSourceOutput($filePath);
1086b7e227cSYamadaMiz
1096b7e227cSYamadaMiz        echo "event: end\n";
1106b7e227cSYamadaMiz        echo "data: Compilation complete\n\n";
1116b7e227cSYamadaMiz        ob_flush();
1126b7e227cSYamadaMiz        flush();
1136b7e227cSYamadaMiz    }
1146b7e227cSYamadaMiz
11547595f9cSYamadaMiz    // view用のコンパイルリクエスト処理
11647595f9cSYamadaMiz    private function handleViewCompileRequest()
11747595f9cSYamadaMiz    {
11847595f9cSYamadaMiz        global $INPUT;
11947595f9cSYamadaMiz        $content = $INPUT->post->str('content');
12047595f9cSYamadaMiz
12147595f9cSYamadaMiz        $filePath = $this->createTempFile($content);
12247595f9cSYamadaMiz
12347595f9cSYamadaMiz        session_start();
12447595f9cSYamadaMiz        $_SESSION['view_filepath'] = $filePath;
12547595f9cSYamadaMiz
12647595f9cSYamadaMiz        $this->sendAjaxResponse(true, 'Mizar content processed successfully');
12747595f9cSYamadaMiz    }
12847595f9cSYamadaMiz
12947595f9cSYamadaMiz    // view用のSSEリクエスト処理
13047595f9cSYamadaMiz    private function handleViewSSERequest()
13147595f9cSYamadaMiz    {
13247595f9cSYamadaMiz        header('Content-Type: text/event-stream');
13347595f9cSYamadaMiz        header('Cache-Control: no-cache');
13447595f9cSYamadaMiz
13547595f9cSYamadaMiz        session_start();
13647595f9cSYamadaMiz        if (!isset($_SESSION['view_filepath'])) {
13747595f9cSYamadaMiz            echo "data: Mizar file path not found in session\n\n";
13847595f9cSYamadaMiz            ob_flush();
13947595f9cSYamadaMiz            flush();
14047595f9cSYamadaMiz            return;
14147595f9cSYamadaMiz        }
14247595f9cSYamadaMiz
14347595f9cSYamadaMiz        $filePath = $_SESSION['view_filepath'];
1449fc5dc4bSYamadaMiz        $this->streamViewCompileOutput($filePath);
14547595f9cSYamadaMiz
14647595f9cSYamadaMiz        echo "event: end\n";
14747595f9cSYamadaMiz        echo "data: Compilation complete\n\n";
14847595f9cSYamadaMiz        ob_flush();
14947595f9cSYamadaMiz        flush();
15047595f9cSYamadaMiz    }
15147595f9cSYamadaMiz
15247595f9cSYamadaMiz    // Mizarコンテンツの抽出
15347595f9cSYamadaMiz    private function extractMizarContent($pageContent)
15447595f9cSYamadaMiz    {
1556b7e227cSYamadaMiz        $pattern = '/<mizar\s+([^>]+)>(.*?)<\/mizar>/s';
1566b7e227cSYamadaMiz        preg_match_all($pattern, $pageContent, $matches, PREG_SET_ORDER);
1576b7e227cSYamadaMiz
1586b7e227cSYamadaMiz        if (empty($matches)) {
15947595f9cSYamadaMiz            return null;
1606b7e227cSYamadaMiz        }
1616b7e227cSYamadaMiz
16247595f9cSYamadaMiz        $fileName = trim($matches[0][1]);
1636b7e227cSYamadaMiz        $combinedContent = '';
1646b7e227cSYamadaMiz
1656b7e227cSYamadaMiz        foreach ($matches as $match) {
1666b7e227cSYamadaMiz            if (trim($match[1]) !== $fileName) {
1676b7e227cSYamadaMiz                return ['error' => 'File name mismatch in <mizar> tags'];
1686b7e227cSYamadaMiz            }
1696b7e227cSYamadaMiz            $combinedContent .= trim($match[2]) . "\n";
1706b7e227cSYamadaMiz        }
1716b7e227cSYamadaMiz
1726b7e227cSYamadaMiz        return ['fileName' => $fileName, 'content' => $combinedContent];
1736b7e227cSYamadaMiz    }
1746b7e227cSYamadaMiz
17547595f9cSYamadaMiz    // Mizarコンテンツの保存
17647595f9cSYamadaMiz    private function saveMizarContent($mizarData)
17747595f9cSYamadaMiz    {
1786b7e227cSYamadaMiz        $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\');
1796b7e227cSYamadaMiz        $filePath = $workPath . "/TEXT/" . $mizarData['fileName'];
1806b7e227cSYamadaMiz        file_put_contents($filePath, $mizarData['content']);
1816b7e227cSYamadaMiz        return $filePath;
1826b7e227cSYamadaMiz    }
1836b7e227cSYamadaMiz
18447595f9cSYamadaMiz    // source用の出力をストリーム
18547595f9cSYamadaMiz    private function streamSourceOutput($filePath)
18647595f9cSYamadaMiz    {
1876b7e227cSYamadaMiz        $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\');
1886b7e227cSYamadaMiz        chdir($workPath);
1896b7e227cSYamadaMiz
1906b7e227cSYamadaMiz        $command = "miz2prel " . escapeshellarg($filePath);
1916b7e227cSYamadaMiz        $process = proc_open($command, array(1 => array("pipe", "w")), $pipes);
1926b7e227cSYamadaMiz
1936b7e227cSYamadaMiz        if (is_resource($process)) {
1946b7e227cSYamadaMiz            while ($line = fgets($pipes[1])) {
1956b7e227cSYamadaMiz                echo "data: " . $line . "\n\n";
1966b7e227cSYamadaMiz                ob_flush();
1976b7e227cSYamadaMiz                flush();
1986b7e227cSYamadaMiz            }
1996b7e227cSYamadaMiz            fclose($pipes[1]);
2009fc5dc4bSYamadaMiz
2019fc5dc4bSYamadaMiz            // エラー処理の追加
2029fc5dc4bSYamadaMiz            $errFilename = str_replace('.miz', '.err', $filePath);
2039fc5dc4bSYamadaMiz            if ($this->handleCompilationErrors($errFilename, rtrim($this->getConf('mizar_share_dir'), '/\\') . '/mizar.msg')) {
2049fc5dc4bSYamadaMiz                // エラーがあった場合は処理を終了
2059fc5dc4bSYamadaMiz                proc_close($process);
2069fc5dc4bSYamadaMiz                return;
2079fc5dc4bSYamadaMiz            }
2089fc5dc4bSYamadaMiz
2096b7e227cSYamadaMiz            proc_close($process);
2106b7e227cSYamadaMiz        }
2116b7e227cSYamadaMiz    }
2126b7e227cSYamadaMiz
21347595f9cSYamadaMiz    // view用の一時ファイル作成
21447595f9cSYamadaMiz    private function createTempFile($content)
21547595f9cSYamadaMiz    {
2166b7e227cSYamadaMiz        $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/';
21747595f9cSYamadaMiz        $uniqueName = str_replace('.', '_', uniqid('tmp', true));
2186b7e227cSYamadaMiz        $tempFilename = $workPath . $uniqueName . ".miz";
2196b7e227cSYamadaMiz        file_put_contents($tempFilename, $content);
2206b7e227cSYamadaMiz        return $tempFilename;
2216b7e227cSYamadaMiz    }
2226b7e227cSYamadaMiz
22347595f9cSYamadaMiz    //  Clear all temporary files in the TEXT folder
22447595f9cSYamadaMiz    private function clearTempFiles()
22547595f9cSYamadaMiz    {
22647595f9cSYamadaMiz        $workPath = rtrim($this->getConf('mizar_work_dir'), '/\\') . '/TEXT/';
22747595f9cSYamadaMiz        $files = glob($workPath . '*');  // TEXTフォルダ内のすべてのファイルを取得
22847595f9cSYamadaMiz
22947595f9cSYamadaMiz        $errors = [];
23047595f9cSYamadaMiz        foreach ($files as $file) {
23147595f9cSYamadaMiz            if (is_file($file)) {
23247595f9cSYamadaMiz                // ファイルが使用中かどうか確認
23347595f9cSYamadaMiz                if (!$this->is_file_locked($file)) {
23447595f9cSYamadaMiz                    $retries = 3; // 最大3回リトライ
23547595f9cSYamadaMiz                    while ($retries > 0) {
23647595f9cSYamadaMiz                        if (unlink($file)) {
23747595f9cSYamadaMiz                            break; // 削除成功
23847595f9cSYamadaMiz                        }
2395cbf3a53SYamadaMiz                        $errors[] = "Error deleting $file: " . error_get_last()['message'];
24047595f9cSYamadaMiz                        $retries--;
24147595f9cSYamadaMiz                        sleep(1); // 1秒待ってリトライ
24247595f9cSYamadaMiz                    }
24347595f9cSYamadaMiz                    if ($retries === 0) {
24447595f9cSYamadaMiz                        $errors[] = "Failed to delete: $file";  // 削除失敗
24547595f9cSYamadaMiz                    }
24647595f9cSYamadaMiz                } else {
24747595f9cSYamadaMiz                    $errors[] = "File is locked: $file";  // ファイルがロックされている
24847595f9cSYamadaMiz                }
24947595f9cSYamadaMiz            }
25047595f9cSYamadaMiz        }
25147595f9cSYamadaMiz
25247595f9cSYamadaMiz        if (empty($errors)) {
25347595f9cSYamadaMiz            $this->sendAjaxResponse(true, 'Temporary files cleared successfully');
25447595f9cSYamadaMiz        } else {
25547595f9cSYamadaMiz            $this->sendAjaxResponse(false, 'Some files could not be deleted', $errors);
25647595f9cSYamadaMiz        }
25747595f9cSYamadaMiz    }
25847595f9cSYamadaMiz
25947595f9cSYamadaMiz    // ファイルがロックされているかをチェックする関数
26047595f9cSYamadaMiz    private function is_file_locked($file)
26147595f9cSYamadaMiz    {
26247595f9cSYamadaMiz        $fileHandle = @fopen($file, "r+");
26347595f9cSYamadaMiz
26447595f9cSYamadaMiz        if ($fileHandle === false) {
26547595f9cSYamadaMiz            return true; // ファイルが開けない、つまりロックされているかアクセス権がない
26647595f9cSYamadaMiz        }
26747595f9cSYamadaMiz
26847595f9cSYamadaMiz        $locked = !flock($fileHandle, LOCK_EX | LOCK_NB); // ロックの取得を試みる(非ブロッキングモード)
26947595f9cSYamadaMiz
27047595f9cSYamadaMiz        fclose($fileHandle);
27147595f9cSYamadaMiz        return $locked; // ロックが取得できなければファイルはロックされている
27247595f9cSYamadaMiz    }
27347595f9cSYamadaMiz
2749fc5dc4bSYamadaMiz    // View用コンパイル出力のストリーム
2759fc5dc4bSYamadaMiz    private function streamViewCompileOutput($filePath)
27647595f9cSYamadaMiz    {
2776b7e227cSYamadaMiz        $workPath = $this->getConf('mizar_work_dir');
2786b7e227cSYamadaMiz        $sharePath = rtrim($this->getConf('mizar_share_dir'), '/\\') . '/';
2796b7e227cSYamadaMiz
2806b7e227cSYamadaMiz        chdir($workPath);
2816b7e227cSYamadaMiz
2829fc5dc4bSYamadaMiz        $errFilename = str_replace('.miz', '.err', $filePath);
2836b7e227cSYamadaMiz        $command = "makeenv " . escapeshellarg($filePath);
2846b7e227cSYamadaMiz        $process = proc_open($command, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $pipes);
2856b7e227cSYamadaMiz
2866b7e227cSYamadaMiz        if (is_resource($process)) {
2876b7e227cSYamadaMiz            while ($line = fgets($pipes[1])) {
2886b7e227cSYamadaMiz                echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n";
2896b7e227cSYamadaMiz                ob_flush();
2906b7e227cSYamadaMiz                flush();
2916b7e227cSYamadaMiz            }
2926b7e227cSYamadaMiz            fclose($pipes[1]);
2936b7e227cSYamadaMiz
2946b7e227cSYamadaMiz            while ($line = fgets($pipes[2])) {
2956b7e227cSYamadaMiz                echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n";
2966b7e227cSYamadaMiz                ob_flush();
2976b7e227cSYamadaMiz                flush();
2986b7e227cSYamadaMiz            }
2996b7e227cSYamadaMiz            fclose($pipes[2]);
3006b7e227cSYamadaMiz            proc_close($process);
3016b7e227cSYamadaMiz
3021174571dSYamadaMiz            // makeenvのエラー処理
3039fc5dc4bSYamadaMiz            if ($this->handleCompilationErrors($errFilename, $sharePath . '/mizar.msg')) {
3046b7e227cSYamadaMiz                return;
3056b7e227cSYamadaMiz            }
3066b7e227cSYamadaMiz
3076b7e227cSYamadaMiz            // verifierの実行
3086b7e227cSYamadaMiz            $exePath = rtrim($this->getConf('mizar_exe_dir'), '/\\') . '/';
3096b7e227cSYamadaMiz            $verifierPath = escapeshellarg($exePath . "verifier");
3106b7e227cSYamadaMiz            if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') {
3116b7e227cSYamadaMiz                $verifierPath .= ".exe";
3126b7e227cSYamadaMiz            }
3136b7e227cSYamadaMiz            $verifierCommand = $verifierPath . " -q -l " . escapeshellarg("TEXT/" . basename($filePath));
3146b7e227cSYamadaMiz
3156b7e227cSYamadaMiz            $verifierProcess = proc_open($verifierCommand, array(1 => array("pipe", "w"), 2 => array("pipe", "w")), $verifierPipes);
3166b7e227cSYamadaMiz
3176b7e227cSYamadaMiz            if (is_resource($verifierProcess)) {
3186b7e227cSYamadaMiz                while ($line = fgets($verifierPipes[1])) {
3196b7e227cSYamadaMiz                    echo "data: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n";
3206b7e227cSYamadaMiz                    ob_flush();
3216b7e227cSYamadaMiz                    flush();
3226b7e227cSYamadaMiz                }
3236b7e227cSYamadaMiz                fclose($verifierPipes[1]);
3246b7e227cSYamadaMiz
3256b7e227cSYamadaMiz                while ($line = fgets($verifierPipes[2])) {
3266b7e227cSYamadaMiz                    echo "data: ERROR: " . mb_convert_encoding($line, 'UTF-8', 'SJIS') . "\n\n";
3276b7e227cSYamadaMiz                    ob_flush();
3286b7e227cSYamadaMiz                    flush();
3296b7e227cSYamadaMiz                }
3306b7e227cSYamadaMiz                fclose($verifierPipes[2]);
3316b7e227cSYamadaMiz                proc_close($verifierProcess);
3321174571dSYamadaMiz
3331174571dSYamadaMiz                // verifierのエラー処理
3341174571dSYamadaMiz                if ($this->handleCompilationErrors($errFilename, $sharePath . '/mizar.msg')) {
3351174571dSYamadaMiz                    return;
3361174571dSYamadaMiz                }
3376b7e227cSYamadaMiz            } else {
3386b7e227cSYamadaMiz                echo "data: ERROR: Failed to execute verifier command.\n\n";
3396b7e227cSYamadaMiz                ob_flush();
3406b7e227cSYamadaMiz                flush();
3416b7e227cSYamadaMiz            }
3426b7e227cSYamadaMiz        } else {
3436b7e227cSYamadaMiz            echo "data: ERROR: Failed to execute makeenv command.\n\n";
3446b7e227cSYamadaMiz            ob_flush();
3456b7e227cSYamadaMiz            flush();
3466b7e227cSYamadaMiz        }
3476b7e227cSYamadaMiz    }
3486b7e227cSYamadaMiz
34947595f9cSYamadaMiz    private function getMizarErrorMessages($mizarMsgFile)
35047595f9cSYamadaMiz    {
3516b7e227cSYamadaMiz        $errorMessages = [];
3526b7e227cSYamadaMiz        $lines = file($mizarMsgFile, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
3536b7e227cSYamadaMiz
3546b7e227cSYamadaMiz        $isReadingErrorMsg = false;
3556b7e227cSYamadaMiz        $key = 0;
3566b7e227cSYamadaMiz
3576b7e227cSYamadaMiz        foreach ($lines as $line) {
3586b7e227cSYamadaMiz            if (preg_match('/# (\d+)/', $line, $matches)) {
3596b7e227cSYamadaMiz                $isReadingErrorMsg = true;
3606b7e227cSYamadaMiz                $key = intval($matches[1]);
3616b7e227cSYamadaMiz            } elseif ($isReadingErrorMsg) {
3626b7e227cSYamadaMiz                $errorMessages[$key] = $line;
3636b7e227cSYamadaMiz                $isReadingErrorMsg = false;
3646b7e227cSYamadaMiz            }
3656b7e227cSYamadaMiz        }
3666b7e227cSYamadaMiz
3676b7e227cSYamadaMiz        return $errorMessages;
3686b7e227cSYamadaMiz    }
3696b7e227cSYamadaMiz
37047595f9cSYamadaMiz    private function sendAjaxResponse($success, $message, $data = '')
37147595f9cSYamadaMiz    {
3726b7e227cSYamadaMiz        header('Content-Type: application/json');
3736b7e227cSYamadaMiz        echo json_encode(['success' => $success, 'message' => $message, 'data' => $data]);
3746b7e227cSYamadaMiz        exit;
3756b7e227cSYamadaMiz    }
3769fc5dc4bSYamadaMiz
3779fc5dc4bSYamadaMiz    private function handleCompilationErrors($errFilename, $mizarMsgFilePath)
3789fc5dc4bSYamadaMiz    {
3799fc5dc4bSYamadaMiz        if (file_exists($errFilename)) {
3809fc5dc4bSYamadaMiz            $errors = [];
3819fc5dc4bSYamadaMiz            $errorLines = file($errFilename, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
3829fc5dc4bSYamadaMiz            if (empty($errorLines)) {
3839fc5dc4bSYamadaMiz            }
3849fc5dc4bSYamadaMiz            foreach ($errorLines as $errorLine) {
3859fc5dc4bSYamadaMiz                if (preg_match('/(\d+)\s+(\d+)\s+(\d+)/', $errorLine, $matches)) {
3869fc5dc4bSYamadaMiz                    $errorCode = intval($matches[3]);
3879fc5dc4bSYamadaMiz                    $errors[] = [
3889fc5dc4bSYamadaMiz                        'code' => $errorCode,
3899fc5dc4bSYamadaMiz                        'line' => intval($matches[1]),
3909fc5dc4bSYamadaMiz                        'column' => intval($matches[2]),
3919fc5dc4bSYamadaMiz                        'message' => $this->getMizarErrorMessages($mizarMsgFilePath)[$errorCode] ?? 'Unknown error'
3929fc5dc4bSYamadaMiz                    ];
3939fc5dc4bSYamadaMiz                }
3949fc5dc4bSYamadaMiz            }
3959fc5dc4bSYamadaMiz            if (!empty($errors)) {
3969fc5dc4bSYamadaMiz                echo "event: compileErrors\n";
3979fc5dc4bSYamadaMiz                echo "data: " . json_encode($errors) . "\n\n";
3989fc5dc4bSYamadaMiz                ob_flush();
3999fc5dc4bSYamadaMiz                flush();
4009fc5dc4bSYamadaMiz                return true;
4019fc5dc4bSYamadaMiz            }
4029fc5dc4bSYamadaMiz        }
4039fc5dc4bSYamadaMiz        return false;
4049fc5dc4bSYamadaMiz    }
4059fc5dc4bSYamadaMiz
4069fc5dc4bSYamadaMiz    private function handle_create_combined_file()
4079fc5dc4bSYamadaMiz    {
4089fc5dc4bSYamadaMiz        global $INPUT;
4099fc5dc4bSYamadaMiz
4109fc5dc4bSYamadaMiz        // 投稿されたコンテンツを取得
4119fc5dc4bSYamadaMiz        $combinedContent = $INPUT->post->str('content');
412a0444036SYamadaMiz        $filename = $INPUT->post->str('filename', 'combined_file.miz'); // デフォルトのファイル名を指定
4139fc5dc4bSYamadaMiz
414a0444036SYamadaMiz        // ファイルを保存せず、コンテンツを直接返す
415a0444036SYamadaMiz        if (!empty($combinedContent)) {
416a0444036SYamadaMiz            // ファイルの内容をレスポンスで返す(PHP側でファイルを作成しない)
417a0444036SYamadaMiz            $this->sendAjaxResponse(true, 'File created successfully', [
418a0444036SYamadaMiz                'filename' => $filename,
419a0444036SYamadaMiz                'content' => $combinedContent
420a0444036SYamadaMiz            ]);
421a0444036SYamadaMiz            error_log("File content sent: " . $filename);
4229fc5dc4bSYamadaMiz        } else {
423a0444036SYamadaMiz            $this->sendAjaxResponse(false, 'Content is empty, no file created');
4249fc5dc4bSYamadaMiz        }
4259fc5dc4bSYamadaMiz    }
4266b7e227cSYamadaMiz}