{-| Agda main module. -} module Agda.Main where import Prelude hiding (null) import qualified Control.Exception as E import Control.Monad ( void ) import Control.Monad.Except ( MonadError(..), ExceptT(..), runExceptT ) import Control.Monad.IO.Class ( MonadIO(..) ) import qualified Data.List as List import Data.Maybe import System.Environment import System.Exit import System.Console.GetOpt import qualified System.IO as IO import Paths_Agda ( getDataDir ) import Agda.Interaction.CommandLine import Agda.Interaction.ExitCode (AgdaError(..), exitSuccess, exitAgdaWith) import Agda.Interaction.Options import Agda.Interaction.Options.Help (Help (..)) import Agda.Interaction.EmacsTop (mimicGHCi) import Agda.Interaction.JSONTop (jsonREPL) import Agda.Interaction.FindFile ( SourceFile(SourceFile) ) import qualified Agda.Interaction.Imports as Imp import Agda.TypeChecking.Monad import qualified Agda.TypeChecking.Monad.Benchmark as Bench import Agda.TypeChecking.Errors import Agda.TypeChecking.Warnings import Agda.TypeChecking.Pretty import Agda.Compiler.Backend import Agda.Compiler.Builtin import Agda.VersionCommit import Agda.Utils.FileName (absolute, filePath, AbsolutePath) import Agda.Utils.Monad import Agda.Utils.Null import Agda.Utils.String import qualified Agda.Utils.Benchmark as UtilsBench import Agda.Utils.Impossible -- | The main function runAgda :: [Backend] -> IO () runAgda backends = runAgda' $ builtinBackends ++ backends -- | The main function without importing built-in backends runAgda' :: [Backend] -> IO () runAgda' backends = runTCMPrettyErrors $ do progName <- liftIO getProgName argv <- liftIO getArgs let (z, warns) = runOptM $ parseBackendOptions backends argv defaultOptions mapM_ (warning . OptionWarning) warns conf <- liftIO $ runExceptT $ do (bs, opts) <- ExceptT $ pure z -- The absolute path of the input file, if provided inputFile <- liftIO $ mapM absolute $ optInputFile opts mode <- getMainMode bs inputFile opts return (bs, opts, mode) case conf of Left err -> liftIO $ optionError err Right (bs, opts, mode) -> do when (optTransliterate opts) $ liftIO $ do -- When --interaction or --interaction-json is used, then we -- use UTF-8 when writing to stdout (and when reading from -- stdin). if optGHCiInteraction opts || optJSONInteraction opts then optionError $ "The option --transliterate must not be combined with " ++ "--interaction or --interaction-json" else do -- Transliterate unsupported code points. enc <- IO.mkTextEncoding (show IO.localeEncoding ++ "//TRANSLIT") IO.hSetEncoding IO.stdout enc IO.hSetEncoding IO.stderr enc case mode of MainModePrintHelp hp -> liftIO $ printUsage bs hp MainModePrintVersion -> liftIO $ printVersion bs MainModePrintAgdaDir -> liftIO $ printAgdaDir MainModeRun interactor -> do setTCLens stBackends bs runAgdaWithOptions interactor progName opts -- | Main execution mode data MainMode = MainModeRun (Interactor ()) | MainModePrintHelp Help | MainModePrintVersion | MainModePrintAgdaDir -- | Determine the main execution mode to run, based on the configured backends and command line options. -- | This is pure. getMainMode :: MonadError String m => [Backend] -> Maybe AbsolutePath -> CommandLineOptions -> m MainMode getMainMode configuredBackends maybeInputFile opts | Just hp <- optPrintHelp opts = return $ MainModePrintHelp hp | optPrintVersion opts = return $ MainModePrintVersion | optPrintAgdaDir opts = return $ MainModePrintAgdaDir | otherwise = do mi <- getInteractor configuredBackends maybeInputFile opts -- If there was no selection whatsoever (e.g. just invoked "agda"), we just show help and exit. return $ maybe (MainModePrintHelp GeneralHelp) MainModeRun mi type Interactor a -- Setup/initialization action. -- This is separated so that errors can be reported in the appropriate format. = TCM () -- Type-checking action -> (AbsolutePath -> TCM CheckResult) -- Main transformed action. -> TCM a data FrontendType = FrontEndEmacs | FrontEndJson | FrontEndRepl -- Emacs mode. Note that it ignores the "check" action because it calls typeCheck directly. emacsModeInteractor :: Interactor () emacsModeInteractor setup _check = mimicGHCi setup -- JSON mode. Note that it ignores the "check" action because it calls typeCheck directly. jsonModeInteractor :: Interactor () jsonModeInteractor setup _check = jsonREPL setup -- The deprecated repl mode. replInteractor :: Maybe AbsolutePath -> Interactor () replInteractor = runInteractionLoop -- The interactor to use when there are no frontends or backends specified. defaultInteractor :: AbsolutePath -> Interactor () defaultInteractor file setup check = do setup; void $ check file getInteractor :: MonadError String m => [Backend] -> Maybe AbsolutePath -> CommandLineOptions -> m (Maybe (Interactor ())) getInteractor configuredBackends maybeInputFile opts = case (maybeInputFile, enabledFrontends, enabledBackends) of (Just inputFile, [], _:_) -> return $ Just $ backendInteraction inputFile enabledBackends (Just inputFile, [], []) -> return $ Just $ defaultInteractor inputFile (Nothing, [], []) -> return Nothing -- No backends, frontends, or input files specified. (Nothing, [], _:_) -> throwError $ concat ["No input file specified for ", enabledBackendNames] (_, _:_, _:_) -> throwError $ concat ["Cannot mix ", enabledFrontendNames, " with ", enabledBackendNames] (_, _:_:_, []) -> throwError $ concat ["Must not specify multiple ", enabledFrontendNames] (_, [fe], []) | optOnlyScopeChecking opts -> errorFrontendScopeChecking fe (_, [FrontEndRepl], []) -> return $ Just $ replInteractor maybeInputFile (Nothing, [FrontEndEmacs], []) -> return $ Just $ emacsModeInteractor (Nothing, [FrontEndJson], []) -> return $ Just $ jsonModeInteractor (Just inputFile, [FrontEndEmacs], []) -> errorFrontendFileDisallowed inputFile FrontEndEmacs (Just inputFile, [FrontEndJson], []) -> errorFrontendFileDisallowed inputFile FrontEndJson where -- NOTE: The notion of a backend being "enabled" *just* refers to this top-level interaction mode selection. The -- interaction/interactive front-ends may still invoke available backends even if they are not "enabled". isBackendEnabled (Backend b) = isEnabled b (options b) enabledBackends = filter isBackendEnabled configuredBackends enabledFrontends = concat [ [ FrontEndRepl | optInteractive opts ] , [ FrontEndEmacs | optGHCiInteraction opts ] , [ FrontEndJson | optJSONInteraction opts ] ] -- Constructs messages like "(no backend)", "backend ghc", "backends (ghc, ocaml)" pluralize w [] = concat ["(no ", w, ")"] pluralize w [x] = concat [w, " ", x] pluralize w xs = concat [w, "s (", List.intercalate ", " xs, ")"] enabledBackendNames = pluralize "backend" [ backendName b | Backend b <- enabledBackends ] enabledFrontendNames = pluralize "frontend" (frontendFlagName <$> enabledFrontends) frontendFlagName = ("--" ++) . \case FrontEndEmacs -> "interaction" FrontEndJson -> "interaction-json" FrontEndRepl -> "interactive" errorFrontendScopeChecking fe = throwError $ concat ["The --only-scope-checking flag cannot be combined with ", frontendFlagName fe] errorFrontendFileDisallowed inputFile fe = throwError $ concat ["Must not specify an input file (", filePath inputFile, ") with ", frontendFlagName fe] -- | Run Agda with parsed command line options runAgdaWithOptions :: Interactor a -- ^ Backend interaction -> String -- ^ program name -> CommandLineOptions -- ^ parsed command line options -> TCM a runAgdaWithOptions interactor progName opts = do -- Main function. -- Bill everything to root of Benchmark trie. UtilsBench.setBenchmarking UtilsBench.BenchmarkOn -- Andreas, Nisse, 2016-10-11 AIM XXIV -- Turn benchmarking on provisionally, otherwise we lose track of time spent -- on e.g. LaTeX-code generation. -- Benchmarking might be turned off later by setCommandlineOptions Bench.billTo [] $ interactor initialSetup checkFile `finally_` do -- Print benchmarks. Bench.print -- Print accumulated statistics. printStatistics Nothing =<< useTC lensAccumStatistics where -- Options are fleshed out here so that (most) errors like -- "bad library path" are validated within the interactor, -- so that they are reported with the appropriate protocol/formatting. initialSetup :: TCM () initialSetup = do opts <- addTrustedExecutables opts setCommandLineOptions opts checkFile :: AbsolutePath -> TCM CheckResult checkFile inputFile = do -- Andreas, 2013-10-30 The following 'resetState' kills the -- verbosity options. That does not make sense (see fail/Issue641). -- 'resetState' here does not seem to serve any purpose, -- thus, I am removing it. -- resetState let mode = if optOnlyScopeChecking opts then Imp.ScopeCheck else Imp.TypeCheck result <- Imp.typeCheckMain mode =<< Imp.parseSource (SourceFile inputFile) unless (crMode result == ModuleScopeChecked) $ unlessNullM (applyFlagsToTCWarnings (crWarnings result)) $ \ ws -> typeError $ NonFatalErrors ws let i = crInterface result reportSDoc "main" 50 $ pretty i -- Print accumulated warnings unlessNullM (tcWarnings . classifyWarnings <$> getAllWarnings AllWarnings) $ \ ws -> do let banner = text $ "\n" ++ delimiter "All done; warnings encountered" reportSDoc "warning" 1 $ vcat $ punctuate "\n" $ banner : (prettyTCM <$> ws) return result -- | Print usage information. printUsage :: [Backend] -> Help -> IO () printUsage backends hp = do progName <- getProgName putStr $ usage standardOptions_ progName hp when (hp == GeneralHelp) $ mapM_ (putStr . backendUsage) backends backendUsage :: Backend -> String backendUsage (Backend b) = usageInfo ("\n" ++ backendName b ++ " backend options") $ map void (commandLineFlags b) -- | Print version information. printVersion :: [Backend] -> IO () printVersion backends = do putStrLn $ "Agda version " ++ versionWithCommitInfo mapM_ putStrLn [ " - " ++ name ++ " backend version " ++ ver | Backend Backend'{ backendName = name, backendVersion = Just ver } <- backends ] printAgdaDir :: IO () printAgdaDir = putStrLn =<< getDataDir -- | What to do for bad options. optionError :: String -> IO () optionError err = do prog <- getProgName putStrLn $ "Error: " ++ err ++ "\nRun '" ++ prog ++ " --help' for help on command line options." exitAgdaWith OptionError -- | Run a TCM action in IO; catch and pretty print errors. -- If some error message cannot be printed due to locale issues, then -- one may get the "Error when handling error" error message. There is -- currently no test case for this error, but on some systems one can -- (at the time of writing) trigger it by running @LC_CTYPE=C agda -- --no-libraries Bug.agda@, where @Bug.agda@ contains the following -- code (if there is some other file in the same directory, for -- instance @Bug.lagda@, then the error message may be different): -- -- @ -- _ : Set -- _ = Set -- @ runTCMPrettyErrors :: TCM () -> IO () runTCMPrettyErrors tcm = do r <- runTCMTop ( ( (Nothing <$ tcm) `catchError` \err -> do s2s <- prettyTCWarnings' =<< getAllWarningsOfTCErr err s1 <- prettyError err let ss = filter (not . null) $ s2s ++ [s1] unless (null s1) (liftIO $ putStr $ unlines ss) liftIO $ helpForLocaleError err return (Just TCMError) ) `catchImpossible` \e -> do liftIO $ putStr $ E.displayException e return (Just ImpossibleError) ) `E.catches` -- Catch all exceptions except for those of type ExitCode -- (which are thrown by exitWith) and asynchronous exceptions -- (which are for instance raised when Ctrl-C is used, or if -- the program runs out of heap or stack space). [ E.Handler $ \(e :: ExitCode) -> E.throw e , E.Handler $ \(e :: E.AsyncException) -> E.throw e , E.Handler $ \(e :: E.SomeException) -> do liftIO $ putStr $ E.displayException e return $ Right (Just UnknownError) ] case r of Right Nothing -> exitSuccess Right (Just reason) -> exitAgdaWith reason Left err -> do liftIO $ do putStrLn "\n\nError when handling error:" putStrLn $ tcErrString err helpForLocaleError err exitAgdaWith UnknownError -- | If the error is an IO error, and the error message suggests that -- the problem is related to locales or code pages, print out some -- extra information. helpForLocaleError :: TCErr -> IO () helpForLocaleError e = case e of (IOException _ _ e) | "invalid argument" `List.isInfixOf` show e -> msg _ -> return () where msg = putStr $ unlines [ "" , "This error may be due to the use of a locale or code page that does not" , "support some character used in the program being type-checked." , "" , "If it is, then one option is to use the option --transliterate, in which" , "case unsupported characters are (hopefully) replaced with something else," , "perhaps question marks. However, that could make the output harder to" , "read." , "" , "If you want to fix the problem \"properly\", then you could try one of the" , "following suggestions:" , "" , "* If you are using Windows, try switching to a different code page (for" , " instance by running the command 'CHCP 65001')." , "" , "* If you are using a Unix-like system, try using a different locale. The" , " installed locales are perhaps printed by the command 'locale -a'. If" , " you have a UTF-8 locale installed (for instance sv_SE.UTF-8), then you" , " can perhaps make Agda use this locale by running something like" , " 'LC_ALL=sv_SE.UTF-8 agda <...>'." ]