[Haskell-cafe] How can I use MACID in my existing application?
Jeremy Shaw
jeremy at n-heptane.com
Sun Nov 7 21:29:53 EST 2010
Hello,
Retrofitting MACID will not be trivial because it makes different
assumptions that permeate your code.
It sounds like your game is possibly multithreaded / multiplayer. But,
the Game is stored the StateT monad. So I assume only one thread
updates the GameState ? Is the Game state per-player or is there only
one global game state ?
In your current app, any code in the GameState monad can update the
state at any time, and IO can be freely intermixed. There are no
really boundaries separating one state update from another -- it is a
continual process with no clear separate events.
MACID will be a lot closer to a traditional database where you perform
a distinct 'query' operation which updates or reads the state.
Each update or query you wanted to perform on the state would become a
separate isolated function which gets registered with the transaction
system (using mkMethod). You would then perform those transactions
using the update and query functions (which run in the IO monad). So
you would get rid of the GameState monad, and just have Comm.
MACID also deals with the issue of multiple threads trying to update
the state -- but that may not be a problem you care about if you only
have one thread.
One question is, what exactly are you trying to achieve.
If you simple want to checkpoint your game state now and then, you
could use just happstack-data. It provides versioned binary
serialization and migration. That would allow you to save the entire
state now and then, and migrate the data when the game state format
changed.
MACID builds on that to add the ability to log every 'update' event
that occurs so that you can replay the events if the server crashes
between checkpoints. (It also allows for distributed state across
multiple servers). But in order to log an event, the app has to first
have things when can be clearly identified as a single 'event'. And
you have to be able to replay those events later and arrive at the
same final state you did the first time.
So, in MACID, your events are *written* in the Update and Query monads.
To specific where an 'event' begins and ends, you register some of
those functions with MACID using the mkMethods function. The event
starts when a registered function is called, and ends when that
function returns a value.
In order to be sure that the events can be replayed later and arrive
at the same result, those events can not perform any IO, because the
IO might result in a different answer when replayed.
So, to answer your question: You will not directly call functions in
the Update monad. And you will not integrated the Update monad into
your other monads. Instead you will register the Update and Query
functions via mkMethods, and call them in the IO monad via query and
update. That is likely to be fairly disruptive to your current design.
But it ensures that every event is saved.
If you merely want periodic checkpoints, you can use happstack-data
and just write the state out periodically.
hope this helps!
If I have still not answered your question, or you have others, feel
free to ask!
- jeremy
On Nov 7, 2010, at 10:02 AM, Corentin Dupont wrote:
> Hello Jeremy,
> thanks for your mail.
>
> I am in despair on this problem since days, I would really help your
> help.
> I can't figure out how I can add MACID into my program.
> Here's the problem:
> I already have monads in my program like that:
>
> > type Comm = StateT Communication IO
> >
> > type GameState a = StateT Game Comm a
>
> Many functions make use of GameState.
> The only type that need to be serialized is Game.
> The type Communication contains TChans used for players communication.
> The IO in Comm is necessary to make some print outs and to use an
> interpretor when needed (Hint).
>
> How can this match with the Update type in Happstack?
>
> Thanks a lot for your help.
> Corentin
>
>
> On Fri, Nov 5, 2010 at 3:50 AM, Jeremy Shaw <jeremy at n-heptane.com>
> wrote:
> Hello,
>
> I added a brief section to the happstack crash course on using MACID:
>
> http://www.happstack.com/docs/crashcourse/HappstackState.html
>
> That should hopefully get you started.
>
> The example uses happstack state with happstack server. But there is
> really no connection between the two.
>
> Hope this helps! If you have additional questions, feel free to ask!
>
> - j
>
>
> On Thu, Nov 4, 2010 at 12:48 PM, Dupont Corentin
> <corentin.dupont at gmail.com> wrote:
> > Hello,
> > I'm wondering how can I use Happstack's MACID in my application
> without
> > breaking everything.
> >
> > I have a monad like that:
> >
> > type Comm = StateT Communication IO
> >
> > type GameState a = StateT Game Comm a
> >
> > and many functions like:
> > foo :: GameState ()
> > foo = do
> > lift $ putComm <some message to player's channel>
> > modify <someAction>
> >
> > The state of the game is stored in Game.
> > Comm is used as an abstraction to communicate over several
> channels with
> > players.
> >
> > Whereas MACID asks to use:
> >
> > type Update state = Ev (StateT state STM)
> >
> > How can I use this without modifying everything??
> > I understand that MACID must record the <someAction> from above
> but the
> > message should not.
> >
> > Thanks for help!
> >
> > Corentin
> >
> >
> >
> >
> > _______________________________________________
> > Haskell-Cafe mailing list
> > Haskell-Cafe at haskell.org
> > http://www.haskell.org/mailman/listinfo/haskell-cafe
> >
> >
>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: http://www.haskell.org/pipermail/haskell-cafe/attachments/20101107/2777878c/attachment.html
More information about the Haskell-Cafe
mailing list