Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/geirolz/scope
A functional, compile-time and type-safe models layer separator
https://github.com/geirolz/scope
cats functional-programming lightweight pure scala typesafe
Last synced: 3 months ago
JSON representation
A functional, compile-time and type-safe models layer separator
- Host: GitHub
- URL: https://github.com/geirolz/scope
- Owner: geirolz
- License: apache-2.0
- Created: 2021-11-03T20:54:13.000Z (about 3 years ago)
- Default Branch: main
- Last Pushed: 2024-09-08T00:29:09.000Z (4 months ago)
- Last Synced: 2024-09-08T01:33:41.433Z (4 months ago)
- Topics: cats, functional-programming, lightweight, pure, scala, typesafe
- Language: Scala
- Homepage: https://geirolz.github.io/scope/
- Size: 229 KB
- Stars: 11
- Watchers: 4
- Forks: 2
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
- Codeowners: CODEOWNERS
Awesome Lists containing this project
README
# Scope
[![Build Status](https://github.com/geirolz/scope/actions/workflows/cicd.yml/badge.svg)](https://github.com/geirolz/scope/actions)
[![Sonatype Nexus (Releases)](https://img.shields.io/nexus/r/com.github.geirolz/scope-core_2.13?nexusVersion=2&server=https%3A%2F%2Foss.sonatype.org)](https://mvnrepository.com/artifact/com.github.geirolz/scope-core)
[![Scala Steward badge](https://img.shields.io/badge/Scala_Steward-helping-blue.svg?style=flat&logo=)](https://scala-steward.org)
[![Mergify Status](https://img.shields.io/endpoint.svg?url=https://api.mergify.com/v1/badges/geirolz/scope&style=flat)](https://mergify.io)
[![GitHub license](https://img.shields.io/github/license/geirolz/scope)](https://github.com/geirolz/scope/blob/main/LICENSE)A functional, compile-time and type-safe models layer separator
## How to install
```sbt
libraryDependencies += "com.github.geirolz" % "scope-core" % "0.0.11"
libraryDependencies += "com.github.geirolz" % "scope-generic" % "0.0.11"//optional - for scala 2 and 3
```## How to use
### Defining the ModelMapper
Given
```scala
import scope.*
import scope.syntax.*//datatypes
case class UserId(value: Long)
case class Name(value: String)
case class Surname(value: String)//doman models
case class User(id: UserId, name: Name, surname: Surname)//http rest contracts
case class UserContract(id: Long, name: String, surname: String)
object UserContract{
implicit val modelMapperForUserContract: ModelMapper[Scope.Endpoint, User, UserContract] =
ModelMapper.scoped[Scope.Endpoint](user => {
UserContract(
user.id.value,
user.name.value,
user.surname.value
)
})
}
```##### Side effects
If the conversion has side effects you can use `ModelMapperK` instead.
```scala
import scala.util.Tryimplicit val modelMapperKForUserContract: ModelMapperK[Try, Scope.Endpoint, User, UserContract] =
ModelMapperK.scoped[Scope.Endpoint](user => Try {
UserContract(
user.id.value,
user.name.value,
user.surname.value,
)
})
// modelMapperKForUserContract: ModelMapperK[Try, Scope.Endpoint, User, UserContract] = scope.ModelMapperK@47cbc06c
```##### Same fields different model
Often in order to decouple things we just duplicate the same model changing just the name.
For example we could find `UserContract` form the endpoint and `User` from the domain that are actually equals deferring only on the name.In these case macros can same us some boilerplate, importing the `scope-generic` module you can use `deriveCaseClassIdMap` to derive
the `ModelMapper` that map the object using the same fields. If the objects aren't equals from the signature point of view the compilation will fail.
Keep in mind that this macro only supports the primary constructor, smart constructors are not supported.```scala
case class User(id: UserId, name: Name, surname: Surname)case class UserContract(id: UserId, name: Name, surname: Surname)
object UserContract{
import scope.*
import scope.generic.syntax.*
implicit val modelMapperForUserContract: ModelMapper[Scope.Endpoint, User, UserContract] =
ModelMapper.scoped[Scope.Endpoint].deriveCaseClassIdMap[User, UserContract]
}
```### Using the ModelMapper
To use the ModelMapper you have to provide the right `ScopeContext` implicitlyGiven
```scala
val user: User = User(
UserId(1),
Name("Foo"),
Surname("Bar"),
)
``````scala
implicit val scopeCtx: TypedScopeContext[Scope.Endpoint] = ScopeContext.of[Scope.Endpoint]
// scopeCtx: TypedScopeContext[Scope.Endpoint] = scope.TypedScopeContext@2517b11duser.scoped.as[UserContract]
// res0: UserContract = UserContract(
// id = UserId(value = 1L),
// name = Name(value = "Foo"),
// surname = Surname(value = "Bar")
// )
```##### Side effects
If the conversion has side effects you have to write
```scala
import scala.util.Tryuser.scoped.as[Try[UserContract]]
// res1: Try[UserContract] = Success(
// value = UserContract(
// id = UserId(value = 1L),
// name = Name(value = "Foo"),
// surname = Surname(value = "Bar")
// )
// )
```In this case if you don't have a `ModelMapperK` defined but just a `ModelMapper` if an `Applicative` instance
is available in the scope for your effect `F[_]` the pure `ModelMapper` will be lifted using `Applicative[F].pure(...)`### ScopeContext
If the `ScopeContext` is wrong or is missing the compilation will fail
```scala
implicit val scopeCtx: TypedScopeContext[Scope.Event] = ScopeContext.of[Scope.Event]user.scoped.as[UserContract]
// error: diverging implicit expansion for type scope.ModelMapper[scopeCtx.ScopeType,User,UserContract]
// starting with method liftPureModelMapper in trait ModelMapperKInstances
// user.scoped.as[UserContract]
// ^
```