Browse Source

Merge pull request #1219 from vector-im/feature/bma/notificationDecryption

Feature/bma/notification decryption
pull/1215/head
Benoit Marty 1 year ago committed by GitHub
parent
commit
1a362fd47b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 5
      appnav/src/main/kotlin/io/element/android/appnav/LoggedInFlowNode.kt
  2. 1
      changelog.d/1178.bugfix
  3. 25
      libraries/matrix/api/src/main/kotlin/io/element/android/libraries/matrix/api/sync/StartSyncReason.kt
  4. 4
      libraries/matrix/api/src/main/kotlin/io/element/android/libraries/matrix/api/sync/SyncService.kt
  5. 40
      libraries/matrix/impl/src/main/kotlin/io/element/android/libraries/matrix/impl/sync/RustSyncService.kt
  6. 5
      libraries/matrix/test/src/main/kotlin/io/element/android/libraries/matrix/test/sync/FakeSyncService.kt
  7. 12
      libraries/push/impl/src/main/kotlin/io/element/android/libraries/push/impl/notifications/NotifiableEventResolver.kt
  8. 5
      samples/minimal/src/main/kotlin/io/element/android/samples/minimal/RoomListScreen.kt

5
appnav/src/main/kotlin/io/element/android/appnav/LoggedInFlowNode.kt

