Lightweight, validatable base Value types - aka Microtypes - aka Tinytypes
In Gradle, install the ForkHandles BOM and then this module in the dependency block:
implementation(platform("dev.forkhandles:forkhandles-bom:X.Y.Z"))
implementation("dev.forkhandles:values4k")
Subvert primitive obsession and provide type safety and other facilities for JVM programs.
The problem which we are trying to solve is to avoid illegal values entering into our system. For this, it is best to use strongly typed values, which allow us to both lean on the compiler and improve the developer experience by engaging with IDE tooling.
For example, take this simple function:
fun transferMoneyTo(amount: Int, sortCode: String, accountNumber: String)
The first problem here is that accountNumber
and sortCode
fields are both of type String
, meaning that a coder could accidentally switch these values around and we would not potentially notice until runtime.
The base type provided by this lib is the interface Value<T>
. This is extended by AbstractValue<T>
or one of the typealiases, which are just a simple wrapper around a value
field and can be used for defining your own domain types. Value classes are also supported by just implementing Value<T>
:
class Money(value: Int): AbstractValue<Int>(value)
class AccountNumber(value: String): StringValue(value)
@JvmInline
value class SortCode(override val value: String): Value<String>
fun transferMoneyTo(amount: Money, sortCode: SortCode, accountNumber: AccountNumber)
The next problem is that there is no domain validation on our values. What if someone passed in a negative amount? Or an accountNumber
containing letters instead of digits?
We can fix that by validating to ensure we can never create an illegal value. We want values to fail on construction (at the entry point to our system) instead of deep inside our domain logic. For this we can force construction to go through a ValueFactory
or one of the provided convenience subclasses (IntValueFactory
, LocalDateTimeValueFactory
etc..), passing a Validation
predicate:
class Money private constructor(value: Int) : AbstractValue<Int>(value) {
companion object : ValueFactory<Money, Int>(::Money, 1.minValue, String::toInt)
}
class AccountNumber private constructor(value: String) : StringValue(value) {
companion object : StringValueFactory<AccountNumber>(::AccountNumber, "\\d{8}".regex)
}
// note that private constructors are only available on inline classes
// starting with Kotlin 1.5.0
@JvmInline
value class SortCode private constructor(override val value: String) : Value<String> {
companion object : StringValueFactory<SortCode>(::SortCode, "\\d{6}".regex)
}
Constructing the instances then happens using one of the built-in or user-supplied factories:
Money.of(123) // returns Money(123)
Money.of(0) // throws IllegalArgumentException
SortCode.ofOrNull("123") // returns null
SortCode.ofResult("asdf12") // returns Kotlin Result failure
SortCode.ofResult4k("asdf12") // returns Result4k Failure<Exception>
Money.parse("123") // returns Money(123)
Money.parse("notmoney") // throws IllegalArgumentException
SortCode.parseOrNull("123") // returns null
SortCode.parseResult("asdf12") // returns Kotlin Result failure
SortCode.parseResult4k("asdf12") // returns Failure<Exception>
Validations are modelled as a simple typealias and there are several useful ones bundled with values4k:
typealias Validation<T> = (T) -> Boolean
The last big problem is one of PII data. We need to ensure that sensitive values are never outputted in their raw form into any logging infrastructure where they could be mined for nefarious purposes.
class AccountNumber private constructor(value: String) : StringValue(value, hidden()) {
companion object : StringValueFactory<AccountNumber>(::AccountNumber, "\\d{8}".regex)
}
If we attempt to print our AccountNumber
using toString() now will result in:
********
Masking rules are modelled as a simple typealias and there are several useful ones bundled with values4k:
typealias Masking<T> = T.() -> String
For times where we want to display the underlying value as a String, we can use show()
, which is the natural opposite to parse()
. This is different (and safer than) using toString(), where we will have to deal with the Masking rules. In order to maintain symmetry (and to ensure that we can support inline classes), this method is present on the ValueFactory instance - this looks a little strange but it actually is consistent because the display and parse logic should NOT be part of the Value
itself, but be separated logically.
Money.show(Money.of(123)) // returns "123"