tex.py
上传用户:market2
上传日期:2018-11-18
资源大小:18786k
文件大小:11k
源码类别:

外挂编程

开发平台:

Windows_Unix

  1. """SCons.Tool.tex
  2. Tool-specific initialization for TeX.
  3. There normally shouldn't be any need to import this module directly.
  4. It will usually be imported through the generic SCons.Tool.Tool()
  5. selection method.
  6. """
  7. #
  8. # Copyright (c) 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008 The SCons Foundation
  9. #
  10. # Permission is hereby granted, free of charge, to any person obtaining
  11. # a copy of this software and associated documentation files (the
  12. # "Software"), to deal in the Software without restriction, including
  13. # without limitation the rights to use, copy, modify, merge, publish,
  14. # distribute, sublicense, and/or sell copies of the Software, and to
  15. # permit persons to whom the Software is furnished to do so, subject to
  16. # the following conditions:
  17. #
  18. # The above copyright notice and this permission notice shall be included
  19. # in all copies or substantial portions of the Software.
  20. #
  21. # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
  22. # KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
  23. # WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  24. # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  25. # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  26. # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  27. # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  28. #
  29. __revision__ = "src/engine/SCons/Tool/tex.py 3057 2008/06/09 22:21:00 knight"
  30. import os.path
  31. import re
  32. import string
  33. import SCons.Action
  34. import SCons.Node
  35. import SCons.Node.FS
  36. import SCons.Util
  37. warning_rerun_re = re.compile('(^LaTeX Warning:.*Rerun)|(^Package w+ Warning:.*Rerun)', re.MULTILINE)
  38. rerun_citations_str = "^LaTeX Warning:.*n.*Rerun to get citations correct"
  39. rerun_citations_re = re.compile(rerun_citations_str, re.MULTILINE)
  40. undefined_references_str = '(^LaTeX Warning:.*undefined references)|(^Package w+ Warning:.*undefined citations)'
  41. undefined_references_re = re.compile(undefined_references_str, re.MULTILINE)
  42. openout_aux_re = re.compile(r"\openout.*`(.*.aux)'")
  43. openout_re = re.compile(r"\openout.*`(.*)'")
  44. makeindex_re = re.compile(r"^[^%]*\makeindex", re.MULTILINE)
  45. tableofcontents_re = re.compile(r"^[^%]*\tableofcontents", re.MULTILINE)
  46. bibliography_re = re.compile(r"^[^%]*\bibliography", re.MULTILINE)
  47. # An Action sufficient to build any generic tex file.
  48. TeXAction = None
  49. # An action to build a latex file.  This action might be needed more
  50. # than once if we are dealing with labels and bibtex.
  51. LaTeXAction = None
  52. # An action to run BibTeX on a file.
  53. BibTeXAction = None
  54. # An action to run MakeIndex on a file.
  55. MakeIndexAction = None
  56. def InternalLaTeXAuxAction(XXXLaTeXAction, target = None, source= None, env=None):
  57.     """A builder for LaTeX files that checks the output in the aux file
  58.     and decides how many times to use LaTeXAction, and BibTeXAction."""
  59.     basename = SCons.Util.splitext(str(source[0]))[0]
  60.     basedir = os.path.split(str(source[0]))[0]
  61.     basefile = os.path.split(str(basename))[1]
  62.     abspath = os.path.abspath(basedir)
  63.     targetbase = SCons.Util.splitext(str(target[0]))[0]
  64.     targetdir = os.path.split(str(target[0]))[0]
  65.     # Not sure if these environment changes should go here or make the
  66.     # user do them I undo all but TEXPICTS but there is still the side
  67.     # effect of creating the empty (':') entries in the environment.
  68.     def modify_env_var(env, var, abspath):
  69.         try:
  70.             save = env['ENV'][var]
  71.         except KeyError:
  72.             save = ':'
  73.             env['ENV'][var] = ''
  74.         if SCons.Util.is_List(env['ENV'][var]):
  75.             env['ENV'][var] = [abspath] + env['ENV'][var]
  76.         else:
  77.             env['ENV'][var] = abspath + os.pathsep + env['ENV'][var]
  78.         return save
  79.     texinputs_save = modify_env_var(env, 'TEXINPUTS', abspath)
  80.     bibinputs_save = modify_env_var(env, 'BIBINPUTS', abspath)
  81.     bstinputs_save = modify_env_var(env, 'BSTINPUTS', abspath)
  82.     texpicts_save = modify_env_var(env, 'TEXPICTS', abspath)
  83.     # Create these file names with the target directory since they will
  84.     # be made there.   That's because the *COM variables have the cd
  85.     # command in the prolog.
  86.     bblfilename = os.path.join(targetdir, basefile + '.bbl')
  87.     bblContents = ""
  88.     if os.path.exists(bblfilename):
  89.         bblContents = open(bblfilename, "rb").read()
  90.     idxfilename = os.path.join(targetdir, basefile + '.idx')
  91.     idxContents = ""
  92.     if os.path.exists(idxfilename):
  93.         idxContents = open(idxfilename, "rb").read()
  94.     tocfilename = os.path.join(targetdir, basefile + '.toc')
  95.     tocContents = ""
  96.     if os.path.exists(tocfilename):
  97.         tocContents = open(tocfilename, "rb").read()
  98.     # Run LaTeX once to generate a new aux file and log file.
  99.     XXXLaTeXAction(target, source, env)
  100.     # Decide if various things need to be run, or run again.  We check
  101.     # for the existence of files before opening them--even ones like the
  102.     # aux file that TeX always creates--to make it possible to write tests
  103.     # with stubs that don't necessarily generate all of the same files.
  104.     # Read the log file to find all .aux files
  105.     logfilename = os.path.join(targetbase + '.log')
  106.     auxfiles = []
  107.     if os.path.exists(logfilename):
  108.         content = open(logfilename, "rb").read()
  109.         auxfiles = openout_aux_re.findall(content)
  110.     # Now decide if bibtex will need to be run.
  111.     for auxfilename in auxfiles:
  112.         target_aux = os.path.join(targetdir, auxfilename)
  113.         if os.path.exists(target_aux):
  114.             content = open(target_aux, "rb").read()
  115.             if string.find(content, "bibdata") != -1:
  116.                 bibfile = env.fs.File(targetbase)
  117.                 BibTeXAction(bibfile, bibfile, env)
  118.                 break
  119.     must_rerun_latex = 0
  120.     # Now decide if latex will need to be run again due to table of contents.
  121.     if os.path.exists(tocfilename) and tocContents != open(tocfilename, "rb").read():
  122.         must_rerun_latex = 1
  123.     # Now decide if latex will need to be run again due to bibliography.
  124.     if os.path.exists(bblfilename) and bblContents != open(bblfilename, "rb").read():
  125.         must_rerun_latex = 1
  126.     # Now decide if latex will need to be run again due to index.
  127.     if os.path.exists(idxfilename) and idxContents != open(idxfilename, "rb").read():
  128.         # We must run makeindex
  129.         idxfile = env.fs.File(targetbase)
  130.         MakeIndexAction(idxfile, idxfile, env)
  131.         must_rerun_latex = 1
  132.     if must_rerun_latex == 1:
  133.         XXXLaTeXAction(target, source, env)
  134.     # Now decide if latex needs to be run yet again to resolve warnings.
  135.     logfilename = targetbase + '.log'
  136.     for _ in range(int(env.subst('$LATEXRETRIES'))):
  137.         if not os.path.exists(logfilename):
  138.             break
  139.         content = open(logfilename, "rb").read()
  140.         if not warning_rerun_re.search(content) and 
  141.            not rerun_citations_re.search(content) and 
  142.            not undefined_references_re.search(content):
  143.             break
  144.         XXXLaTeXAction(target, source, env)
  145.     env['ENV']['TEXINPUTS'] = texinputs_save
  146.     env['ENV']['BIBINPUTS'] = bibinputs_save
  147.     env['ENV']['BSTINPUTS'] = bibinputs_save
  148.     # The TEXPICTS enviroment variable is needed by a dvi -> pdf step
  149.     # later on Mac OSX so leave it,
  150.     # env['ENV']['TEXPICTS']  = texpicts_save
  151.     return 0
  152. def LaTeXAuxAction(target = None, source= None, env=None):
  153.     InternalLaTeXAuxAction( LaTeXAction, target, source, env )
  154. LaTeX_re = re.compile("\\document(style|class)")
  155. def is_LaTeX(flist):
  156.     # Scan a file list to decide if it's TeX- or LaTeX-flavored.
  157.     for f in flist:
  158.         content = f.get_contents()
  159.         if LaTeX_re.search(content):
  160.             return 1
  161.     return 0
  162. def TeXLaTeXFunction(target = None, source= None, env=None):
  163.     """A builder for TeX and LaTeX that scans the source file to
  164.     decide the "flavor" of the source and then executes the appropriate
  165.     program."""
  166.     if is_LaTeX(source):
  167.         LaTeXAuxAction(target,source,env)
  168.     else:
  169.         TeXAction(target,source,env)
  170.     return 0
  171. def tex_emitter(target, source, env):
  172.     base = SCons.Util.splitext(str(source[0]))[0]
  173.     targetbase = SCons.Util.splitext(str(target[0]))[0]
  174.     target.append(targetbase + '.aux')
  175.     env.Precious(targetbase + '.aux')
  176.     target.append(targetbase + '.log')
  177.     for f in source:
  178.         content = f.get_contents()
  179.         if tableofcontents_re.search(content):
  180.             target.append(targetbase + '.toc')
  181.             env.Precious(targetbase + '.toc')
  182.         if makeindex_re.search(content):
  183.             target.append(targetbase + '.ilg')
  184.             target.append(targetbase + '.ind')
  185.             target.append(targetbase + '.idx')
  186.             env.Precious(targetbase + '.idx')
  187.         if bibliography_re.search(content):
  188.             target.append(targetbase + '.bbl')
  189.             env.Precious(targetbase + '.bbl')
  190.             target.append(targetbase + '.blg')
  191.     # read log file to get all .aux files
  192.     logfilename = targetbase + '.log'
  193.     dir, base_nodir = os.path.split(targetbase)
  194.     if os.path.exists(logfilename):
  195.         content = open(logfilename, "rb").read()
  196.         out_files = openout_re.findall(content)
  197.         out_files = filter(lambda f, b=base_nodir+'.aux': f != b, out_files)
  198.         if dir != '':
  199.             out_files = map(lambda f, d=dir: d+os.sep+f, out_files)
  200.         target.extend(out_files)
  201.         for f in out_files:
  202.             env.Precious( f )
  203.     return (target, source)
  204. TeXLaTeXAction = None
  205. def generate(env):
  206.     """Add Builders and construction variables for TeX to an Environment."""
  207.     # A generic tex file Action, sufficient for all tex files.
  208.     global TeXAction
  209.     if TeXAction is None:
  210.         TeXAction = SCons.Action.Action("$TEXCOM", "$TEXCOMSTR")
  211.     # An Action to build a latex file.  This might be needed more
  212.     # than once if we are dealing with labels and bibtex.
  213.     global LaTeXAction
  214.     if LaTeXAction is None:
  215.         LaTeXAction = SCons.Action.Action("$LATEXCOM", "$LATEXCOMSTR")
  216.     # Define an action to run BibTeX on a file.
  217.     global BibTeXAction
  218.     if BibTeXAction is None:
  219.         BibTeXAction = SCons.Action.Action("$BIBTEXCOM", "$BIBTEXCOMSTR")
  220.     # Define an action to run MakeIndex on a file.
  221.     global MakeIndexAction
  222.     if MakeIndexAction is None:
  223.         MakeIndexAction = SCons.Action.Action("$MAKEINDEXCOM", "$MAKEINDEXCOMSTR")
  224.     global TeXLaTeXAction
  225.     if TeXLaTeXAction is None:
  226.         TeXLaTeXAction = SCons.Action.Action(TeXLaTeXFunction, strfunction=None)
  227.     import dvi
  228.     dvi.generate(env)
  229.     bld = env['BUILDERS']['DVI']
  230.     bld.add_action('.tex', TeXLaTeXAction)
  231.     bld.add_emitter('.tex', tex_emitter)
  232.     env['TEX']      = 'tex'
  233.     env['TEXFLAGS'] = SCons.Util.CLVar('')
  234.     env['TEXCOM']   = 'cd ${TARGET.dir} && $TEX $TEXFLAGS ${SOURCE.file}'
  235.     # Duplicate from latex.py.  If latex.py goes away, then this is still OK.
  236.     env['LATEX']        = 'latex'
  237.     env['LATEXFLAGS']   = SCons.Util.CLVar('')
  238.     env['LATEXCOM']     = 'cd ${TARGET.dir} && $LATEX $LATEXFLAGS ${SOURCE.file}'
  239.     env['LATEXRETRIES'] = 3
  240.     env['BIBTEX']      = 'bibtex'
  241.     env['BIBTEXFLAGS'] = SCons.Util.CLVar('')
  242.     env['BIBTEXCOM']   = 'cd ${TARGET.dir} && $BIBTEX $BIBTEXFLAGS ${SOURCE.filebase}'
  243.     env['MAKEINDEX']      = 'makeindex'
  244.     env['MAKEINDEXFLAGS'] = SCons.Util.CLVar('')
  245.     env['MAKEINDEXCOM']   = 'cd ${TARGET.dir} && $MAKEINDEX $MAKEINDEXFLAGS ${SOURCE.file}'
  246. def exists(env):
  247.     return env.Detect('tex')