🐥

OSS: kotlin-resultを読んで、contract、高階関数の使いどころを理解する

2022/12/03に公開

株式会社ログラスDevアドベントカレンダー2022の12/3

2024/12/13追記

2024年12月現在、kotlin-resultのv2.0.0により内部の実装が当時のものとは大きく変わっていることをご了承ください。
v2.0.0の内部実装の変更については下記の記事で取り上げていますので合わせてご確認ください。
https://zenn.dev/loglass/articles/b8e6282e6d87cf

kotlin-resultとは?

この記事は株式会社ログラスDevアドベントカレンダー2022の12/3(土)の記事です!

https://qiita.com/advent-calendar/2022/loglass

手頃にKotlinで書かれているOSSのコードを読みたい、でもExposedとかKtorとか大きすぎてどこから読めばいいかわからないよ、、、という皆さん! 
読んで勉強になるシンプルでコンパクトな素晴らしいOSSを紹介します!
それが、kotlin-resultというOSSです。

https://github.com/michaelbull/kotlin-result

kotlin-resultは Rust、Elm、HaskellのResult型にインスパイアされたようなインターフェースを持つライブラリです。
成功したときの値か、失敗したときの値かのどちらかを持つResult型を提供しています。
例外とは違って返り値にエラー情報を明示することで可読性を上げたり、使用者にエラーハンドリングを強制させることができます。

日本語でよくまとまっている記事はこちら(なんと本家のReadmeからも参照されています!スゴイ!)
https://note.com/yasukotelin/n/n6d9e352c344c

軽く使い方を話すと、このような形でOkかErrかをResult型としてリターンできたり、

fun checkPrivileges(user: User, command: Command): Result<Command, CommandError> {
    return if (user.rank >= command.mininimumRank) {
        Ok(command)
    } else {
        Err(CommandError.InsufficientRank(command.name))
    }
}

処理の中で例外が起きたら型としてThrowableを詰めるなどもできます。

val result: Result<Customer, Throwable> = runCatching {
    customerDb.findById(id = 50) // could throw SQLException or similar
}

内部のコードを見ると、 contract機能や高階関数などの機能をキレイに扱っており、コードとしてとても勉強になるので今回紹介することになりました。

目次

  • Result型のコードリーディング
    • seald class
    • ジェネリクスと共変
    • componentN
  • runCatchingのコードリーディング
    • 高階関数
    • callsInPlace contract
    • レシーバを受け取る高階関数 A.(B) -> C
    • infix
  • getのコードリーディング
    • implies contract
    • implies contractについてわからなかったこと

Result型のコードリーディング

ではまずはResult.ktを読んでいきます。

https://github.com/michaelbull/kotlin-result/blob/master/kotlin-result/src/commonMain/kotlin/com/github/michaelbull/result/Result.kt

ポイントをかいつまんで解説していきます。

sealed class

public sealed class Result<out V, out E> 

public class Ok<out V>(public val value: V) : Result<V, Nothing>()

public class Err<out E>(public val error: E) : Result<Nothing, E>()

sealed classを用いた代数的データ型でResult型を定義しています。
Result型がOk型の場合は成功値の value: V を Err型の場合はエラー値の error: E を取るようになっています。

ジェネリクスと共変

public sealed class Result<out V, out E> 

val result: Result<Any, Throwable> = Ok(1)
val result: Result<Any, Throwable> = Err(Exception("error"))

ジェネリクスのVとEにはout(共変)がついていますね。このおかげでVのサブクラスとEのサブクラスも型として受け入れることができます。

componentN

public sealed class Result<out V, out E> {

    public abstract operator fun component1(): V?
    public abstract operator fun component2(): E?
}

componentN は Destructuring Declarations と呼ばれるもので、 val (a, b) のように分解できるものです。

https://kotlinlang.org/docs/destructuring-declarations.html

Pairなどの構文がイメージしやすいと思います。

val (first, second) = Pair(1, "A")

ちなみにdata classでは自動で宣言されています

data class Person(val name: String, val age: Int)

