----------------------------------------------------------------------------- -- -- Module : Main -- Copyright : (c) Phil Freeman 2013 -- License : MIT -- -- Maintainer : Phil Freeman -- Stability : experimental -- Portability : -- -- | -- ----------------------------------------------------------------------------- {-# LANGUAGE DataKinds, GeneralizedNewtypeDeriving #-} module Main where import Control.Applicative import Control.Monad.Error import Data.Version (showVersion) import System.Console.CmdTheLine import System.Directory (createDirectoryIfMissing) import System.FilePath (takeDirectory) import System.Exit (exitSuccess, exitFailure) import System.IO (stderr) import Text.Parsec (ParseError) import qualified Language.PureScript as P import qualified Paths_purescript as Paths import qualified System.IO.UTF8 as U readInput :: Maybe [FilePath] -> IO (Either ParseError [(FilePath, P.Module)]) readInput Nothing = do text <- getContents return $ map ((,) undefined) <$> P.runIndentParser "" P.parseModules text readInput (Just input) = fmap collect $ forM input $ \inputFile -> do text <- U.readFile inputFile return $ (inputFile, P.runIndentParser inputFile P.parseModules text) where collect :: [(FilePath, Either ParseError [P.Module])] -> Either ParseError [(FilePath, P.Module)] collect = fmap concat . sequence . map (\(fp, e) -> fmap (map ((,) fp)) e) compile :: FilePath -> P.Options P.Compile -> Bool -> [FilePath] -> Maybe FilePath -> Maybe FilePath -> Bool -> IO () compile prelude opts stdin input output externs usePrefix = do modules <- readInput stdInOrInputFiles case modules of Left err -> do U.hPutStr stderr $ show err exitFailure Right ms -> do case P.compile opts (map snd ms) prefix of Left err -> do U.hPutStrLn stderr err exitFailure Right (js, exts, _) -> do case output of Just path -> mkdirp path >> U.writeFile path js Nothing -> U.putStrLn js case externs of Just path -> mkdirp path >> U.writeFile path exts Nothing -> return () exitSuccess where stdInOrInputFiles :: Maybe [FilePath] stdInOrInputFiles | stdin = Nothing | P.optionsNoPrelude opts = Just input | otherwise = Just $ prelude : input prefix = if usePrefix then ["Generated by psc version " ++ showVersion Paths.version] else [] mkdirp :: FilePath -> IO () mkdirp = createDirectoryIfMissing True . takeDirectory useStdIn :: Term Bool useStdIn = value . flag $ (optInfo [ "s", "stdin" ]) { optDoc = "Read from standard input" } inputFiles :: Term [FilePath] inputFiles = value $ posAny [] $ posInfo { posDoc = "The input .ps files" } outputFile :: Term (Maybe FilePath) outputFile = value $ opt Nothing $ (optInfo [ "o", "output" ]) { optDoc = "The output .js file" } externsFile :: Term (Maybe FilePath) externsFile = value $ opt Nothing $ (optInfo [ "e", "externs" ]) { optDoc = "The output .e.ps file" } noTco :: Term Bool noTco = value $ flag $ (optInfo [ "no-tco" ]) { optDoc = "Disable tail call optimizations" } noPrelude :: Term Bool noPrelude = value $ flag $ (optInfo [ "no-prelude" ]) { optDoc = "Omit the Prelude" } noMagicDo :: Term Bool noMagicDo = value $ flag $ (optInfo [ "no-magic-do" ]) { optDoc = "Disable the optimization that overloads the do keyword to generate efficient code specifically for the Eff monad." } runMain :: Term (Maybe String) runMain = value $ defaultOpt (Just "Main") Nothing $ (optInfo [ "main" ]) { optDoc = "Generate code to run the main method in the specified module." } noOpts :: Term Bool noOpts = value $ flag $ (optInfo [ "no-opts" ]) { optDoc = "Skip the optimization phase." } browserNamespace :: Term String browserNamespace = value $ opt "PS" $ (optInfo [ "browser-namespace" ]) { optDoc = "Specify the namespace that PureScript modules will be exported to when running in the browser." } dceModules :: Term [String] dceModules = value $ optAll [] $ (optInfo [ "m", "module" ]) { optDoc = "Enables dead code elimination, all code which is not a transitive dependency of a specified module will be removed. This argument can be used multiple times." } codeGenModules :: Term [String] codeGenModules = value $ optAll [] $ (optInfo [ "codegen" ]) { optDoc = "A list of modules for which Javascript and externs should be generated. This argument can be used multiple times." } verboseErrors :: Term Bool verboseErrors = value $ flag $ (optInfo [ "v", "verbose-errors" ]) { optDoc = "Display verbose error messages" } noPrefix :: Term Bool noPrefix = value $ flag $ (optInfo ["no-prefix" ]) { optDoc = "Do not include comment header"} options :: Term (P.Options P.Compile) options = P.Options <$> noPrelude <*> noTco <*> noMagicDo <*> runMain <*> noOpts <*> verboseErrors <*> additionalOptions where additionalOptions = P.CompileOptions <$> browserNamespace <*> dceModules <*> codeGenModules term :: FilePath -> Term (IO ()) term prelude = compile prelude <$> options <*> useStdIn <*> inputFiles <*> outputFile <*> externsFile <*> (not <$> noPrefix) termInfo :: TermInfo termInfo = defTI { termName = "psc" , version = showVersion Paths.version , termDoc = "Compiles PureScript to Javascript" } main :: IO () main = do prelude <- P.preludeFilename run (term prelude, termInfo)