mirror of
https://github.com/unclechu/gRPC-haskell.git
synced 2025-01-12 12:09:46 +01:00
917 lines
36 KiB
Haskell
917 lines
36 KiB
Haskell
{-# LANGUAGE DataKinds #-}
|
|
{-# LANGUAGE GeneralizedNewtypeDeriving #-}
|
|
{-# LANGUAGE LambdaCase #-}
|
|
{-# LANGUAGE OverloadedLists #-}
|
|
{-# LANGUAGE OverloadedStrings #-}
|
|
{-# LANGUAGE PatternSynonyms #-}
|
|
{-# LANGUAGE RecordWildCards #-}
|
|
{-# LANGUAGE ScopedTypeVariables #-}
|
|
{-# LANGUAGE TupleSections #-}
|
|
{-# OPTIONS_GHC -fno-warn-orphans #-}
|
|
|
|
module LowLevelTests where
|
|
|
|
import Control.Concurrent (threadDelay)
|
|
import Control.Concurrent.Async
|
|
import Control.Monad
|
|
import Control.Monad.Managed
|
|
import Data.ByteString (ByteString,
|
|
isPrefixOf,
|
|
isSuffixOf)
|
|
import Data.List (find)
|
|
import qualified Data.Map.Strict as M
|
|
import qualified Data.Set as S
|
|
import GHC.Exts (fromList, toList)
|
|
import Network.GRPC.Unsafe.ChannelArgs (Arg(..))
|
|
import Network.GRPC.LowLevel
|
|
import qualified Network.GRPC.LowLevel.Call.Unregistered as U
|
|
import qualified Network.GRPC.LowLevel.Client.Unregistered as U
|
|
import Network.GRPC.LowLevel.GRPC (threadDelaySecs)
|
|
import qualified Network.GRPC.LowLevel.Server.Unregistered as U
|
|
import qualified Pipes as P
|
|
import Test.Tasty
|
|
import Test.Tasty.HUnit as HU (Assertion,
|
|
assertBool,
|
|
assertEqual,
|
|
assertFailure,
|
|
testCase,
|
|
(@?=))
|
|
|
|
lowLevelTests :: TestTree
|
|
lowLevelTests = testGroup "Unit tests of low-level Haskell library"
|
|
[ testGRPCBracket
|
|
, testCompletionQueueCreateDestroy
|
|
, testClientCreateDestroy
|
|
, testClientCall
|
|
, testClientTimeoutNoServer
|
|
, testServerCreateDestroy
|
|
, testMixRegisteredUnregistered
|
|
, testPayload
|
|
, testSSL
|
|
, testAuthMetadataTransfer
|
|
, testServerAuthProcessorCancel
|
|
, testPayloadUnregistered
|
|
, testServerCancel
|
|
, testGoaway
|
|
, testSlowServer
|
|
, testServerCallExpirationCheck
|
|
, testCustomUserAgent
|
|
, testClientCompression
|
|
, testClientServerCompression
|
|
, testClientMaxReceiveMessageLengthChannelArg
|
|
, testClientStreaming
|
|
, testClientStreamingUnregistered
|
|
, testServerStreaming
|
|
, testServerStreamingUnregistered
|
|
, testBiDiStreaming
|
|
, testBiDiStreamingUnregistered
|
|
]
|
|
|
|
testGRPCBracket :: TestTree
|
|
testGRPCBracket =
|
|
testCase "Start/stop GRPC" $ runManaged $ void mgdGRPC
|
|
|
|
testCompletionQueueCreateDestroy :: TestTree
|
|
testCompletionQueueCreateDestroy =
|
|
testCase "Create/destroy CQ" $ runManaged $ do
|
|
g <- mgdGRPC
|
|
liftIO (withCompletionQueue g nop)
|
|
|
|
testClientCreateDestroy :: TestTree
|
|
testClientCreateDestroy =
|
|
clientOnlyTest "start/stop" nop
|
|
|
|
testClientTimeoutNoServer :: TestTree
|
|
testClientTimeoutNoServer =
|
|
clientOnlyTest "request timeout when server DNE" $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
r <- clientRequest c rm 1 "Hello" mempty
|
|
r @?= Left GRPCIOTimeout
|
|
|
|
testServerCreateDestroy :: TestTree
|
|
testServerCreateDestroy =
|
|
serverOnlyTest "start/stop" (["/foo"],[],[],[]) nop
|
|
|
|
testMixRegisteredUnregistered :: TestTree
|
|
testMixRegisteredUnregistered =
|
|
csTest "server uses unregistered calls to handle unknown endpoints"
|
|
client
|
|
server
|
|
(["/foo"],[],[],[])
|
|
where
|
|
client c = do
|
|
rm1 <- clientRegisterMethodNormal c "/foo"
|
|
rm2 <- clientRegisterMethodNormal c "/bar"
|
|
clientRequest c rm1 1 "Hello" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspBody @?= "reply test"
|
|
initMD @?= dummyMeta
|
|
trailMD @?= dummyMeta
|
|
clientRequest c rm2 1 "bad endpoint" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspBody @?= ""
|
|
return ()
|
|
server s = do
|
|
concurrently regThread unregThread
|
|
return ()
|
|
where regThread = do
|
|
let rm = head (normalMethods s)
|
|
_r <- serverHandleNormalCall s rm dummyMeta $ \c -> do
|
|
payload c @?= "Hello"
|
|
return ("reply test", dummyMeta, StatusOk, "")
|
|
return ()
|
|
unregThread = do
|
|
U.serverHandleNormalCall s mempty $ \call _ -> do
|
|
U.callMethod call @?= "/bar"
|
|
return ("", mempty, StatusOk,
|
|
StatusDetails "Wrong endpoint")
|
|
return ()
|
|
|
|
-- TODO: There seems to be a race here (and in other client/server pairs, of
|
|
-- course) about what gets reported when there is a failure. E.g., if one of the
|
|
-- Assertions fails in the request processing block for the server, we /may/ get
|
|
-- that error reported accurately as a call cancellation on the client, rather
|
|
-- than a useful error about the failure on the server. Maybe we'll need to
|
|
-- tweak EH behavior / async use.
|
|
testPayload :: TestTree
|
|
testPayload =
|
|
csTest "registered normal request/response" client server (["/foo"],[],[],[])
|
|
where
|
|
clientMD = [ ("foo_key", "foo_val")
|
|
, ("bar_key", "bar_val")
|
|
, ("bar_key", "bar_repeated_val")]
|
|
client c = do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 10 "Hello!" clientMD >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "reply test"
|
|
details @?= "details string"
|
|
initMD @?= dummyMeta
|
|
trailMD @?= dummyMeta
|
|
server s = do
|
|
let rm = head (normalMethods s)
|
|
r <- serverHandleNormalCall s rm dummyMeta $ \c -> do
|
|
payload c @?= "Hello!"
|
|
checkMD "Server metadata mismatch" clientMD (metadata c)
|
|
return ("reply test", dummyMeta, StatusOk, "details string")
|
|
r @?= Right ()
|
|
|
|
testSSL :: TestTree
|
|
testSSL =
|
|
csTest' "request/response using SSL" client server
|
|
where
|
|
clientConf = stdClientConf
|
|
{clientSSLConfig = Just (ClientSSLConfig
|
|
(Just "tests/ssl/localhost.crt")
|
|
Nothing
|
|
Nothing)
|
|
}
|
|
client = TestClient clientConf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 10 "hi" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "reply test"
|
|
|
|
serverConf' = defServerConf
|
|
{ sslConfig = Just (ServerSSLConfig
|
|
Nothing
|
|
"tests/ssl/localhost.key"
|
|
"tests/ssl/localhost.crt"
|
|
SslDontRequestClientCertificate
|
|
Nothing)
|
|
}
|
|
server = TestServer serverConf' $ \s -> do
|
|
r <- U.serverHandleNormalCall s mempty $ \U.ServerCall{..} body -> do
|
|
body @?= "hi"
|
|
return ("reply test", mempty, StatusOk, "")
|
|
r @?= Right ()
|
|
|
|
-- NOTE: With auth plugin tests, if an auth plugin decides someone isn't
|
|
-- authenticated, then the call never happens from the perspective of
|
|
-- the server, so the server will continue to block waiting for a call. So, if
|
|
-- these tests hang forever, it's because auth failed and the server is still
|
|
-- waiting for a successfully authenticated call to come in.
|
|
|
|
testServerAuthProcessorCancel :: TestTree
|
|
testServerAuthProcessorCancel =
|
|
csTest' "request rejection by auth processor" client server
|
|
where
|
|
clientConf = stdClientConf
|
|
{clientSSLConfig = Just (ClientSSLConfig
|
|
(Just "tests/ssl/localhost.crt")
|
|
Nothing
|
|
Nothing)
|
|
}
|
|
client = TestClient clientConf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
r <- clientRequest c rm 10 "hi" mempty
|
|
-- TODO: using checkReqRslt on this first result causes the test to hang!
|
|
r @?= Left (GRPCIOBadStatusCode StatusUnauthenticated "denied!")
|
|
clientRequest c rm 10 "hi" [("foo","bar")] >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "reply test"
|
|
|
|
serverProcessor = Just $ \_ m -> do
|
|
let (status, details) = if M.member "foo" (unMap m)
|
|
then (StatusOk, "")
|
|
else (StatusUnauthenticated, "denied!")
|
|
return $ AuthProcessorResult mempty mempty status details
|
|
|
|
serverConf' = defServerConf
|
|
{ sslConfig = Just (ServerSSLConfig
|
|
Nothing
|
|
"tests/ssl/localhost.key"
|
|
"tests/ssl/localhost.crt"
|
|
SslDontRequestClientCertificate
|
|
serverProcessor)
|
|
}
|
|
server = TestServer serverConf' $ \s -> do
|
|
r <- U.serverHandleNormalCall s mempty $ \U.ServerCall{..} _body -> do
|
|
checkMD "Handler only sees requests with good metadata"
|
|
[("foo","bar")]
|
|
metadata
|
|
return ("reply test", mempty, StatusOk, "")
|
|
r @?= Right ()
|
|
|
|
testAuthMetadataTransfer :: TestTree
|
|
testAuthMetadataTransfer =
|
|
csTest' "Auth metadata changes sent from client to server" client server
|
|
where
|
|
plugin :: ClientMetadataCreate
|
|
plugin authMetaCtx = do
|
|
let authCtx = (channelAuthContext authMetaCtx)
|
|
|
|
addAuthProperty authCtx (AuthProperty "foo1" "bar1")
|
|
newProps <- getAuthProperties authCtx
|
|
let addedProp = find ((== "foo1") . authPropName) newProps
|
|
addedProp @?= Just (AuthProperty "foo1" "bar1")
|
|
return $ ClientMetadataCreateResult [("foo","bar")] StatusOk ""
|
|
clientConf = stdClientConf
|
|
{clientSSLConfig = Just (ClientSSLConfig
|
|
(Just "tests/ssl/localhost.crt")
|
|
Nothing
|
|
(Just plugin))
|
|
}
|
|
client = TestClient clientConf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 10 "hi" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "reply test"
|
|
|
|
serverProcessor :: Maybe ProcessMeta
|
|
serverProcessor = Just $ \authCtx m -> do
|
|
let expected = fromList [("foo","bar")]
|
|
|
|
props <- getAuthProperties authCtx
|
|
let clientProp = find ((== "foo1") . authPropName) props
|
|
assertBool "server plugin doesn't see auth properties set by client"
|
|
(clientProp == Nothing)
|
|
checkMD "server plugin sees metadata added by client plugin" expected m
|
|
return $ AuthProcessorResult mempty mempty StatusOk ""
|
|
|
|
serverConf' = defServerConf
|
|
{ sslConfig = Just (ServerSSLConfig
|
|
Nothing
|
|
"tests/ssl/localhost.key"
|
|
"tests/ssl/localhost.crt"
|
|
SslDontRequestClientCertificate
|
|
serverProcessor)
|
|
}
|
|
server = TestServer serverConf' $ \s -> do
|
|
r <- U.serverHandleNormalCall s mempty $ \U.ServerCall{..} body -> do
|
|
body @?= "hi"
|
|
return ("reply test", mempty, StatusOk, "")
|
|
r @?= Right ()
|
|
|
|
-- TODO: auth metadata doesn't propagate from parent calls to child calls.
|
|
-- Once we implement our own system for doing so, update this test and add it
|
|
-- to the tests list.
|
|
testAuthMetadataPropagate :: TestTree
|
|
testAuthMetadataPropagate = testCase "auth metadata inherited by children" $ do
|
|
c <- async client
|
|
s1 <- async server
|
|
s2 <- async server2
|
|
wait c
|
|
wait s1
|
|
wait s2
|
|
return ()
|
|
where
|
|
clientPlugin _ =
|
|
return $ ClientMetadataCreateResult [("foo","bar")] StatusOk ""
|
|
clientConf = stdClientConf
|
|
{clientSSLConfig = Just (ClientSSLConfig
|
|
(Just "tests/ssl/localhost.crt")
|
|
Nothing
|
|
(Just clientPlugin))
|
|
}
|
|
client = do
|
|
threadDelaySecs 3
|
|
withGRPC $ \g -> withClient g clientConf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 10 "hi" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "reply test"
|
|
|
|
server1ServerPlugin _ctx md = do
|
|
checkMD "server1 sees client's auth metadata." [("foo","bar")] md
|
|
-- TODO: add response meta to check, and consume meta to see what happens.
|
|
return $ AuthProcessorResult mempty mempty StatusOk ""
|
|
|
|
server1ServerConf = defServerConf
|
|
{sslConfig = Just (ServerSSLConfig
|
|
Nothing
|
|
"tests/ssl/localhost.key"
|
|
"tests/ssl/localhost.crt"
|
|
SslDontRequestClientCertificate
|
|
(Just server1ServerPlugin)),
|
|
methodsToRegisterNormal = ["/foo"]
|
|
}
|
|
|
|
server1ClientPlugin _ =
|
|
return $ ClientMetadataCreateResult [("foo1","bar1")] StatusOk ""
|
|
|
|
server1ClientConf = stdClientConf
|
|
{clientSSLConfig = Just (ClientSSLConfig
|
|
(Just "tests/ssl/localhost.crt")
|
|
Nothing
|
|
(Just server1ClientPlugin)),
|
|
clientServerPort = 50052
|
|
}
|
|
|
|
server = do
|
|
threadDelaySecs 2
|
|
withGRPC $ \g -> withServer g server1ServerConf $ \s ->
|
|
withClient g server1ClientConf $ \c -> do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty $ \call -> do
|
|
rmc <- clientRegisterMethodNormal c "/foo"
|
|
res <- clientRequestParent c (Just call) rmc 10 "hi" mempty
|
|
case res of
|
|
Left _ ->
|
|
error "got bad result from server2"
|
|
Right (NormalRequestResult{..}) ->
|
|
return (rspBody, mempty, StatusOk, "")
|
|
|
|
server2ServerPlugin _ctx md = do
|
|
print md
|
|
checkMD "server2 sees server1's auth metadata." [("foo1","bar1")] md
|
|
--TODO: this assert fails
|
|
checkMD "server2 sees client's auth metadata." [("foo","bar")] md
|
|
return $ AuthProcessorResult mempty mempty StatusOk ""
|
|
|
|
server2ServerConf = defServerConf
|
|
{sslConfig = Just (ServerSSLConfig
|
|
Nothing
|
|
"tests/ssl/localhost.key"
|
|
"tests/ssl/localhost.crt"
|
|
SslDontRequestClientCertificate
|
|
(Just server2ServerPlugin)),
|
|
methodsToRegisterNormal = ["/foo"],
|
|
port = 50052
|
|
}
|
|
|
|
server2 = withGRPC $ \g -> withServer g server2ServerConf $ \s -> do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty $ \_call -> do
|
|
return ("server2 reply", mempty, StatusOk, "")
|
|
|
|
testServerCancel :: TestTree
|
|
testServerCancel =
|
|
csTest "server cancel call" client server (["/foo"],[],[],[])
|
|
where
|
|
client c = do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
Left (GRPCIOBadStatusCode s _) <- clientRequest c rm 10 "" mempty
|
|
s @?= StatusCancelled
|
|
server s = do
|
|
let rm = head (normalMethods s)
|
|
r <- serverHandleNormalCall s rm mempty $ \c -> do
|
|
serverCallCancel c StatusCancelled ""
|
|
return (mempty, mempty, StatusCancelled, "")
|
|
r @?= Right ()
|
|
|
|
testServerStreaming :: TestTree
|
|
testServerStreaming =
|
|
csTest "server streaming" client server ([],[],["/feed"],[])
|
|
where
|
|
clientInitMD = [("client","initmd")]
|
|
serverInitMD = [("server","initmd")]
|
|
clientPay = "FEED ME!"
|
|
pays = ["ONE", "TWO", "THREE", "FOUR"] :: [ByteString]
|
|
|
|
client c = do
|
|
rm <- clientRegisterMethodServerStreaming c "/feed"
|
|
eea <- clientReader c rm 10 clientPay clientInitMD $ \_cc initMD recv -> do
|
|
checkMD "Server initial metadata mismatch" serverInitMD initMD
|
|
forM_ pays $ \p -> recv `is` Right (Just p)
|
|
recv `is` Right Nothing
|
|
eea @?= Right (dummyMeta, StatusOk, "dtls")
|
|
|
|
server s = do
|
|
let rm = head (sstreamingMethods s)
|
|
r <- serverWriter s rm serverInitMD $ \sc send -> do
|
|
checkMD "Server request metadata mismatch" clientInitMD (metadata sc)
|
|
payload sc @?= clientPay
|
|
forM_ pays $ \p -> send p `is` Right ()
|
|
return (dummyMeta, StatusOk, "dtls")
|
|
r @?= Right ()
|
|
|
|
-- TODO: these unregistered streaming tests are basically the same as the
|
|
-- registered ones. Reduce duplication.
|
|
-- TODO: Once client-side unregistered streaming functions are added, switch
|
|
-- to using them in these tests.
|
|
testServerStreamingUnregistered :: TestTree
|
|
testServerStreamingUnregistered =
|
|
csTest "unregistered server streaming" client server ([],[],[],[])
|
|
where
|
|
clientInitMD = [("client","initmd")]
|
|
serverInitMD = [("server","initmd")]
|
|
clientPay = "FEED ME!"
|
|
pays = ["ONE", "TWO", "THREE", "FOUR"] :: [ByteString]
|
|
|
|
client c = do
|
|
rm <- clientRegisterMethodServerStreaming c "/feed"
|
|
eea <- clientReader c rm 10 clientPay clientInitMD $ \_cc initMD recv -> do
|
|
checkMD "Server initial metadata mismatch" serverInitMD initMD
|
|
forM_ pays $ \p -> recv `is` Right (Just p)
|
|
recv `is` Right Nothing
|
|
eea @?= Right (dummyMeta, StatusOk, "dtls")
|
|
|
|
server s = U.withServerCallAsync s $ \call -> do
|
|
r <- U.serverWriter s call serverInitMD $ \sc send -> do
|
|
checkMD "Server request metadata mismatch" clientInitMD (metadata sc)
|
|
payload sc @?= clientPay
|
|
forM_ pays $ \p -> send p `is` Right ()
|
|
return (dummyMeta, StatusOk, "dtls")
|
|
r @?= Right ()
|
|
|
|
testClientStreaming :: TestTree
|
|
testClientStreaming =
|
|
csTest "client streaming" client server ([],["/slurp"],[],[])
|
|
where
|
|
clientInitMD = [("a","b")]
|
|
serverInitMD = [("x","y")]
|
|
trailMD = dummyMeta
|
|
serverRsp = "serverReader reply"
|
|
serverDtls = "deets"
|
|
serverStatus = StatusOk
|
|
pays = ["P_ONE", "P_TWO", "P_THREE"] :: [ByteString]
|
|
|
|
client c = do
|
|
rm <- clientRegisterMethodClientStreaming c "/slurp"
|
|
eea <- clientWriter c rm 10 clientInitMD $ \send -> do
|
|
-- liftIO $ checkMD "Server initial metadata mismatch" serverInitMD initMD
|
|
forM_ pays $ \p -> send p `is` Right ()
|
|
eea @?= Right (Just serverRsp, serverInitMD, trailMD, serverStatus, serverDtls)
|
|
|
|
server s = do
|
|
let rm = head (cstreamingMethods s)
|
|
eea <- serverReader s rm serverInitMD $ \sc recv -> do
|
|
checkMD "Client request metadata mismatch" clientInitMD (metadata sc)
|
|
forM_ pays $ \p -> recv `is` Right (Just p)
|
|
recv `is` Right Nothing
|
|
return (Just serverRsp, trailMD, serverStatus, serverDtls)
|
|
eea @?= Right ()
|
|
|
|
testClientStreamingUnregistered :: TestTree
|
|
testClientStreamingUnregistered =
|
|
csTest "unregistered client streaming" client server ([],[],[],[])
|
|
where
|
|
clientInitMD = [("a","b")]
|
|
serverInitMD = [("x","y")]
|
|
trailMD = dummyMeta
|
|
serverRsp = "serverReader reply"
|
|
serverDtls = "deets"
|
|
serverStatus = StatusOk
|
|
pays = ["P_ONE", "P_TWO", "P_THREE"] :: [ByteString]
|
|
|
|
client c = do
|
|
rm <- clientRegisterMethodClientStreaming c "/slurp"
|
|
eea <- clientWriter c rm 10 clientInitMD $ \send -> do
|
|
-- liftIO $ checkMD "Server initial metadata mismatch" serverInitMD initMD
|
|
forM_ pays $ \p -> send p `is` Right ()
|
|
eea @?= Right (Just serverRsp, serverInitMD, trailMD, serverStatus, serverDtls)
|
|
|
|
server s = U.withServerCallAsync s $ \call -> do
|
|
eea <- U.serverReader s call serverInitMD $ \sc recv -> do
|
|
checkMD "Client request metadata mismatch" clientInitMD (metadata sc)
|
|
forM_ pays $ \p -> recv `is` Right (Just p)
|
|
recv `is` Right Nothing
|
|
return (Just serverRsp, trailMD, serverStatus, serverDtls)
|
|
eea @?= Right ()
|
|
|
|
testBiDiStreaming :: TestTree
|
|
testBiDiStreaming =
|
|
csTest "bidirectional streaming" client server ([],[],[],["/bidi"])
|
|
where
|
|
clientInitMD = [("bidi-streaming","client")]
|
|
serverInitMD = [("bidi-streaming","server")]
|
|
trailMD = dummyMeta
|
|
serverStatus = StatusOk
|
|
serverDtls = "deets"
|
|
|
|
client c = do
|
|
rm <- clientRegisterMethodBiDiStreaming c "/bidi"
|
|
eea <- clientRW c rm 10 clientInitMD $ \_cc getMD recv send writesDone -> do
|
|
either clientFail (checkMD "Server rsp metadata mismatch" serverInitMD) =<< getMD
|
|
send "cw0" `is` Right ()
|
|
recv `is` Right (Just "sw0")
|
|
send "cw1" `is` Right ()
|
|
recv `is` Right (Just "sw1")
|
|
recv `is` Right (Just "sw2")
|
|
writesDone `is` Right ()
|
|
recv `is` Right Nothing
|
|
eea @?= Right (trailMD, serverStatus, serverDtls)
|
|
|
|
server s = do
|
|
let rm = head (bidiStreamingMethods s)
|
|
eea <- serverRW s rm serverInitMD $ \sc recv send -> do
|
|
checkMD "Client request metadata mismatch" clientInitMD (metadata sc)
|
|
recv `is` Right (Just "cw0")
|
|
send "sw0" `is` Right ()
|
|
recv `is` Right (Just "cw1")
|
|
send "sw1" `is` Right ()
|
|
send "sw2" `is` Right ()
|
|
recv `is` Right Nothing
|
|
return (trailMD, serverStatus, serverDtls)
|
|
eea @?= Right ()
|
|
|
|
testBiDiStreamingUnregistered :: TestTree
|
|
testBiDiStreamingUnregistered =
|
|
csTest "unregistered bidirectional streaming" client server ([],[],[],[])
|
|
where
|
|
clientInitMD = [("bidi-streaming","client")]
|
|
serverInitMD = [("bidi-streaming","server")]
|
|
trailMD = dummyMeta
|
|
serverStatus = StatusOk
|
|
serverDtls = "deets"
|
|
|
|
client c = do
|
|
rm <- clientRegisterMethodBiDiStreaming c "/bidi"
|
|
eea <- clientRW c rm 10 clientInitMD $ \_cc getMD recv send writesDone -> do
|
|
either clientFail (checkMD "Server rsp metadata mismatch" serverInitMD) =<< getMD
|
|
send "cw0" `is` Right ()
|
|
recv `is` Right (Just "sw0")
|
|
send "cw1" `is` Right ()
|
|
recv `is` Right (Just "sw1")
|
|
recv `is` Right (Just "sw2")
|
|
writesDone `is` Right ()
|
|
recv `is` Right Nothing
|
|
eea @?= Right (trailMD, serverStatus, serverDtls)
|
|
|
|
server s = U.withServerCallAsync s $ \call -> do
|
|
eea <- U.serverRW s call serverInitMD $ \sc recv send -> do
|
|
checkMD "Client request metadata mismatch" clientInitMD (metadata sc)
|
|
recv `is` Right (Just "cw0")
|
|
send "sw0" `is` Right ()
|
|
recv `is` Right (Just "cw1")
|
|
send "sw1" `is` Right ()
|
|
send "sw2" `is` Right ()
|
|
recv `is` Right Nothing
|
|
return (trailMD, serverStatus, serverDtls)
|
|
eea @?= Right ()
|
|
|
|
--------------------------------------------------------------------------------
|
|
-- Unregistered tests
|
|
|
|
testClientCall :: TestTree
|
|
testClientCall =
|
|
clientOnlyTest "create/destroy call" $ \c -> do
|
|
r <- U.withClientCall c "/foo" 10 $ const $ return $ Right ()
|
|
r @?= Right ()
|
|
|
|
testServerCall :: TestTree
|
|
testServerCall =
|
|
serverOnlyTest "create/destroy call" ([],[],[],[]) $ \s -> do
|
|
r <- U.withServerCall s $ const $ return $ Right ()
|
|
r @?= Left GRPCIOTimeout
|
|
|
|
testPayloadUnregistered :: TestTree
|
|
testPayloadUnregistered =
|
|
csTest "unregistered normal request/response" client server ([],[],[],[])
|
|
where
|
|
client c =
|
|
U.clientRequest c "/foo" 10 "Hello!" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "reply test"
|
|
details @?= "details string"
|
|
server s = do
|
|
r <- U.serverHandleNormalCall s mempty $ \U.ServerCall{..} body -> do
|
|
body @?= "Hello!"
|
|
callMethod @?= "/foo"
|
|
return ("reply test", mempty, StatusOk, "details string")
|
|
r @?= Right ()
|
|
|
|
testGoaway :: TestTree
|
|
testGoaway =
|
|
csTest "Client handles server shutdown gracefully"
|
|
client
|
|
server
|
|
(["/foo"],[],[],[])
|
|
where
|
|
client c = do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 10 "" mempty
|
|
clientRequest c rm 10 "" mempty
|
|
eer <- clientRequest c rm 1 "" mempty
|
|
assertBool "Client handles server shutdown gracefully" $ case eer of
|
|
Left (GRPCIOBadStatusCode StatusUnavailable _) -> True
|
|
Left (GRPCIOBadStatusCode StatusDeadlineExceeded "Deadline Exceeded") -> True
|
|
Left GRPCIOTimeout -> True
|
|
_ -> False
|
|
|
|
server s = do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty dummyHandler
|
|
serverHandleNormalCall s rm mempty dummyHandler
|
|
return ()
|
|
|
|
testSlowServer :: TestTree
|
|
testSlowServer =
|
|
csTest "Client handles slow server response" client server (["/foo"],[],[],[])
|
|
where
|
|
client c = do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
result <- clientRequest c rm 1 "" mempty
|
|
result @?= Left (GRPCIOBadStatusCode StatusDeadlineExceeded "Deadline Exceeded")
|
|
server s = do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty $ \_ -> do
|
|
threadDelay (2*10^(6 :: Int))
|
|
return dummyResp
|
|
return ()
|
|
|
|
testServerCallExpirationCheck :: TestTree
|
|
testServerCallExpirationCheck =
|
|
csTest "Check for call expiration" client server (["/foo"],[],[],[])
|
|
where
|
|
client c = do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
void $ clientRequest c rm 3 "" mempty
|
|
server s = do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty $ \c -> do
|
|
exp1 <- serverCallIsExpired c
|
|
assertBool "Call isn't expired when handler starts" $ not exp1
|
|
threadDelaySecs 1
|
|
exp2 <- serverCallIsExpired c
|
|
assertBool "Call isn't expired after 1 second" $ not exp2
|
|
threadDelaySecs 3
|
|
exp3 <- serverCallIsExpired c
|
|
assertBool "Call is expired after 4 seconds" exp3
|
|
return dummyResp
|
|
return ()
|
|
|
|
testCustomUserAgent :: TestTree
|
|
testCustomUserAgent =
|
|
csTest' "Server sees custom user agent prefix/suffix" client server
|
|
where
|
|
clientArgs = [UserAgentPrefix "prefix!", UserAgentSuffix "suffix!"]
|
|
client =
|
|
TestClient (ClientConfig "localhost" 50051 clientArgs Nothing Nothing) $
|
|
\c -> do rm <- clientRegisterMethodNormal c "/foo"
|
|
void $ clientRequest c rm 4 "" mempty
|
|
server = TestServer (serverConf (["/foo"],[],[],[])) $ \s -> do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty $ \c -> do
|
|
ua <- case toList $ (unMap $ metadata c) M.! "user-agent" of
|
|
[] -> fail "user-agent missing from metadata."
|
|
[ua] -> return ua
|
|
_ -> fail "multiple user-agent keys."
|
|
assertBool "User agent prefix is present" $ isPrefixOf "prefix!" ua
|
|
assertBool "User agent suffix is present" $ isSuffixOf "suffix!" ua
|
|
return dummyResp
|
|
return ()
|
|
|
|
testClientCompression :: TestTree
|
|
testClientCompression =
|
|
csTest' "client-only compression: no errors" client server
|
|
where
|
|
client =
|
|
TestClient (ClientConfig
|
|
"localhost"
|
|
50051
|
|
[CompressionAlgArg GrpcCompressDeflate]
|
|
Nothing
|
|
Nothing) $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
void $ clientRequest c rm 1 "hello" mempty
|
|
server = TestServer (serverConf (["/foo"],[],[],[])) $ \s -> do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm mempty $ \c -> do
|
|
payload c @?= "hello"
|
|
return dummyResp
|
|
return ()
|
|
|
|
testClientServerCompression :: TestTree
|
|
testClientServerCompression =
|
|
csTest' "client/server compression: no errors" client server
|
|
where
|
|
cconf = ClientConfig "localhost"
|
|
50051
|
|
[CompressionAlgArg GrpcCompressDeflate]
|
|
Nothing
|
|
Nothing
|
|
client = TestClient cconf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 1 "hello" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "hello"
|
|
details @?= ""
|
|
initMD @?= dummyMeta
|
|
trailMD @?= dummyMeta
|
|
return ()
|
|
sconf = ServerConfig "localhost"
|
|
50051
|
|
["/foo"] [] [] []
|
|
[CompressionAlgArg GrpcCompressDeflate]
|
|
Nothing
|
|
server = TestServer sconf $ \s -> do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm dummyMeta $ \sc -> do
|
|
payload sc @?= "hello"
|
|
return ("hello", dummyMeta, StatusOk, StatusDetails "")
|
|
return ()
|
|
|
|
testClientServerCompressionLvl :: TestTree
|
|
testClientServerCompressionLvl =
|
|
csTest' "client/server compression: no errors" client server
|
|
where
|
|
cconf = ClientConfig "localhost"
|
|
50051
|
|
[CompressionLevelArg GrpcCompressLevelHigh]
|
|
Nothing
|
|
Nothing
|
|
client = TestClient cconf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 1 "hello" mempty >>= do
|
|
checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= "hello"
|
|
details @?= ""
|
|
initMD @?= dummyMeta
|
|
trailMD @?= dummyMeta
|
|
return ()
|
|
sconf = ServerConfig "localhost"
|
|
50051
|
|
["/foo"] [] [] []
|
|
[CompressionLevelArg GrpcCompressLevelLow]
|
|
Nothing
|
|
server = TestServer sconf $ \s -> do
|
|
let rm = head (normalMethods s)
|
|
serverHandleNormalCall s rm dummyMeta $ \sc -> do
|
|
payload sc @?= "hello"
|
|
return ("hello", dummyMeta, StatusOk, StatusDetails "")
|
|
return ()
|
|
|
|
testClientMaxReceiveMessageLengthChannelArg :: TestTree
|
|
testClientMaxReceiveMessageLengthChannelArg = do
|
|
testGroup "max receive message length channel arg (client channel)"
|
|
[ csTest' "payload size < small bound succeeds" shouldSucceed server
|
|
, csTest' "payload size > small bound fails" shouldFail server
|
|
]
|
|
where
|
|
-- The server always sends a 4-byte payload
|
|
pay = "four"
|
|
server = TestServer (ServerConfig "localhost" 50051 ["/foo"] [] [] [] [] Nothing) $ \s -> do
|
|
let rm = head (normalMethods s)
|
|
void $ serverHandleNormalCall s rm mempty $ \sc -> do
|
|
payload sc @?= pay
|
|
pure (pay, mempty, StatusOk, StatusDetails "")
|
|
|
|
clientMax n k = TestClient conf $ \c -> do
|
|
rm <- clientRegisterMethodNormal c "/foo"
|
|
clientRequest c rm 1 pay mempty >>= k
|
|
where
|
|
conf = ClientConfig "localhost" 50051 [MaxReceiveMessageLength n] Nothing Nothing
|
|
|
|
-- Expect success when the max recv payload size is set to 4 bytes, and we
|
|
-- are sent 4.
|
|
shouldSucceed = clientMax 4 $ checkReqRslt $ \NormalRequestResult{..} -> do
|
|
rspCode @?= StatusOk
|
|
rspBody @?= pay
|
|
details @?= ""
|
|
|
|
-- Expect failure when the max recv payload size is set to 3 bytes, and we
|
|
-- are sent 4.
|
|
shouldFail = clientMax 3 $ \case
|
|
Left (GRPCIOBadStatusCode StatusResourceExhausted _)
|
|
-> pure ()
|
|
rsp
|
|
-> clientFail ("Expected failure response, but got: " ++ show rsp)
|
|
|
|
--------------------------------------------------------------------------------
|
|
-- Utilities and helpers
|
|
|
|
is :: (Eq a, Show a, MonadIO m) => m a -> a -> m ()
|
|
is act x = act >>= liftIO . (@?= x)
|
|
|
|
dummyMeta :: MetadataMap
|
|
dummyMeta = [("foo","bar")]
|
|
|
|
dummyResp :: (ByteString, MetadataMap, StatusCode, StatusDetails)
|
|
dummyResp = ("", mempty, StatusOk, StatusDetails "")
|
|
|
|
dummyHandler :: ServerCall a
|
|
-> IO (ByteString, MetadataMap, StatusCode, StatusDetails)
|
|
dummyHandler _ = return dummyResp
|
|
|
|
dummyResult' :: StatusDetails
|
|
-> IO (ByteString, MetadataMap, StatusCode, StatusDetails)
|
|
dummyResult' = return . (mempty, mempty, StatusOk, )
|
|
|
|
nop :: Monad m => a -> m ()
|
|
nop = const (return ())
|
|
|
|
serverOnlyTest :: TestName
|
|
-> ([MethodName],[MethodName],[MethodName],[MethodName])
|
|
-> (Server -> IO ())
|
|
-> TestTree
|
|
serverOnlyTest nm ms =
|
|
testCase ("Server - " ++ nm) . runTestServer . TestServer (serverConf ms)
|
|
|
|
clientOnlyTest :: TestName -> (Client -> IO ()) -> TestTree
|
|
clientOnlyTest nm =
|
|
testCase ("Client - " ++ nm) . runTestClient . stdTestClient
|
|
|
|
csTest :: TestName
|
|
-> (Client -> IO ())
|
|
-> (Server -> IO ())
|
|
-> ([MethodName],[MethodName],[MethodName],[MethodName])
|
|
-> TestTree
|
|
csTest nm c s ms =
|
|
csTest' nm (stdTestClient c) (TestServer (serverConf ms) s)
|
|
|
|
csTest' :: TestName -> TestClient -> TestServer -> TestTree
|
|
csTest' nm tc ts =
|
|
testCase ("Client/Server - " ++ nm)
|
|
$ void (s `concurrently` c)
|
|
where
|
|
-- We use a small delay to give the server a head start
|
|
c = threadDelay 100000 >> runTestClient tc
|
|
s = runTestServer ts
|
|
|
|
-- | @checkMD msg expected actual@ fails when keys from @expected@ are not in
|
|
-- @actual@, or when values differ for matching keys.
|
|
checkMD :: String -> MetadataMap -> MetadataMap -> Assertion
|
|
checkMD desc expected actual =
|
|
assertEqual desc expected' (actual' `S.intersection` expected')
|
|
where
|
|
expected' = fromList . toList $ expected
|
|
actual' = fromList . toList $ actual
|
|
|
|
checkReqRslt :: Show a => (b -> Assertion) -> Either a b -> Assertion
|
|
checkReqRslt = either clientFail
|
|
|
|
-- | The consumer which asserts that the next value it consumes is equal to the
|
|
-- given value; string parameter used as in 'assertEqual'.
|
|
assertConsumeEq :: (Eq a, Show a) => String -> a -> P.Consumer a IO ()
|
|
assertConsumeEq s v = P.lift . assertEqual s v =<< P.await
|
|
|
|
clientFail :: Show a => a -> Assertion
|
|
clientFail = assertFailure . ("Client error: " ++). show
|
|
|
|
data TestClient = TestClient ClientConfig (Client -> IO ())
|
|
|
|
runTestClient :: TestClient -> IO ()
|
|
runTestClient (TestClient conf f) =
|
|
runManaged $ mgdGRPC >>= mgdClient conf >>= liftIO . f
|
|
|
|
stdTestClient :: (Client -> IO ()) -> TestClient
|
|
stdTestClient = TestClient stdClientConf
|
|
|
|
stdClientConf :: ClientConfig
|
|
stdClientConf = ClientConfig "localhost" 50051 [] Nothing Nothing
|
|
|
|
data TestServer = TestServer ServerConfig (Server -> IO ())
|
|
|
|
runTestServer :: TestServer -> IO ()
|
|
runTestServer (TestServer conf f) =
|
|
runManaged $ mgdGRPC >>= mgdServer conf >>= liftIO . f
|
|
|
|
defServerConf :: ServerConfig
|
|
defServerConf = ServerConfig "localhost" 50051 [] [] [] [] [] Nothing
|
|
|
|
serverConf :: ([MethodName],[MethodName],[MethodName],[MethodName])
|
|
-> ServerConfig
|
|
serverConf (ns, cs, ss, bs) =
|
|
defServerConf {methodsToRegisterNormal = ns,
|
|
methodsToRegisterClientStreaming = cs,
|
|
methodsToRegisterServerStreaming = ss,
|
|
methodsToRegisterBiDiStreaming = bs}
|
|
|
|
mgdGRPC :: Managed GRPC
|
|
mgdGRPC = managed withGRPC
|
|
|
|
mgdClient :: ClientConfig -> GRPC -> Managed Client
|
|
mgdClient conf g = managed $ withClient g conf
|
|
|
|
mgdServer :: ServerConfig -> GRPC -> Managed Server
|
|
mgdServer conf g = managed $ withServer g conf
|