val (name, age) = person 

componentNが定義されていることで Result型を V? と E? に分割することができます。

val result: Result<Customer, Throwable> = runCatching {
    customerDb.findById(id = 50) // could throw SQLException or similar
}
val (customer, error) = result

型を明示的に書くとこうなります

val (customer: Customer, error: Throwable) = result

ところで、これらはabstract関数なためサブクラスのOk型とErr型には実装が必要です。
Ok型はどのようになっているのでしょうか?

public class Ok<out V>(public val value: V) : Result<V, Nothing>() {

    override fun component1(): V = value
    override fun component2(): Nothing? = null
}

component1 と component2 の返す型が変わっていることに注意です。
V? とE? だったものが V と Nothing? になっていますね。

あれ、関数の返す型で実装時に変更できたっけ?→サブクラスならできます。

まずVはV?のサブクラスです。

val i1: Int = 1
val i2: Int? = i1 // 代入可能

また、 Nothing は すべてのクラスのサブクラスなため、 Nothing? は E?のサブクラスだと言えます。

val nothing: Nothing? = null
val i: Int? = nothing // 代入可能

このように定義することでOk型の場合は型をいきなり確定させることできます。

val (value: Int, error: Nothing?) = Ok(1)
val (value: Int) = Ok(1)
val (value) = Ok(1)

一方でErr型は以下のようになります。

public class Err<out E>(public val error: E) : Result<Nothing, E>() {

    override fun component1(): Nothing? = null
    override fun component2(): E = error
}

先ほどと逆ですね。
そのためこのように使うことができます。

val (value: Nothing?, error: Exception) = Err(Exception("error"))
val (_, error: Exception) = Err(Exception("error"))
val (_, error) = Err(Exception("error"))

runCatchingのコードリーディング

次は runCatchingのコードを見ていきます。

https://github.com/michaelbull/kotlin-result/blob/master/kotlin-result/src/commonMain/kotlin/com/github/michaelbull/result/Factory.kt

使い方は下記のようなイメージです。

val result: Result<Customer, Throwable> = runCatching {
    customerDb.findById(id = 50) // could throw SQLException or similar
}

実装はトップレベル関数として定義されています。

public inline fun <V> runCatching(block: () -> V): Result<V, Throwable> {
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }

    return try {
        Ok(block())
    } catch (e: Throwable) {
        Err(e)
    }
}

高階関数

runCatching は 関数を引数として受け取る関数(高階関数)となっていますね。

public inline fun <V> runCatching(block: () -> V): Result<V, Throwable>

引数として受け取った block関数はそのまま内部のTry節のなかで評価されています。
block関数は最後にV型を返すので最終的には Result<V, Throwable> を返すようになります。

public inline fun <V> runCatching(block: () -> V): Result<V, Throwable> {
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }

    return try {
        Ok(block())
    } catch (e: Throwable) {
        Err(e)
    }
}

callsInPlace contract

runCatghingではcontractも使われていました。

contractはkotlinコンパイラに想定しうる実行時エラーは起きえないことを教えることで、無駄なエラーハンドリングをなくす機能です。
以下の記事がうまくまとまっています。
https://qiita.com/ko2ic/items/5b8079dc76d38f437631

ではrunCatchingではどのようにcontractが使われているのでしょうか?

public inline fun <V> runCatching(block: () -> V): Result<V, Throwable> {
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    // 中略
}

callsInPlace(block, InvocationKind.EXACTLY_ONCE) はblockという関数は必ず一度評価されるという意味です。

https://kotlinlang.org/api/latest/jvm/stdlib/kotlin.contracts/-contract-builder/calls-in-place.html

これがどう嬉しいの?という話を説明していきます。
たとえば var を使った場合、その変数を使う場合は初期化を済ます必要があります。

var i: Int
println(i)
// => Variable 'i' must be initialized

関数というのは場合によっては内部で処理が分岐する場合もあるので var を使った場合必ず初期化できるとは限りません。

fun runSomething(block: () -> Unit) {
    block()
}

