Why types matter
@volpegabriel87
@gvolpe
April 2020
Scala Love Conf
@volpegabriel87
@gvolpe
About me
A motivating example
@volpegabriel87
@gvolpe
def showName(username: String, name: String, email: String): String =
s"""
Hi $name! Your username is $username
and your email is $email.
"""
val program: IO[Unit] =
putStrLn(
showName("gvolpe@github.com", "12345", "foo")
)
Dealing with Strings
A motivating example
@volpegabriel87
@gvolpe
@volpegabriel87
@gvolpe
Introducing types
case class UserNameV(value: String) extends AnyVal
case class NameV(value: String) extends AnyVal
case class EmailV(value: String) extends AnyVal
val program: IO[Unit] =
putStrLn(
showNameV(
UserNameV("gvolpe@github.com"),
NameV("12345"),
EmailV("foo")
)
)
def showNameV(username: UserNameV, name: NameV, email: EmailV): String =
s"""
Hi ${name.value}! Your username is ${username.value}
and your email is ${email.value}.
"""
Value classes
Let's do better!
@volpegabriel87
@gvolpe
def mkUsername(value: String): Option[UserNameV] =
if (value.nonEmpty) UserNameV(value).some else None
def mkName(value: String): Option[NameV] =
if (value.nonEmpty) NameV(value).some else None
// Let's pretend we validate it properly
def mkEmail(value: String): Option[EmailV] =
if (value.contains("@")) EmailV(value).some else None
val program: IO[Unit] =
(
mkUsername("gvolpe").liftTo[IO](EmptyError),
mkName("George").liftTo[IO](EmptyError),
mkEmail("123").liftTo[IO](InvalidEmail)
).parMapN(showNameT)
.flatMap(putStrLn)
Smart Constructors
Let's do better!
@volpegabriel87
@gvolpe
λ > root[ERROR] TypesDemo$InvalidEmail$
root ... finished with exit code 1
Smart Constructors
RUNTIME VALIDATION
Case classes gotcha
@volpegabriel87
@gvolpe
Copy method
val program: IO[Unit] =
(
mkUsername("gjl").liftTo[IO](EmptyError),
mkName("George").liftTo[IO](EmptyError),
mkEmail("gjl@foo.com").liftTo[IO](InvalidEmail)
).parMapN {
case (u, n, e) =>
showNameV(u.copy(value = ""), n, e)
}.flatMap(putStrLn)
Ouch!
A neat trick
@volpegabriel87
@gvolpe
Sealed abstract case class
sealed abstract case class UserNameP(value: String)
object UserNameP {
def apply(value: String): Option[UserNameP] =
if (value.nonEmpty) new UserNameP(value) {}.some else None
}
// same for name
sealed abstract case class EmailP(value: String)
object EmailP {
def apply(value: String): Option[EmailP] =
if (value.contains("@")) new EmailP(value) {}.some else None
}
val program: IO[Unit] =
(
UserNameP("jr"),
NameP("Joe Reef"),
EmailP("joe@bar.com")
).tupled.fold(IO.unit) {
case (u, n, e) => putStrLn(showNameP(u, n, e))
}
@volpegabriel87
@gvolpe
A better solution
@newtype case class UserNameT(value: String)
@newtype case class NameT(value: String)
@newtype case class EmailT(value: String)
val program: IO[Unit] =
putStrLn(
showNameT(
UserNameT("gvolpe@github.com"),
NameT("12345"),
EmailT("")
)
)
def showNameT(username: UserNameT, name: NameT, email: EmailT): String =
s"""
Hi ${name.value}! Your username is ${username.value}
and your email is ${email.value}.
"""
Newtypes
@volpegabriel87
@gvolpe
A better solution
Newtypes
def mkUsername(value: String): Option[UserNameT] =
if (value.nonEmpty) UserNameT(value).some else None
def mkName(value: String): Option[NameT] =
if (value.nonEmpty) NameT(value).some else None
def mkEmail(value: String): Option[EmailT] =
if (value.contains("@")) EmailT(value).some else None
@volpegabriel87
@gvolpe
A better solution
Newtypes
val program: IO[Unit] =
(
mkUsername("gvolpe").liftTo[IO](EmptyError),
mkName("George").liftTo[IO](EmptyError),
mkEmail("123").liftTo[IO](InvalidEmail)
).parMapN {
case (u, n, e) =>
showNameT(u, n, e)
}.flatMap(putStrLn)
case (u, n, e) =>
showNameT(UserName(""), n, e)
Refinement Types
@volpegabriel87
@gvolpe
import eu.timepit.refined._ // and more ...
type UserNameR = NonEmptyString
type NameR = NonEmptyString
type EmailR = String Refined Contains['@']
Because we can do better
Refinement Types
@volpegabriel87
@gvolpe
val program: IO[Unit] =
putStrLn(
showNameR("jr", "Joe", "123#com")
)
Because we can do better
def showNameR(username: UserNameR, name: NameR, email: EmailR): String =
s"""
Hi ${name.value}! Your username is ${username.value}
and your email is ${email.value}.
"""
@volpegabriel87
@gvolpe
COMPILE TIME VALIDATION
[error] /foo/one.scala:77:30: Predicate (!(1 == @) && !(2 == @)) did not fail.
[error] showNameR("jr", "Joe", "123#com")
[error] ^
[error] one error found
[error] (Compile / compileIncremental) Compilation failed
[error] Total time: 2 s, completed Nov 29, 2019 5:12:33 PM
Refinement Types
Because we can do better
✔️
Refinement Types
@volpegabriel87
@gvolpe
Runtime validation
case class MyType(a: NonEmptyString, b: Int Refined Greater[5])
def validate(a: String, b: Int): Either[String, MyType] =
for {
x <- refineV[NonEmpty](a)
y <- refineV[Greater[5]](b)
} yield MyType(x, y)
validate("", 4)
Left(Predicate isEmpty() did not fail.)
def refineV[T, P](t: T)(
implicit v: Validate[T, P]
): Either[String, Refined[T, P]]
Refinement Types
@volpegabriel87
@gvolpe
Runtime validation via Validated
case class MyType(a: NonEmptyString, b: Int Refined Greater[5])
// Round-trip Either-Validated conversion via Parallel
def validate(a: String, b: Int): Either[String, MyType] =
(refineV[NonEmpty](a), refineV[Greater[5]](b))
.parMapN(MyType.apply)
validate("", 4)
Left(Predicate isEmpty() did not fail.Predicate failed: (4 > 5).)
Refinement Types
@volpegabriel87
@gvolpe
Runtime validation via Validated NEL
case class MyType(a: NonEmptyString, b: Int Refined Greater[5])
// Round-trip EitherNel-ValidatedNel conversion via Parallel
def validate(a: String, b: Int): EitherNel[String, MyType] =
(refineV[NonEmpty](a).toEitherNel, refineV[Greater[5]](b).toEitherNel)
.parMapN(MyType.apply)
validate("", 4)
Left(
NonEmptyList(
Predicate isEmpty() did not fail.,
Predicate failed: (4 > 5).
)
)
Refinement Types
@volpegabriel87
@gvolpe
-
Logical Predicates
- Not, And, Or
-
Numeric Predicates
- LessThan
- GreaterThan
- EqualTo
- From, To, FromTo
- Positive, Negative
-
Foldable Predicates
- SizeEqualTo
- NonEmpty
Newtypes + Refined
@volpegabriel87
@gvolpe
Because we can excel
@newtype case class UserName(value: NonEmptyString)
@newtype case class Name(value: NonEmptyString)
@newtype case class Email(value: String Refined Contains['@'])
import eu.timepit.refined._ // and more ...
type UserNameR = NonEmptyString
type NameR = NonEmptyString
type EmailR = String Refined Contains['@']
Newtypes + Refined
@volpegabriel87
@gvolpe
Because we can excel
def showNameTR(username: UserName, name: Name, email: Email): String =
s"""
Hi ${name.value}! Your username is ${username.value}
and your email is ${email.value}.
"""
val program: IO[Unit] =
putStrLn(
showNameTR(
UserName("jr"),
Name("John"),
Email("foo@bar.com")
)
)
Newtypes + Refined
@volpegabriel87
@gvolpe
Because we can excel
def program(u: String, n: String, e: String): IO[Unit] =
putStrLn (
(
refineV[NonEmpty](u).toEitherNel.map(UserName.apply),
refineV[NonEmpty](n).toEitherNel.map(Name.apply),
refineV[Contains['@']](e).toEitherNel.map(Email.apply)
).parMapN(showNameTR)
)
def program(u: String, n: String, e: String): IO[Unit] =
putStrLn (
(
validate[UserName](u),
validate[Name](n),
validate[Email](e)
).parMapN(showNameTR)
)
Newtypes + Refined
@volpegabriel87
@gvolpe
Because we can excel
final class NewtypeRefinedPartiallyApplied[A] {
def apply[T, P](raw: T)(
implicit c: Coercible[Refined[T, P], A],
v: Validate[T, P]
): EitherNel[String, A] =
refineV[P](raw).toEitherNel.map(_.coerce[A])
}
def validate[A]: NewtypeRefinedPartiallyApplied[A] =
new NewtypeRefinedPartiallyApplied[A]
Summary
@volpegabriel87
@gvolpe
- Newtypes: zero-cost wrappers.
- Refined: type-level predicates.
- EitherNEL + Parallel: seamless runtime validation.
✔️ No boilerplate
✔️ No smart constructors
✔️ Compile-time validation
✔️ Strongly-typed functions
Questions?
@volpegabriel87
@gvolpe
Why types matter
By Gabriel Volpe
Why types matter
Why types matter
- 2,861