Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/typelevel/fs2-grpc

gRPC implementation for FS2/cats-effect
https://github.com/typelevel/fs2-grpc

Last synced: 2 months ago
JSON representation

gRPC implementation for FS2/cats-effect

Awesome Lists containing this project

README

        

# fs2-grpc - gRPC implementation for FS2/cats-effect

[![Discord](https://img.shields.io/discord/632277896739946517.svg?label=&logo=discord&logoColor=ffffff&color=404244&labelColor=6A7EC2)](https://discord.gg/CADXBp3kxj) [![CI](https://github.com/typelevel/fs2-grpc/actions/workflows/ci.yml/badge.svg)](https://github.com/typelevel/fs2-grpc/actions/workflows/ci.yml) [![Latest version](https://index.scala-lang.org/typelevel/fs2-grpc/sbt-fs2-grpc/latest.svg?color=orange&v=1)](https://index.scala-lang.org/typelevel/fs2-grpc/sbt-fs2-grpc)
![Code of Consuct](https://img.shields.io/badge/Code%20of%20Conduct-Scala-blue.svg) [![Scala Steward badge](https://img.shields.io/badge/Scala_Steward-helping-blue.svg?style=flat&logo=)](https://scala-steward.org)

## SBT configuration

`project/plugins.sbt`:
```scala
addSbtPlugin("org.typelevel" % "sbt-fs2-grpc" % "")
```

`build.sbt`:
```scala
enablePlugins(Fs2Grpc)
```

Depending if you wish to use `grpc-netty` or `grpc-okhttp`, add one of the following dependencies:
```
libraryDependencies += "io.grpc" % "grpc-netty-shaded" % scalapb.compiler.Version.grpcJavaVersion
```
or
```
libraryDependencies += "io.grpc" % "grpc-okhttp" % scalapb.compiler.Version.grpcJavaVersion
```

## Protocol buffer files

The protobuf files should be stored in the directory `/src/main/protobuf`.

## Multiple projects

If the generated code is used by multiple projects, you may build the client/server code in a common project which other projects depend on. For example:

```scala
lazy val protobuf =
project
.in(file("protobuf"))
.enablePlugins(Fs2Grpc)

lazy val client =
project
.in(file("client"))
.dependsOn(protobuf)

lazy val server =
project
.in(file("server"))
.dependsOn(protobuf)
```

## Creating a client

A `ManagedChannel` is the type used by `grpc-java` to manage a connection to a particular server. This library provides syntax for `ManagedChannelBuilder` which creates a `Resource` which can manage the shutdown of the channel, by calling `.resource[F]` where `F` has an instance of the `Sync` typeclass. This implementation will do a drain of the channel, and attempt to shut down the channel, forcefully closing after 30 seconds. An example of the syntax using `grpc-netty` is:

```scala
import io.grpc.netty.shaded.io.grpc.netty.NettyChannelBuilder
import fs2.grpc.syntax.all._

val managedChannelResource: Resource[IO, ManagedChannel] =
NettyChannelBuilder
.forAddress("127.0.0.1", 9999)
.resource[IO]
```

The syntax also offers the method `resourceWithShutdown` which takes a function `ManagedChannel => F[Unit]` which is used to manage the shutdown. This may be used where requirements before shutdown do not match the default behaviour.

The generated code provides a method `stubResource[F]`, for any `F` which has a `Async` instance, and it takes a parameter of type `Channel`. It returns a `Resource` with an implementation of the service (in a trait), which can be used to make calls.

Moreover, the generated code provides method overloads that take `ClientOptions` used for configuring calls.

```scala
def runProgram(stub: MyFs2Grpc[IO]): IO[Unit] = ???

val run: IO[Unit] = managedChannelResource
.flatMap(ch => MyFs2Grpc.stubResource[IO](ch))
.use(runProgram)
```

## Creating a server

The generated code provides a method `bindServiceResource[F]`, for any `F` which has a `Async` instance, and it takes an implementation of the service (in a trait), which is used to serve responses to RPC calls. It returns a `Resource[F, ServerServiceDefinition]` which is given to the server builder when setting up the service. Furthermore, the generated code provides method overloads that take `ServerOptions` used for configuring service calls.

A `Server` is the type used by `grpc-java` to manage the server connections and lifecycle. This library provides syntax for `ServerBuilder`, which mirrors the pattern for the client. An example is:

```scala
import io.grpc.netty.shaded.io.grpc.netty.NettyServerBuilder
import fs2.grpc.syntax.all._

val helloService: Resource[IO, ServerServiceDefinition] =
MyFs2Grpc.bindServiceResource[IO](new MyImpl())

def run(service: ServerServiceDefinition) = NettyServerBuilder
.forPort(9999)
.addService(service)
.resource[IO]
.evalMap(server => IO(server.start()))
.useForever

helloService.use(run)
```

## Code generation options

To alter code generation, you can set some flags with `scalapbCodeGeneratorOptions`, e.g.:

```scala
scalapbCodeGeneratorOptions += CodeGeneratorOption.FlatPackage
```

The full set of options available are:

- `CodeGeneratorOption.FlatPackage` - If true, the compiler does not append the proto base file name
- `CodeGeneratorOption.JavaConversions` - Enable Java conversions for protobuf
- `CodeGeneratorOption.Grpc` (included by default) - generate grpc bindings based on Observables
- `CodeGeneratorOption.Fs2Grpc` (included by default) - generate grpc bindings for FS2/cats
- `CodeGeneratorOption.SingleLineToProtoString` - `toProtoString` generates single line
- `CodeGeneratorOption.AsciiFormatToString` - `toString` uses `toProtoString` functionality

## Pass additional protoc options

```scala
PB.protocOptions in Compile := Seq("-xyz")
```

### Tool Sponsorship

Development of fs2-grpc is generously supported in part by [YourKit](https://www.yourkit.com) through the use of their excellent Java profiler.