diff --git a/cabal.project b/cabal.project index 12e47d21..f6279c21 100644 --- a/cabal.project +++ b/cabal.project @@ -3,6 +3,7 @@ packages: , ./tutorials/quickstart , ./tutorials/basics , ./tutorials/lowlevel + , ./tutorials/metadata package grapesy tests: True diff --git a/grapesy/demo-server/Demo/Server/Service/Greeter.hs b/grapesy/demo-server/Demo/Server/Service/Greeter.hs index 48098b22..33d14abc 100644 --- a/grapesy/demo-server/Demo/Server/Service/Greeter.hs +++ b/grapesy/demo-server/Demo/Server/Service/Greeter.hs @@ -41,7 +41,7 @@ sayHelloStreamReply = mkRpcHandlerNoInitialMetadata $ \call -> do setResponseInitialMetadata call $ SayHelloMetadata (Just "initial-md-value") -- The client expects the metadata well before the first output - _ <- initiateResponse call + initiateResponse call req <- recvFinalInput call diff --git a/grapesy/interop/Interop/Server/Common.hs b/grapesy/interop/Interop/Server/Common.hs index 9aa73127..38df58ba 100644 --- a/grapesy/interop/Interop/Server/Common.hs +++ b/grapesy/interop/Interop/Server/Common.hs @@ -41,7 +41,7 @@ constructResponseMetadata call = do -- Send initial metadata setResponseInitialMetadata call initMeta - _initiated <- initiateResponse call + initiateResponse call -- Return the final metadata to be sent at the end of the call return trailMeta diff --git a/grapesy/src/Network/GRPC/Client/Call.hs b/grapesy/src/Network/GRPC/Client/Call.hs index 8fe8fce9..45bd7121 100644 --- a/grapesy/src/Network/GRPC/Client/Call.hs +++ b/grapesy/src/Network/GRPC/Client/Call.hs @@ -105,7 +105,7 @@ data Call rpc = SupportsClientRpc rpc => Call { -- If there are still /inbound/ messages upon leaving the scope of 'withRPC' no -- exception is raised (but the call is nonetheless still closed, and the server -- handler will be informed that the client has disappeared). -withRPC :: forall m rpc a. +withRPC :: forall rpc m a. (MonadMask m, MonadIO m, SupportsClientRpc rpc, HasCallStack) => Connection -> CallParams rpc -> Proxy rpc -> (Call rpc -> m a) -> m a withRPC conn callParams proxy k = fmap fst $ @@ -386,7 +386,7 @@ sendInputWithMeta Call{callChannel} msg = liftIO $ do -- 'GrpcStatus' here: a status of 'GrpcOk' carries no information, and any other -- status will result in a 'GrpcException'. Calling 'recvOutput' again after -- receiving the trailers is a bug and results in a 'RecvAfterFinal' exception. -recvOutput :: forall m rpc. +recvOutput :: forall rpc m. (MonadIO m, HasCallStack) => Call rpc -> m (StreamElem (ResponseTrailingMetadata rpc) (Output rpc)) @@ -444,8 +444,10 @@ recvOutputWithMeta = recvBoth -- returns any replies. -- * The response metadata /will/ be available before the first output from the -- server, and may indeed be available /well/ before. -recvResponseMetadata :: forall rpc. Call rpc -> IO (ResponseMetadata rpc) -recvResponseMetadata call@Call{} = +recvResponseMetadata :: forall rpc m. + MonadIO m + => Call rpc -> m (ResponseMetadata rpc) +recvResponseMetadata call@Call{} = liftIO $ recvInitialResponse call >>= aux where aux :: @@ -472,12 +474,13 @@ recvResponseMetadata call@Call{} = -- rather than thrown as an exception. -- -- Most applications will never need to use this function. -recvInitialResponse :: - Call rpc - -> IO ( Either (TrailersOnly' HandledSynthesized) +recvInitialResponse :: forall rpc m. + MonadIO m + => Call rpc + -> m ( Either (TrailersOnly' HandledSynthesized) (ResponseHeaders' HandledSynthesized) ) -recvInitialResponse Call{callChannel} = +recvInitialResponse Call{callChannel} = liftIO $ fmap inbHeaders <$> Session.getInboundHeaders callChannel {------------------------------------------------------------------------------- @@ -516,7 +519,7 @@ sendEndOfInput call = sendInput call $ NoMoreElems NoMetadata -- of \"Trailers-Only\" amounts to a protocol error; if the server /does/ use -- \"Trailers-Only\", this throws a 'ProtoclException' -- ('UnexpectedTrailersOnly'). -recvResponseInitialMetadata :: forall m rpc. +recvResponseInitialMetadata :: forall rpc m. MonadIO m => Call rpc -> m (ResponseInitialMetadata rpc) @@ -534,7 +537,7 @@ recvResponseInitialMetadata call@Call{} = liftIO $ do -- | Receive the next output -- -- Throws 'ProtocolException' if there are no more outputs. -recvNextOutput :: forall m rpc. +recvNextOutput :: forall rpc m. (MonadIO m, HasCallStack) => Call rpc -> m (Output rpc) recvNextOutput call@Call{} = liftIO $ do @@ -555,7 +558,7 @@ recvNextOutput call@Call{} = liftIO $ do -- -- NOTE: If the first output we receive from the server is not marked as final, -- we will block until we receive the end-of-stream indication. -recvFinalOutput :: forall m rpc. +recvFinalOutput :: forall rpc m. (MonadIO m, HasCallStack) => Call rpc -> m (Output rpc, ResponseTrailingMetadata rpc) @@ -577,7 +580,7 @@ recvFinalOutput call@Call{} = liftIO $ do -- | Receive trailers -- -- Throws 'ProtocolException' if we received an output. -recvTrailers :: forall m rpc. +recvTrailers :: forall rpc m. (MonadIO m, HasCallStack) => Call rpc -> m (ResponseTrailingMetadata rpc) recvTrailers call@Call{} = liftIO $ do @@ -594,7 +597,7 @@ recvTrailers call@Call{} = liftIO $ do Internal auxiliary: deal with final message -------------------------------------------------------------------------------} -recvBoth :: forall m rpc. +recvBoth :: forall rpc m. (HasCallStack, MonadIO m) => Call rpc -> m (StreamElem ProperTrailers' (InboundMeta, Output rpc)) @@ -612,7 +615,7 @@ recvBoth Call{callChannel} = liftIO $ flatten (Right streamElem) = streamElem -recvEither :: forall m rpc. +recvEither :: forall rpc m. (HasCallStack, MonadIO m) => Call rpc -> m (Either ProperTrailers' (InboundMeta, Output rpc)) diff --git a/grapesy/src/Network/GRPC/Common/StreamElem.hs b/grapesy/src/Network/GRPC/Common/StreamElem.hs index 1c517ac6..959aa632 100644 --- a/grapesy/src/Network/GRPC/Common/StreamElem.hs +++ b/grapesy/src/Network/GRPC/Common/StreamElem.hs @@ -141,7 +141,7 @@ whileNext_ f g = go FinalElem a b -> g a >> return b NoMoreElems b -> return b --- | Invoke the callback until it returns 'NoNextElem', collecting results +-- | Invoke the callback until 'FinalElem' or 'NoMoreElems', collecting results collect :: forall m b a. Monad m => m (StreamElem b a) -> m ([a], b) collect f = first reverse . swap <$> flip runStateT [] aux diff --git a/grapesy/src/Network/GRPC/Server/Call.hs b/grapesy/src/Network/GRPC/Server/Call.hs index f848da5e..e8ddf741 100644 --- a/grapesy/src/Network/GRPC/Server/Call.hs +++ b/grapesy/src/Network/GRPC/Server/Call.hs @@ -401,8 +401,8 @@ sendOutputWithMeta :: forall rpc. -> StreamElem (ResponseTrailingMetadata rpc) (OutboundMeta, Output rpc) -> IO () sendOutputWithMeta call@Call{callChannel} msg = do - _updated <- initiateResponse call - msg' <- bitraverse mkTrailers return msg + initiateResponse call + msg' <- bitraverse mkTrailers return msg Session.send callChannel msg' -- This /must/ be called before leaving the scope of 'runHandler' (or we @@ -480,14 +480,16 @@ setResponseInitialMetadata Call{ callResponseMetadata Low-level API -------------------------------------------------------------------------------} --- | Initiate the response +-- | Initiate the response. -- --- This will cause the initial response metadata to be sent --- (see also 'setResponseMetadata'). +-- This will cause the initial response metadata to be sent (see also +-- 'setResponseMetadata'). -- --- Returns 'False' if the response was already initiated. -initiateResponse :: HasCallStack => Call rpc -> IO Bool -initiateResponse Call{callResponseKickoff} = +-- Does nothing if the response was already initated (that is, the response +-- headers, or trailers in the case of 'sendTrailersOnly', have already been +-- sent). +initiateResponse :: HasCallStack => Call rpc -> IO () +initiateResponse Call{callResponseKickoff} = void $ atomically $ tryPutTMVar callResponseKickoff $ KickoffRegular callStack -- | Use the gRPC @Trailers-Only@ case for non-error responses diff --git a/grapesy/test-grapesy/Test/Driver/Dialogue/Execution.hs b/grapesy/test-grapesy/Test/Driver/Dialogue/Execution.hs index 441a66f0..c07811ae 100644 --- a/grapesy/test-grapesy/Test/Driver/Dialogue/Execution.hs +++ b/grapesy/test-grapesy/Test/Driver/Dialogue/Execution.hs @@ -380,7 +380,7 @@ serverLocal clock call = \(LocalSteps steps) -> do case action of Initiate metadata -> liftIO $ do Server.setResponseInitialMetadata call metadata - void $ Server.initiateResponse call + Server.initiateResponse call return True Send x -> do peerHealth <- get diff --git a/tutorials/lowlevel/app/Server.hs b/tutorials/lowlevel/app/Server.hs index 4ac4ef1a..6d3a3bb9 100644 --- a/tutorials/lowlevel/app/Server.hs +++ b/tutorials/lowlevel/app/Server.hs @@ -68,6 +68,16 @@ methods db = $ RawMethod (mkRpcHandler $ routeChat ) $ NoMoreMethods + +-- Alternative way to define the handlers, avoiding 'fromMethods' +_handlers :: DB -> [SomeRpcHandler IO] +_handlers db = [ + someRpcHandler . mkRpcHandler $ getFeature db + , someRpcHandler . mkRpcHandler $ listFeatures db + , someRpcHandler . mkRpcHandler $ recordRoute db + , someRpcHandler . mkRpcHandler $ routeChat + ] + main :: IO () main = do db <- getDB diff --git a/tutorials/metadata/LICENSE b/tutorials/metadata/LICENSE new file mode 100644 index 00000000..54362a91 --- /dev/null +++ b/tutorials/metadata/LICENSE @@ -0,0 +1,31 @@ +Copyright (c) 2023-2024, Well-Typed LLP and Anduril Industries Inc. + +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + + * Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following + disclaimer in the documentation and/or other materials provided + with the distribution. + + * Neither the name of Well-Typed LLP, the name of Anduril + Industries Inc., nor the names of other contributors may be + used to endorse or promote products derived from this software + without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/tutorials/metadata/README.md b/tutorials/metadata/README.md new file mode 100644 index 00000000..0058acb8 --- /dev/null +++ b/tutorials/metadata/README.md @@ -0,0 +1 @@ +# Metadata tutorial diff --git a/tutorials/metadata/Setup.hs b/tutorials/metadata/Setup.hs new file mode 100644 index 00000000..bf45b62e --- /dev/null +++ b/tutorials/metadata/Setup.hs @@ -0,0 +1,3 @@ +import Data.ProtoLens.Setup + +main = defaultMainGeneratingProtos "proto" \ No newline at end of file diff --git a/tutorials/metadata/app/Client.hs b/tutorials/metadata/app/Client.hs new file mode 100644 index 00000000..b24cbdde --- /dev/null +++ b/tutorials/metadata/app/Client.hs @@ -0,0 +1,98 @@ +module Client (main) where + +import Control.Monad.Catch +import Control.Monad.State +import Crypto.Hash.SHA256 qualified as SHA256 +import Data.ByteString (ByteString) +import Data.ByteString qualified as BS +import Data.Text qualified as Text +import System.Environment +import System.IO + +import Network.GRPC.Client +import Network.GRPC.Common +import Network.GRPC.Common.Protobuf +import Network.GRPC.Common.StreamElem qualified as StreamElem + +import Proto.API.Fileserver + +{------------------------------------------------------------------------------- + Very simple progress bar +-------------------------------------------------------------------------------} + +newtype ProgressT m a = WrapProgressT { + unwrapProgressT :: StateT [Integer] m a + } + deriving newtype (Functor, Applicative, Monad, MonadTrans, MonadIO) + +instance MonadState s m => MonadState s (ProgressT m) where + state = lift . state + +runProgressT :: (MonadMask m, MonadIO m) => Integer -> ProgressT m a -> m a +runProgressT totalSize = + (`finally` liftIO (putStrLn " done")) + . flip evalStateT dots + . unwrapProgressT + where + numDots = 50 + stepSize = totalSize `div` fromIntegral numDots + dots = take numDots $ iterate (+ stepSize) stepSize + +updateProgressBar :: forall m. MonadIO m => Integer -> ProgressT m () +updateProgressBar currentSize = do + WrapProgressT $ StateT go + liftIO $ hFlush stdout + where + go :: [Integer] -> m ((), [Integer]) + go [] = return ((), []) + go (d:ds) + | d <= currentSize = liftIO (putChar '.') >> go ds + | otherwise = return ((), d:ds) + +{------------------------------------------------------------------------------- + RPC +-------------------------------------------------------------------------------} + +processPartial :: + Handle + -> Proto Partial + -> ProgressT (StateT (SHA256.Ctx, Integer) IO) () +processPartial h partial = do + liftIO $ BS.hPut h chunk + accSize <- state $ \(ctx, currentSize) -> + let currentSize' = currentSize + fromIntegral (BS.length chunk) + in (currentSize', (SHA256.update ctx chunk, currentSize')) + updateProgressBar accSize + where + chunk :: ByteString + chunk = partial ^. #chunk + +download :: Connection -> String -> String -> IO () +download conn inp out = do + withRPC conn def (Proxy @(Protobuf Fileserver "download")) $ \call -> do + sendFinalInput call $ defMessage & #name .~ Text.pack inp + + -- Wait for initial metadata, telling us how big the file is + DownloadStart{downloadSize} <- recvResponseInitialMetadata call + + -- Process each chunk + (DownloadDone{downloadHash = theirHash}, (ourHash, _)) <- + withFile out WriteMode $ \h -> + flip runStateT (SHA256.init, 0) . runProgressT downloadSize $ + StreamElem.whileNext_ (recvOutput call) (processPartial h) + + -- Check hash + putStrLn $ "Hash match: " ++ show (theirHash == SHA256.finalize ourHash) + +{------------------------------------------------------------------------------- + Main application +-------------------------------------------------------------------------------} + +main :: IO () +main = do + [inp, out] <- getArgs + withConnection def server $ \conn -> do + download conn inp out + where + server :: Server + server = ServerInsecure $ Address "127.0.0.1" defaultInsecurePort Nothing diff --git a/tutorials/metadata/app/Server.hs b/tutorials/metadata/app/Server.hs new file mode 100644 index 00000000..b2a749ad --- /dev/null +++ b/tutorials/metadata/app/Server.hs @@ -0,0 +1,72 @@ +module Server (main) where + +import Crypto.Hash.SHA256 qualified as SHA256 +import Data.ByteString qualified as BS +import Data.ByteString.Builder.Extra (defaultChunkSize) +import Data.Text qualified as Text +import System.Directory (getFileSize) +import System.IO + +import Network.GRPC.Common +import Network.GRPC.Common.Protobuf +import Network.GRPC.Server +import Network.GRPC.Server.Protobuf +import Network.GRPC.Server.Run +import Network.GRPC.Server.StreamType + +import Proto.API.Fileserver + +{------------------------------------------------------------------------------- + Individual handlers +-------------------------------------------------------------------------------} + +-- Note: this just allows to download any file, not intended to be secure! +download :: Call (Protobuf Fileserver "download") -> IO () +download call = do + req :: Proto File <- recvFinalInput call + let fp :: FilePath + fp = Text.unpack (req ^. #name) + + fileSize <- getFileSize fp + setResponseInitialMetadata call $ DownloadStart fileSize + initiateResponse call + + withFile fp ReadMode $ \h -> do + let loop :: SHA256.Ctx -> IO () + loop ctx = do + chunk <- BS.hGet h defaultChunkSize + eof <- hIsEOF h + + let resp :: Proto Partial + resp = defMessage & #chunk .~ chunk + + ctx' :: SHA256.Ctx + ctx' = SHA256.update ctx chunk + + if eof then + sendFinalOutput call (resp, DownloadDone $ SHA256.finalize ctx') + else do + sendNextOutput call resp + loop ctx' + + loop SHA256.init + +{------------------------------------------------------------------------------- + Server top-level +-------------------------------------------------------------------------------} + +methods :: Methods IO (ProtobufMethodsOf Fileserver) +methods = + RawMethod (mkRpcHandlerNoInitialMetadata download) + $ NoMoreMethods + + +main :: IO () +main = + runServerWithHandlers def config $ fromMethods methods + where + config :: ServerConfig + config = ServerConfig { + serverInsecure = Just (InsecureConfig Nothing defaultInsecurePort) + , serverSecure = Nothing + } diff --git a/tutorials/metadata/metadata.cabal b/tutorials/metadata/metadata.cabal new file mode 100644 index 00000000..36727661 --- /dev/null +++ b/tutorials/metadata/metadata.cabal @@ -0,0 +1,103 @@ +cabal-version: 3.0 +name: metadata +synopsis: Metadata tutorial for grapesy +version: 0.1.0 +license: BSD-3-Clause +license-file: LICENSE +author: Edsko de Vries +maintainer: edsko@well-typed.com +build-type: Custom +extra-source-files: proto/fileserver.proto +tested-with: GHC==8.10.7 + , GHC==9.2.8 + , GHC==9.4.8 + , GHC==9.6.4 + , GHC==9.8.2 + +custom-setup + setup-depends: + base >= 4.14 && < 5 + , Cabal >= 3.0 && < 4 + , proto-lens-setup >= 0.4 && < 0.5 + +common lang + build-depends: base >= 4.14 && < 5 + default-language: Haskell2010 + ghc-options: -Wall + + if impl(ghc >= 9.0) + ghc-options: + -Wunused-packages + + default-extensions: + DataKinds + DerivingStrategies + FlexibleInstances + GeneralizedNewtypeDeriving + ImportQualifiedPost + MultiParamTypeClasses + NamedFieldPuns + OverloadedLabels + OverloadedStrings + ScopedTypeVariables + TypeApplications + TypeFamilies + UndecidableInstances + +library + import: lang + hs-source-dirs: src + build-tool-depends: proto-lens-protoc:proto-lens-protoc + + build-depends: + , bytestring >= 0.10 && < 0.13 + , exceptions >= 0.10 && < 0.11 + , grapesy >= 0.1 && < 0.2 + , proto-lens-runtime >= 0.7 && < 0.8 + exposed-modules: + Proto.API.Fileserver + other-modules: + Proto.Fileserver + autogen-modules: + Proto.Fileserver + +executable fileserver_server + import: lang + main-is: Server.hs + hs-source-dirs: app + ghc-options: -main-is Server + + build-depends: + -- internal + , metadata + build-depends: + -- inherited + , bytestring + , cryptohash-sha256 + , grapesy + build-depends: + -- additional + , cryptohash-sha256 >= 0.11 && < 0.12 + , directory >= 1.3 && < 1.4 + , text >= 1.2 && < 2.2 + +executable fileserver_client + import: lang + main-is: Client.hs + hs-source-dirs: app + ghc-options: -main-is Client + + build-depends: + -- internal + , metadata + build-depends: + -- inherited + , bytestring + , cryptohash-sha256 + , exceptions + , grapesy + build-depends: + -- additional + , cryptohash-sha256 >= 0.11 && < 0.12 + , mtl >= 2.2 && < 2.4 + , text >= 1.2 && < 2.2 diff --git a/tutorials/metadata/proto/fileserver.proto b/tutorials/metadata/proto/fileserver.proto new file mode 100644 index 00000000..e15c2d7b --- /dev/null +++ b/tutorials/metadata/proto/fileserver.proto @@ -0,0 +1,15 @@ +syntax = "proto3"; + +package fileserver; + +service Fileserver { + rpc Download (File) returns (stream Partial) {} +} + +message File { + string name = 1; +} + +message Partial { + bytes chunk = 1; +} diff --git a/tutorials/metadata/src/Proto/API/Fileserver.hs b/tutorials/metadata/src/Proto/API/Fileserver.hs new file mode 100644 index 00000000..ac808a22 --- /dev/null +++ b/tutorials/metadata/src/Proto/API/Fileserver.hs @@ -0,0 +1,73 @@ +module Proto.API.Fileserver ( + module Proto.Fileserver + -- * Metadata + , DownloadStart(..) + , DownloadDone(..) + ) where + +import Control.Monad.Catch (throwM) +import Data.ByteString (ByteString) +import Data.ByteString.Char8 qualified as C8 +import Text.Read (readMaybe) + +import Network.GRPC.Common +import Network.GRPC.Common.Protobuf + +import Proto.Fileserver + +{------------------------------------------------------------------------------- + Metadata +-------------------------------------------------------------------------------} + +type instance RequestMetadata (Protobuf Fileserver "download") = NoMetadata +type instance ResponseInitialMetadata (Protobuf Fileserver "download") = DownloadStart +type instance ResponseTrailingMetadata (Protobuf Fileserver "download") = DownloadDone + +data DownloadStart = DownloadStart { + downloadSize :: Integer + } + deriving stock (Show) + +data DownloadDone = DownloadDone { + downloadHash :: ByteString + } + deriving stock (Show) + +{------------------------------------------------------------------------------- + Serialization (server-side) +-------------------------------------------------------------------------------} + +instance BuildMetadata DownloadStart where + buildMetadata DownloadStart{downloadSize} = [ + CustomMetadata "download-size" $ C8.pack (show downloadSize) + ] + +instance BuildMetadata DownloadDone where + buildMetadata DownloadDone{downloadHash} = [ + CustomMetadata "download-hash-bin" downloadHash + ] + +instance StaticMetadata DownloadDone where + metadataHeaderNames _ = ["download-hash-bin"] + +{------------------------------------------------------------------------------- + Deserialization (client-side) +-------------------------------------------------------------------------------} + +instance ParseMetadata DownloadStart where + parseMetadata md = + case md of + [CustomMetadata "download-size" value] + | Just downloadSize <- readMaybe (C8.unpack value) + -> return $ DownloadStart{downloadSize} + _otherwise + -> throwM $ UnexpectedMetadata md + +instance ParseMetadata DownloadDone where + parseMetadata md = + case md of + [CustomMetadata "download-hash-bin" downloadHash] + -> return $ DownloadDone{downloadHash} + _otherwise + -> throwM $ UnexpectedMetadata md +