Ecosyste.ms: Awesome

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

Awesome Lists | Featured Topics | Projects

https://github.com/neandertech/smithy4s-deriving

Experimental Scala 3 library that allows to automatically derive instances of the smithy4s abstractions from scala constructs.
https://github.com/neandertech/smithy4s-deriving

derivation scala scala3 smithy smithy4s

Last synced: 3 months ago
JSON representation

Experimental Scala 3 library that allows to automatically derive instances of the smithy4s abstractions from scala constructs.

Awesome Lists containing this project

README

        

# smithy4s-deriving

`smithy4s-deriving` is a **Scala-3 only** experimental library that allows to automatically derive instances of the [smithy4s](https://disneystreaming.github.io/smithy4s/) abstractions from scala constructs.

If `smithy4s` is a tool that promotes a spec-first approach to API design, providing a code-generator that feeds of smithy specifications, the runtime interpreters it provides are written against a set of abstractions that are not inherently tied to code-generators.

`smithy4s-deriving` provides a code-first alternative way to code-generation to interact with these interpreters, by using handcrafted data-types and interfaces written directly in Scala as the source of truth, thus giving access to a large number of features provided by Smithy4s, with a lower barrier of entry. In particular, this enables usage in scala-cli projects.

This project takes some inspiration from :

* Jamie Thompson's https://github.com/bishabosha/ops-mirror
* Jakub Kozlowski's https://github.com/polyvariant/respectfully/

## Installation

Scala 3.4.1 or newer is required.

SBT :

```
libraryDependencies += "tech.neander" %% "smithy4s-deriving" %
addCompilerPlugin("tech.neander" %% "smithy4s-deriving-compiler" % )
```

scala-cli :

```
//> using dep "tech.neander::smithy4s-deriving:"
//> using plugin "tech.neander::smithy4s-deriving-compiler:"
```

You'll typically need the following imports to use the derivation :

```scala
import smithy4s.*
import smithy4s.deriving.{given, *}
import smithy4s.deriving.aliases.* // for syntactically pleasant annotations
import scala.annotation.experimental // the derivation of API uses experimental metaprogramming features, at this time.

import smithy.api.* // if you want to use hints from the official smithy standard library
import alloy.* // if you want to use hints from the alloy library
```

The `smithy4s-deriving` library is provided for Scala JVM, Scala JS (1.16+) and Scala Native (0.4.x), and is currently compatible with smithy4s 0.18.16+.

## Examples

head other to [the examples](./modules/examples/shared/src/main/scala/) directory to get a feel for how to use it.

## Derivation of case-classes schemas

`smithy4s-deriving` allows for deriving schemas from case classes.

```scala
import smithy4s.*
import smithy4s.deriving.{given, *}

case class Person(firstName: String, lastName: String) derives Schema
```

This allows to access a bunch of serialisation [utilities](https://disneystreaming.github.io/smithy4s/docs/02.1-serialisation/serialisation) and other schema-driven features from smithy4s.

It is possible to customise the behaviour of serialisers in a similar way that you'd have in spec-first smithy4s, by annotating data-types / fields with the `@hints` annotation, and passing it instances of datatypes that were generated from smithy-traits by smithy4s. For instance :

```scala
package example

import smithy.api.*

case class Person(@hints(JsonName("first-name")) firstName: String = "John", @hints(JsonName("last-name")) lastName = "Doe") derives Schema
```

is semantically equivalent to :

```smithy
namespace example

structure Person {
@jsonName("first-name")
@required
firstName: String = "John"

@jsonName("last-name")
@required
lastName: String = "Doe"
}
```

### NB :

* All case class fields must have implicit (given) schemas available
* Defaults are supported
* Scaladoc is converted to `@documentation` hints

## Derivation of ADTs schemas

`smithy4s-deriving` allows for deriving schemas from ADTs.

```scala
import smithy4s.*
import smithy4s.deriving.{given, *}

enum Foo derives Schema {
case Bar(x: Int, y: Boolean)
case Baz(z: String)
}
```

It is possible to :
* use the `@hints` annotation on ADTs and their members.
* use the `@hints.member` annotation on ADT members to distinguish whether hints should go to the member or the target shape at the smithy level.
* use the `@wrapper` on single-field case classes in order to prevent a layer of "structure" schema from being created.

For instance :

```scala
package example

import smithy4s.*
import smithy4s.deriving.{given, *}

import smithy.api.*

enum Foo derives Schema {
@hints(Documentation("Some docs"))
@hints.member(JsonName("bar")) // note the different annotation to target the smithy member
case Bar(x: Int, y: Boolean)

@hints.member(JsonName("baz"))
@wrapper // note the wrapper annotation here
case Baz(z: String)
}
```

Is equivalent to the combination of these smithy specs :

```smithy
namespace example

use example.foo#Bar

union Foo {
@jsonName("baz")
Bar: Bar
@jsonName("bar")
Baz: String
}
```

and

```smithy
namespace example.foo

///Some docs
structure Bar {
@required
x: Integer
@required
y: String
}
```

## Derivation of Services from interfaces/classes

```scala
import smithy4s.*
import smithy4s.deriving.{given, *}
import scala.annotation.experimental

@experimental
trait HelloWorldService derives API {

def hello(name: String, location: Option[String]) : IO[String]

}
```

This allows to access whatever interpreters are provided by smithy4s or its downstream libraries. These interpreters are how smithy4s integrates with various libraries and protocols.

* It is possible to use hints on interfaces to customise interpreter behaviour.
* It is also possible to use the `@errors` annotation to tie error handling to either services

For instance:

```scala
package example

import smithy4s.*
import smithy4s.deriving.{given, *}
import smithy.api.*
import alloy.*
import scala.annotation.experimental

@hints(HttpError(403))
case class Bounce(message: String) extends Throwable derives Schema
@hints(HttpError(500))
case class Crash(cause: String) extends Throwable derives Schema

@experimental
@hints(SimpleRestJson())
trait HelloWorldService derives API {

@errors[(BadLocation, Crash)]
@hints(Http("GET", "/hello/{name}", 200))
def hello(
@hints(HttpLabel()) name: String,
@hints(HttpQuery("from")) location: Option[String]
) : IO[String]

}
```

Is semantically equivalent to the combination of these smithy specs :

```smithy
$version: "2"

namespace example

use alloy#simpleRestJson
use example.helloWorldService#hello

@simpleRestJson
service Foo {
operations: [hello]
}

@error("client")
@httpError(403)
structure Bounce {
@required
message: String
}

@error("server")
@httpError(500)
structure Crash {
@required
cause: String
}
```

and

```smithy
$version: "2"

namespace example.helloWorldService

use example#Bounce
use example#Crash

@http(method: GET, uri: "/hello/{name}", code: 200)
operation hello {
input := {
@required
@httpLabel
name: String

@httpQuery("from")
location: String
}
output := {
@httpPayload
value: String
}

errors: [Bounce, Crash]
}
```

### NB :

* All parameters of methods and all output types (within the effect) must have implicit (given) schemas available.
* Defaults are supported
* Scaladoc is converted to `@documentation` hints

## More concise annotations

To reduce the verbosity induced by the `hints` annotation, it is possible to define custom annotations that create the responsibility of
creating hints, as such :

```scala
import smithy4s.Hints
import smithy4s.deriving.HintsProvider

case class httpGet(uri: String, status: Int) extends HintsProvider {
def hints = Hints(Http(NonEmptyString("GET"), NonEmptyString(uri), status))
}
```

A few of these more-concise annotations are provided out-of-the-box in the `smithy4s.deriving.aliases` package.

## Difference between smithy4s.deriving.API and and smithy4s.Service

As you may have noted, instead of deriving `smithy4s.Service`, we're deriving `smithy4s.deriving.API` instead. That is because the `Service` abstraction expects a polymorphic type, whereas our interface is monomorphic. Therefore, the `API` construct allows to turn instances of our interface into a "virtual" interface that does abide by the kind that `smithy4s.Service` expects. Because of this slight mismatch, the user is expected to perform a call to the `.liftService` extension on the instance of the interface when wiring it into interpreters that are coming from smithy4s, such as :

```scala
SimpleRestJsonBuilder
.routes(new HelloWorldService().liftService[IO])
.resource
.map(_.orNotFound)
```

The derivation works for monomorphic interfaces (and concrete classes) that carry methods that are homogenous in effect type. This means that the derivation will fail for an interface like this.

```scala
trait Foo {
def bar() : IO[Boolean]
def baz() : Int
}
```

It will however work for direct-style interfaces, such as :

```scala
trait Foo {
def bar() : Boolean
def baz() : Int
}
```

or for any other mono-functor effect (`IO`, `Future`, type aliases to `type Result[A] = Either[String, A]`, etc).

### Transformations

It is possible to apply generic transformations to an implementation of a service :

```scala
class Foo() derives API {
def bar(x: Int) : IO[Int] = IO(x)
}

val addDelay = new PolyFunction[IO, IO]{
def apply[A](io: IO[A]) : IO[A] = IO.sleep(1.second) *> io
}

new Foo().transform(addDelay)
```

A lot more is possible to achieve, but I don't have time to write much docs.

### Stubs

smithy4s-deriving, combined with the `export` keyword introduced by Scala 3, makes it reasonably easy to implement mocks/stubs for interfaces that have many methods :

```scala
trait Foo() derives API {
def foo(x: Int): Try[Int]

def bar(x: Int): Try[Int]
}

// creates a stub that will implement all methods by returning `Failure(Boom)`
val stub = API[Foo].default[Try](Failure(Boom))
val instance = new Foo {
export stub.{foo => _, *}
override def foo(x: Int): Try[Int] = Success(x + 2)
}
```

### Compile time validation

The `smithy4s-deriving-compiler` permits the validation of API/Schema usage within the compilation cycle. At the time of writing this, the plugin works by looking up derived `API` instances and crawling through the schemas from there, which implies that standalone `Schema` instances that are not (transitively) tied to `API` instances are not validated at compile time.

### Re-creating a smithy-model from the derived constructs (JVM only)

It is possible to automatically recreate a smithy model from the derived abstractions, and to run the smithy validators. One could use this in a unit test, for instance, to verify the correctness of their services according to the rules of smithy.

See an example of how to do that [here](./modules/examples/jvm/src/main/scala/printSpecs.scala).

### Known limitations

* Default parameters are captured in schemas of case classes, but not for methods, unless the `-Yretain-trees` compiler option is set.
* `API` derivation is using experimental features from the Scala meta-programming tooling, which implies an invasive (but justified) requirement to annotate stuff with `@experimental`
* The [smithy to openapi](https://disneystreaming.github.io/smithy4s/docs/protocols/simple-rest-json/openapi) conversion feature provided by smithy4s happens at build-time, via the build plugins. This unfortunately implies that users wanting to use smithy4s-deriving will not benefit from the simpler one-liner allowing to serve swagger-ui.