var i: Int
runSomething { i = 1 }
println(i)
// => Variable 'i' must be initialized

しかし callsInPlace(block, InvocationKind.EXACTLY_ONCE) を使うと必ず一度実行するとコンパイルに伝えることができます。

fun runSomething(block: () -> Unit) {
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    
    block()
}

var i: Int
runSomething { i = 1 }
println(i)
// => 1

これがcallsInPlaceのよいところです。そのためrunCatchingも同様で以下のようなことができます。

var i: Int
runCatching { i = 1 }
println(i)
// => 1

レシーバーを受け取る高階関数 A.(B) -> C

kotlin-resultには2つのrunCatchingが存在します。以下の2つです。
内部の実装は同じです。インターフェースだけ違います。
ここまでは前者の方を説明していました。

public inline fun <V> runCatching(block: () -> V): Result<V, Throwable>

public inline infix fun <T, V> T.runCatching(block: T.() -> V): Result<V, Throwable>

後者の方は何が違うのでしょうか?
まずは拡張関数を使っていますね。

https://kotlinlang.org/docs/extensions.html

任意の型Tに対してrunCatchingメソッドを生やしています。そのため以下のようにIntのインスタンスに対してメソッドを呼ぶことができます。

1.runCatching { doSomething() }

引数のblockも違いますね。 block: T.() -> V のような型となっています。このT.()はレシーバーを受け取れるという関数の定義です。

https://kotlinlang.org/docs/lambdas.html#function-types

そのため以下のようなことができます。

class Person(
    val name: String
) {
    fun hello() {
        return "Hello, I am ${name}"
    }
}

val person = Person("taro")
val result: Result<String, Throwable> = person.runCatching { this.hello() }

thisを使ってpersonを受け取っていますね。
もちろんこのthisは省略可能です。

val result: Result<String, Throwable> = person.runCatching { hello() }

似たような関数としてはletがありますが、こちらはレシーバーではなく引数として受け取る形になっています。

public inline fun <T, R> T.let(block: (T) -> R): R {
    contract {
        callsInPlace(block, InvocationKind.EXACTLY_ONCE)
    }
    return block(this)
}

val person = Person("taro")
person.let { p -> p.hello() }

runCatchingでは block: T.() -> R だったものが、 letでは block: (T) -> R になっていることを注意してください(※比較のためジェネリクス名を揃えています)。

差分としては微々たるものですが、レシーバーを受け取るタイプの場合block: T.() -> Rは以下のように関数を丸ごと渡せることが便利かもしれないです。

class Person(
    val name: String
) {
    fun hello() {
        return "Hello, I am ${name}"
    }
}

val person = Person("taro")
// Person::hello をそのまま渡す。
val result: Result<String, Throwable> = person.runCatching(Person::hello)

そう考えると Person::hello という関数の型は Person.() -> String と表すことが可能ですね。

// 代入可能
val personHello: Person.() -> String = Person::hello

余談ですが、この新しく定義した personHello は拡張関数のように振る舞うことが可能です。

val personHello: Person.() -> String = Person::hello
val person = Person("taro")
person.personHello()

めちゃくちゃマニアックな話ではありますが、内部構造を理解する上では面白いですね。

infix

とてもマイナー記法かと思いますが、infixはドットを省略してメソッド呼び出しできる記法です。
ちょっとだけオシャレに書けます。

public inline infix fun <T, V> T.runCatching(block: T.() -> V): Result<V, Throwable>

// .ドットを省略
1 runCatching { doSomething }

getのコードリーディング

mapやflatMapなどいろいろ面白いのはあるのですが、最後はgetを読んでいきます。

https://github.com/michaelbull/kotlin-result/blob/master/kotlin-result/src/commonMain/kotlin/com/github/michaelbull/result/Get.kt

getはOk型なら成功値のvalueをNullableで返します。

val result: Result<String, Throwable> = runCatching { "a" }
val value: String? = result.get()

getはResult型の拡張関数として定義されています。

