module Utils
( module Utils
, HSE.Located(..)
, HSE.rebracket1
)where
import Control.Monad
import Control.Monad.Trans.State
import Data.Data
import Data.Default
import Data.Functor.Identity
import Data.Generics.Uniplate.Data
import Data.List
import Data.Map (Map)
import Data.Set (Set)
import qualified Data.Set as Set
import Debug.Hoed.Pure hiding (Module)
import Language.Haskell.Exts
import qualified Language.Haskell.Exts.Util as HSE
#ifdef DEBUG
import Language.Haskell.Exts.Observe ()
#endif
newtype S = S {getSrcSpanInfo :: SrcSpanInfo}
deriving (Data, Typeable)
instance Eq S where _ == _ = True
instance Ord S where compare _ _ = EQ
instance Show S where show _ = "<loc>"
instance Default S where
def = S noSrcSpan
instance Observable S where
observer = observeOpaque "<loc>"
constrain = constrainBase
freeVars
:: (Observable a, HSE.FreeVars a, S ~ HSE.LocType a)
=> a -> Set (Name S)
freeVars = observe "freeVars" HSE.freeVars
freeVarss
:: (Observable a, HSE.AllVars a, S ~ HSE.LocType a)
=> a -> Set (Name S)
freeVarss = observe "freeVarss" (HSE.free . HSE.allVars)
definedVars
:: (Observable a, HSE.AllVars a, S ~ HSE.LocType a)
=> a -> Set (Name S)
definedVars = observe "definedVars" (HSE.bound . HSE.allVars)
same :: Pat S -> Exp S -> Bool
same = observe "same" same'
same' :: Pat S -> Exp S -> Bool
same' (PApp _ n1 []) (Con _ n2) = n1 == n2
same' (PVar l n1) (Var _ n2) = UnQual l n1 == n2
same' (PTuple _ Boxed []) y = same (PApp (ann y) (unit_con_name (ann y)) []) y
same' (PTuple _ Boxed [pv]) y = same pv y
same' y (Tuple _ Boxed []) = same y (unit_con(ann y))
same' y (Tuple _ Boxed [pv]) = same y pv
same' (PTuple _ boxed ps) (Tuple _ boxed' es) =
length ps == length es && boxed == boxed' && and (zipWith same ps es)
same' (PAsPat _ n _) (Var _ (UnQual _ n')) = n == n'
same' (PAsPat _ _ p) e = same p e
same' (PParen _ p) e = same p e
same' p (Paren _ e) = same p e
same' _ _ = False
times :: Int -> (a -> a) -> a -> a
times n f x = iterate f x !! n
hidePat :: Set (Name S) -> Pat S -> Pat S
hidePat vs = transform (go vs) where
go vs p@(PVar l n)
| n `Set.member` vs = PWildCard l
| otherwise = p
go vs (PAsPat _ n p)
| n `Set.member` vs = go vs p
go _ x = x
pair :: Exp S -> Exp S -> Exp S
pair e1 e2 = Tuple (ann e1) Boxed [e1, e2]
pairP :: Pat S -> Pat S -> Pat S
pairP p1 p2 = PTuple (ann p1) Boxed [hidePat (definedVars p2) p1, p2]
left, right :: Exp S -> Exp S
left x = App (ann x) left_exp (Paren def x)
right x = App (ann x) right_exp (Paren def x)
returnCmd :: Exp S -> Exp S
returnCmd x = LeftArrApp (ann x) returnA_exp x
compose_op, choice_op :: QOp S
returnA_exp, arr_exp, first_exp :: Exp S
left_exp, right_exp, app_exp, loop_exp :: Exp S
unqualId :: String -> Exp S
unqualId id = Var def $ UnQual def (Ident def id)
unqualOp :: String -> QOp S
unqualOp id = QVarOp def $ UnQual def (Symbol def id)
unqualCon :: String -> Exp S
unqualCon id = Con def $ UnQual def (Symbol def id)
arr_exp = unqualId "arr"
compose_op = unqualOp ">>>"
first_exp = unqualId "first"
returnA_exp = unqualId "returnA"
choice_op = unqualOp "|||"
left_exp = unqualCon "Left"
right_exp = unqualCon "Right"
app_exp = unqualId "app"
loop_exp = unqualId "loop"
irrPat :: Pat S -> Pat S
irrPat p@PVar{} = p
irrPat (PParen l p) = PParen l (irrPat p)
irrPat (PAsPat l n p) = PAsPat l n (irrPat p)
irrPat p@PWildCard{} = p
irrPat p@PIrrPat{} = p
irrPat p = PIrrPat (ann p) p
observeSt
:: (Observable a, Observable b, Observable c, Observable s)
=> String -> (a -> b -> State s c) -> a -> b -> State s c
observeSt name f a b = StateT $ \s -> Identity $ observe name f' a b s
where
f' a b = runState (f a b)
instance (Eq a, Show a) => Observable (Set a) where
constrain = constrainBase
observer = observeBase
instance (Eq a, Eq k, Show a, Show k) => Observable (Map k a) where
constrain = constrainBase
observer = observeBase
instance Observable (Exp S) where
observer = observePretty
instance Observable (Name S) where
observer = observePretty
instance Observable (QName S) where
observer = observePretty
instance Observable [Stmt S] where
observer lit cxt =
seq lit $ send (bracket $ intercalate ";" $ fmap prettyPrint lit) (return lit) cxt
instance Observable (Stmt S) where
observer = observePretty
instance Observable (Pat S) where
observer = observePretty
instance Observable (QOp S) where
observer = observePretty
instance Observable (Op S) where
observer = observePretty
instance Observable (Rhs S) where
observer = observePretty
instance Observable (Alt S) where
observer = observePretty
instance Observable (Set (Name S)) where
constrain = constrainBase
observer x cxt =
seq x $ send (between "[" "]"$ intercalate "," $ prettyPrint <$> map void (Set.toList x)) (return x) cxt
observePretty lit cxt =
seq lit $ send (between "<" ">" $ prettyPrint lit) (return lit) cxt
bracket :: [Char] -> [Char]
between open close s = open ++ s ++ close
bracket = between "[" "]"