@ -60,6 +60,7 @@ import io.element.android.libraries.di.SessionScope
import io.element.android.libraries.matrix.api.MatrixClient import io.element.android.libraries.matrix.api.MatrixClient
import io.element.android.libraries.matrix.api.core.MAIN_SPACE import io.element.android.libraries.matrix.api.core.MAIN_SPACE
import io.element.android.libraries.matrix.api.core.RoomId import io.element.android.libraries.matrix.api.core.RoomId
import io.element.android.libraries.matrix.api.sync.StartSyncReason
import io.element.android.libraries.matrix.api.sync.SyncState import io.element.android.libraries.matrix.api.sync.SyncState
import io.element.android.libraries.push.api.notifications.NotificationDrawerManager import io.element.android.libraries.push.api.notifications.NotificationDrawerManager
import io.element.android.services.appnavstate.api.AppNavigationStateService import io.element.android.services.appnavstate.api.AppNavigationStateService
@ -124,7 +125,7 @@ class LoggedInFlowNode @AssistedInject constructor(
onStop = { onStop = {
//Counterpart startSync is done in observeSyncStateAndNetworkStatus method. //Counterpart startSync is done in observeSyncStateAndNetworkStatus method.
coroutineScope.launch { coroutineScope.launch {
syncService.stopSync() syncService.stopSync(StartSyncReason.AppInForeground)
} }
}, },
onDestroy = { onDestroy = {
@ -150,7 +151,7 @@ class LoggedInFlowNode @AssistedInject constructor(
.collect { (syncState, networkStatus) -> .collect { (syncState, networkStatus) ->
Timber.d("Sync state: $syncState, network status: $networkStatus") Timber.d("Sync state: $syncState, network status: $networkStatus")
if (syncState != SyncState.Running && networkStatus == NetworkStatus.Online) { if (syncState != SyncState.Running && networkStatus == NetworkStatus.Online) {
syncService.startSync() syncService.startSync(StartSyncReason.AppInForeground)
} }
} }
} }

1
changelog.d/1178.bugfix

@ -0,0 +1 @@
Ensure notification for Event from encrypted room get decrypted content.

25
libraries/matrix/api/src/main/kotlin/io/element/android/libraries/matrix/api/sync/StartSyncReason.kt

@ -0,0 +1,25 @@
/*
* Copyright (c) 2023 New Vector Ltd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.element.android.libraries.matrix.api.sync
import io.element.android.libraries.matrix.api.core.EventId
import io.element.android.libraries.matrix.api.core.RoomId
sealed interface StartSyncReason {
data object AppInForeground : StartSyncReason
data class Notification(val roomId: RoomId, val eventId: EventId) : StartSyncReason
}

4
libraries/matrix/api/src/main/kotlin/io/element/android/libraries/matrix/api/sync/SyncService.kt

@ -22,12 +22,12 @@ interface SyncService {
/** /**
* Tries to start the sync. If already syncing it has no effect. * Tries to start the sync. If already syncing it has no effect.
*/ */
suspend fun startSync(): Result<Unit> suspend fun startSync(reason: StartSyncReason): Result<Unit>
/** /**
* Tries to stop the sync. If service is not syncing it has no effect. * Tries to stop the sync. If service is not syncing it has no effect.
*/ */
suspend fun stopSync(): Result<Unit> suspend fun stopSync(reason: StartSyncReason): Result<Unit>
/** /**
* Flow of [SyncState]. Will be updated as soon as the current [SyncState] changes. * Flow of [SyncState]. Will be updated as soon as the current [SyncState] changes.

40
libraries/matrix/impl/src/main/kotlin/io/element/android/libraries/matrix/impl/sync/RustSyncService.kt

@ -16,6 +16,7 @@
package io.element.android.libraries.matrix.impl.sync package io.element.android.libraries.matrix.impl.sync
import io.element.android.libraries.matrix.api.sync.StartSyncReason
import io.element.android.libraries.matrix.api.sync.SyncService import io.element.android.libraries.matrix.api.sync.SyncService
import io.element.android.libraries.matrix.api.sync.SyncState import io.element.android.libraries.matrix.api.sync.SyncState
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
@ -25,6 +26,8 @@ import kotlinx.coroutines.flow.distinctUntilChanged
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.onEach import kotlinx.coroutines.flow.onEach
import kotlinx.coroutines.flow.stateIn import kotlinx.coroutines.flow.stateIn
import kotlinx.coroutines.sync.Mutex
import kotlinx.coroutines.sync.withLock
import org.matrix.rustcomponents.sdk.SyncServiceInterface import org.matrix.rustcomponents.sdk.SyncServiceInterface
import org.matrix.rustcomponents.sdk.SyncServiceState import org.matrix.rustcomponents.sdk.SyncServiceState
import timber.log.Timber import timber.log.Timber
@ -33,19 +36,36 @@ class RustSyncService(
private val innerSyncService: SyncServiceInterface, private val innerSyncService: SyncServiceInterface,
sessionCoroutineScope: CoroutineScope sessionCoroutineScope: CoroutineScope
) : SyncService { ) : SyncService {
private val mutex = Mutex()
private val startSyncReasonSet = mutableSetOf<StartSyncReason>()
override suspend fun startSync() = runCatching { override suspend fun startSync(reason: StartSyncReason): Result<Unit> {
Timber.i("Start sync") return mutex.withLock {
innerSyncService.start() startSyncReasonSet.add(reason)
}.onFailure { runCatching {
Timber.d("Start sync failed: $it") Timber.d("Start sync")
innerSyncService.start()
}.onFailure {
Timber.e("Start sync failed: $it")
}
}
} }
override suspend fun stopSync() = runCatching { override suspend fun stopSync(reason: StartSyncReason): Result<Unit> {
Timber.i("Stop sync") return mutex.withLock {
innerSyncService.stop() startSyncReasonSet.remove(reason)
}.onFailure { if (startSyncReasonSet.isEmpty()) {
Timber.d("Stop sync failed: $it") runCatching {
Timber.d("Stop sync")
innerSyncService.stop()
}.onFailure {
Timber.e("Stop sync failed: $it")
}
} else {
Timber.d("Stop sync skipped, still $startSyncReasonSet")
Result.success(Unit)
}
}
} }
override val syncState: StateFlow<SyncState> = override val syncState: StateFlow<SyncState> =

5
libraries/matrix/test/src/main/kotlin/io/element/android/libraries/matrix/test/sync/FakeSyncService.kt

@ -16,6 +16,7 @@
package io.element.android.libraries.matrix.test.sync package io.element.android.libraries.matrix.test.sync
import io.element.android.libraries.matrix.api.sync.StartSyncReason
import io.element.android.libraries.matrix.api.sync.SyncService import io.element.android.libraries.matrix.api.sync.SyncService
import io.element.android.libraries.matrix.api.sync.SyncState import io.element.android.libraries.matrix.api.sync.SyncState
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.MutableStateFlow
@ -29,12 +30,12 @@ class FakeSyncService : SyncService {
syncStateFlow.value = SyncState.Error syncStateFlow.value = SyncState.Error
} }
override suspend fun startSync(): Result<Unit> { override suspend fun startSync(reason: StartSyncReason): Result<Unit> {
syncStateFlow.value = SyncState.Running syncStateFlow.value = SyncState.Running
return Result.success(Unit) return Result.success(Unit)
} }
override suspend fun stopSync(): Result<Unit> { override suspend fun stopSync(reason: StartSyncReason): Result<Unit> {
syncStateFlow.value = SyncState.Terminated syncStateFlow.value = SyncState.Terminated
return Result.success(Unit) return Result.success(Unit)
} }

12
libraries/push/impl/src/main/kotlin/io/element/android/libraries/push/impl/notifications/NotifiableEventResolver.kt

@ -26,6 +26,7 @@ import io.element.android.libraries.matrix.api.core.UserId
import io.element.android.libraries.matrix.api.notification.NotificationContent import io.element.android.libraries.matrix.api.notification.NotificationContent
import io.element.android.libraries.matrix.api.notification.NotificationData import io.element.android.libraries.matrix.api.notification.NotificationData
import io.element.android.libraries.matrix.api.room.RoomMembershipState import io.element.android.libraries.matrix.api.room.RoomMembershipState
import io.element.android.libraries.matrix.api.sync.StartSyncReason
import io.element.android.libraries.matrix.api.timeline.item.event.AudioMessageType import io.element.android.libraries.matrix.api.timeline.item.event.AudioMessageType
import io.element.android.libraries.matrix.api.timeline.item.event.EmoteMessageType import io.element.android.libraries.matrix.api.timeline.item.event.EmoteMessageType
import io.element.android.libraries.matrix.api.timeline.item.event.FileMessageType import io.element.android.libraries.matrix.api.timeline.item.event.FileMessageType
@ -44,6 +45,7 @@ import io.element.android.libraries.push.impl.notifications.model.NotifiableMess
import io.element.android.libraries.ui.strings.CommonStrings import io.element.android.libraries.ui.strings.CommonStrings
import io.element.android.services.toolbox.api.strings.StringProvider import io.element.android.services.toolbox.api.strings.StringProvider
import io.element.android.services.toolbox.api.systemclock.SystemClock import io.element.android.services.toolbox.api.systemclock.SystemClock
import kotlinx.coroutines.delay
import timber.log.Timber import timber.log.Timber
import javax.inject.Inject import javax.inject.Inject
@ -65,6 +67,14 @@ class NotifiableEventResolver @Inject constructor(
// Restore session // Restore session
val client = matrixClientProvider.getOrRestore(sessionId).getOrNull() ?: return null val client = matrixClientProvider.getOrRestore(sessionId).getOrNull() ?: return null
val notificationService = client.notificationService() val notificationService = client.notificationService()
// Restart the sync service to ensure that the crypto sync handle the toDevice Events.
client.syncService().startSync(StartSyncReason.Notification(roomId, eventId))
// Wait for toDevice Event to be processed
// FIXME This delay can be removed when the Rust SDK will handle internal retry to get
// clear notification content.
delay(300)
val notificationData = notificationService.getNotification( val notificationData = notificationService.getNotification(
userId = sessionId, userId = sessionId,
roomId = roomId, roomId = roomId,
@ -73,6 +83,8 @@ class NotifiableEventResolver @Inject constructor(
Timber.tag(loggerTag.value).e(it, "Unable to resolve event: $eventId.") Timber.tag(loggerTag.value).e(it, "Unable to resolve event: $eventId.")
}.getOrNull() }.getOrNull()
client.syncService().stopSync(StartSyncReason.Notification(roomId, eventId))
// TODO this notificationData is not always valid at the moment, sometimes the Rust SDK can't fetch the matching event // TODO this notificationData is not always valid at the moment, sometimes the Rust SDK can't fetch the matching event
return notificationData?.asNotifiableEvent(sessionId) return notificationData?.asNotifiableEvent(sessionId)
} }

5
samples/minimal/src/main/kotlin/io/element/android/samples/minimal/RoomListScreen.kt

@ -39,6 +39,7 @@ import io.element.android.libraries.eventformatter.impl.StateContentFormatter
import io.element.android.libraries.matrix.api.MatrixClient import io.element.android.libraries.matrix.api.MatrixClient
import io.element.android.libraries.matrix.api.core.RoomId import io.element.android.libraries.matrix.api.core.RoomId
import io.element.android.libraries.matrix.api.room.RoomMembershipObserver import io.element.android.libraries.matrix.api.room.RoomMembershipObserver
import io.element.android.libraries.matrix.api.sync.StartSyncReason
import io.element.android.services.toolbox.impl.strings.AndroidStringProvider import io.element.android.services.toolbox.impl.strings.AndroidStringProvider
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import kotlinx.coroutines.runBlocking import kotlinx.coroutines.runBlocking
@ -109,12 +110,12 @@ class RoomListScreen(
DisposableEffect(Unit) { DisposableEffect(Unit) {
Timber.w("Start sync!") Timber.w("Start sync!")
runBlocking { runBlocking {
matrixClient.syncService().startSync() matrixClient.syncService().startSync(StartSyncReason.AppInForeground)
} }
onDispose { onDispose {
Timber.w("Stop sync!") Timber.w("Stop sync!")
runBlocking { runBlocking {
matrixClient.syncService().stopSync() matrixClient.syncService().stopSync(StartSyncReason.AppInForeground)
} }
} }
} }

Loading…
Cancel
Save