Skip to content

lowmelvin/hammer-scala

Repository files navigation

Hammer-Scala

Hammer-Scala is a Scala 3 utility library that allows quick and painless data conversion between different product types.

To include Hammer in your project, add the following to your dependencies:

libraryDependencies += "com.melvinlow" %% "hammer" % <version>

Quick Start

Assume you have an Algebraic Data Type (ADT) like this:

case class AccountEntity(id: String, email: String, name: String, secret: String, createdAt: Instant)

val entity = AccountEntity("123", "[email protected]", "nobo", "should-be-hashed", Instant.now)

And you want to convert it to something like this:

case class Account(id: String, name: String, email: String)

With Hammer, you can make this conversion in one function call:

import com.melvinlow.hammer.instances.auto.given
import com.melvinlow.hammer.syntax.all.*
import com.melvinlow.hammer.*

entity.hammerTo[Account]
// res0: Account = Account(
//   id = "123",
//   name = "nobo",
//   email = "[email protected]"
// )

Introduction

In many scenarios, we create case classes that are simpler versions of others. For example, you might have a comprehensive model representation for your database and a leaner version for your API consumers.

Manually constructing these leaner versions can be tedious and error-prone:

case class Octagon(a: Int, b: Int, c: Int, d: Int, e: Int, f: Int, g: Int, h: Int)
case class Hexagon(b: Int, c: Int, d: Int, e: Int, f: Int, g: Int, h: Int)

val octagon = Octagon(1, 2, 3, 4, 5, 6, 7, 8)
// octagon: Octagon = Octagon(
//   a = 1,
//   b = 2,
//   c = 3,
//   d = 4,
//   e = 5,
//   f = 6,
//   g = 7,
//   h = 8
// )

val hexagon = Hexagon(octagon.b, octagon.c, octagon.d, octagon.e, octagon.f, octagon.g, octagon.h)
// hexagon: Hexagon = Hexagon(b = 2, c = 3, d = 4, e = 5, f = 6, g = 7, h = 8)

Hammer uses generic programming techniques to automate the process:

octagon.hammerTo[Hexagon]
// res1: Hexagon = Hexagon(b = 2, c = 3, d = 4, e = 5, f = 6, g = 7, h = 8)

It can handle missing and swapped fields:

case class A(x: String, y: String, z: String)
case class B(z: String, x: String)

A("x", "y", "z").hammerTo[B]
// res2: B = B(z = "z", x = "x")

And also nested fields:

case class CompanyEntity(name: String, createdAt: Instant)
case class PersonEntity(name: String, company: CompanyEntity, createdAt: Instant)

case class Company(name: String)
case class Person(name: String, company: Company)

PersonEntity("John", CompanyEntity("Scala", Instant.now), Instant.now)
  .hammerTo[Person]
// res3: Person = Person(name = "John", company = Company(name = "Scala"))

It can also convert types, such as for auto-unboxing wrapper types:

opaque type EmailAddress = String
object EmailAddress {
  def apply(email: String) = email
  extension (email: EmailAddress) def underlying: String = email

  // Define how to hammer an email address to a string
  given Hammer[EmailAddress, String] = (e: EmailAddress) => e.underlying
}

case class Boxed(email: EmailAddress)
case class Unboxed(email: String)

Boxed(EmailAddress("[email protected]")).hammerTo[Unboxed]
// res4: Unboxed = Unboxed(email = "[email protected]")

Importantly, it will error at compile time if conversion is not possible:

case class Cat(name: String, voice: "Meow")
case class Dog(name: String, voice: "Bark")

Cat("Peanuts", "Meow").hammerTo[Dog]
// error: 
// No given instance of type com.melvinlow.hammer.Hammer[("Meow" : String), ("Bark" : String)] was found.
// I found:
// 
//     com.melvinlow.hammer.instances.auto.given_Hammer_I_O[("Meow" : String),
//       ("Bark" : String)](
//       /* missing */summon[deriving.Mirror.ProductOf[("Meow" : String)]], ???)
// 
// But Failed to synthesize an instance of type deriving.Mirror.ProductOf[("Meow" : String)]: class String is not a generic product because it is not a case class.

Usage

Simply include the correct imports and call the hammerTo method:

import com.melvinlow.hammer.instances.auto.given
import com.melvinlow.hammer.syntax.all.*

Advanced Usage

Hammer includes patching functionality that allows you to override specific fields. To do this, call the hammerWith extension method:

case class HumanEntity(name: String, age: Int)
case class Human(name: String, age: Int)

// Override the name field
HumanEntity("Hami", 18).hammerWith[Human](Patch["name"]("Arno"))
// res6: Human = Human(name = "Arno", age = 18)

The same method can also be used to convert a case class to a richer case class by providing the values of the missing fields:

case class Engineer(name: String)
case class EngineerEntity(name: String, createdAt: Instant, updatedAt: Instant)

Engineer("Lynn").hammerWith[EngineerEntity](
  Patch["createdAt"](Instant.now),
  Patch["updatedAt"](Instant.now)
)
// res7: EngineerEntity = EngineerEntity(
//   name = "Lynn",
//   createdAt = 2023-07-31T06:14:57.426081Z,
//   updatedAt = 2023-07-31T06:14:57.426083Z
// )

As shown, you are required to provide the desired output type (can be inferred) along with a variable number of Patch objects that each contain the field name and value to inject. The Patch objects can be specified in any order.

For comparison, these two usages are equivalent:

octagon.hammerTo[Hexagon]
// res8: Hexagon = Hexagon(b = 2, c = 3, d = 4, e = 5, f = 6, g = 7, h = 8)

octagon.hammerWith[Hexagon]()
// res9: Hexagon = Hexagon(b = 2, c = 3, d = 4, e = 5, f = 6, g = 7, h = 8)

Finally, note that only top level patching is supported--it is not possible to inject a value somewhere deep into a nested case class.

Typeclasses and Extensions

To automatically convert from type I to O, provide an instance of Hammer[I, O]:

import com.melvinlow.hammer.*

trait Hammer[I, O] {
  def hammer(input: I): O
}

For example, you could automatically lift values to Option:

given [I]: Hammer[I, Option[I]] = (i: I) => Some(i)

case class F(x: Int)
case class G(x: Option[Int])

F(1).hammerTo[G]
// res11: G = G(x = Some(value = 1))

Underneath the hood, Hammer simply looks for a matching field name where a Hammer[I, O] is provided. The default imports include an identity hammer (i.e., Hammer[I, I]) so that if two fields have the same name and type, they are always compatible.