2008-08-10 17:33:20 +00:00
|
|
|
{-# LANGUAGE CPP #-}
|
2008-08-10 17:33:07 +00:00
|
|
|
{-
|
|
|
|
Copyright (C) 2008 John MacFarlane <jgm@berkeley.edu>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
-}
|
|
|
|
|
|
|
|
{- |
|
2008-08-10 17:33:20 +00:00
|
|
|
Module : Text.Pandoc.PDF
|
2008-08-10 17:33:07 +00:00
|
|
|
Copyright : Copyright (C) 2006-7 John MacFarlane
|
|
|
|
License : GNU GPL, version 2 or above
|
|
|
|
|
|
|
|
Maintainer : John MacFarlane <jgm@berkeley.edu>
|
|
|
|
Stability : alpha
|
|
|
|
Portability : portable
|
|
|
|
|
2008-08-10 17:33:20 +00:00
|
|
|
Functions for producing a PDF file from LaTeX.
|
2008-08-10 17:33:07 +00:00
|
|
|
-}
|
2008-08-10 17:33:20 +00:00
|
|
|
module Text.Pandoc.PDF ( saveLaTeXAsPDF ) where
|
|
|
|
import Data.List ( isInfixOf )
|
2008-08-10 17:33:07 +00:00
|
|
|
import System.Directory
|
2008-08-10 17:33:20 +00:00
|
|
|
import System.FilePath ( (</>), (<.>), takeBaseName )
|
2008-08-10 17:33:07 +00:00
|
|
|
import System.Process ( runProcess, waitForProcess )
|
|
|
|
import System.Exit
|
2008-08-10 17:33:20 +00:00
|
|
|
import System.Environment ( getEnvironment )
|
2008-08-10 17:33:07 +00:00
|
|
|
import Text.Pandoc.Shared ( withTempDir )
|
2008-08-10 17:33:20 +00:00
|
|
|
import Prelude hiding ( writeFile, readFile, putStrLn )
|
|
|
|
import System.IO ( stderr, openFile, IOMode (..) )
|
2008-08-10 17:33:07 +00:00
|
|
|
#ifdef _UTF8STRING
|
|
|
|
import System.IO.UTF8
|
|
|
|
#else
|
|
|
|
import Text.Pandoc.UTF8
|
|
|
|
#endif
|
|
|
|
|
2008-08-10 17:33:20 +00:00
|
|
|
-- | Produce an PDF file from LaTeX.
|
|
|
|
saveLaTeXAsPDF :: FilePath -- ^ Pathname of PDF file to be produced.
|
|
|
|
-> FilePath -- ^ Relative directory of source file.
|
|
|
|
-> String -- ^ LaTeX document.
|
|
|
|
-> IO ()
|
|
|
|
saveLaTeXAsPDF destinationPDFPath sourceDirRelative latex = do
|
|
|
|
-- check for pdflatex and bibtex in path:
|
|
|
|
latexPathMaybe <- findExecutable "pdflatex"
|
|
|
|
bibtexPathMaybe <- findExecutable "bibtex"
|
|
|
|
let latexPath = case latexPathMaybe of
|
|
|
|
Nothing -> error $ "The 'pdflatex' command, which is needed to build an PDF file, was not found."
|
2008-08-10 17:33:07 +00:00
|
|
|
Just x -> x
|
2008-08-10 17:33:20 +00:00
|
|
|
let bibtexPath = case bibtexPathMaybe of
|
|
|
|
Nothing -> error $ "The 'bibtex' command, which is needed to build an PDF file, was not found."
|
|
|
|
Just x -> x
|
|
|
|
sourceDirAbsolute <- getCurrentDirectory >>= return . (</> sourceDirRelative) >>= canonicalizePath
|
|
|
|
withTempDir "pandoc-pdf" $ \tempDir -> do
|
|
|
|
env <- getEnvironment
|
|
|
|
let env' = ("TEXINPUTS", ".:" ++ sourceDirAbsolute ++ ":") : env
|
|
|
|
let baseName = "input"
|
|
|
|
writeFile (tempDir </> baseName <.> "tex") latex
|
|
|
|
let runLatex = runProgram latexPath ["-interaction=nonstopmode", baseName] tempDir env'
|
|
|
|
let runBibtex = runProgram bibtexPath [baseName] tempDir env'
|
|
|
|
messages1 <- runLatex
|
|
|
|
let logPath = tempDir </> baseName <.> "log"
|
|
|
|
tocExists <- doesFileExist (tempDir </> baseName <.> "toc")
|
|
|
|
logContents <- readFile logPath
|
|
|
|
let undefinedRefs = "There were undefined references" `isInfixOf` logContents
|
|
|
|
let needsBibtex = "itation" `isInfixOf` logContents
|
|
|
|
if needsBibtex
|
|
|
|
then runBibtex >>= hPutStr stderr . unlines
|
|
|
|
else return ()
|
|
|
|
if tocExists || undefinedRefs
|
|
|
|
then do
|
|
|
|
messages2 <- runLatex
|
|
|
|
logContents' <- readFile logPath
|
|
|
|
let stillUndefinedRefs = "There were undefined references" `isInfixOf` logContents'
|
|
|
|
if stillUndefinedRefs
|
|
|
|
then runLatex >>= hPutStr stderr . unlines
|
|
|
|
else hPutStr stderr $ unlines messages2
|
|
|
|
else
|
|
|
|
hPutStr stderr $ unlines messages1
|
|
|
|
let pdfPath = tempDir </> baseName <.> "pdf"
|
|
|
|
pdfExists <- doesFileExist pdfPath
|
|
|
|
if pdfExists
|
|
|
|
then copyFile pdfPath destinationPDFPath
|
|
|
|
else error "The PDF could not be created."
|
2008-08-10 17:33:07 +00:00
|
|
|
|
2008-08-10 17:33:20 +00:00
|
|
|
runProgram :: FilePath -- ^ pathname of executable
|
|
|
|
-> [String] -- ^ arguments
|
|
|
|
-> FilePath -- ^ working directory
|
|
|
|
-> [(String, String)] -- ^ environment
|
|
|
|
-> IO [String]
|
|
|
|
runProgram cmdPath arguments workingDir env = do
|
|
|
|
let runOutputPath = workingDir </> "output" <.> "tmp"
|
|
|
|
runOutput <- openFile runOutputPath WriteMode
|
|
|
|
ph <- runProcess cmdPath arguments (Just workingDir) (Just env) Nothing (Just runOutput) (Just runOutput)
|
|
|
|
ec <- waitForProcess ph -- requires compilation with -threaded
|
|
|
|
case ec of
|
|
|
|
ExitSuccess -> return []
|
|
|
|
_ -> do
|
|
|
|
output <- readFile runOutputPath
|
|
|
|
if (takeBaseName cmdPath) == "bibtex"
|
|
|
|
then return $! lines output
|
|
|
|
else do
|
|
|
|
return $!
|
|
|
|
(if "`ucs.sty' not found" `isInfixOf` output
|
|
|
|
then ["Please install the 'unicode' package from CTAN:",
|
|
|
|
" http://www.ctan.org/tex-archive/macros/latex/contrib/unicode/"]
|
|
|
|
else []) ++
|
|
|
|
(if "`ulem.sty' not found" `isInfixOf` output
|
|
|
|
then ["Please install the 'ulem' package from CTAN:",
|
|
|
|
" http://www.ctan.org/tex-archive/macros/latex/contrib/misc/"]
|
|
|
|
else []) ++
|
|
|
|
(if "`graphicx.sty' not found" `isInfixOf` output
|
|
|
|
then ["Please install the 'graphicx' package from CTAN:",
|
|
|
|
" http://www.ctan.org/tex-archive/macros/latex/required/graphics/"]
|
|
|
|
else []) ++
|
|
|
|
(if "`fancyhdr.sty' not found" `isInfixOf` output
|
|
|
|
then ["Please install the 'fancyhdr' package from CTAN:",
|
|
|
|
" http://www.ctan.org/tex-archive/macros/latex/contrib/fancyhdr/"]
|
|
|
|
else []) ++
|
|
|
|
(if "`array.sty' not found" `isInfixOf` output
|
|
|
|
then ["Please install the 'array' package from CTAN:",
|
|
|
|
" http://www.ctan.org/tex-archive/macros/latex/required/tools/"]
|
|
|
|
else []) ++
|
|
|
|
(filter isUseful $ lines output)
|
|
|
|
where isUseful ln = take 1 ln == "!" ||
|
|
|
|
take 2 ln == "l." ||
|
|
|
|
"Error" `isInfixOf` ln ||
|
|
|
|
"error" `isInfixOf` ln
|
2008-08-10 17:33:07 +00:00
|
|
|
|