-- | Main file for FixImports that uses the default formatting. It reads -- a config file from the current directory. -- -- More documentation in "FixImports". {-# LANGUAGE DisambiguateRecordFields #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE ScopedTypeVariables #-} module Main where import qualified Control.Exception as Exception import Control.Monad (when) import qualified Data.Set as Set import qualified Data.Text as Text import qualified Data.Text.IO as Text.IO import qualified Data.Version as Version import qualified System.Console.GetOpt as GetOpt import qualified System.Environment as Environment import qualified System.Exit as Exit import qualified System.IO as IO import qualified Config import qualified FixImports import qualified Paths_fix_imports import qualified Types import qualified Util main :: IO () main = do -- I need the module path to search for modules relative to it first. I -- could figure it out from the parsed module name, but a main module may -- not have a name. (modulePath, flags) <- parseArgs =<< Environment.getArgs (config, errors) <- readConfig ".fix-imports" if null errors then mainConfig config flags modulePath else do Text.IO.hPutStrLn IO.stderr $ Text.unlines errors Exit.exitFailure readConfig :: FilePath -> IO (Config.Config, [Text.Text]) readConfig = fmap (maybe (Config.empty, []) Config.parse) . Util.catchENOENT . Text.IO.readFile mainConfig :: Config.Config -> [Flag] -> FilePath -> IO () mainConfig config flags modulePath = do let (verbose, debug, includes) = extractFlags flags source <- IO.getContents config <- return $ config { Config._includes = includes ++ Config._includes config , Config._debug = debug } fixed <- FixImports.fixModule config modulePath source `Exception.catch` (\(exc :: Exception.SomeException) -> return $ Left $ "exception: " ++ show exc) case fixed of Left err -> do IO.putStr source IO.hPutStrLn IO.stderr $ "error: " ++ err Exit.exitFailure Right (FixImports.Result source added removed metrics) -> do IO.putStr source let names = Util.join ", " . map Types.moduleName . Set.toList (addedMsg, removedMsg) = (names added, names removed) mDone <- FixImports.metric metrics "done" Config.debug config $ Text.stripEnd $ FixImports.showMetrics (mDone : metrics) when (verbose && not (null addedMsg) || not (null removedMsg)) $ IO.hPutStrLn IO.stderr $ Util.join "; " $ filter (not . null) [ if null addedMsg then "" else "added: " ++ addedMsg , if null removedMsg then "" else "removed: " ++ removedMsg ] Exit.exitSuccess data Flag = Debug | Include String | Verbose deriving (Eq, Show) options :: [GetOpt.OptDescr Flag] options = [ GetOpt.Option [] ["debug"] (GetOpt.NoArg Debug) "print debugging info on stderr" , GetOpt.Option ['i'] [] (GetOpt.ReqArg Include "path") "add to module include path" , GetOpt.Option ['v'] [] (GetOpt.NoArg Verbose) "print added and removed modules on stderr" ] parseArgs :: [String] -> IO (String, [Flag]) parseArgs args = case GetOpt.getOpt GetOpt.Permute options args of (flags, [modulePath], []) -> return (modulePath, flags) (_, [], errs) -> usage $ concat errs _ -> usage "too many args" extractFlags :: [Flag] -> (Bool, Bool, [FilePath]) extractFlags flags = ( Verbose `elem` flags , Debug `elem` flags , "." : [p | Include p <- flags] ) -- Includes always have the current directory first. usage :: String -> IO a usage msg = do name <- Environment.getProgName IO.hPutStr IO.stderr $ GetOpt.usageInfo (msg ++ "\n" ++ name ++ " Module.hs