{-# LANGUAGE DataKinds #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE PolyKinds #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-}

module Polysemy.Several
  ( HList (..),
    TypeMap,
    Append,
    runSeveral,
  )
where

import Data.Kind
import Polysemy

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

-- | A list capable of storing values of different types. Creating an HList
-- looks like.
--
-- > 1 ::: "test" ::: True ::: HNil
infixr 5 :::

data HList a where
  HNil :: HList '[]
  (:::) :: a -> HList (b :: [Type]) -> HList (a ': b)

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

-- | A map function over type level lists. For example, the following two
-- lines are equivalent:
--
-- > TypeMap Reader [Int, String, False]
-- > [Reader Int, Reader String, Reader Bool]
type family TypeMap (f :: a -> b) (xs :: [a]) where
  TypeMap _ '[] = '[]
  TypeMap f (x ': xs) = f x ': TypeMap f xs

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

-- | Type-level append.
--
-- > Append [Int, String] [Bool]
-- > [Int, String, Bool]
type family Append (a :: [t]) (b :: [t]) where
  Append '[] b = b
  Append (a ': as) b = a ': Append as b

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

-- | A helper function for building new runners which accept HLists intsead of
-- individual elements. If you would normally write
--
-- > f 5 . f "Text" . f True
--
-- then this function can turn that into
--
-- > runSeveral f (True ::: "Text" ::: 5 ::: HNil)
--
-- For example, a runReaders function could be implemented as:
--
-- > runReaders :: HList t -> Sem (TypeConcat (TypeMap Reader t) r) a -> Sem r a
-- > runReaders = runSeveral runReader
--
-- @since 0.1.0.0
runSeveral ::
  (forall r' k x. k -> Sem (e k ': r') x -> Sem r' x) ->
  HList t ->
  Sem (Append (TypeMap e t) r) a ->
  Sem r a
runSeveral :: (forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x)
-> HList t -> Sem (Append (TypeMap e t) r) a -> Sem r a
runSeveral forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x
f (a
a ::: HList b
as) = (forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x)
-> HList b -> Sem (Append (TypeMap e b) r) a -> Sem r a
forall (e :: * -> Effect) (t :: [*]) (r :: [Effect]) a.
(forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x)
-> HList t -> Sem (Append (TypeMap e t) r) a -> Sem r a
runSeveral forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x
f HList b
as (Sem (Append (TypeMap e b) r) a -> Sem r a)
-> (Sem (e a : Append (TypeMap e b) r) a
    -> Sem (Append (TypeMap e b) r) a)
-> Sem (e a : Append (TypeMap e b) r) a
-> Sem r a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. a
-> Sem (e a : Append (TypeMap e b) r) a
-> Sem (Append (TypeMap e b) r) a
forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x
f a
a
runSeveral forall (r' :: [Effect]) k x. k -> Sem (e k : r') x -> Sem r' x
_ HList t
HNil = Sem (Append (TypeMap e t) r) a -> Sem r a
forall a. a -> a
id