module Test.Monad(main) where import Test.Type import Development.Shake.Internal.Core.Monad import Data.Either.Extra import Data.IORef import Control.Concurrent import Control.Exception import Control.Monad import Control.Monad.IO.Class run :: ro -> rw -> RAW () () ro rw a -> IO a run ro rw m = do res <- newEmptyMVar runRAW return ro rw m $ void . tryPutMVar res either throwIO return =<< readMVar res main = testSimple $ do let conv x = mapLeft fromException x :: Either (Maybe ArithException) Int let dump rw = liftIO . (=== rw) =<< getRW -- test the basics plus exception handling run 1 "test" $ do dump "test" putRW "more" dump "more" res <- tryRAW $ do dump "more" modifyRW (++ "x") dump "morex" return 100 liftIO $ conv res === Right 100 dump "morex" putRW "new" dump "new" res <- tryRAW $ do dump "new" modifyRW (++ "z") dump "newz" throwRAW Overflow error "Should not have reached here" return 9 liftIO $ conv res === Left (Just Overflow) dump "newz" catchRAW (catchRAW (throwRAW Overflow) $ \_ -> modifyRW (++ "x")) $ \_ -> modifyRW (++ "y") dump "newzx" catchRAW (catchRAW (throwRAW Overflow) $ \e -> modifyRW (++ "x") >> throwRAW e) $ \_ -> modifyRW (++ "y") dump "newzxxy" -- test capture run 1 "test" $ do i <- captureRAW $ \k -> k $ Right 1 liftIO $ i === 1 i <- tryRAW $ captureRAW $ \k -> k $ Left $ toException Overflow liftIO $ conv i === Left (Just Overflow) captureRAW $ \k -> k $ Right () i <- tryRAW $ throwRAW Underflow liftIO $ conv i === Left (Just Underflow) -- catch does not scope too far res <- try $ run 1 "test" $ fmap (either show id) $ tryRAW $ captureRAW $ \_ -> throwIO Overflow res === Left Overflow res <- try $ run 1 "test" $ do captureRAW $ \_ -> throwIO Overflow return "x" res === Left Overflow -- test for GHC bug 11555 runRAW return 1 "test" (throw Overflow :: RAW () () Int String ()) $ \res -> mapLeft fromException res === Left (Just Overflow) -- catch works properly if continuation called multiple times ref <- newIORef [] run 1 "test" $ flip catchRAW (const $ liftIO $ modifyIORef ref ('x':)) $ do captureRAW $ \k -> do k $ Right () k $ Right () k $ Left $ toException Overflow k $ Right () k $ Left $ toException Overflow flip catchRAW (const $ liftIO $ modifyIORef ref ('y':)) $ throwRAW $ toException Overflow (===) "xyxyy" =<< readIORef ref -- what if we throw an exception inside the continuation of run ref <- newIORef 0 res <- try $ runRAW return 1 "test" (return 1) $ \_ -> do modifyIORef ref (+1) throwIO Overflow res === Left Overflow (=== 1) =<< readIORef ref