module Settings(readSettings, readPragma, classify, defaultHintName) where
import HSE.All
import Type
import Data.Char
import Data.List
import System.FilePath
import Util
readSettings :: FilePath -> [FilePath] -> IO [Setting]
readSettings dataDir xs = do
(builtin,mods) <- fmap unzipEither $ concatMapM (readHints dataDir) xs
return $ map Builtin builtin ++ concatMap (concatMap readSetting . concatMap getEquations . moduleDecls) mods
readHints :: FilePath -> FilePath -> IO [Either String Module_]
readHints dataDir file = do
y <- fromParseResult `fmap` parseFile parseFlags{implies=True} file
ys <- concatMapM (f . fromNamed . importModule) $ moduleImports y
return $ Right y:ys
where
f x | "HLint.Builtin." `isPrefixOf` x = return [Left $ drop 14 x]
| "HLint." `isPrefixOf` x = readHints dataDir $ dataDir </> drop 6 x <.> "hs"
| otherwise = readHints dataDir $ x <.> "hs"
classify :: [Setting] -> Idea -> Idea
classify xs i = if isParseError i then i else i{rank = foldl' (rerank i) (rank i) xs}
where
rerank :: Idea -> Rank -> Setting -> Rank
rerank i r c | matchHint (hintS c) (hint i) && matchFunc (funcS c) (func i) = rankS c
| otherwise = r
matchHint = (~=)
matchFunc (x1,x2) (y1,y2) = (x1~=y1) && (x2~=y2)
x ~= y = null x || x == y
defaultHintName = "Use alternative"
readSetting :: Decl_ -> [Setting]
readSetting (FunBind _ [Match _ (Ident _ (getRank -> Just rank)) pats (UnGuardedRhs _ bod) bind])
| InfixApp _ lhs op rhs <- bod, opExp op ~= "==>" =
[MatchExp rank (if null names then defaultHintName else head names) (fromParen lhs) (fromParen rhs) (readSide $ childrenBi bind)]
| otherwise = [Classify rank n func | n <- names2, func <- readFuncs bod]
where
names = getNames pats bod
names2 = ["" | null names] ++ names
readSetting x@WarnPragmaDecl{} | Just y <- readPragma x = y
readSetting (PatBind an (PVar _ name) _ bod bind) = readSetting $ FunBind an [Match an name [PLit an (String an "" "")] bod bind]
readSetting (FunBind an xs) | length xs /= 1 = concatMap (readSetting . FunBind an . return) xs
readSetting (SpliceDecl an (App _ (Var _ x) (Lit _ y))) = readSetting $ FunBind an [Match an (toNamed $ fromNamed x) [PLit an y] (UnGuardedRhs an $ Lit an $ String an "" "") Nothing]
readSetting x = errorOn x "bad hint"
readPragma :: Decl_ -> Maybe [Setting]
readPragma x@(WarnPragmaDecl _ [(names,warn)])
| not $ "HLint:" `isPrefixOf` warn = Nothing
| Just rank <- getRank a = Just $ map (Classify rank (dropWhile isSpace b)) ns2
| otherwise = errorOn x "bad classify pragma"
where ns = if null names then [""] else map fromNamed names
ns2 = [if n == "module_" then ("","") else ("",n) | n <- ns]
(a,b) = break isSpace $ dropWhile isSpace $ drop 6 warn
readPragma (WarnPragmaDecl an xs) = concatMapM (readPragma . WarnPragmaDecl an . return) xs
readPragma _ = Nothing
readSide :: [Decl_] -> Maybe Exp_
readSide [] = Nothing
readSide [PatBind _ PWildCard{} Nothing (UnGuardedRhs _ bod) Nothing] = Just bod
readSide (x:_) = errorOn x "bad side condition"
readFuncs :: Exp_ -> [FuncName]
readFuncs (App _ x y) = readFuncs x ++ readFuncs y
readFuncs (Lit _ (String _ "" _)) = [("","")]
readFuncs (Var _ (UnQual _ name)) = [("",fromNamed name)]
readFuncs (Var _ (Qual _ (ModuleName _ mod) name)) = [(mod, fromNamed name)]
readFuncs (Con _ (UnQual _ name)) = [(fromNamed name,""),("",fromNamed name)]
readFuncs (Con _ (Qual _ (ModuleName _ mod) name)) = [(mod ++ "." ++ fromNamed name,""),(mod,fromNamed name)]
readFuncs x = errorOn x "bad classification rule"
errorOn val msg = exitMessage $
showSrcLoc (getPointLoc $ ann val) ++
" Error while reading hint file, " ++ msg ++ "\n" ++
prettyPrint val
getNames :: [Pat_] -> Exp_ -> [String]
getNames ps _ | ps /= [] && all isPString ps = map fromPString ps
getNames [] (InfixApp _ lhs op rhs) | opExp op ~= "==>" = map ("Use "++) names
where
lnames = map f $ childrenS lhs
rnames = map f $ childrenS rhs
names = filter (not . isUnifyVar) $ (rnames \\ lnames) ++ rnames
f (Ident _ x) = x
f (Symbol _ x) = x
getNames _ _ = []
getRank :: String -> Maybe Rank
getRank "ignore" = Just Ignore
getRank "warn" = Just Warning
getRank "warning" = Just Warning
getRank "error" = Just Error
getRank _ = Nothing