Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Sample http4s REST client/server with client macro derivation #552

Merged
merged 39 commits into from
Mar 8, 2019
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
ed9cf04
Initial steps for http integration (#203)
juanpedromoreno Mar 20, 2018
93f8ace
Implemented sample client and server REST handlers to be generated
Apr 19, 2018
b03637c
Added monix.Observable implementation
Apr 24, 2018
2a1cc42
Implemented error handling for unary and streaming REST services (#258)
May 4, 2018
a74604e
Tentative fix for the hanging Monix-Observable tests
May 9, 2018
6424dab
Merge branch 'master' into feature/182-http-support-from-protocols
Nov 21, 2018
8caee87
Undo Single Abstract Method syntax to restore 2.11 compatibility
Nov 21, 2018
65d705d
Merge master into branch
L-Lavigne Jan 24, 2019
98df299
Add auto-derived HTTP client implementation, move packages
L-Lavigne Jan 28, 2019
a216c63
Fix Monix/FS2 conversions using updated dependency
L-Lavigne Jan 28, 2019
051519d
fixes Scala 2.11 compilation error
Feb 18, 2019
8eb36b4
fixes unit tests to prove http client derivation
Feb 20, 2019
c172539
removes some println
Feb 20, 2019
20aae64
adds more tests
Feb 21, 2019
15abe94
removes the macro params that can be inferred
Feb 22, 2019
5d5b726
builds the client according to the typology of the request
Feb 22, 2019
683147c
fixes macro
Feb 26, 2019
54990d8
advances with client derivation
Feb 26, 2019
fe34e47
adds more unit test to prove the derived http client
Feb 26, 2019
1132fc2
restores the derivation of the rpc server, without the refactoring
Feb 26, 2019
97d1c73
re-applies part of the refactoring little by little
Feb 26, 2019
6db7ab3
applies the refactoring again with the fix
Feb 27, 2019
2a9cd3d
derived the simplest http route that only serves GET calls
Feb 28, 2019
06f2b83
derived the stream reaquests http server
Mar 2, 2019
3888be7
fixes the binding pattern in POST routes
Mar 3, 2019
965fdd4
adds tests to cover all the possible types of endpoints
Mar 4, 2019
d6ce882
removes unused imports
Mar 4, 2019
3133ec5
removed unused HttpMethod
Mar 4, 2019
34c0504
upgraded http4s and moved Utils
Mar 5, 2019
795eb12
Merge branch 'master' into feature/182-http-support-from-protocols
juanpedromoreno Mar 5, 2019
c922e0f
solves all the comments in code review
Mar 6, 2019
9b0a8f3
expressed type as FQN and propagated encoder/decoders constraints
Mar 7, 2019
fb91483
removes the import of monix.Scheduler in the macro
Mar 7, 2019
854b0d6
replaces executionContext by Schedule at some points
Mar 7, 2019
f6eab68
adds _root_ to ExecutionContext
Mar 7, 2019
ad90c2d
Apply suggestions from code review
juanpedromoreno Mar 7, 2019
f1f60ff
removes circe-generic
Mar 7, 2019
94dba66
Merge remote-tracking branch 'origin/feature/182-http-support-from-pr…
Mar 7, 2019
bf6e853
replaces Throwable by UnexpectedError and its encoder/decoder
Mar 8, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 37 additions & 32 deletions modules/http/src/main/scala/higherkindness/mu/http/implicits.scala
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ package higherkindness.mu.http
import cats.ApplicativeError
import cats.effect._
import cats.implicits._
import cats.syntax.either._
import fs2.{RaiseThrowable, Stream}
import io.grpc.Status.Code._
import org.typelevel.jawn.ParseException
Expand All @@ -34,6 +35,29 @@ import scala.util.control.NoStackTrace

object implicits {

implicit val unexpectedErrorEncoder: Encoder[UnexpectedError] = new Encoder[UnexpectedError] {
final def apply(a: UnexpectedError): Json = Json.obj(
("className", Json.fromString(a.className)),
("msg", a.msg.fold(Json.Null)(s => Json.fromString(s)))
)
}

implicit val unexpectedErrorDecoder: Decoder[UnexpectedError] = new Decoder[UnexpectedError] {
final def apply(c: HCursor): Decoder.Result[UnexpectedError] =
for {
className <- c.downField("className").as[String]
msg <- c.downField("msg").as[Option[String]]
} yield UnexpectedError(className, msg)
}

implicit def EitherDecoder[A, B](implicit a: Decoder[A], b: Decoder[B]): Decoder[Either[A, B]] =
a.map(Left.apply) or b.map(Right.apply)

implicit def EitherEncoder[A, B](implicit ea: Encoder[A], eb: Encoder[B]): Encoder[Either[A, B]] =
new Encoder[Either[A, B]] {
final def apply(a: Either[A, B]): Json = a.fold(_.asJson, _.asJson)
}

implicit class MessageOps[F[_]](private val message: Message[F]) extends AnyVal {

def jsonBodyAsStream[A](
Expand All @@ -55,47 +79,18 @@ object implicits {

implicit class ResponseOps[F[_]](private val response: Response[F]) {

implicit def EitherDecoder[A, B](
implicit a: Decoder[A],
b: Decoder[B]): Decoder[Either[A, B]] = {
val l: Decoder[Either[A, B]] = a.map(Left.apply)
val r: Decoder[Either[A, B]] = b.map(Right.apply)
l or r
}

implicit private val throwableDecoder: Decoder[Throwable] =
Decoder.decodeTuple2[String, String].map {
case (cls, msg) =>
Class
.forName(cls)
.getConstructor(classOf[String])
.newInstance(msg)
.asInstanceOf[Throwable]
}

def asStream[A](
implicit decoder: Decoder[A],
F: ApplicativeError[F, Throwable],
R: RaiseThrowable[F]): Stream[F, A] =
if (response.status.code != Ok.code) Stream.raiseError(ResponseError(response.status))
else response.jsonBodyAsStream[Either[Throwable, A]].rethrow
else response.jsonBodyAsStream[Either[UnexpectedError, A]].rethrow
}

implicit class Fs2StreamOps[F[_], A](private val stream: Stream[F, A]) {

implicit def EitherEncoder[A, B](implicit ea: Encoder[A], eb: Encoder[B]): Encoder[Either[A, B]] =
new Encoder[Either[A, B]] {
final def apply(a: Either[A, B]): Json = a match {
case Left(a) => a.asJson
case Right(b) => b.asJson
}
}

implicit val throwableEncoder: Encoder[Throwable] = new Encoder[Throwable] {
def apply(ex: Throwable): Json = (ex.getClass.getName, ex.getMessage).asJson
}

def asJsonEither(implicit encoder: Encoder[A]): Stream[F, Json] = stream.attempt.map(_.asJson)
def asJsonEither(implicit encoder: Encoder[A]): Stream[F, Json] =
stream.attempt.map(_.bimap(_.toUnexpected, identity).asJson)
}

implicit class FResponseOps[F[_]: Sync](private val response: F[Response[F]])
Expand All @@ -121,8 +116,18 @@ object implicits {
def handleResponseError[F[_]: Sync](errorResponse: Response[F]): F[Throwable] =
errorResponse.bodyAsText.compile.foldMonoid.map(body =>
ResponseError(errorResponse.status, Some(body).filter(_.nonEmpty)))

implicit class ThrowableOps(self: Throwable) {
def toUnexpected: UnexpectedError =
UnexpectedError(self.getClass.getName, Option(self.getMessage))
}

}

final case class UnexpectedError(className: String, msg: Option[String])
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The name is a bit generic - most errors are unexpected. The way we're using those, they're closer to "RequestException"s, or 4XX-status ResponseExceptions. Can we combine those error types or rename this one?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(resolved offline)

extends RuntimeException(className + msg.fold("")(": " + _))
with NoStackTrace

final case class ResponseError(status: Status, msg: Option[String] = None)
extends RuntimeException(status + msg.fold("")(": " + _))
with NoStackTrace
Original file line number Diff line number Diff line change
Expand Up @@ -19,22 +19,18 @@ package higherkindness.mu.rpc.http
import cats.effect.{IO, _}
import fs2.Stream
import fs2.interop.reactivestreams._
import higherkindness.mu.http.{HttpServer, ResponseError, RouteMap}
import higherkindness.mu.http.{HttpServer, ResponseError, RouteMap, UnexpectedError}
import higherkindness.mu.rpc.common.RpcBaseTestSuite
import monix.reactive.Observable
import io.circe.generic.auto._
import org.http4s._
import org.http4s.client.blaze.BlazeClientBuilder
import org.http4s.server.blaze._
import org.scalatest._
import org.scalatestplus.scalacheck.ScalaCheckDrivenPropertyChecks

import scala.concurrent.duration._

class GreeterDerivedRestTests
extends RpcBaseTestSuite
with ScalaCheckDrivenPropertyChecks
with BeforeAndAfter {
class GreeterDerivedRestTests extends RpcBaseTestSuite with BeforeAndAfter {

val host = "localhost"
val port = 8080
Expand Down Expand Up @@ -153,17 +149,17 @@ class GreeterDerivedRestTests
val request = HelloRequest("")
val responses =
BlazeClientBuilder[IO](ec).stream.flatMap(fs2Client.sayHelloAll(request)(_))
the[IllegalArgumentException] thrownBy responses.compile.toList
.unsafeRunSync() should have message "empty greeting"
the[UnexpectedError] thrownBy responses.compile.toList
.unsafeRunSync() should have message "java.lang.IllegalArgumentException: empty greeting"
}

"handle errors with Observable streaming response" in {
val request = HelloRequest("")
val responses = BlazeClientBuilder[IO](ec).stream
.flatMap(monixClient.sayHelloAll(request)(_).toReactivePublisher.toStream[IO])
the[IllegalArgumentException] thrownBy responses.compile.toList
the[UnexpectedError] thrownBy responses.compile.toList
.unsafeRunTimed(10.seconds)
.getOrElse(sys.error("Stuck!")) should have message "empty greeting"
.getOrElse(sys.error("Stuck!")) should have message "java.lang.IllegalArgumentException: empty greeting"
}

"serve a POST request with bidirectional fs2 streaming" in {
Expand Down Expand Up @@ -199,17 +195,6 @@ class GreeterDerivedRestTests
.getOrElse(sys.error("Stuck!")) shouldBe Nil
}

"serve ScalaCheck-generated POST requests with bidirectional Observable streaming" in {
forAll { strings: List[String] =>
val requests = Observable.fromIterable(strings.map(HelloRequest))
val responses = BlazeClientBuilder[IO](ec).stream
.flatMap(monixClient.sayHellosAll(requests)(_).toReactivePublisher.toStream[IO])
responses.compile.toList
.unsafeRunTimed(10.seconds)
.getOrElse(sys.error("Stuck!")) shouldBe strings.map(HelloResponse)
}
}

}

}
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ package higherkindness.mu.rpc.http
import cats.effect.{IO, _}
import fs2.Stream
import fs2.interop.reactivestreams._
import higherkindness.mu.http.ResponseError
import higherkindness.mu.http.{ResponseError, UnexpectedError}
import higherkindness.mu.rpc.common.RpcBaseTestSuite
import higherkindness.mu.http.implicits._
import io.circe.Json
Expand All @@ -34,14 +34,10 @@ import org.http4s.server.blaze._
import org.scalatest._
import org.http4s.implicits._
import org.http4s.server.Router
import org.scalatestplus.scalacheck.ScalaCheckDrivenPropertyChecks

import scala.concurrent.duration._

class GreeterRestTests
extends RpcBaseTestSuite
with ScalaCheckDrivenPropertyChecks
with BeforeAndAfter {
class GreeterRestTests extends RpcBaseTestSuite with BeforeAndAfter {

val Hostname = "localhost"
val Port = 8080
Expand Down Expand Up @@ -72,7 +68,7 @@ class GreeterRestTests
s"/$Fs2ServicePrefix" -> fs2Service,
s"/$MonixServicePrefix" -> monixService).orNotFound)

var serverTask: Fiber[IO, Nothing] = _ // sorry
var serverTask: Fiber[IO, Nothing] = _
before(serverTask = server.resource.use(_ => IO.never).start.unsafeRunSync())
after(serverTask.cancel)

Expand Down Expand Up @@ -208,17 +204,17 @@ class GreeterRestTests
val request = HelloRequest("")
val responses =
BlazeClientBuilder[IO](ec).stream.flatMap(fs2ServiceClient.sayHelloAll(request)(_))
the[IllegalArgumentException] thrownBy responses.compile.toList
.unsafeRunSync() should have message "empty greeting"
the[UnexpectedError] thrownBy responses.compile.toList
.unsafeRunSync() should have message "java.lang.IllegalArgumentException: empty greeting"
}

"handle errors with Observable streaming response" in {
val request = HelloRequest("")
val responses = BlazeClientBuilder[IO](ec).stream
.flatMap(monixServiceClient.sayHelloAll(request)(_).toReactivePublisher.toStream[IO])
the[IllegalArgumentException] thrownBy responses.compile.toList
the[UnexpectedError] thrownBy responses.compile.toList
.unsafeRunTimed(10.seconds)
.getOrElse(sys.error("Stuck!")) should have message "empty greeting"
.getOrElse(sys.error("Stuck!")) should have message "java.lang.IllegalArgumentException: empty greeting"
}

"serve a POST request with bidirectional fs2 streaming" in {
Expand Down Expand Up @@ -254,15 +250,5 @@ class GreeterRestTests
.getOrElse(sys.error("Stuck!")) shouldBe Nil
}

"serve ScalaCheck-generated POST requests with bidirectional Observable streaming" in {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just curious, were there any issues with the ScalaCheck tests?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(resolved offline)

forAll { strings: List[String] =>
val requests = Observable.fromIterable(strings.map(HelloRequest))
val responses = BlazeClientBuilder[IO](ec).stream
.flatMap(monixServiceClient.sayHellosAll(requests)(_).toReactivePublisher.toStream[IO])
responses.compile.toList
.unsafeRunTimed(10.seconds)
.getOrElse(sys.error("Stuck!")) shouldBe strings.map(HelloResponse)
}
}
}
}
1 change: 0 additions & 1 deletion project/ProjectPlugin.scala
Original file line number Diff line number Diff line change
Expand Up @@ -137,7 +137,6 @@ object ProjectPlugin extends AutoPlugin {
%%("monix", V.monix),
%%("http4s-blaze-client", V.http4s) % Test,
%%("circe-generic") % Test,
%%("scalacheck") % Test,
"ch.qos.logback" % "logback-classic" % V.logback % Test
)
)
Expand Down