(This is the concrete case behind How do I conditionally declare an instance?)
In my small project players can create/join/watch a game session or list the ongoing sessions. A very simple Yesod app exposes some endpoints to match the use-cases.
Step 1) I want to abstract away the concept of "being able to hold multiple ongoing games":
class Host m k s where
create :: s -> m k
read :: k -> m s
write :: k -> s -> m ()
update :: (s -> s) -> k -> m ()
delete :: k -> m ()
An instance of Host m k s
means m
can host games (states) of s
, identified (keyed) by k
.
Step 2) A very practical host is a shared Map
:
newtype InMemory k s a = InMemory {run :: TVar (M.Map k s) -> STM a}
A TVar
holding a Map k s
is captured in InMemory
. InMemory
keyed with Int
can host :
instance Host (InMemory Int a) Int a where
create s = InMemory $ \tvar -> do
id <- succ . M.size <$> readTVar tvar
modifyTVar tvar (M.insert id s)
return id
read id = InMemory $ \tvar -> do
s <- M.lookup id <$> readTVar tvar
maybe err return s
where
err = throwSTM $ NotFound id
write id s = modify $ M.insert id s -- modify is just a small helper, returns InMemory
update f id = modify $ M.update (Just . f) id
delete id = modify $ M.delete id
Step 3) Any m
that can produce the correct tvar is a host as well. For example a Yesod app can make the HandlerFor
a host:
newtype App = App {rooms :: TVar (M.Map Int Game)} -- (1)
instance MonadReader (HandlerFor App) (M.Map Int Game) where ... -- (2) HandlerFor App can produce the correct tvar
-- (1) && (2) ==> the HandlerFor App is a Host
So let's have this notion:
-- smt is a just a small helper, needs MonadIO
-- this instance declaration is *wrong* and not what I want to say
instance (C.MonadIO m, R.MonadReader (TVar (M.Map k s)) m) => Host m GameId s where
create s = GameId <$> stm (create s)
write (GameId id) s = stm (write id s)
update f (GameId id) = stm (update f id)
delete (GameId id) = stm (delete id)
read (GameId id) = stm (read id)
Idea is any m
who can produce the right piece of data can host.
Now here comes the trouble. This:
instance R.MonadReader (TVar (M.Map k s)) m => Host m GameId s where
is not saying "IF m
satisfies the context THEN it can host", it is saying "ALL m
that are host MUST satisfy the context" which is not what I want to say. So I'm stuck in expressing the following in Haskell:
IF
m
satisfies the context THEN it can host, if it does not then this line has abs. no effect whatsoever. Don't force the context onm
and it is not an error ifm
can host but does not satisfy the context.
There is a question of its own about this very topic (how to opt-in for a instance but leave undisturbed otherwise) and apparently this is not possible to express in Haskell. So question is:
- What are my options here ?
- In my thinking process (1,2 and 3) where did I go wrong that got me to a dead end ?
- What is the right way of thinking for this ?
The first thing you should consider in any problem like this is whether having a class makes sense in the first place. The fact that both Haskell and OO languages use the same
class
keyword can easily trick one into thinking they have the same uses, but (as was written many times before) Haskell typeclasses are really fundamentally different beasts from OO classes or even interfaces.And in many cases, the most sensible translation of an OO class is simply as a struct, which after all can contain functions corresponding to OO methods, just as well as traditional "scalar" values. Any class-y code can be translated to this style, with the main change being that you explicitly pass around “instances” and any inferences between them:
Because instances are now first-class citizens, there is no problem making something like the Scala one you thought of:
This does exactly what you asked for: it can produce a
Host
form
, provided thatm
fulfills the mentioned constraints – without any extrapolation that this is what should be done for everym
.Of course, this struct-approach does have its downsides though: not only can you pass around instances explicitly now, you actually must do it always. And in some cases this can incur a lot of boilerplate. One workaround is to use
-XImplicitParams
, which basically lets you treat a variable as if it were a typeclass constraint. This is not often done, and indeed has been criticised – the problem basically goes back to again using class-tooling for something that doesn't behave like a class, much like your original attempt.Also, changing from classes to data basically makes the whole code weaker typed. You could mix e.g. the
create
andread
method from different “instances”, which could well create complete mayhem.A different option is to keep the decisions on the type level like, using
class
mechanisms, but still have a means of uniquely distinguishing between different instances. This can be done with phantom types. In your example, it seems sensible to tie the unique distuishing role to the key values, which after all already serve a related purpose anyway.Now before you write an instance, you declare a unique tag identifying this kind of instance:
and similarly
This approach still requires to pass around some explicit information, but basically you only need to once be explicit which sort of instance you want, then this is automatically synchronised to every connected piece of code via the keys' types. A nice way of doing that is to use
-XTypeApplications
oncreate
: