module Eventful.Projection
( Projection (..)
, latestProjection
, allProjections
, getLatestProjection
)
where
import Data.Foldable (foldl')
import Data.List (scanl')
import Eventful.Store.Class
import Eventful.UUID
data Projection state event
= Projection
{ projectionSeed :: state
, projectionEventHandler :: state -> event -> state
}
latestProjection :: (Foldable t) => Projection state event -> t event -> state
latestProjection (Projection seed handler) = foldl' handler seed
allProjections :: Projection state event -> [event] -> [state]
allProjections (Projection seed handler) = scanl' handler seed
getLatestProjection
:: (Monad m)
=> EventStore serialized m
-> Projection proj serialized
-> UUID
-> m (proj, EventVersion)
getLatestProjection store proj uuid = do
events <- getEvents store uuid Nothing
let
latestVersion = maxEventVersion events
latestProj = latestProjection proj $ storedEventEvent <$> events
return (latestProj, latestVersion)
where
maxEventVersion [] = 1
maxEventVersion es = maximum $ storedEventVersion <$> es