Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
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.
- Host: GitHub
- URL: https://github.com/neandertech/smithy4s-deriving
- Owner: neandertech
- License: apache-2.0
- Created: 2024-04-25T16:23:47.000Z (9 months ago)
- Default Branch: main
- Last Pushed: 2024-07-16T17:33:06.000Z (6 months ago)
- Last Synced: 2024-10-07T06:04:52.873Z (4 months ago)
- Topics: derivation, scala, scala3, smithy, smithy4s
- Language: Scala
- Homepage: https://neander.tech/2024-04-27-specs-first-and-code-first
- Size: 107 KB
- Stars: 23
- Watchers: 1
- Forks: 2
- Open Issues: 3
-
Metadata Files:
- Readme: README.md
- License: LICENSE.txt
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 exampleimport 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 examplestructure 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 exampleimport 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 exampleuse 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 servicesFor instance:
```scala
package exampleimport 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.HintsProvidercase 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.