public fun <V, E> Result<V, E>.get(): V? {
    contract {
        returnsNotNull() implies (this@get is Ok<V>)
        returns(null) implies (this@get is Err<E>)
    }

    return when (this) {
        is Ok -> value
        is Err -> null
    }
}

型判定でOk型ならvalueを返していますね。そしてErrならnullを返しています。

implies contract

またでました。contractです。

public fun <V, E> Result<V, E>.get(): V? {
    contract {
        returnsNotNull() implies (this@get is Ok<V>)
        returns(null) implies (this@get is Err<E>)
    }
    // 中略
}

一つ目から見ていきます。

returnsNotNull() implies (this@get is Ok<V>)

こちらは「getがnullを返さない場合、getのレシーバーは Ok<V> 型である」という意味です。
Ok型にSmart Castができるということですね。

https://kotlinlang.org/api/latest/jvm/stdlib/kotlin.contracts/-simple-effect/implies.html

https://kotlinlang.org/api/latest/jvm/stdlib/kotlin.contracts/-contract-builder/returns-not-null.html

this@getもあまり聞きなれない記法ですね。公式ドキュメントを貼っておきます。
https://kotlinlang.org/docs/this-expressions.html

Ok型にSmart Castができるということで以下のようなことができます。

class Person(
    val name
)

val result = runCatching { Person("taro") }

if (result.get() != null) {
    println(result.value.name) // Smart Castで型が確定している
}

if節のなかでresultを明示的にキャストせず、 Ok<Person> 型として扱うことができます。
しかし現状のKotlinでは以下のように一度変数に受けるとSmart Castがうまく作動しないようです。
この辺も後々対応されていくような感じはします(感じがするだけ)。

val result = runCatching { Person("taro") }
val person: Person? = result.get()

if (person != null) {
    println(result.value.name)
    // コンパイルエラー
}

次はこちらのcontractですね。

returns(null) implies (this@get is Err<E>)

https://kotlinlang.org/api/latest/jvm/stdlib/kotlin.contracts/-contract-builder/returns.html

こちらは「getがnullを返す場合、getのレシーバーは Err<E> 型である」という意味です。
先ほどと逆ですね。

そのためこのようなことができます。

val result = runCatching { Person("taro") }

if (result.get() == null) {
    println(result.error) // Smart Castで型が確定している
}

implies contractについて分からなかったこと

上記のようなSmart Castは同じモジュール内ならできたのですが、なぜか別モジュールとしてインポートするとコンパイラがうまく動作しませんでした。
kotlin-resultをcloneして同モジュール内でgetを使うとSmart Castが作動するが、外部ライブラリとしてインポートすると作動しないということですね。

import com.github.michaelbull.result.runCatching

val result = runCatching { Person("taro") }

if (result.get() != null) {
    println(result.value.name)
    // なぜかコンパイルエラーになる
}

impliesについてはExperimentalContractsだからか?と思ったのですが、公式ドキュメントにはそのようなことは記載されていませんでした。

https://kotlinlang.org/api/latest/jvm/stdlib/kotlin.contracts/-experimental-contracts/

Any usage of a declaration annotated with @ExperimentalContracts must be accepted either by annotating that usage with the OptIn annotation, e.g. @OptIn(ExperimentalContracts::class), or by using the compiler argument -opt-in=kotlin.contracts.ExperimentalContracts.

と書いているので、OptInを記述してコンパイラを走らせたりしたのですがうまくいかず。。Stableになったらまた試してみます。
ちなみに上述の callsInPlace contract については問題なく動作しました。
どなたか詳しい方いれば是非お教えください、、、!

おわり

今回のOSS解説はこのへんで終わりにします。
他にも面白い関数があるのでぜひみてみてください。
kotlin-resultについては今回解説したところを抑えておけば基本読み進められると思います。

これにて 株式会社ログラスDevアドベントカレンダー2022の12/3(土) の記事は終了です。
明日はログラス開発組織のフルスタック大エースエンジニアの @urmot の記事です!お楽しみに!

https://twitter.com/urmot2

株式会社ログラス テックブログ

Discussion