more secure tmp file creation for latex
This commit is contained in:
parent
395bf4fd99
commit
43b2a1b4f0
1 changed files with 31 additions and 34 deletions
|
@ -29,7 +29,7 @@
|
|||
|
||||
import os
|
||||
import random
|
||||
from tempfile import gettempdir
|
||||
from tempfile import mkstemp, mkdtemp
|
||||
from subprocess import Popen, PIPE
|
||||
|
||||
import logging
|
||||
|
@ -57,24 +57,6 @@ def check_blacklist(str_):
|
|||
return True
|
||||
return False
|
||||
|
||||
def get_tmpfile_name():
|
||||
random.seed()
|
||||
nb = 0
|
||||
while(nb < 100):
|
||||
int_ = random.randint(0, 10000)
|
||||
filename = os.path.join(gettempdir(), 'gajimtex_' + int_.__str__())
|
||||
# Check if a file to not overwrite it
|
||||
ok = True
|
||||
extensions = ['.tex', '.log', '.aux', '.dvi']
|
||||
for ext in extensions:
|
||||
if os.path.exists(filename + ext):
|
||||
ok = False
|
||||
break
|
||||
if ok:
|
||||
return filename
|
||||
nb += 1
|
||||
return filename
|
||||
|
||||
def write_latex(filename, str_):
|
||||
texstr = '\\documentclass[12pt]{article}\\usepackage[dvips]{graphicx}'
|
||||
texstr += '\\usepackage{amsmath}\\usepackage{amssymb}'
|
||||
|
@ -91,12 +73,13 @@ def write_latex(filename, str_):
|
|||
# a wrapper for Popen so that no window gets opened on Windows
|
||||
# (i think this is the reason we're using Popen rather than just system())
|
||||
# stdout goes to a pipe so that it can be read
|
||||
def popen_nt_friendly(command):
|
||||
def popen_nt_friendly(command, directory):
|
||||
if os.name == 'nt':
|
||||
# CREATE_NO_WINDOW
|
||||
return Popen(command, creationflags=0x08000000, cwd=gettempdir(), stdout=PIPE)
|
||||
return Popen(command, creationflags=0x08000000, cwd=directory,
|
||||
stdout=PIPE)
|
||||
else:
|
||||
return Popen(command, cwd=gettempdir(), stdout=PIPE)
|
||||
return Popen(command, cwd=directory, stdout=PIPE)
|
||||
|
||||
def check_for_latex_support():
|
||||
"""
|
||||
|
@ -112,16 +95,16 @@ def check_for_latex_support():
|
|||
except LatexError:
|
||||
return False
|
||||
|
||||
def try_run(argv):
|
||||
def try_run(argv, directory):
|
||||
try:
|
||||
p = popen_nt_friendly(argv)
|
||||
p = popen_nt_friendly(argv, directory)
|
||||
out = p.communicate()[0]
|
||||
log.info(out)
|
||||
return p.wait()
|
||||
except Exception, e:
|
||||
return _('Error executing "%(command)s": %(error)s') % {
|
||||
'command': " ".join(argv),
|
||||
'error': helpers.decode_string(str(e))}
|
||||
'command': " ".join(argv),
|
||||
'error': helpers.decode_string(str(e))}
|
||||
|
||||
|
||||
def latex_to_image(str_):
|
||||
|
@ -137,32 +120,41 @@ def latex_to_image(str_):
|
|||
return []
|
||||
except AttributeError:
|
||||
# interface may not be available when we test latext at startup
|
||||
return ['-fg', 'rgb 0.0 0.0 0.0']
|
||||
return {'hex': ['+level-colors', '0x000000'],
|
||||
'tex': ['-fg', 'rgb 0.0 0.0 0.0']}[fmt]
|
||||
|
||||
# filter latex code with bad commands
|
||||
if check_blacklist(str_):
|
||||
# we triggered the blacklist, immediately return None
|
||||
return None
|
||||
|
||||
tmpfile = get_tmpfile_name()
|
||||
try:
|
||||
tmpdir = mkdtemp(prefix='gajimtex')
|
||||
tmppng = mkstemp(prefix='gajim_tex', suffix='.png')[1]
|
||||
except Exception:
|
||||
raise LatexError('could not securely create one or more temporary files'
|
||||
' for LaTeX conversion')
|
||||
|
||||
tmpfile = os.path.join(tmpdir, 'gajim_tex')
|
||||
|
||||
# build latex string
|
||||
write_latex(os.path.join(tmpfile + '.tex'), str_)
|
||||
write_latex(tmpfile + '.tex', str_)
|
||||
|
||||
# convert TeX to dvi
|
||||
exitcode = try_run(['latex', '--interaction=nonstopmode',
|
||||
tmpfile + '.tex'])
|
||||
exitcode = try_run(['latex', '--interaction=nonstopmode', tmpfile + '.tex'],
|
||||
tmpdir)
|
||||
|
||||
if exitcode == 0:
|
||||
# convert dvi to png
|
||||
latex_png_dpi = gajim.config.get('latex_png_dpi')
|
||||
exitcode = try_run(['dvipng'] + fg_str('tex') + ['-T', 'tight', '-D',
|
||||
latex_png_dpi, tmpfile + '.dvi', '-o', tmpfile + '.png'])
|
||||
latex_png_dpi, tmpfile + '.dvi', '-o', tmpfile + '.png'], tmpdir)
|
||||
|
||||
if exitcode:
|
||||
# dvipng failed, try convert
|
||||
exitcode = try_run(['convert'] + fg_str('hex') + ['-trim',
|
||||
'-density', latex_png_dpi, tmpfile + '.dvi', tmpfile + '.png'])
|
||||
'-density', latex_png_dpi, tmpfile + '.dvi', tmpfile + '.png'],
|
||||
tmpdir)
|
||||
|
||||
# remove temp files created by us and TeX
|
||||
extensions = ['.tex', '.log', '.aux', '.dvi']
|
||||
|
@ -172,10 +164,15 @@ def latex_to_image(str_):
|
|||
except Exception:
|
||||
pass
|
||||
|
||||
if exitcode == 0:
|
||||
os.rename(tmpfile + '.png', tmppng)
|
||||
|
||||
os.rmdir(tmpdir)
|
||||
|
||||
if isinstance(exitcode, (unicode, str)):
|
||||
raise LatexError(exitcode)
|
||||
|
||||
if exitcode == 0:
|
||||
result = tmpfile + '.png'
|
||||
result = tmppng
|
||||
|
||||
return result
|
||||
|
|
Loading…
Add table
Reference in a new issue