forked from slindley/effect-handlers
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHandlers.hs
35 lines (30 loc) · 1.04 KB
/
Handlers.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
{- Open handlers -}
{-# LANGUAGE TypeFamilies,
MultiParamTypeClasses,
GADTs,
TypeOperators,
RankNTypes,
FunctionalDependencies,
PolyKinds
#-}
module Handlers where
type family Return (opApp :: *) :: *
type family Result (h :: *) :: *
class ((h :: *) `Handles` (op :: j -> k -> *)) (e :: j) | h op -> e where
clause :: op e u -> (Return (op e u) -> h -> Result h) -> h -> Result h
newtype Comp h a = Comp {handle :: (a -> h -> Result h) -> h -> Result h}
doOp :: (h `Handles` op) e => op e u -> Comp h (Return (op e u))
doOp op = Comp (\k h -> clause op k h)
-- doOp op = Comp (\k -> clause op k)
-- doOp op = Comp (clause op)
-- doOp = Comp . clause
-- We are careful not to use this equivalent implementation because it
-- leads to an enormous slow-down. Pointless programming in GHC is
-- dangerous!
--
-- doOp = Comp . clause
instance Monad (Comp h) where
return v = Comp (\k h -> k v h)
Comp c >>= f = Comp (\k h -> c (\x h' -> handle (f x) k h') h)
instance Functor (Comp h) where
fmap f c = c >>= \x -> return (f x)