X-Git-Url: http://git.lukelau.me/?p=opengl.git;a=blobdiff_plain;f=src%2FLanguage%2FHaskell%2FLSP%2FTest%2FReplay.hs;h=ad26858ee39632d4a5e8260a83576ef0a65a93b6;hp=1e361542b71f39b1356491b2232f1f119e072165;hb=bd6901688e6c9d8332fea161260d32666885f9ed;hpb=493d20ada6e48a8987e00a5ec92a1b31fe3c9b8c diff --git a/src/Language/Haskell/LSP/Test/Replay.hs b/src/Language/Haskell/LSP/Test/Replay.hs index 1e36154..ad26858 100644 --- a/src/Language/Haskell/LSP/Test/Replay.hs +++ b/src/Language/Haskell/LSP/Test/Replay.hs @@ -7,7 +7,6 @@ where import Prelude hiding (id) import Control.Concurrent -import Control.Exception import Control.Monad.IO.Class import qualified Data.ByteString.Lazy.Char8 as B import qualified Data.Text as T @@ -35,7 +34,7 @@ import Language.Haskell.LSP.Test.Server -- named "session.log". replaySession :: String -- ^ The command to run the server. -> FilePath -- ^ The recorded session directory. - -> IO Bool + -> IO () replaySession serverExe sessionDir = do entries <- B.lines <$> B.readFile (sessionDir "session.log") @@ -55,19 +54,18 @@ replaySession serverExe sessionDir = do reqSema <- newEmptyMVar rspSema <- newEmptyMVar - passVar <- newEmptyMVar :: IO (MVar Bool) + passSema <- newEmptyMVar + mainThread <- myThreadId - threadId <- forkIO $ + sessionThread <- liftIO $ forkIO $ runSessionWithHandles serverIn serverOut - (listenServer serverMsgs requestMap reqSema rspSema passVar) + (listenServer serverMsgs requestMap reqSema rspSema passSema mainThread) def sessionDir (sendMessages clientMsgs reqSema rspSema) - - result <- takeMVar passVar - killThread threadId - return result + takeMVar passSema + killThread sessionThread where isClientMsg (FromClient _ _) = True @@ -125,30 +123,23 @@ isNotification (NotShowMessage _) = True isNotification (NotCancelRequestFromServer _) = True isNotification _ = False -listenServer :: [FromServerMessage] -> RequestMap -> MVar LspId -> MVar LspIdRsp -> MVar Bool -> Handle -> Session () -listenServer [] _ _ _ passVar _ = liftIO $ putMVar passVar True -listenServer expectedMsgs reqMap reqSema rspSema passVar serverOut = do - - let handler :: IOException -> IO B.ByteString - handler _ = putMVar passVar False >> return B.empty +listenServer :: [FromServerMessage] -> RequestMap -> MVar LspId -> MVar LspIdRsp -> MVar () -> ThreadId -> Handle -> Session () +listenServer [] _ _ _ passSema _ _ = liftIO $ putMVar passSema () +listenServer expectedMsgs reqMap reqSema rspSema passSema mainThreadId serverOut = do - msgBytes <- liftIO $ catch (getNextMessage serverOut) handler + msgBytes <- liftIO $ getNextMessage serverOut let msg = decodeFromServerMsg reqMap msgBytes handleServerMessage request response notification msg if shouldSkip msg - then listenServer expectedMsgs reqMap reqSema rspSema passVar serverOut + then listenServer expectedMsgs reqMap reqSema rspSema passSema mainThreadId serverOut else if inRightOrder msg expectedMsgs - then listenServer (delete msg expectedMsgs) reqMap reqSema rspSema passVar serverOut - else liftIO $ do - putStrLn "Out of order" - putStrLn "Got:" - print msg - putStrLn "Expected one of:" - mapM_ print $ takeWhile (not . isNotification) expectedMsgs - print $ head $ dropWhile isNotification expectedMsgs - putMVar passVar False + then listenServer (delete msg expectedMsgs) reqMap reqSema rspSema passSema mainThreadId serverOut + else let expectedMsgs = takeWhile (not . isNotification) expectedMsgs + ++ [head $ dropWhile isNotification expectedMsgs] + exc = ReplayOutOfOrderException msg expectedMsgs + in liftIO $ throwTo mainThreadId exc where response :: ResponseMessage a -> Session ()