diff options
author | Keuin <[email protected]> | 2023-06-01 20:14:00 +0800 |
---|---|---|
committer | Keuin <[email protected]> | 2023-06-01 20:14:00 +0800 |
commit | 77c5ef83402aa28fc6d3a9d2982ccb03fea70d2f (patch) | |
tree | e86ca5b3191060a9966cf92e300a6c6679129e45 /texgen.py | |
parent | 093c815e9e3ee8aecf92cb1de7b51337ca28c16e (diff) |
texgen (WIP)
Diffstat (limited to 'texgen.py')
-rw-r--r-- | texgen.py | 65 |
1 files changed, 65 insertions, 0 deletions
diff --git a/texgen.py b/texgen.py new file mode 100644 index 0000000..851df83 --- /dev/null +++ b/texgen.py @@ -0,0 +1,65 @@ +import asyncio +import contextlib +import hashlib +import os +import shutil +import typing +import subprocess + + +def temp_dir(path: str): + os.mkdir(path) + try: + yield path + finally: + # shutil.rmtree(path) + pass + + +class TexGenerationError(Exception): + pass + + +class TexGenerator: + + def __init__(self, storage_path: str, temp_path: str, task_timeout): + self._storage_path = os.path.abspath(storage_path) + self._temp_path = os.path.abspath(temp_path) + self._task_timeout = task_timeout + for p in (self._temp_path, self._storage_path): + try: + os.mkdir(p) + except FileExistsError: + pass + + async def xelatex(self, tex_source: str) -> str: + tex_hash = hashlib.sha256(tex_source.encode('utf-8')).hexdigest() + cache_file_path = os.path.join(self._storage_path, f'{tex_hash}.pdf') + if os.path.exists(cache_file_path): + return cache_file_path + with temp_dir(os.path.join(self._temp_path, os.urandom(24).hex())) as workdir: + job_name = 'texput' + with subprocess.Popen( + [ + 'xelatex', + '-interaction=nonstopmode', + '-halt-on-error', + # this seems not working, as the output file name is always set to default `texput.pdf` + f'-jobname={job_name}', + ], + cwd=workdir, + stdin=subprocess.PIPE, + stdout=subprocess.PIPE, + stderr=subprocess.STDOUT, + text=True, + shell=True, + ) as proc: + stdout, stderr = proc.communicate(input=tex_source, timeout=self._task_timeout) + print('STDOUT', stdout) + print('STDERR', stderr) + if proc.returncode != 0: + raise TexGenerationError(f'xelatex process exited with non-zero code {proc.returncode}') + + os.rename(os.path.join(workdir, f'{job_name}.pdf'), cache_file_path) + return cache_file_path |