Skip to content

Commit

Permalink
Quick start tutorial
Browse files Browse the repository at this point in the history
Also improve some of the docs in the main library
  • Loading branch information
edsko committed Oct 18, 2024
1 parent 47ee6e2 commit f06ff6e
Show file tree
Hide file tree
Showing 14 changed files with 238 additions and 5 deletions.
2 changes: 1 addition & 1 deletion cabal.project
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
packages: ./grapesy
packages: ./grapesy, ./tutorials/quickstart

package grapesy
tests: True
Expand Down
6 changes: 5 additions & 1 deletion grapesy/src/Network/GRPC/Client/Call.hs
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,10 @@ data Call rpc = SupportsClientRpc rpc => Call {

-- | Scoped RPC call
--
-- This is the low-level API for making RPC calls, providing full flexibility.
-- You may wish to consider using the infrastructure from
-- "Network.GRPC.Client.StreamType.IO" instead.
--
-- Typical usage:
--
-- > withRPC conn def (Proxy @ListFeatures) $ \call -> do
Expand All @@ -99,7 +103,7 @@ data Call rpc = SupportsClientRpc rpc => Call {
-- considered closed and the cancellation exception is not raised. Under normal
-- circumstances (with well-behaved server handlers) this should not arise.
-- (The gRPC specification itself is not very specific about this case; see
-- discussion at https://stackoverflow.com/questions/55511528/should-grpc-server-side-half-closing-implicitly-terminate-the-client.)
-- discussion at <https://stackoverflow.com/questions/55511528/should-grpc-server-side-half-closing-implicitly-terminate-the-client>.)
--
-- 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
Expand Down
5 changes: 5 additions & 0 deletions grapesy/src/Network/GRPC/Client/Connection.hs
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,8 @@ import Network.GRPC.Util.TLS qualified as Util.TLS

-- | Open connection to server
--
-- See 'withConnection'.
--
-- Before we can send RPC requests, we have to connect to a specific server
-- first. Once we have opened a connection to that server, we can send as many
-- RPC requests over that one connection as we wish. 'Connection' abstracts over
Expand Down Expand Up @@ -263,6 +265,9 @@ data Server =

-- | Open connection to the server
--
-- See 'Network.GRPC.Client.withRPC' for making individual RPCs on the new
-- connection.
--
-- The connection to the server is set up asynchronously; the first call to
-- 'withRPC' will block until the connection has been established.
--
Expand Down
1 change: 1 addition & 0 deletions grapesy/src/Network/GRPC/Common/Protobuf.hs
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ module Network.GRPC.Common.Protobuf (
, (.~)
, (^.)
-- ** "Data.ProtoLens"
, StreamingType(..)
, HasField(..)
, FieldDefault(..)
, defMessage
Expand Down
7 changes: 7 additions & 0 deletions grapesy/src/Network/GRPC/Server.hs
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,13 @@ import Network.GRPC.Util.HTTP2.Stream (ClientDisconnected(..))
-- The server can be run using the standard infrastructure offered by the
-- @http2@ package, but "Network.GRPC.Server.Run" provides some convenience
-- functions.
--
-- If you are using Protobuf (or if you have another way to compute a list of
-- methods at the type level), you may wish to use the infrastructure from
-- "Network.GRPC.Server.StreamType" (in particular,
-- 'Network.GRPC.Server.StreamType.fromMethods' or
-- 'Network.GRPC.Server.StreamType.fromServices') to construct the set of
-- handlers.
mkGrpcServer :: ServerParams -> [SomeRpcHandler IO] -> IO HTTP2.Server
mkGrpcServer params@ServerParams{serverTopLevel} handlers = do
ctxt <- newServerContext params
Expand Down
29 changes: 26 additions & 3 deletions grapesy/src/Network/GRPC/Server/StreamType.hs
Original file line number Diff line number Diff line change
Expand Up @@ -209,15 +209,20 @@ fromNextElem _ = NextElem.toStreamElem def
--
-- This will reveal types such as this:
--
-- > _getFeature :: NonStreamingHandler IO (Protobuf RouteGuide "getFeature")
-- > _getFeature :: ServerHandler' NonStreaming IO (Protobuf RouteGuide "getFeature")
--
-- Finally, if we then refine the skeleton to
-- which we can simplify to
--
-- > _getFeature :: ServerHandler IO (Protobuf RouteGuide "getFeature")
--
-- (the non-primed version of 'ServerHandler' infers the streaming type from
-- the RPC, when possible). Finally, if we then refine the skeleton to
--
-- > Method (mkNonStreaming $ _getFeature)
--
-- ghc will tell us
--
-- > _getFeature :: Point -> IO Feature
-- > _getFeature :: Proto Point -> IO (Proto Feature)
data Methods (m :: Type -> Type) (rpcs :: [k]) where
-- | All methods of the service handled
NoMoreMethods :: Methods m '[]
Expand Down Expand Up @@ -298,6 +303,15 @@ fromMethod =
SServerStreaming -> someRpcHandler . fromStreamingHandler
SBiDiStreaming -> someRpcHandler . fromStreamingHandler

-- | List handlers for all methods of a given service.
--
-- This can be used to verify at the type level that all methods of the given
-- service are handled. A typical definition of the 'Methods' might have a type
-- such as this:
--
-- > methods :: Methods IO (ProtobufMethodsOf RouteGuide)
--
-- See also 'fromServices' if you are defining more than one service.
fromMethods :: forall m rpcs.
MonadIO m
=> Methods m rpcs -> [SomeRpcHandler m]
Expand All @@ -309,6 +323,15 @@ fromMethods = go
go (RawMethod m ms) = someRpcHandler m : go ms
go (UnsupportedMethod ms) = go ms

-- | List handlers for all methods of all services.
--
-- This can be used to verify at the type level that all methods of all services
-- are handled. A typical definition of the 'Services' might have a type such as
-- this:
--
-- > services :: Services IO (ProtobufServices '[Greeter, RouteGuide])
--
-- See also 'fromMethods' if you are only defining one service.
fromServices :: forall m servs.
MonadIO m
=> Services m servs -> [SomeRpcHandler m]
Expand Down
31 changes: 31 additions & 0 deletions tutorials/quickstart/LICENSE
Original file line number Diff line number Diff line change
@@ -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.
18 changes: 18 additions & 0 deletions tutorials/quickstart/LICENSE.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
The protobuf file is a modified version of `examples/protos/helloworld.proto`
from the official gRPC repository at https://github.com/grpc/grpc.

Its license is reproduced below:

// Copyright 2015 gRPC authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
3 changes: 3 additions & 0 deletions tutorials/quickstart/Setup.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import Data.ProtoLens.Setup

main = defaultMainGeneratingProtos "proto"
18 changes: 18 additions & 0 deletions tutorials/quickstart/app/Client.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
module Client (main) where

import Network.GRPC.Client
import Network.GRPC.Client.StreamType.IO
import Network.GRPC.Common
import Network.GRPC.Common.Protobuf

import Proto.API.Helloworld

main :: IO ()
main =
withConnection def server $ \conn -> do
let req = defMessage & #name .~ "you"
resp <- nonStreaming conn (rpc @(Protobuf Greeter "sayHello")) req
print resp
where
server :: Server
server = ServerInsecure $ Address "127.0.0.1" defaultInsecurePort Nothing
29 changes: 29 additions & 0 deletions tutorials/quickstart/app/Server.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
module Server (main) where

import Network.GRPC.Common
import Network.GRPC.Common.Protobuf
import Network.GRPC.Server.Protobuf
import Network.GRPC.Server.Run
import Network.GRPC.Server.StreamType

import Proto.API.Helloworld

methods :: Methods IO (ProtobufMethodsOf Greeter)
methods =
Method (mkNonStreaming sayHello)
$ NoMoreMethods

sayHello :: Proto HelloRequest -> IO (Proto HelloReply)
sayHello req = do
let resp = defMessage & #message .~ "Hello, " <> req ^. #name
return resp

main :: IO ()
main =
runServerWithHandlers def config $ fromMethods methods
where
config :: ServerConfig
config = ServerConfig {
serverInsecure = Just (InsecureConfig Nothing defaultInsecurePort)
, serverSecure = Nothing
}
17 changes: 17 additions & 0 deletions tutorials/quickstart/proto/helloworld.proto
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
syntax = "proto3";

// The greeting service definition.
service Greeter {
// Sends a greeting
rpc SayHello (HelloRequest) returns (HelloReply) {}
}

// The request message containing the user's name.
message HelloRequest {
string name = 1;
}

// The response message containing the greetings
message HelloReply {
string message = 1;
}
62 changes: 62 additions & 0 deletions tutorials/quickstart/quickstart.cabal
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
cabal-version: 3.0
name: quickstart
version: 0.1.0
license: BSD-3-Clause
license-file: LICENSE
author: Edsko de Vries
maintainer: [email protected]
build-type: Custom
extra-source-files: proto/helloworld.proto

custom-setup
setup-depends:
base
, Cabal
, proto-lens-setup

common lang
build-depends: base >= 4.14 && < 5
default-language: Haskell2010
ghc-options: -Wall

default-extensions:
DataKinds
OverloadedLabels
OverloadedStrings
TypeApplications
TypeFamilies

library
import: lang
hs-source-dirs: src
build-tool-depends: proto-lens-protoc:proto-lens-protoc

build-depends:
, grapesy
, proto-lens-runtime
exposed-modules:
Proto.API.Helloworld
other-modules:
Proto.Helloworld
autogen-modules:
Proto.Helloworld

executable greeter_server
import: lang
main-is: Server.hs
hs-source-dirs: app
ghc-options: -main-is Server

build-depends:
, grapesy
, quickstart

executable greeter_client
import: lang
main-is: Client.hs
hs-source-dirs: app
ghc-options: -main-is Client

build-depends:
, grapesy
, quickstart
15 changes: 15 additions & 0 deletions tutorials/quickstart/src/Proto/API/Helloworld.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
module Proto.API.Helloworld (
module Proto.Helloworld
) where

import Proto.Helloworld
import Network.GRPC.Common.Protobuf
import Network.GRPC.Common

{-------------------------------------------------------------------------------
Metadata
-------------------------------------------------------------------------------}

type instance RequestMetadata (Protobuf Greeter meth) = NoMetadata
type instance ResponseInitialMetadata (Protobuf Greeter meth) = NoMetadata
type instance ResponseTrailingMetadata (Protobuf Greeter meth) = NoMetadata

0 comments on commit f06ff6e

Please sign in to comment.