From 95fd7190e4ba3771c08e72c109cdbd932e6a0256 Mon Sep 17 00:00:00 2001 From: ganfra Date: Sat, 13 Oct 2018 11:18:49 +0200 Subject: [PATCH] Better coroutines management --- .idea/caches/build_file_checksums.ser | Bin 653 -> 653 bytes .idea/codeStyles/codeStyleConfig.xml | 5 ++ build.gradle | 4 +- .../im/vector/matrix/android/api/Matrix.kt | 5 +- .../android/internal/auth/AuthModule.kt | 2 +- .../internal/auth/DefaultAuthenticator.kt | 49 +++++++++++------- .../android/internal/di/MatrixModule.kt | 3 +- .../android/internal/di/MoshiProvider.kt | 2 +- .../internal/events/sync/RoomSyncHandler.kt | 28 ++++++---- .../internal/events/sync/SyncModule.kt | 2 +- .../internal/events/sync/Synchronizer.kt | 33 ++++++------ .../android/internal/network/Request.kt | 12 ++--- 12 files changed, 81 insertions(+), 64 deletions(-) create mode 100644 .idea/codeStyles/codeStyleConfig.xml diff --git a/.idea/caches/build_file_checksums.ser b/.idea/caches/build_file_checksums.ser index f844065a2b5b117fb29e9fb47bb347954b242e71..b8a09f1794794a39dcb3fc6f4aa34972183cdfdb 100644 GIT binary patch delta 36 ucmV+<0NekK1&sxem;`LH^**tj5&;l=aQc)}vc`RFu5@E~jDW(Ez5&lBIuR`Z delta 36 ucmV+<0NekK1&sxem;_ + + + \ No newline at end of file diff --git a/build.gradle b/build.gradle index 0d0a9a596f..6487402879 100644 --- a/build.gradle +++ b/build.gradle @@ -1,7 +1,7 @@ // Top-level build file where you can add configuration options common to all sub-projects/modules. buildscript { - ext.kotlin_version = '1.3.0-rc-116' + ext.kotlin_version = '1.3.0-rc-146' ext.koin_version = '1.0.1' repositories { google() @@ -9,7 +9,7 @@ buildscript { maven { url 'https://dl.bintray.com/kotlin/kotlin-eap' } } dependencies { - classpath 'com.android.tools.build:gradle:3.2.0' + classpath 'com.android.tools.build:gradle:3.2.1' classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version" // NOTE: Do not place your application dependencies here; they belong diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/api/Matrix.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/api/Matrix.kt index 855a4d3d4c..0badf5c4c5 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/api/Matrix.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/api/Matrix.kt @@ -4,7 +4,6 @@ import android.content.Context import im.vector.matrix.android.BuildConfig import im.vector.matrix.android.api.auth.Authenticator import im.vector.matrix.android.api.session.Session -import im.vector.matrix.android.api.thread.MainThreadExecutor import im.vector.matrix.android.internal.auth.AuthModule import im.vector.matrix.android.internal.di.MatrixModule import im.vector.matrix.android.internal.di.NetworkModule @@ -13,7 +12,6 @@ import org.koin.standalone.StandAloneContext.loadKoinModules import org.koin.standalone.inject import timber.log.Timber import timber.log.Timber.DebugTree -import java.util.concurrent.Executor class Matrix(matrixOptions: MatrixOptions) : KoinComponent { @@ -38,5 +36,4 @@ class Matrix(matrixOptions: MatrixOptions) : KoinComponent { } -data class MatrixOptions(val context: Context, - val mainExecutor: Executor = MainThreadExecutor()) +data class MatrixOptions(val context: Context) diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/AuthModule.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/AuthModule.kt index 8b61de0a56..76b61b6b73 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/AuthModule.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/AuthModule.kt @@ -18,7 +18,7 @@ class AuthModule(private val context: Context) : Module { override fun invoke(): ModuleDefinition = module { single { - DefaultAuthenticator(get(), get(), get(), get()) as Authenticator + DefaultAuthenticator(get(), get(), get()) as Authenticator } single(name = AUTH_BOX_STORE) { diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/DefaultAuthenticator.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/DefaultAuthenticator.kt index 751fd40ff1..f468a55720 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/DefaultAuthenticator.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/auth/DefaultAuthenticator.kt @@ -1,7 +1,7 @@ package im.vector.matrix.android.internal.auth +import arrow.core.Either import arrow.core.leftIfNull -import com.squareup.moshi.Moshi import im.vector.matrix.android.api.MatrixCallback import im.vector.matrix.android.api.auth.Authenticator import im.vector.matrix.android.api.auth.data.HomeServerConnectionConfig @@ -17,10 +17,10 @@ import im.vector.matrix.android.internal.util.CancelableCoroutine import im.vector.matrix.android.internal.util.MatrixCoroutineDispatchers import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext import retrofit2.Retrofit class DefaultAuthenticator(private val retrofitBuilder: Retrofit.Builder, - private val jsonMapper: Moshi, private val coroutineDispatchers: MatrixCoroutineDispatchers, private val sessionParamsStore: SessionParamsStore) : Authenticator { @@ -35,28 +35,37 @@ class DefaultAuthenticator(private val retrofitBuilder: Retrofit.Builder, } } - override fun authenticate(homeServerConnectionConfig: HomeServerConnectionConfig, login: String, password: String, callback: MatrixCallback): Cancelable { - val authAPI = buildAuthAPI(homeServerConnectionConfig) + override fun authenticate(homeServerConnectionConfig: HomeServerConnectionConfig, + login: String, + password: String, + callback: MatrixCallback): Cancelable { val job = GlobalScope.launch(coroutineDispatchers.main) { - val loginParams = PasswordLoginParams.userIdentifier(login, password, "Mobile") - executeRequest { - apiCall = authAPI.login(loginParams) - moshi = jsonMapper - dispatcher = coroutineDispatchers.io - }.leftIfNull { - Failure.Unknown(IllegalArgumentException("Credentials shouldn't not be null")) - }.map { - val sessionParams = SessionParams(it, homeServerConnectionConfig) - sessionParamsStore.save(sessionParams) - sessionParams - }.map { - DefaultSession(it) - }.bimap( - { callback.onFailure(it) }, { callback.onSuccess(it) } - ) + val sessionOrFailure = authenticate(homeServerConnectionConfig, login, password) + sessionOrFailure.bimap({ callback.onFailure(it) }, { callback.onSuccess(it) }) } return CancelableCoroutine(job) + + } + + private suspend fun authenticate(homeServerConnectionConfig: HomeServerConnectionConfig, + login: String, + password: String): Either = withContext(coroutineDispatchers.io) { + + val authAPI = buildAuthAPI(homeServerConnectionConfig) + val loginParams = PasswordLoginParams.userIdentifier(login, password, "Mobile") + executeRequest { + apiCall = authAPI.login(loginParams) + }.leftIfNull { + Failure.Unknown(IllegalArgumentException("Credentials shouldn't not be null")) + }.map { + val sessionParams = SessionParams(it, homeServerConnectionConfig) + sessionParamsStore.save(sessionParams) + sessionParams + }.map { + DefaultSession(it) + } + } private fun buildAuthAPI(homeServerConnectionConfig: HomeServerConnectionConfig): AuthAPI { diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MatrixModule.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MatrixModule.kt index f3d0aebe8b..7d4cfbf728 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MatrixModule.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MatrixModule.kt @@ -1,6 +1,7 @@ package im.vector.matrix.android.internal.di import im.vector.matrix.android.api.MatrixOptions +import im.vector.matrix.android.api.thread.MainThreadExecutor import im.vector.matrix.android.internal.util.MatrixCoroutineDispatchers import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.IO @@ -19,7 +20,7 @@ class MatrixModule(private val options: MatrixOptions) : Module { } single { - MatrixCoroutineDispatchers(io = Dispatchers.IO, computation = Dispatchers.IO, main = options.mainExecutor.asCoroutineDispatcher()) + MatrixCoroutineDispatchers(io = Dispatchers.IO, computation = Dispatchers.IO, main = MainThreadExecutor().asCoroutineDispatcher()) } }.invoke() } \ No newline at end of file diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MoshiProvider.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MoshiProvider.kt index 1a38a172a8..a037f51c49 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MoshiProvider.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/di/MoshiProvider.kt @@ -11,4 +11,4 @@ object MoshiProvider { return moshi } -} \ No newline at end of file +} diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/RoomSyncHandler.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/RoomSyncHandler.kt index eea6799ef4..e68344e015 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/RoomSyncHandler.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/RoomSyncHandler.kt @@ -1,5 +1,6 @@ package im.vector.matrix.android.internal.events.sync +import im.vector.matrix.android.api.events.Event import im.vector.matrix.android.internal.database.mapper.EventMapper import im.vector.matrix.android.internal.database.model.ChunkEntity import im.vector.matrix.android.internal.database.model.EventEntity @@ -23,8 +24,8 @@ class RoomSyncHandler( return } val roomEntities = ArrayList() - roomSyncByRoom.forEach { roomId, roomSync -> - val roomEntity = handleJoinedRoom(roomId, roomSync) + roomSyncByRoom.forEach { + val roomEntity = handleJoinedRoom(it.key, it.value) roomEntities.add(roomEntity) } roomBox.put(roomEntities) @@ -40,22 +41,27 @@ class RoomSyncHandler( } roomEntity.membership = RoomEntity.Membership.JOINED if (roomSync.timeline != null) { - val chunkEntity = ChunkEntity() - chunkEntity.prevToken = roomSync.timeline.prevBatch - roomSync.timeline.events - .map { event -> EventMapper.map(event) } - .forEach { chunkEntity.events.add(it) } + val chunkEntity = eventListToChunk(roomSync.timeline.events, roomSync.timeline.prevBatch) roomEntity.chunks.add(chunkEntity) } if (roomSync.state != null) { - val chunkEntity = ChunkEntity() - roomSync.state.events - .map { event -> EventMapper.map(event) } - .forEach { chunkEntity.events.add(it) } + val chunkEntity = eventListToChunk(roomSync.state.events) roomEntity.chunks.add(chunkEntity) } return roomEntity } + private fun eventListToChunk(eventList: List, + prevToken: String? = null, + nextToken: String? = null): ChunkEntity { + val chunkEntity = ChunkEntity() + chunkEntity.prevToken = prevToken + chunkEntity.nextToken = nextToken + eventList + .map { event -> EventMapper.map(event) } + .forEach { chunkEntity.events.add(it) } + return chunkEntity + } + } \ No newline at end of file diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/SyncModule.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/SyncModule.kt index 6566290ea6..cea92d7fe4 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/SyncModule.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/SyncModule.kt @@ -24,7 +24,7 @@ class SyncModule : Module { } scope(DefaultSession.SCOPE) { - Synchronizer(get(), get(), get(), get()) + Synchronizer(get(), get(), get()) } }.invoke() diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/Synchronizer.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/Synchronizer.kt index 5373381416..a0f6b3c9b9 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/Synchronizer.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/events/sync/Synchronizer.kt @@ -1,6 +1,5 @@ package im.vector.matrix.android.internal.events.sync -import com.squareup.moshi.Moshi import im.vector.matrix.android.api.MatrixCallback import im.vector.matrix.android.api.util.Cancelable import im.vector.matrix.android.internal.events.sync.data.SyncResponse @@ -11,30 +10,32 @@ import im.vector.matrix.android.internal.util.CancelableCoroutine import im.vector.matrix.android.internal.util.MatrixCoroutineDispatchers import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.launch +import kotlinx.coroutines.withContext class Synchronizer(private val syncAPI: SyncAPI, private val coroutineDispatchers: MatrixCoroutineDispatchers, - private val jsonMapper: Moshi, private val syncResponseHandler: SyncResponseHandler) { fun synchronize(callback: MatrixCallback): Cancelable { val job = GlobalScope.launch(coroutineDispatchers.main) { - val params = HashMap() - val filterBody = FilterBody() - FilterUtil.enableLazyLoading(filterBody, true) - params["timeout"] = "0" - params["filter"] = filterBody.toJSONString() - val syncResponse = executeRequest { - apiCall = syncAPI.sync(params) - moshi = jsonMapper - dispatcher = coroutineDispatchers.io - }.map { - syncResponseHandler.handleResponse(it, null, false) - it - } - syncResponse.bimap({ callback.onFailure(it) }, { callback.onSuccess(it) }) + val syncOrFailure = synchronize() + syncOrFailure.bimap({ callback.onFailure(it) }, { callback.onSuccess(it) }) } return CancelableCoroutine(job) } + private suspend fun synchronize() = withContext(coroutineDispatchers.io) { + val params = HashMap() + val filterBody = FilterBody() + FilterUtil.enableLazyLoading(filterBody, true) + params["timeout"] = "0" + params["filter"] = filterBody.toJSONString() + executeRequest { + apiCall = syncAPI.sync(params) + }.map { + syncResponseHandler.handleResponse(it, null, false) + it + } + } + } \ No newline at end of file diff --git a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/network/Request.kt b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/network/Request.kt index a6692f38bc..4184e2cf43 100644 --- a/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/network/Request.kt +++ b/matrix-sdk-android/src/main/java/im/vector/matrix/android/internal/network/Request.kt @@ -4,9 +4,9 @@ import arrow.core.Either import com.squareup.moshi.Moshi import im.vector.matrix.android.api.failure.Failure import im.vector.matrix.android.api.failure.MatrixError -import kotlinx.coroutines.CoroutineDispatcher +import im.vector.matrix.android.internal.di.MoshiProvider import kotlinx.coroutines.Deferred -import kotlinx.coroutines.withContext +import kotlinx.coroutines.coroutineScope import okhttp3.ResponseBody import retrofit2.Response import java.io.IOException @@ -15,13 +15,11 @@ suspend inline fun executeRequest(block: Request.() -> Unit) = Requ class Request { + var moshi: Moshi = MoshiProvider.providesMoshi() lateinit var apiCall: Deferred> - lateinit var moshi: Moshi - lateinit var dispatcher: CoroutineDispatcher - - suspend fun execute(): Either = withContext(dispatcher) { - return@withContext try { + suspend fun execute(): Either = coroutineScope { + try { val response = apiCall.await() if (response.isSuccessful) { val result = response.body()