Httpクライアントのテスト(モック)

Estimated reading time: 1 minute

Ktorは、HttpClientのMockEngineを公開します。 このエンジンにより、実際にエンドポイントに接続せずにHTTP呼び出しをシミュレートできます。 リクエストを処理してレスポンスを生成できるコードブロックを設定できます。

This engine is defined in the class io.ktor.client.engine.mock.MockEngine in the artifact io.ktor:ktor-client-mock:$ktor_version,io.ktor:ktor-client-mock-jvm:$ktor_version,io.ktor:ktor-client-mock-js:$ktor_version,io.ktor:ktor-client-mock-native:$ktor_version.
dependencies { api "io.ktor:ktor-client-mock:$ktor_version" api "io.ktor:ktor-client-mock-jvm:$ktor_version" api "io.ktor:ktor-client-mock-js:$ktor_version" api "io.ktor:ktor-client-mock-native:$ktor_version" }
dependencies { testImplementation("io.ktor:ktor-client-mock:$ktor_version") testImplementation("io.ktor:ktor-client-mock-jvm:$ktor_version") testImplementation("io.ktor:ktor-client-mock-js:$ktor_version") testImplementation("io.ktor:ktor-client-mock-native:$ktor_version") }
<project> ... <dependencies> <dependency> <groupId>io.ktor</groupId> <artifactId>ktor-client-mock</artifactId> <version>${ktor.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>io.ktor</groupId> <artifactId>ktor-client-mock-jvm</artifactId> <version>${ktor.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>io.ktor</groupId> <artifactId>ktor-client-mock-js</artifactId> <version>${ktor.version}</version> <scope>test</scope> </dependency> <dependency> <groupId>io.ktor</groupId> <artifactId>ktor-client-mock-native</artifactId> <version>${ktor.version}</version> <scope>test</scope> </dependency> </dependencies> </project>

使い方

使い方は簡単です。MockEngineクラスには、MockEngineConfigaddHandlerメソッドがあり、リクエストを処理するブロック/コールバックを受け取ります。 このコールバックは、HttpRequestをパラメーターとして受け取り、HttpResponseDataを返す必要があります。レスポンスを作成する多くのヘルパーメソッドがあります。

完全なAPIの説明とヘルパーメソッドのリストは、こちらにあります。

以下がサンプルです:

val client = HttpClient(MockEngine) {
    engine {
        addHandler { request ->
            when (request.url.fullUrl) {
                "https://example.org/" -> {
                    val responseHeaders = headersOf("Content-Type" to listOf(ContentType.Text.Plain.toString()))
                    respond("Hello, world", headers = responseHeaders)
                }
                else -> error("Unhandled ${request.url.fullUrl}")
            }
        }
    }
}

private val Url.hostWithPortIfRequired: String get() = if (port == protocol.defaultPort) host else hostWithPort
private val Url.fullUrl: String get() = "${protocol.name}://$hostWithPortIfRequired$fullPath"