-- Copyright (c) Facebook, Inc. and its affiliates. -- -- This source code is licensed under the MIT license found in the -- LICENSE file in the root directory of this source tree. -- {-# LANGUAGE ApplicativeDo #-} {-# LANGUAGE OverloadedStrings #-} {-# LANGUAGE RecordWildCards #-} {-# LANGUAGE TupleSections #-} module Retrie.Options ( -- * Options Options , Options_(..) , ExecutionMode(..) , defaultOptions , parseOptions -- * Internal , buildGrepChain , forFn , getOptionsParser , getTargetFiles , parseRewritesInternal , parseVerbosity , ProtoOptions , resolveOptions , GrepCommands(..) ) where import Control.Concurrent.Async (mapConcurrently) import Control.Monad (when, foldM) import Data.Bool import Data.Char (isAlphaNum, isSpace) import Data.Default as D import Data.Foldable (toList) import Data.Functor.Identity import Data.List import Data.HashSet (HashSet) import qualified Data.HashSet as HashSet import Data.Traversable import Options.Applicative import System.Directory import System.FilePath import System.Process import System.Random.Shuffle import Retrie.CPP import Retrie.Debug import Retrie.Elaborate import Retrie.ExactPrint import Retrie.Fixity import Retrie.GroundTerms import Retrie.GHC import Retrie.Pretty import Retrie.Rewrites import Retrie.Types import Retrie.Universe import Retrie.Util -- | Command-line options. type Options = Options_ [Rewrite Universe] AnnotatedImports -- | Parse options using the given 'FixityEnv'. parseOptions :: LibDir -> FixityEnv -> IO Options parseOptions libdir fixityEnv = do p <- getOptionsParser fixityEnv opts <- execParser (info (p <**> helper) fullDesc) resolveOptions libdir opts -- | Create 'Rewrite's from string specifications of rewrites. -- We expose this from "Retrie" with a nicer type signature as -- 'Retrie.Options.parseRewrites'. We have it here so we can use it with -- 'ProtoOptions'. parseRewritesInternal :: LibDir -> Options_ a b -> [RewriteSpec] -> IO [Rewrite Universe] parseRewritesInternal libdir Options{..} = parseRewriteSpecs libdir parser fixityEnv where parser fp = parseCPPFile (parseContent libdir fixityEnv) (targetDir fp) -- | Controls the ultimate action taken by 'apply'. The default action is -- 'ExecRewrite'. data ExecutionMode = ExecDryRun -- ^ Pretend to do rewrites, show diff. | ExecRewrite -- ^ Perform rewrites. | ExecExtract -- ^ Print the resulting expression for each match. | ExecSearch -- ^ Print the matched expressions. deriving (Show) data Options_ rewrites imports = Options { additionalImports :: imports -- ^ Imports specified by the command-line flag '--import'. , colorise :: ColoriseFun -- ^ Function used to colorize results of certain execution modes. , elaborations :: rewrites -- ^ Rewrites which are applied to the left-hand side of the actual rewrites. , executionMode :: ExecutionMode -- ^ Controls behavior of 'apply'. See 'ExecutionMode'. , extraIgnores :: [FilePath] -- ^ Specific files that should be ignored. Paths should be relative to -- 'targetDir'. , fixityEnv :: FixityEnv -- ^ Fixity information for operators used during parsing (of rewrites and -- target modules). Defaults to base fixities. , iterateN :: Int -- ^ Iterate the given rewrites or 'Retrie' computation up to this many -- times. Iteration may stop before the limit if no changes are made during -- a given iteration. , noDefaultElaborations :: Bool -- ^ Do not apply any of the built in elaborations in 'defaultElaborations'. , randomOrder :: Bool -- ^ Whether to randomize the order of target modules before rewriting them. , rewrites :: rewrites -- ^ Rewrites specified by command-line flags such as '--adhoc'. , roundtrips :: [RoundTrip] -- ^ Paths that should be roundtripped through ghc-exactprint to debug. -- Specified by the '--roundtrip' command-line flag. , singleThreaded :: Bool -- ^ Whether to concurrently rewrite target modules. -- Mostly useful for viewing debugging output without interleaving it. , targetDir :: FilePath -- ^ Directory that contains the code being targeted for rewriting. , targetFiles :: [FilePath] -- ^ Instead of targeting all Haskell files in 'targetDir', only target -- specific files. Paths should be relative to 'targetDir'. , verbosity :: Verbosity -- ^ How much should be output on 'stdout'. } -- | Construct default options for the given target directory. defaultOptions :: (Default rewrites, Default imports) => FilePath -> Options_ rewrites imports defaultOptions fp = Options { additionalImports = D.def , colorise = noColor , elaborations = D.def , executionMode = ExecRewrite , extraIgnores = [] , fixityEnv = mempty , iterateN = 1 , noDefaultElaborations = False , randomOrder = False , rewrites = D.def , roundtrips = [] , singleThreaded = False , targetDir = fp , targetFiles = [] , verbosity = Normal } -- | Get the options parser. The returned 'ProtoOptions' should be passed -- to 'resolveOptions' to get final 'Options'. getOptionsParser :: FixityEnv -> IO (Parser ProtoOptions) getOptionsParser fEnv = do dOpts <- defaultOptions <$> getCurrentDirectory return $ buildParser dOpts { fixityEnv = fEnv } buildParser :: ProtoOptions -> Parser ProtoOptions buildParser dOpts = do singleThreaded <- switch $ mconcat [ long "single-threaded" , showDefault , help "Don't try to parallelize things (for debugging)." ] targetDir <- option str $ mconcat [ long "target" , short 't' , metavar "PATH" , action "directory" -- complete with directory , value (targetDir dOpts) , showDefault , help "Path to target with rewrites." ] targetFiles <- many $ option str $ mconcat [ long "target-file" , metavar "PATH" , action "file" -- complete with filenames , help "Target specific file for rewriting." ] verbosity <- parseVerbosity (verbosity dOpts) additionalImports <- many $ option str $ mconcat [ long "import" , metavar "IMPORT" , help "Add given import statement to modules that are modified by a rewrite." ] extraIgnores <- many $ option str $ mconcat [ long "ignore" , metavar "PATH" , action "file" -- complete with filenames , help "Ignore specific file while rewriting." ] colorise <- fmap (bool noColor addColor) $ switch $ mconcat [ long "color" , help "Highlight matches with color." ] noDefaultElaborations <- switch $ mconcat [ long "no-default-elaborations" , showDefault , help "Don't apply any of the default elaborations to rewrites." ] randomOrder <- switch $ mconcat [ long "random-order" , help "Randomize the order of targeted modules." ] iterateN <- option auto $ mconcat [ long "iterate" , short 'i' , metavar "N" , value 1 , help "Iterate rewrites up to N times." ] executionMode <- parseMode rewrites <- parseRewriteSpecOptions elaborations <- parseElaborations roundtrips <- parseRoundtrips return Options{ fixityEnv = fixityEnv dOpts, ..} parseElaborations :: Parser [RewriteSpec] parseElaborations = concat <$> traverse many [ fmap Adhoc $ option str $ mconcat [ long "elaborate" , metavar "EQUATION" , help "Elaborate the left-hand side of rewrites using the given equation." ] , fmap AdhocType $ option str $ mconcat [ long "elaborate-type" , metavar "EQUATION" , help "Elaborate the left-hand side of rewrites using the given equation." ] , fmap AdhocPattern $ option str $ mconcat [ long "elaborate-pattern" , metavar "EQUATION" , help "Elaborate the left-hand side of rewrites using the given equation." ] ] parseRewriteSpecOptions :: Parser [RewriteSpec] parseRewriteSpecOptions = concat <$> traverse many [ fmap Unfold $ option str $ mconcat [ long "unfold" , short 'u' , metavar "NAME" , help "Unfold given fully-qualified name." ] , fmap Fold $ option str $ mconcat [ long "fold" , short 'f' , metavar "NAME" , help "Fold given fully-qualified name." ] , fmap RuleForward $ option str $ mconcat [ long "rule-forward" , short 'l' , metavar "NAME" , help "Apply fully-qualified RULE name left-to-right." ] , fmap RuleBackward $ option str $ mconcat [ long "rule-backward" , short 'r' , metavar "NAME" , help "Apply fully-qualified RULE name right-to-left." ] , fmap TypeForward $ option str $ mconcat [ long "type-forward" , metavar "NAME" , help "Apply fully-qualified type synonym name left-to-right." ] , fmap TypeBackward $ option str $ mconcat [ long "type-backward" , metavar "NAME" , help "Apply fully-qualified type synonym name right-to-left." ] , fmap Adhoc $ option str $ mconcat [ long "adhoc" , metavar "EQUATION" , help "Apply an adhoc equation of the form: forall vs. lhs = rhs" ] , fmap AdhocType $ option str $ mconcat [ long "adhoc-type" , metavar "EQUATION" , help "Apply an adhoc type equation of the form: lhs = rhs" ] , fmap PatternForward $ option str $ mconcat [ long "pattern-forward" , metavar "NAME" , help "Apply fully-qualified pattern synonym name left-to-right." ] , fmap PatternBackward $ option str $ mconcat [ long "pattern-backward" , metavar "NAME" , help "Apply fully-qualified pattern synonym name right-to-left." ] , fmap AdhocPattern $ option str $ mconcat [ long "adhoc-pattern" , metavar "EQUATION" , help "Apply an adhoc pattern equation of the form: lhs = rhs" ] ] parseMode :: Parser ExecutionMode parseMode = parseDryRun <|> parseExtract <|> parseSearch <|> pure ExecRewrite parseDryRun :: Parser ExecutionMode parseDryRun = flag' ExecDryRun $ mconcat [ long "dry-run" , help "Don't overwrite files. Print rewrite results." ] parseExtract :: Parser ExecutionMode parseExtract = flag' ExecExtract $ mconcat [ long "extract" , help "Find the left-hand side, display the instantiated right-hand side." ] parseSearch :: Parser ExecutionMode parseSearch = flag' ExecSearch $ mconcat [ long "search" , help "Search for left-hand side of the rewrite and show matches." ] -- | Parser for 'Verbosity'. parseVerbosity :: Verbosity -> Parser Verbosity parseVerbosity defaultV = option (eitherReader verbosityReader) $ mconcat [ long "verbosity" , short 'v' , value defaultV , showDefault , help verbosityHelp ] verbosityReader :: String -> Either String Verbosity verbosityReader "0" = Right Silent verbosityReader "1" = Right Normal verbosityReader "2" = Right Loud verbosityReader _ = Left $ "invalid verbosity. Valid values: " ++ verbosityHelp verbosityHelp :: String verbosityHelp = "0: silent, 1: normal, 2: loud (implies --single-threaded)" ------------------------------------------------------------------------------- -- | Options that have been parsed, but not fully resolved. type ProtoOptions = Options_ [RewriteSpec] [String] -- | Resolve 'ProtoOptions' into 'Options'. Parses rewrites into 'Rewrite's, -- parses imports, validates options, and extends 'fixityEnv' with any -- declared fixities in the target directory. resolveOptions :: LibDir -> ProtoOptions -> IO Options resolveOptions libdir protoOpts = do absoluteTargetDir <- makeAbsolute (targetDir protoOpts) opts@Options{..} <- addLocalFixities libdir protoOpts { targetDir = absoluteTargetDir } parsedImports <- parseImports libdir additionalImports debugPrint verbosity "Imports:" $ runIdentity $ fmap astA $ transformA parsedImports $ \ imps -> do -- anns <- getAnnsT return $ map exactPrint imps rrs <- parseRewritesInternal libdir opts rewrites es <- parseRewritesInternal libdir opts $ (if noDefaultElaborations then [] else defaultElaborations) ++ elaborations elaborated <- elaborateRewritesInternal fixityEnv es rrs return Options { additionalImports = parsedImports , elaborations = es , rewrites = elaborated , singleThreaded = singleThreaded || verbosity == Loud , .. } -- | Find all fixity declarations in targetDir and add them to fixity env. addLocalFixities :: LibDir -> Options_ a b -> IO (Options_ a b) addLocalFixities libdir opts = do -- do not limit search for infix decls to only targetFiles let opts' = opts { targetFiles = [] } -- "infix" will find infixl and infixr as well files <- getTargetFiles opts' [HashSet.singleton "infix"] fixFns <- forFn opts files $ \ fp -> do ms <- toList <$> parseCPPFile (parseContentNoFixity libdir) fp return $ extendFixityEnv [ (rdrFS nm, fixity) | m <- ms , (L _ nm, fixity) <- fixityDecls (unLoc (astA m)) ] return opts { fixityEnv = foldr ($) (fixityEnv opts) fixFns } -- | 'forM', but concurrency and input order controled by 'Options'. forFn :: Options_ x y -> [a] -> (a -> IO b) -> IO [b] forFn Options{..} c f | randomOrder = fn f =<< shuffleM c | otherwise = fn f c where fn | singleThreaded = mapM | otherwise = mapConcurrently -- | Find all files to target for rewriting. getTargetFiles :: Options_ a b -> [GroundTerms] -> IO [FilePath] -- Always include at least one set of ground terms -- This selects all files if the list of rewrites is empty getTargetFiles opts [] = getTargetFiles opts [mempty] getTargetFiles Options{..} gtss = do ignorePred <- maybe onIgnoreErr return =<< vcsIgnorePred verbosity targetDir let ignore fp = ignorePred fp || extraIgnorePred fp fpSets <- forM (dedup gtss) $ \ gts -> do -- See Note [Ground Terms] fps <- runGrepChain targetDir verbosity (buildGrepChain targetDir gts targetFiles) let r = filter (not . ignore) $ map (normalise . (targetDir )) fps debugPrint verbosity "Files:" r return $ HashSet.fromList r return $ HashSet.toList $ mconcat fpSets where dedup = HashSet.toList . HashSet.fromList extraIgnorePred = let fps = [ normalise (targetDir f) | f <- extraIgnores ] in \fp -> any (`isPrefixOf` fp) fps onIgnoreErr = do when (verbosity > Silent) $ putStrLn "Reading VCS ignore failed! Continuing without ignoring." return $ const False -- | Return a chain of grep commands to find files with relevant groundTerms -- If filesGiven is empty, use all *.hs files under targetDir buildGrepChain :: FilePath -> HashSet String -> [FilePath] -> GrepCommands buildGrepChain targetDir gts filesGiven = GrepCommands {initialFileSet=filesGiven, commandChain=commands} where commands = if null filesGiven then commandsWithoutFiles else commandsWithFiles commandsWithFiles = case terms of [] -> [] -- no processing gs -> map normalGrep gs commandsWithoutFiles = case terms of [] -> [findCmd] -- all .hs files g:gs -> recursiveGrep g : map normalGrep gs -- start with recursive grep findCmd = unwords ["find", quotePath (addTrailingPathSeparator targetDir), "-iname", hsExtension] recursiveGrep g = unwords ["grep", "-R", "--include=" ++ hsExtension, "-l", esc g, quotePath targetDir] normalGrep gt = unwords ["grep", "-l", esc gt] -- Limit the number of the shell command we build by only selecting -- up to 10 ground terms. The goal is to filter file list down to -- a manageable size. It doesn't have to be exact. terms = take 10 $ filter p $ HashSet.toList gts p [] = False p (c:cs) | isSpace c = p cs | otherwise = isAlphaNum c hsExtension = "\"*.hs\"" esc s = "'" ++ intercalate "[[:space:]]\\+" (words $ escChars s) ++ "'" escChars = concatMap escChar escChar c | c `elem` magicChars = "\\" <> [c] | otherwise = [c] magicChars :: [Char] magicChars = "*?[#˜=%\\" type CommandLine = String data GrepCommands = GrepCommands { initialFileSet :: [FilePath], commandChain :: [CommandLine] } deriving (Eq, Show) runGrepChain :: FilePath -> Verbosity -> GrepCommands -> IO [FilePath] runGrepChain targetDir verbosity GrepCommands{..} = foldM (commandStep targetDir verbosity) initialFileSet commandChain -- | run a command with a list of files as quoted arguments commandStep :: FilePath -> Verbosity -> [FilePath]-> CommandLine -> IO [FilePath] commandStep targetDir verbosity files cmd = doCmd targetDir verbosity (cmd <> formatPaths files) where formatPaths [] = "" formatPaths xs = " " <> unwords (map quotePath xs) quotePath :: FilePath -> FilePath quotePath x = "'" <> x <> "'" doCmd :: FilePath -> Verbosity -> String -> IO [FilePath] doCmd targetDir verbosity shellCmd = do debugPrint verbosity "shellCmd:" [shellCmd] let cmd = (shell shellCmd) { cwd = Just targetDir } (_ec, fps, _) <- readCreateProcessWithExitCode cmd "" return $ lines fps