{-# LANGUAGE DerivingVia #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE QuantifiedConstraints #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE StandaloneDeriving #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeInType #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE UnboxedTuples #-}
{-# LANGUAGE UndecidableInstances #-}
{-# LANGUAGE PatternSynonyms #-}
{-# OPTIONS_GHC -Wno-orphans #-}
{-# OPTIONS_GHC -Wno-simplifiable-class-constraints #-}
{-# OPTIONS_HADDOCK hide #-}
module Capability.Reader.Internal.Strategies
( MonadReader(..)
, ReadStatePure(..)
, ReadState(..)
) where
import Capability.Accessors
import Capability.Reader.Internal.Class
import Capability.State.Internal.Class
import Capability.Source.Internal.Strategies
import Control.Lens (over, view)
import Control.Monad.Catch (MonadMask, bracket)
import qualified Control.Monad.Reader.Class as Reader
import Control.Monad.Trans.Class (lift)
import Control.Monad.Trans.Control (MonadTransControl(..))
import Data.Coerce (Coercible, coerce)
import qualified Data.Generics.Product.Fields as Generic
import qualified Data.Generics.Product.Positions as Generic
import GHC.Exts (Proxy#)
instance Reader.MonadReader r m => HasReader tag r (MonadReader m) where
local_
:: forall a. Proxy# tag -> (r -> r) -> MonadReader m a -> MonadReader m a
local_ _ = coerce @((r -> r) -> m a -> m a) Reader.local
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# tag -> (r -> a) -> MonadReader m a
reader_ _ = coerce @((r -> a) -> m a) Reader.reader
{-# INLINE reader_ #-}
instance HasState tag r m => HasReader tag r (ReadStatePure m) where
local_ :: forall a.
Proxy# tag -> (r -> r) -> ReadStatePure m a -> ReadStatePure m a
local_ _ f = coerce @(m a -> m a) $ \m -> do
r <- state @tag $ \r -> (r, f r)
m <* put @tag r
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# tag -> (r -> a) -> ReadStatePure m a
reader_ _ = coerce @((r -> a) -> m a) $ gets @tag
{-# INLINE reader_ #-}
instance
(HasState tag r m, MonadMask m)
=> HasReader tag r (ReadState m)
where
local_ :: forall a.
Proxy# tag -> (r -> r) -> ReadState m a -> ReadState m a
local_ _ f = coerce @(m a -> m a) $ \action ->
let
setAndSave = state @tag $ \r -> (r, f r)
restore r = put @tag r
in
bracket setAndSave restore $ \_ -> action
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# tag -> (r -> a) -> ReadState m a
reader_ _ = coerce @((r -> a) -> m a) $ gets @tag
{-# INLINE reader_ #-}
instance
( Coercible from to, HasReader tag from m
, forall x y. Coercible x y => Coercible (m x) (m y) )
=> HasReader tag to (Coerce to m)
where
local_
:: forall a. Proxy# tag -> (to -> to) -> Coerce to m a -> Coerce to m a
local_ tag = coerce @((from -> from) -> m a -> m a) $ local_ tag
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# tag -> (to -> a) -> Coerce to m a
reader_ tag = coerce @((from -> a) -> m a) $ reader_ tag
{-# INLINE reader_ #-}
instance HasReader oldtag r m => HasReader newtag r (Rename oldtag m) where
local_ :: forall a.
Proxy# newtag -> (r -> r) -> Rename oldtag m a -> Rename oldtag m a
local_ _ = coerce @((r -> r) -> m a -> m a) $ local @oldtag
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# newtag -> (r -> a) -> Rename oldtag m a
reader_ _ = coerce @((r -> a) -> m a) $ reader @oldtag
{-# INLINE reader_ #-}
instance
( tag ~ field, Generic.HasField' field record v, HasReader oldtag record m )
=> HasReader tag v (Field field oldtag m)
where
local_ :: forall a.
Proxy# tag
-> (v -> v)
-> Field field oldtag m a
-> Field field oldtag m a
local_ _ = coerce @((v -> v) -> m a -> m a) $
local @oldtag . over (Generic.field' @field)
{-# INLINE local_ #-}
reader_ :: forall a.
Proxy# tag
-> (v -> a)
-> Field field oldtag m a
reader_ _ f = coerce @(m a) $
reader @oldtag $ f . view (Generic.field' @field)
{-# INLINE reader_ #-}
instance
( tag ~ pos, Generic.HasPosition' pos struct v, HasReader oldtag struct m )
=> HasReader tag v (Pos pos oldtag m)
where
local_ :: forall a.
Proxy# tag
-> (v -> v)
-> Pos pos oldtag m a
-> Pos pos oldtag m a
local_ _ = coerce @((v -> v) -> m a -> m a) $
local @oldtag . over (Generic.position' @pos)
{-# INLINE local_ #-}
reader_ :: forall a.
Proxy# tag
-> (v -> a)
-> Pos pos oldtag m a
reader_ _ f = coerce @(m a) $
reader @oldtag $ f . view (Generic.position' @pos)
{-# INLINE reader_ #-}
instance (HasReader tag r m, MonadTransControl t, Monad (t m))
=> HasReader tag r (Lift (t m))
where
local_
:: forall a. Proxy# tag -> (r -> r) -> Lift (t m) a -> Lift (t m) a
local_ _ f = coerce @(t m a -> t m a) $
\m -> liftWith (\run -> local @tag f $ run m) >>= restoreT . pure
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# tag -> (r -> a) -> Lift (t m) a
reader_ _ = coerce @((r -> a) -> t m a) $ lift . reader @tag
{-# INLINE reader_ #-}
deriving via ((t2 :: (* -> *) -> * -> *) ((t1 :: (* -> *) -> * -> *) m))
instance
( forall x. Coercible (m x) (t2 (t1 m) x)
, Monad m, HasReader tag r (t2 (t1 m)) )
=> HasReader tag r ((t2 :.: t1) m)