From 53e57ba8086d5e93773cca237be7d1f6964a7293 Mon Sep 17 00:00:00 2001 From: tomridder <1402709211@qq.com> Date: Wed, 5 Apr 2023 15:59:49 +0800 Subject: [PATCH] Add support for Kotlin's Result --- gradle/libs.versions.toml | 2 + retrofit/kotlin-test/build.gradle | 2 + .../test/java/retrofit2/KotlinSuspendTest.kt | 45 +++++++++- .../java/retrofit2/HttpServiceMethod.java | 10 ++- .../retrofit2/ResultCallAdapterFactory.kt | 88 +++++++++++++++++++ 5 files changed, 143 insertions(+), 4 deletions(-) create mode 100644 retrofit/src/main/java/retrofit2/ResultCallAdapterFactory.kt diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 2084330ec9..0bfe7b1bc8 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -44,6 +44,8 @@ androidxTestRunner = { module = "androidx.test:runner", version = "1.4.0" } rxjava = { module = "io.reactivex:rxjava", version = "1.3.8" } rxjava2 = { module = "io.reactivex.rxjava2:rxjava", version = "2.2.21" } rxjava3 = { module = "io.reactivex.rxjava3:rxjava", version = "3.1.6" } +rxjavaAdapter = { module = "com.squareup.retrofit2:adapter-rxjava2", version = "2.9.0"} +rxjavaAdapter2 = { module = "com.squareup.retrofit2:converter-gson", version = "2.9.0"} reactiveStreams = { module = "org.reactivestreams:reactive-streams", version = "1.0.4" } scalaLibrary = { module = "org.scala-lang:scala-library", version = "2.13.10" } gson = { module = "com.google.code.gson:gson", version = "2.10.1" } diff --git a/retrofit/kotlin-test/build.gradle b/retrofit/kotlin-test/build.gradle index b39ed4e17c..6733122e58 100644 --- a/retrofit/kotlin-test/build.gradle +++ b/retrofit/kotlin-test/build.gradle @@ -8,4 +8,6 @@ dependencies { testImplementation libs.mockwebserver testImplementation libs.kotlinStdLib testImplementation libs.kotlinCoroutines + testImplementation libs.rxjavaAdapter + testImplementation libs.rxjavaAdapter2 } diff --git a/retrofit/kotlin-test/src/test/java/retrofit2/KotlinSuspendTest.kt b/retrofit/kotlin-test/src/test/java/retrofit2/KotlinSuspendTest.kt index 260fd7ab98..e203fcd034 100644 --- a/retrofit/kotlin-test/src/test/java/retrofit2/KotlinSuspendTest.kt +++ b/retrofit/kotlin-test/src/test/java/retrofit2/KotlinSuspendTest.kt @@ -26,8 +26,6 @@ import okhttp3.mockwebserver.MockWebServer import okhttp3.mockwebserver.SocketPolicy.DISCONNECT_AFTER_REQUEST import okhttp3.mockwebserver.SocketPolicy.NO_RESPONSE import org.assertj.core.api.Assertions.assertThat -import org.junit.Assert.assertTrue -import org.junit.Assert.fail import org.junit.Ignore import org.junit.Rule import org.junit.Test @@ -39,6 +37,8 @@ import java.io.IOException import java.lang.reflect.ParameterizedType import java.lang.reflect.Type import kotlin.coroutines.CoroutineContext +import org.junit.Assert.* +import retrofit2.converter.gson.GsonConverterFactory class KotlinSuspendTest { @get:Rule val server = MockWebServer() @@ -50,6 +50,9 @@ class KotlinSuspendTest { @GET("/") suspend fun unit() @HEAD("/") suspend fun headUnit() + @GET("user") + suspend fun getUser(): Result + @GET("/{a}/{b}/{c}") suspend fun params( @Path("a") a: String, @@ -58,6 +61,8 @@ class KotlinSuspendTest { ): String } + data class User(val id:Int,val name: String,val email:String) + @Test fun body() { val retrofit = Retrofit.Builder() .baseUrl(server.url("/")) @@ -353,6 +358,42 @@ class KotlinSuspendTest { } } + @Test + fun testSuccessfulResponse() = runBlocking { + val responseBody = """ + { + "id": 1, + "name": "John Doe", + "email": "john.doe@example.com" + } + """.trimIndent() + server.enqueue(MockResponse().setResponseCode(200).setBody(responseBody)) + val retrofit = Retrofit.Builder() + .baseUrl(server.url("/")) + .addCallAdapterFactory(ResultCallAdapterFactory()) + .addConverterFactory(GsonConverterFactory.create()) + .build() + val service = retrofit.create(Service::class.java) + val result = service.getUser() + assertEquals(1, result.getOrNull()?.id) + assertEquals("John Doe", result.getOrNull()?.name) + assertEquals("john.doe@example.com", result.getOrNull()?.email) + } + + @Test + fun testErrorResponse() = runBlocking { + server.enqueue(MockResponse().setResponseCode(404)) + val retrofit = Retrofit.Builder() + .baseUrl(server.url("/")) + .addCallAdapterFactory(ResultCallAdapterFactory()) + .addConverterFactory(GsonConverterFactory.create()) + .build() + val service = retrofit.create(Service::class.java) + val result = service.getUser() + assert(result.isFailure) + } + + @Suppress("EXPERIMENTAL_OVERRIDE") private object DirectUnconfinedDispatcher : CoroutineDispatcher() { override fun isDispatchNeeded(context: CoroutineContext): Boolean = false diff --git a/retrofit/src/main/java/retrofit2/HttpServiceMethod.java b/retrofit/src/main/java/retrofit2/HttpServiceMethod.java index 2ca5c7f0ae..42b72ce4dd 100644 --- a/retrofit/src/main/java/retrofit2/HttpServiceMethod.java +++ b/retrofit/src/main/java/retrofit2/HttpServiceMethod.java @@ -23,6 +23,8 @@ import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; import javax.annotation.Nullable; + +import kotlin.Result; import kotlin.Unit; import kotlin.coroutines.Continuation; import okhttp3.ResponseBody; @@ -52,15 +54,19 @@ static HttpServiceMethod parseAnnotatio // Unwrap the actual body type from Response. responseType = Utils.getParameterUpperBound(0, (ParameterizedType) responseType); continuationWantsResponse = true; + adapterType = new Utils.ParameterizedTypeImpl(null, Call.class, responseType); } else { + if ((getRawType(responseType).isAssignableFrom(Result.class))) { + adapterType = responseType; + } else { + adapterType = new Utils.ParameterizedTypeImpl(null, Call.class, responseType); + } continuationIsUnit = Utils.isUnit(responseType); // TODO figure out if type is nullable or not // Metadata metadata = method.getDeclaringClass().getAnnotation(Metadata.class) // Find the entry for method // Determine if return type is nullable or not } - - adapterType = new Utils.ParameterizedTypeImpl(null, Call.class, responseType); annotations = SkipCallbackExecutorImpl.ensurePresent(annotations); } else { adapterType = method.getGenericReturnType(); diff --git a/retrofit/src/main/java/retrofit2/ResultCallAdapterFactory.kt b/retrofit/src/main/java/retrofit2/ResultCallAdapterFactory.kt new file mode 100644 index 0000000000..b842250601 --- /dev/null +++ b/retrofit/src/main/java/retrofit2/ResultCallAdapterFactory.kt @@ -0,0 +1,88 @@ +package retrofit2 + +import java.io.IOException +import java.lang.reflect.ParameterizedType +import java.lang.reflect.Type +import okhttp3.Request +import okio.Timeout + +class ResultCallAdapterFactory : CallAdapter.Factory() { + override fun get( + returnType: Type, + annotations: Array, + retrofit: Retrofit + ): CallAdapter<*, *>? { + if (getRawType(returnType) != Result::class.java) { + return null + } + + check(returnType is ParameterizedType) { + "Result must have a generic type (e.g., Result)" + } + + val responseType = getParameterUpperBound(0, returnType) + return ResultCallAdapter(responseType) + } +} + +class ResultCallAdapter( + private val responseType: Type +) : CallAdapter>> { + + override fun responseType(): Type { + return responseType + } + + override fun adapt(call: Call): Call> { + return ResultCall(call) + } +} + +class ResultCall(private val delegate: Call) : Call> { + + override fun enqueue(callback: Callback>) { + delegate.enqueue(object : Callback { + override fun onResponse(call: Call, response: Response) { + val result = if (response.isSuccessful) { + Result.success(response.body()!!) + } else { + Result.failure(HttpException(response)) + } + callback.onResponse(this@ResultCall, Response.success(result)) + } + + override fun onFailure(call: Call, t: Throwable) { + callback.onResponse(this@ResultCall, Response.success(Result.failure(t))) + } + }) + } + + override fun execute(): Response> { + return try { + val response = delegate.execute() + val result = if (response.isSuccessful) { + Result.success(response.body()!!) + } else { + Result.failure(IOException("Unexpected error: ${response.errorBody()?.string()}")) + } + Response.success(result) + } catch (e: IOException) { + Response.success(Result.failure(e)) + } catch (e: Exception) { + Response.success(Result.failure(e)) + } + } + override fun isExecuted(): Boolean = delegate.isExecuted + + override fun clone(): ResultCall = ResultCall(delegate.clone()) + + override fun isCanceled(): Boolean = delegate.isCanceled + + override fun cancel() = delegate.cancel() + + override fun request(): Request = delegate.request() + + override fun timeout(): Timeout = delegate.timeout() +} + +