{-# LANGUAGE PackageImports #-}
{-# LANGUAGE PatternGuards, ScopedTypeVariables, RecordWildCards #-}
module Hint.Import(importHint) where
import Control.Applicative
import Data.Tuple.Extra
import Hint.Type
import Refact.Types hiding (ModuleName)
import qualified Refact.Types as R
import Data.List.Extra
import Data.Maybe
import Prelude
import "ghc-lib-parser" FastString
import "ghc-lib-parser" BasicTypes
import "ghc-lib-parser" RdrName
import "ghc-lib-parser" Module
import "ghc-lib-parser" HsSyn as GHC
import qualified "ghc-lib-parser" SrcLoc as GHC
import GHC.Util
importHint :: ModuHint
importHint _ ModuleEx {ghcModule=GHC.L _ HsModule{hsmodImports=ms}} =
concatMap (reduceImports . snd) (
groupSort [((n, pkg), i) | i <- ms
, not $ ideclSource (unloc i)
, let i' = unloc i
, let n = unloc $ ideclName i'
, let pkg = unpackFS . sl_fs <$> ideclPkgQual i']) ++
concatMap stripRedundantAlias ms ++
concatMap preferHierarchicalImports ms
reduceImports :: [LImportDecl GhcPs] -> [Idea]
reduceImports ms =
[rawIdea Hint.Type.Warning "Use fewer imports"
(ghcSpanToHSE (getloc $ head ms)) (f ms) (Just $ f x) [] rs
| Just (x, rs) <- [simplify ms]]
where f = unlines . map unsafePrettyPrint
simplify :: [LImportDecl GhcPs]
-> Maybe ([LImportDecl GhcPs], [Refactoring R.SrcSpan])
simplify [] = Nothing
simplify (x : xs) = case simplifyHead x xs of
Nothing -> first (x:) <$> simplify xs
Just (xs, rs) -> Just $ maybe (xs, rs) (second (++ rs)) $ simplify xs
simplifyHead :: LImportDecl GhcPs
-> [LImportDecl GhcPs]
-> Maybe ([LImportDecl GhcPs], [Refactoring R.SrcSpan])
simplifyHead x (y : ys) = case combine x y of
Nothing -> first (y:) <$> simplifyHead x ys
Just (xy, rs) -> Just (xy : ys, rs)
simplifyHead x [] = Nothing
combine :: LImportDecl GhcPs
-> LImportDecl GhcPs
-> Maybe (LImportDecl GhcPs, [Refactoring R.SrcSpan])
combine x@(GHC.L _ x') y@(GHC.L _ y')
| qual, as, specs = Just (x, [Delete Import (toSS' y)])
| qual, as
, Just (False, xs) <- ideclHiding x'
, Just (False, ys) <- ideclHiding y' =
let newImp = noloc x'{ideclHiding = Just (False, noloc (unloc xs ++ unloc ys))}
in Just (newImp, [Replace Import (toSS' x) [] (unsafePrettyPrint (unloc newImp))
, Delete Import (toSS' y)])
| qual, as, isNothing (ideclHiding x') || isNothing (ideclHiding y') =
let (newImp, toDelete) = if isNothing (ideclHiding x') then (x, y) else (y, x)
in Just (newImp, [Delete Import (toSS' toDelete)])
| not (ideclQualified x'), qual, specs, length ass == 1 =
let (newImp, toDelete) = if isJust (ideclAs x') then (x, y) else (y, x)
in Just (newImp, [Delete Import (toSS' toDelete)])
| otherwise = Nothing
where
qual = ideclQualified x' == ideclQualified y'
as = ideclAs x' `GHC.Util.eqMaybe` ideclAs y'
ass = mapMaybe ideclAs [x', y']
specs = transformBi (const noSrcSpan) (ideclHiding x') ==
transformBi (const noSrcSpan) (ideclHiding y')
stripRedundantAlias :: LImportDecl GhcPs -> [Idea]
stripRedundantAlias x@(GHC.L loc i@GHC.ImportDecl {..})
| Just (unloc ideclName) == fmap unloc ideclAs =
[suggest' "Redundant as" x (GHC.L loc i{ideclAs=Nothing}) [RemoveAsKeyword (toSS' x)]]
stripRedundantAlias _ = []
preferHierarchicalImports :: LImportDecl GhcPs -> [Idea]
preferHierarchicalImports x@(GHC.L loc i@GHC.ImportDecl{ideclName=(GHC.L _ n),ideclPkgQual=Nothing})
| n == mkModuleName "IO" && isNothing (ideclHiding i) =
[rawIdeaN Suggestion "Use hierarchical imports" (ghcSpanToHSE loc)
(trimStart $ unsafePrettyPrint i) (
Just $ unlines $ map (trimStart . unsafePrettyPrint)
[ f "System.IO" Nothing, f "System.IO.Error" Nothing
, f "Control.Exception" $ Just (False, noloc [mkLIE x | x <- ["bracket","bracket_"]])]) []]
| Just y <- lookup (moduleNameString n) newNames =
let newModuleName = y ++ "." ++ moduleNameString n
r = [Replace R.ModuleName (toSS' x) [] newModuleName] in
[suggest' "Use hierarchical imports"
x (noloc (desugarQual i){ideclName=noloc (mkModuleName newModuleName)}) r]
where
f a b = (desugarQual i){ideclName=noloc (mkModuleName a), ideclHiding=b}
mkLIE :: String -> LIE GhcPs
mkLIE n = noloc $ IEVar noext (noloc (IEName (noloc (mkVarUnqual (fsLit n)))))
desugarQual :: GHC.ImportDecl GhcPs -> GHC.ImportDecl GhcPs
desugarQual i
| ideclQualified i && isNothing (ideclAs i) = i{ideclAs = Just (ideclName i)}
| otherwise = i
preferHierarchicalImports _ = []
newNames :: [(String, String)]
newNames = let (*) = flip (,) in
["Control" * "Monad"
,"Data" * "Char"
,"Data" * "List"
,"Data" * "Maybe"
,"Data" * "Ratio"
,"System" * "Directory"
]