{-# LANGUAGE AllowAmbiguousTypes #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE FunctionalDependencies #-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE KindSignatures #-}
{-# LANGUAGE MagicHash #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE QuantifiedConstraints #-}
{-# LANGUAGE RankNTypes #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
{-# LANGUAGE TypeFamilies #-}
{-# LANGUAGE TypeOperators #-}
{-# LANGUAGE TypeInType #-}
{-# OPTIONS_HADDOCK hide #-}
module Capability.Reader.Internal.Class
( HasReader(..)
, ask
, asks
, local
, reader
, magnify
, Reified(..)
) where
import Capability.Constraints
import Capability.Derive (derive)
import Capability.Reflection
import Capability.Source.Internal.Class
import Data.Coerce (Coercible, coerce)
import Data.Kind (Type)
import GHC.Exts (Proxy#, proxy#)
class (Monad m, HasSource tag r m)
=> HasReader (tag :: k) (r :: Type) (m :: Type -> Type) | tag m -> r
where
local_ :: Proxy# tag -> (r -> r) -> m a -> m a
reader_ :: Proxy# tag -> (r -> a) -> m a
ask :: forall tag r m. HasReader tag r m => m r
ask :: m r
ask = forall k (tag :: k) a (m :: * -> *). HasSource tag a m => m a
forall a (m :: * -> *). HasSource tag a m => m a
await @tag
{-# INLINE ask #-}
asks :: forall tag r m a. HasReader tag r m => (r -> a) -> m a
asks :: (r -> a) -> m a
asks = forall k (tag :: k) r (m :: * -> *) a.
HasSource tag r m =>
(r -> a) -> m a
forall r (m :: * -> *) a. HasSource tag r m => (r -> a) -> m a
awaits @tag
{-# INLINE asks #-}
local :: forall tag r m a. HasReader tag r m => (r -> r) -> m a -> m a
local :: (r -> r) -> m a -> m a
local = Proxy# tag -> (r -> r) -> m a -> m a
forall k (tag :: k) r (m :: * -> *) a.
HasReader tag r m =>
Proxy# tag -> (r -> r) -> m a -> m a
local_ (Proxy# tag
forall k (a :: k). Proxy# a
proxy# @tag)
{-# INLINE local #-}
reader :: forall tag r m a. HasReader tag r m => (r -> a) -> m a
reader :: (r -> a) -> m a
reader = Proxy# tag -> (r -> a) -> m a
forall k (tag :: k) r (m :: * -> *) a.
HasReader tag r m =>
Proxy# tag -> (r -> a) -> m a
reader_ (Proxy# tag
forall k (a :: k). Proxy# a
proxy# @tag)
{-# INLINE reader #-}
magnify :: forall innertag t (cs :: [Capability]) inner m a.
( forall x. Coercible (t m x) (m x)
, HasReader innertag inner (t m)
, All cs m)
=> (forall m'. All (HasReader innertag inner ': cs) m' => m' a) -> m a
magnify :: (forall (m' :: * -> *).
All (HasReader innertag inner : cs) m' =>
m' a)
-> m a
magnify =
forall (m :: * -> *) a.
(forall x. Coercible (t m x) (m x),
All '[HasReader innertag inner] (t m), All cs m) =>
(forall (m' :: * -> *).
(All '[HasReader innertag inner] m', All cs m') =>
m' a)
-> m a
forall (t :: (* -> *) -> * -> *)
(derived :: [(* -> *) -> Constraint])
(ambient :: [(* -> *) -> Constraint]) (m :: * -> *) a.
(forall x. Coercible (t m x) (m x), All derived (t m),
All ambient m) =>
(forall (m' :: * -> *). (All derived m', All ambient m') => m' a)
-> m a
derive @t @'[HasReader innertag inner] @cs
{-# INLINE magnify #-}
data instance Reified tag (HasReader tag r) m = ReifiedReader
{ Reified tag (HasReader tag r) m -> Reified tag (HasSource tag r) m
_readerSource :: Reified tag (HasSource tag r) m,
Reified tag (HasReader tag r) m -> forall a. (r -> r) -> m a -> m a
_local :: forall a. (r -> r) -> m a -> m a,
Reified tag (HasReader tag r) m -> forall a. (r -> a) -> m a
_reader :: forall a. (r -> a) -> m a
}
instance
( Monad m,
Reifies s (Reified tag (HasReader tag r) m)
) =>
HasSource tag r (Reflected s (HasReader tag r) m)
where
await_ :: Proxy# tag -> Reflected s (HasReader tag r) m r
await_ Proxy# tag
_ = m r -> Reflected s (HasReader tag r) m r
coerce (m r -> Reflected s (HasReader tag r) m r)
-> m r -> Reflected s (HasReader tag r) m r
forall a b. (a -> b) -> a -> b
$ Reified tag (HasSource tag r) m -> m r
forall k (tag :: k) a (m :: * -> *).
Reified tag (HasSource tag a) m -> m a
_await (Reified tag (HasSource tag r) m -> m r)
-> Reified tag (HasSource tag r) m -> m r
forall a b. (a -> b) -> a -> b
$ Reified tag (HasReader tag r) m -> Reified tag (HasSource tag r) m
forall k (tag :: k) r (m :: * -> *).
Reified tag (HasReader tag r) m -> Reified tag (HasSource tag r) m
_readerSource (Reified tag (HasReader tag r) m
-> Reified tag (HasSource tag r) m)
-> Reified tag (HasReader tag r) m
-> Reified tag (HasSource tag r) m
forall a b. (a -> b) -> a -> b
$ forall (tag :: k) (c :: (* -> *) -> Constraint) (m :: * -> *).
Reifies s (Reified tag c m) =>
Reified tag c m
forall k1 k2 (s :: k1) (tag :: k2) (c :: (* -> *) -> Constraint)
(m :: * -> *).
Reifies s (Reified tag c m) =>
Reified tag c m
reified @s
{-# INLINE await_ #-}
instance
( Monad m,
Reifies s (Reified tag (HasReader tag r) m)
) =>
HasReader tag r (Reflected s (HasReader tag r) m)
where
local_ :: forall a. Proxy# tag -> (r -> r) -> Reflected s (HasReader tag r) m a -> Reflected s (HasReader tag r) m a
local_ :: Proxy# tag
-> (r -> r)
-> Reflected s (HasReader tag r) m a
-> Reflected s (HasReader tag r) m a
local_ Proxy# tag
_ = forall b.
Coercible ((r -> r) -> m a -> m a) b =>
((r -> r) -> m a -> m a) -> b
coerce @((r -> r) -> m a -> m a) (((r -> r) -> m a -> m a)
-> (r -> r)
-> Reflected s (HasReader tag r) m a
-> Reflected s (HasReader tag r) m a)
-> ((r -> r) -> m a -> m a)
-> (r -> r)
-> Reflected s (HasReader tag r) m a
-> Reflected s (HasReader tag r) m a
forall a b. (a -> b) -> a -> b
$ Reified tag (HasReader tag r) m -> forall a. (r -> r) -> m a -> m a
forall k (tag :: k) r (m :: * -> *).
Reified tag (HasReader tag r) m -> forall a. (r -> r) -> m a -> m a
_local (forall (tag :: k) (c :: (* -> *) -> Constraint) (m :: * -> *).
Reifies s (Reified tag c m) =>
Reified tag c m
forall k1 k2 (s :: k1) (tag :: k2) (c :: (* -> *) -> Constraint)
(m :: * -> *).
Reifies s (Reified tag c m) =>
Reified tag c m
reified @s)
{-# INLINE local_ #-}
reader_ :: forall a. Proxy# tag -> (r -> a) -> Reflected s (HasReader tag r) m a
reader_ :: Proxy# tag -> (r -> a) -> Reflected s (HasReader tag r) m a
reader_ Proxy# tag
_ = forall b. Coercible ((r -> a) -> m a) b => ((r -> a) -> m a) -> b
coerce @((r -> a) -> m a) (((r -> a) -> m a)
-> (r -> a) -> Reflected s (HasReader tag r) m a)
-> ((r -> a) -> m a)
-> (r -> a)
-> Reflected s (HasReader tag r) m a
forall a b. (a -> b) -> a -> b
$ Reified tag (HasReader tag r) m -> forall a. (r -> a) -> m a
forall k (tag :: k) r (m :: * -> *).
Reified tag (HasReader tag r) m -> forall a. (r -> a) -> m a
_reader (forall (tag :: k) (c :: (* -> *) -> Constraint) (m :: * -> *).
Reifies s (Reified tag c m) =>
Reified tag c m
forall k1 k2 (s :: k1) (tag :: k2) (c :: (* -> *) -> Constraint)
(m :: * -> *).
Reifies s (Reified tag c m) =>
Reified tag c m
reified @s)
{-# INLINE reader_ #-}