Skip to content

Finch 0.11

Compare
Choose a tag to compare
@vkostyukov vkostyukov released this 13 Dec 19:06
· 1194 commits to master since this release

NOTE: If you're upgrading from 0.10, see changes in the milestone releases as well:

Goodbye Scala 2.10 and Java 7

We've finally decided to drop 2.10 and Java 7 support and align with the most recent version of Finagle (6.40 at this point). See #686. For the reference, Finagle and Co dropped 2.10 support in 6.35 (five releases ago).

Server Sent Events

Finch now provides very basic support to SSE. See #655 and the new cookbook example (thanks @rpless).

Long story short, serving AsyncStream[io.finch.sse.ServerSentEvent[A]], for which, cats.Show[A] is defined will stream generated events back to the client until the connection is closed or the stream is empty.

New Decoding Machinery

In 0.11-M1 we've made a decent progress on making encoding in Finch less magical by introducing a type-level content type. This removed a lot of ambiguity when more than one implicit encoder present in the scope.

This time, we did a similar trick for Decode type class. Now it embeds a type-level string indicating a content-type this decoder can decode. This highlighted an obvious difference between decoding query-string param as Int and HTTP body as JSON. These are clearly two different use case that should be handled separately. Thus there is a new type class for decoding HTTP entities (not bodies) in Finch: DecodeEntity that (1) doesn't embed a content-type and (2) decodes from String. Decode, on the other hand, know about content-type and (now) decodes from Buf (as it should). See #663 for more details.

This work not only made the decoding story in Finch more explicit (clear separation of concerns, decoders for HTTP bodies are now resolved w.r.t. their content-types) but also allowed a performance optimization for body* endpoints. Given that we now can decode directly from Buf, we can eliminate one extra copy of going from Buf to String (to satisfy the DecodeEntity input type). At this point, we managed to improve the throughput on out end-to-end test by 12.5%. See #671 (thanks @rpless) for more details.

All these benefits came with the cost of breaking API. See API changes for more details.

Decoding with Content-Type

Now that we have type-level content-type in place, we can enforce the implicit resolution to pick the right decoder for an expected request. We introduce a new API for body* endpoints that also accept the desired content-type. See #695 for more details.

NOTE: body.as[User] is still supported and defaults to body[User, Application.Json].

// before (deprecated in this version)
body.as[User]

// after
body[User, Application.Json]
// or using an alias method
jsonBody[User]

Previously, it was also possible to run as[User] to decode a User from JSON sent as a query string param or header. This indeed really powerful and allows some questionable design patterns, which are not necessary useful. Instead of being super generic here, we're trying to reduce the number of ways things could be built. This not only makes them easy to reason about but also quite efficient (b/c you now, specialization).

That said, we're promoting a new way of decoding HTTP payloads. Instead of using .as[A], we make it less powerful and more explicit. By limiting the responsibility of the Decode type-class, we tight it directly with HTTP payloads. This means constructing body* endpoints could be done in a single step, instead of producing 3-nested structures hence reduce allocations.

Quick experiments showed that we could save 15% of running time and 20% of allocations by just beeing explicit (json2 is the new body[A, Application.Json], json is body.as[A]).

[info] BodyBenchmark.json                       avgt    6  4824.580 ± 1205.444   ns/op
[info] BodyBenchmark.json:·gc.alloc.rate.norm   avgt    6  5896.004 ±  147.449    B/op
[info] BodyBenchmark.json2                      avgt    6  4179.209 ±  673.098   ns/op
[info] BodyBenchmark.json2:·gc.alloc.rate.norm  avgt    6  4936.004 ±   73.723    B/op

[info] BodyBenchmark.jsonOption                       avgt    6  4335.755 ±  150.928   ns/op
[info] BodyBenchmark.jsonOption:·gc.alloc.rate.norm   avgt    6  5712.004 ±    0.001    B/op
[info] BodyBenchmark.jsonOption2                      avgt    6  4050.681 ±  685.263   ns/op
[info] BodyBenchmark.jsonOption2:·gc.alloc.rate.norm  avgt    6  4940.004 ±   36.862    B/op   

Fixing Mistakes in Errors

Some of the Finch's core components, including its error types, were designed a couple of years ago. Mistakes were made. We acknowledge it and fixing them now. See #694 for the full discussion.

Here is the summary of what's changed:

  • A misleading type for error accumulation (RequestErrros) now represents a flat non-empty list of Finch's own errors (previously, a recursive Seq of generic Throwables). Technically, the new type tells exactly what happens at runtime (which is exactly why need types) - we always flatten errors while collecting, not nest them.
  • Now product endpoint only accumulates Finch's own errors and fails-fast with the first non-Finch error observed. We think this is a sane default behavior given that's not safe to keep evaluating endpoints while one of them failed with an unknown reasons that could have side-affected an entire application.

Lift Your Stuff

Endpoint.liftX is a collection of factory methods allowing to build Finch Endpoints out of anything. Presumably, these could be useful for wrapping functions returning arbitrary value within an [Endpoint] context.

// The following endpoint will recompute a random integer on each request.
val nextInt: Endpoint[Int] = Endpoint.lift(scala.util.random.nextInt)

Behaviour Changes

  • Finch now defines a very basic instance of Encode.Aux[Exception, ?] that is polymorphic to the content type. This means, if no Encode[Exception] is provided for a given content-type, Finch app will still compile using this default instance (see #683).

API Changes

  • body and bodyOption endpoints now return Buf instead of String. Thus body.as[User] should still work as expected, but there is a new stringBody instance that might be used in place of body where a UTF-8 string of an HTTP body is expected.
  • Encode instance for Either is removed (this shouldn't be defined in Finch). See #689.

Bug Fixes