{-# LANGUAGE BlockArguments #-}
{-# LANGUAGE TemplateHaskell #-}

-----------------------------------------------------------------------------

-----------------------------------------------------------------------------

-- |
-- Module      :  Disco.Effects.Counter
-- Copyright   :  disco team and contributors
-- Maintainer  :  byorgey@gmail.com
--
-- SPDX-License-Identifier: BSD-3-Clause
--
-- Polysemy effect for integer counter.
module Disco.Effects.Counter where

import Polysemy
import Polysemy.State

data Counter m a where
  -- | Return the next integer in sequence.
  Next :: Counter m Integer

makeSem ''Counter

-- | Dispatch a counter effect, starting the counter from the given
--   Integer.
runCounter' :: Integer -> Sem (Counter ': r) a -> Sem r a
runCounter' :: forall (r :: EffectRow) a.
Integer -> Sem (Counter : r) a -> Sem r a
runCounter' Integer
i =
  forall s (r :: EffectRow) a. s -> Sem (State s : r) a -> Sem r a
evalState Integer
i
    forall b c a. (b -> c) -> (a -> b) -> a -> c
. forall (e1 :: Effect) (e2 :: Effect) (r :: EffectRow) a.
FirstOrder e1 "reinterpret" =>
(forall (rInitial :: EffectRow) x.
 e1 (Sem rInitial) x -> Sem (e2 : r) x)
-> Sem (e1 : r) a -> Sem (e2 : r) a
reinterpret \case
      Counter (Sem rInitial) x
Next -> do
        x
n <- forall s (r :: EffectRow). Member (State s) r => Sem r s
get
        forall s (r :: EffectRow). Member (State s) r => s -> Sem r ()
put (x
n forall a. Num a => a -> a -> a
+ x
1)
        forall (m :: * -> *) a. Monad m => a -> m a
return x
n

-- | Dispatch a counter effect, starting the counter from zero.
runCounter :: Sem (Counter ': r) a -> Sem r a
runCounter :: forall (r :: EffectRow) a. Sem (Counter : r) a -> Sem r a
runCounter = forall (r :: EffectRow) a.
Integer -> Sem (Counter : r) a -> Sem r a
runCounter' Integer
0