{-# LANGUAGE GeneralizedNewtypeDeriving #-}

-- This file is part of the Wire Server implementation.
--
-- Copyright (C) 2022 Wire Swiss GmbH <opensource@wire.com>
--
-- This program is free software: you can redistribute it and/or modify it under
-- the terms of the GNU Affero General Public License as published by the Free
-- Software Foundation, either version 3 of the License, or (at your option) any
-- later version.
--
-- This program is distributed in the hope that it will be useful, but WITHOUT
-- ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
-- FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
-- details.
--
-- You should have received a copy of the GNU Affero General Public License along
-- with this program. If not, see <https://www.gnu.org/licenses/>.

module Bilge.TestSession where

import Control.Monad.Catch (MonadCatch, MonadMask, MonadThrow)
import Control.Monad.State (StateT)
import Control.Monad.State qualified as ST
import Imports
import Network.Wai qualified as Wai
import Network.Wai.Test qualified as WaiTest
import Network.Wai.Test.Internal qualified as WaiTest

newtype SessionT m a = SessionT {forall (m :: * -> *) a.
SessionT m a -> ReaderT Application (StateT ClientState m) a
unSessionT :: ReaderT Wai.Application (StateT WaiTest.ClientState m) a}
  deriving newtype ((forall a b. (a -> b) -> SessionT m a -> SessionT m b)
-> (forall a b. a -> SessionT m b -> SessionT m a)
-> Functor (SessionT m)
forall a b. a -> SessionT m b -> SessionT m a
forall a b. (a -> b) -> SessionT m a -> SessionT m b
forall (m :: * -> *) a b.
Functor m =>
a -> SessionT m b -> SessionT m a
forall (m :: * -> *) a b.
Functor m =>
(a -> b) -> SessionT m a -> SessionT m b
forall (f :: * -> *).
(forall a b. (a -> b) -> f a -> f b)
-> (forall a b. a -> f b -> f a) -> Functor f
$cfmap :: forall (m :: * -> *) a b.
Functor m =>
(a -> b) -> SessionT m a -> SessionT m b
fmap :: forall a b. (a -> b) -> SessionT m a -> SessionT m b
$c<$ :: forall (m :: * -> *) a b.
Functor m =>
a -> SessionT m b -> SessionT m a
<$ :: forall a b. a -> SessionT m b -> SessionT m a
Functor, Functor (SessionT m)
Functor (SessionT m) =>
(forall a. a -> SessionT m a)
-> (forall a b.
    SessionT m (a -> b) -> SessionT m a -> SessionT m b)
-> (forall a b c.
    (a -> b -> c) -> SessionT m a -> SessionT m b -> SessionT m c)
-> (forall a b. SessionT m a -> SessionT m b -> SessionT m b)
-> (forall a b. SessionT m a -> SessionT m b -> SessionT m a)
-> Applicative (SessionT m)
forall a. a -> SessionT m a
forall a b. SessionT m a -> SessionT m b -> SessionT m a
forall a b. SessionT m a -> SessionT m b -> SessionT m b
forall a b. SessionT m (a -> b) -> SessionT m a -> SessionT m b
forall a b c.
(a -> b -> c) -> SessionT m a -> SessionT m b -> SessionT m c
forall (m :: * -> *). Monad m => Functor (SessionT m)
forall (m :: * -> *) a. Monad m => a -> SessionT m a
forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> SessionT m b -> SessionT m a
forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> SessionT m b -> SessionT m b
forall (m :: * -> *) a b.
Monad m =>
SessionT m (a -> b) -> SessionT m a -> SessionT m b
forall (m :: * -> *) a b c.
Monad m =>
(a -> b -> c) -> SessionT m a -> SessionT m b -> SessionT m c
forall (f :: * -> *).
Functor f =>
(forall a. a -> f a)
-> (forall a b. f (a -> b) -> f a -> f b)
-> (forall a b c. (a -> b -> c) -> f a -> f b -> f c)
-> (forall a b. f a -> f b -> f b)
-> (forall a b. f a -> f b -> f a)
-> Applicative f
$cpure :: forall (m :: * -> *) a. Monad m => a -> SessionT m a
pure :: forall a. a -> SessionT m a
$c<*> :: forall (m :: * -> *) a b.
Monad m =>
SessionT m (a -> b) -> SessionT m a -> SessionT m b
<*> :: forall a b. SessionT m (a -> b) -> SessionT m a -> SessionT m b
$cliftA2 :: forall (m :: * -> *) a b c.
Monad m =>
(a -> b -> c) -> SessionT m a -> SessionT m b -> SessionT m c
liftA2 :: forall a b c.
(a -> b -> c) -> SessionT m a -> SessionT m b -> SessionT m c
$c*> :: forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> SessionT m b -> SessionT m b
*> :: forall a b. SessionT m a -> SessionT m b -> SessionT m b
$c<* :: forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> SessionT m b -> SessionT m a
<* :: forall a b. SessionT m a -> SessionT m b -> SessionT m a
Applicative, Applicative (SessionT m)
Applicative (SessionT m) =>
(forall a b. SessionT m a -> (a -> SessionT m b) -> SessionT m b)
-> (forall a b. SessionT m a -> SessionT m b -> SessionT m b)
-> (forall a. a -> SessionT m a)
-> Monad (SessionT m)
forall a. a -> SessionT m a
forall a b. SessionT m a -> SessionT m b -> SessionT m b
forall a b. SessionT m a -> (a -> SessionT m b) -> SessionT m b
forall (m :: * -> *). Monad m => Applicative (SessionT m)
forall (m :: * -> *) a. Monad m => a -> SessionT m a
forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> SessionT m b -> SessionT m b
forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> (a -> SessionT m b) -> SessionT m b
forall (m :: * -> *).
Applicative m =>
(forall a b. m a -> (a -> m b) -> m b)
-> (forall a b. m a -> m b -> m b)
-> (forall a. a -> m a)
-> Monad m
$c>>= :: forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> (a -> SessionT m b) -> SessionT m b
>>= :: forall a b. SessionT m a -> (a -> SessionT m b) -> SessionT m b
$c>> :: forall (m :: * -> *) a b.
Monad m =>
SessionT m a -> SessionT m b -> SessionT m b
>> :: forall a b. SessionT m a -> SessionT m b -> SessionT m b
$creturn :: forall (m :: * -> *) a. Monad m => a -> SessionT m a
return :: forall a. a -> SessionT m a
Monad, Monad (SessionT m)
Monad (SessionT m) =>
(forall e a. (HasCallStack, Exception e) => e -> SessionT m a)
-> MonadThrow (SessionT m)
forall e a. (HasCallStack, Exception e) => e -> SessionT m a
forall (m :: * -> *).
Monad m =>
(forall e a. (HasCallStack, Exception e) => e -> m a)
-> MonadThrow m
forall (m :: * -> *). MonadThrow m => Monad (SessionT m)
forall (m :: * -> *) e a.
(MonadThrow m, HasCallStack, Exception e) =>
e -> SessionT m a
$cthrowM :: forall (m :: * -> *) e a.
(MonadThrow m, HasCallStack, Exception e) =>
e -> SessionT m a
throwM :: forall e a. (HasCallStack, Exception e) => e -> SessionT m a
MonadThrow, MonadThrow (SessionT m)
MonadThrow (SessionT m) =>
(forall e a.
 (HasCallStack, Exception e) =>
 SessionT m a -> (e -> SessionT m a) -> SessionT m a)
-> MonadCatch (SessionT m)
forall e a.
(HasCallStack, Exception e) =>
SessionT m a -> (e -> SessionT m a) -> SessionT m a
forall (m :: * -> *).
MonadThrow m =>
(forall e a.
 (HasCallStack, Exception e) =>
 m a -> (e -> m a) -> m a)
-> MonadCatch m
forall (m :: * -> *). MonadCatch m => MonadThrow (SessionT m)
forall (m :: * -> *) e a.
(MonadCatch m, HasCallStack, Exception e) =>
SessionT m a -> (e -> SessionT m a) -> SessionT m a
$ccatch :: forall (m :: * -> *) e a.
(MonadCatch m, HasCallStack, Exception e) =>
SessionT m a -> (e -> SessionT m a) -> SessionT m a
catch :: forall e a.
(HasCallStack, Exception e) =>
SessionT m a -> (e -> SessionT m a) -> SessionT m a
MonadCatch, MonadCatch (SessionT m)
MonadCatch (SessionT m) =>
(forall b.
 HasCallStack =>
 ((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
 -> SessionT m b)
-> (forall b.
    HasCallStack =>
    ((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
    -> SessionT m b)
-> (forall a b c.
    HasCallStack =>
    SessionT m a
    -> (a -> ExitCase b -> SessionT m c)
    -> (a -> SessionT m b)
    -> SessionT m (b, c))
-> MonadMask (SessionT m)
forall b.
HasCallStack =>
((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
-> SessionT m b
forall a b c.
HasCallStack =>
SessionT m a
-> (a -> ExitCase b -> SessionT m c)
-> (a -> SessionT m b)
-> SessionT m (b, c)
forall (m :: * -> *). MonadMask m => MonadCatch (SessionT m)
forall (m :: * -> *) b.
(MonadMask m, HasCallStack) =>
((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
-> SessionT m b
forall (m :: * -> *) a b c.
(MonadMask m, HasCallStack) =>
SessionT m a
-> (a -> ExitCase b -> SessionT m c)
-> (a -> SessionT m b)
-> SessionT m (b, c)
forall (m :: * -> *).
MonadCatch m =>
(forall b. HasCallStack => ((forall a. m a -> m a) -> m b) -> m b)
-> (forall b.
    HasCallStack =>
    ((forall a. m a -> m a) -> m b) -> m b)
-> (forall a b c.
    HasCallStack =>
    m a -> (a -> ExitCase b -> m c) -> (a -> m b) -> m (b, c))
-> MonadMask m
$cmask :: forall (m :: * -> *) b.
(MonadMask m, HasCallStack) =>
((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
-> SessionT m b
mask :: forall b.
HasCallStack =>
((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
-> SessionT m b
$cuninterruptibleMask :: forall (m :: * -> *) b.
(MonadMask m, HasCallStack) =>
((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
-> SessionT m b
uninterruptibleMask :: forall b.
HasCallStack =>
((forall a. SessionT m a -> SessionT m a) -> SessionT m b)
-> SessionT m b
$cgeneralBracket :: forall (m :: * -> *) a b c.
(MonadMask m, HasCallStack) =>
SessionT m a
-> (a -> ExitCase b -> SessionT m c)
-> (a -> SessionT m b)
-> SessionT m (b, c)
generalBracket :: forall a b c.
HasCallStack =>
SessionT m a
-> (a -> ExitCase b -> SessionT m c)
-> (a -> SessionT m b)
-> SessionT m (b, c)
MonadMask, Monad (SessionT m)
Monad (SessionT m) =>
(forall a. IO a -> SessionT m a) -> MonadIO (SessionT m)
forall a. IO a -> SessionT m a
forall (m :: * -> *).
Monad m =>
(forall a. IO a -> m a) -> MonadIO m
forall (m :: * -> *). MonadIO m => Monad (SessionT m)
forall (m :: * -> *) a. MonadIO m => IO a -> SessionT m a
$cliftIO :: forall (m :: * -> *) a. MonadIO m => IO a -> SessionT m a
liftIO :: forall a. IO a -> SessionT m a
MonadIO, Monad (SessionT m)
Monad (SessionT m) =>
(forall a. String -> SessionT m a) -> MonadFail (SessionT m)
forall a. String -> SessionT m a
forall (m :: * -> *).
Monad m =>
(forall a. String -> m a) -> MonadFail m
forall (m :: * -> *). MonadFail m => Monad (SessionT m)
forall (m :: * -> *) a. MonadFail m => String -> SessionT m a
$cfail :: forall (m :: * -> *) a. MonadFail m => String -> SessionT m a
fail :: forall a. String -> SessionT m a
MonadFail)

instance MonadTrans SessionT where
  lift :: forall (m :: * -> *) a. Monad m => m a -> SessionT m a
lift = ReaderT Application (StateT ClientState m) a -> SessionT m a
forall (m :: * -> *) a.
ReaderT Application (StateT ClientState m) a -> SessionT m a
SessionT (ReaderT Application (StateT ClientState m) a -> SessionT m a)
-> (m a -> ReaderT Application (StateT ClientState m) a)
-> m a
-> SessionT m a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. StateT ClientState m a
-> ReaderT Application (StateT ClientState m) a
forall (m :: * -> *) a. Monad m => m a -> ReaderT Application m a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift (StateT ClientState m a
 -> ReaderT Application (StateT ClientState m) a)
-> (m a -> StateT ClientState m a)
-> m a
-> ReaderT Application (StateT ClientState m) a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. m a -> StateT ClientState m a
forall (m :: * -> *) a. Monad m => m a -> StateT ClientState m a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift

liftSession :: (MonadIO m) => WaiTest.Session a -> SessionT m a
liftSession :: forall (m :: * -> *) a. MonadIO m => Session a -> SessionT m a
liftSession Session a
session = ReaderT Application (StateT ClientState m) a -> SessionT m a
forall (m :: * -> *) a.
ReaderT Application (StateT ClientState m) a -> SessionT m a
SessionT (ReaderT Application (StateT ClientState m) a -> SessionT m a)
-> ReaderT Application (StateT ClientState m) a -> SessionT m a
forall a b. (a -> b) -> a -> b
$ do
  Application
app <- ReaderT Application (StateT ClientState m) Application
forall r (m :: * -> *). MonadReader r m => m r
ask
  ClientState
clientState <- StateT ClientState m ClientState
-> ReaderT Application (StateT ClientState m) ClientState
forall (m :: * -> *) a. Monad m => m a -> ReaderT Application m a
forall (t :: (* -> *) -> * -> *) (m :: * -> *) a.
(MonadTrans t, Monad m) =>
m a -> t m a
lift StateT ClientState m ClientState
forall s (m :: * -> *). MonadState s m => m s
ST.get
  let resultInState :: StateT ClientState IO a
resultInState = Session a -> Application -> StateT ClientState IO a
forall r (m :: * -> *) a. ReaderT r m a -> r -> m a
runReaderT Session a
session Application
app
  let resultInIO :: IO a
resultInIO = StateT ClientState IO a -> ClientState -> IO a
forall (m :: * -> *) s a. Monad m => StateT s m a -> s -> m a
ST.evalStateT StateT ClientState IO a
resultInState ClientState
clientState
  IO a -> ReaderT Application (StateT ClientState m) a
forall a. IO a -> ReaderT Application (StateT ClientState m) a
forall (m :: * -> *) a. MonadIO m => IO a -> m a
liftIO IO a
resultInIO