module Language.Haskell.Inspector where
import Language.Haskell.Parser
import Language.Haskell.Syntax
import Data.Maybe (fromMaybe, isJust)
import Control.Monad (join)
import Data.List (find)
import Language.Haskell.Explorer
type Binding = String
type Code = String
type Inspection = Binding -> Code -> Bool
hasComposition :: Inspection
hasComposition = isBindingEO f
where f (O (HsQVarOp (UnQual (HsSymbol ".")))) = True
f _ = False
hasGuards :: Inspection
hasGuards = isBindingRhs f
where f (HsGuardedRhss _) = True
f _ = False
hasIf :: Inspection
hasIf = isBindingEO f
where f (E (HsIf _ _ _)) = True
f _ = False
hasConditional :: Inspection
hasConditional target code = hasIf target code || hasGuards target code
hasLambda :: Inspection
hasLambda = isBindingEO f
where f (E (HsLambda _ _ _)) = True
f _ = False
hasDirectRecursion :: Inspection
hasDirectRecursion binding = hasUsage binding binding
hasUsage :: String -> Inspection
hasUsage target = isBindingEO f
where f (O (HsQVarOp name)) = isTarget name
f (E (HsVar name)) = isTarget name
f _ = False
isTarget (Qual _ n) = isTarget' n
isTarget (UnQual n) = isTarget' n
isTarget _ = False
isTarget' (HsSymbol t) = t == target
isTarget' (HsIdent t) = t == target
hasComprehension :: Inspection
hasComprehension = isBindingEO f
where f (E (HsListComp _ _)) = True
f _ = False
hasBinding :: Inspection
hasBinding binding = isJust . findBindingRhs binding
isParseable :: Code -> Bool
isParseable = testWithCode (const True)
isBindingEO f = isBindingRhs isExpr
where isExpr rhs = exploreExprs f $ topExprs rhs
isBindingRhs f = testWithBindingRhs (any f)
testWithBindingRhs :: ([HsRhs] -> Bool) -> Binding -> Code -> Bool
testWithBindingRhs f binding = orFalse . withBindingRhs f binding
withBindingRhs :: ([HsRhs] -> a) -> Binding -> Code -> Maybe a
withBindingRhs f binding = fmap f . findBindingRhs binding
findBindingRhs binding = fmap rhsForBinding . join . withCode (find isBinding)
where isBinding (HsPatBind _ (HsPVar (HsIdent name)) _ _) = name == binding
isBinding (HsFunBind cases) = any isBindingInMatch cases
isBinding _ = False
isBindingInMatch (HsMatch _ (HsIdent name) _ _ _ ) = name == binding
isBindingInMatch _ = False
rhsForBinding :: HsDecl -> [HsRhs]
rhsForBinding (HsPatBind _ _ rhs localDecls) = concatRhs rhs localDecls
rhsForBinding (HsFunBind cases) = cases >>= \(HsMatch _ _ _ rhs localDecls) -> concatRhs rhs localDecls
rhsForBinding _ = []
concatRhs rhs l = [rhs] ++ concatMap rhsForBinding l
testWithCode f = orFalse . withCode f
withCode :: ([HsDecl] -> a) -> Code -> Maybe a
withCode f code | ParseOk (HsModule _ _ _ _ decls) <- parseModule code = Just (f decls)
| otherwise = Nothing
orFalse = fromMaybe False