Catching crash when offline during stop of a live location share
This commit is contained in:
parent
e55c378683
commit
9eba3034db
@ -53,7 +53,7 @@ interface LocationSharingService {
|
|||||||
/**
|
/**
|
||||||
* Stops sharing live location in the room.
|
* Stops sharing live location in the room.
|
||||||
*/
|
*/
|
||||||
suspend fun stopLiveLocationShare()
|
suspend fun stopLiveLocationShare(): UpdateLiveLocationShareResult
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a LiveData on the list of current running live location shares.
|
* Returns a LiveData on the list of current running live location shares.
|
||||||
|
@ -75,7 +75,7 @@ internal class DefaultLocationSharingService @AssistedInject constructor(
|
|||||||
return startLiveLocationShareTask.execute(params)
|
return startLiveLocationShareTask.execute(params)
|
||||||
}
|
}
|
||||||
|
|
||||||
override suspend fun stopLiveLocationShare() {
|
override suspend fun stopLiveLocationShare(): UpdateLiveLocationShareResult {
|
||||||
val params = StopLiveLocationShareTask.Params(
|
val params = StopLiveLocationShareTask.Params(
|
||||||
roomId = roomId,
|
roomId = roomId,
|
||||||
)
|
)
|
||||||
|
@ -22,6 +22,7 @@ import org.matrix.android.sdk.api.session.events.model.Event
|
|||||||
import org.matrix.android.sdk.api.session.events.model.EventType
|
import org.matrix.android.sdk.api.session.events.model.EventType
|
||||||
import org.matrix.android.sdk.api.session.events.model.toContent
|
import org.matrix.android.sdk.api.session.events.model.toContent
|
||||||
import org.matrix.android.sdk.api.session.events.model.toModel
|
import org.matrix.android.sdk.api.session.events.model.toModel
|
||||||
|
import org.matrix.android.sdk.api.session.room.location.UpdateLiveLocationShareResult
|
||||||
import org.matrix.android.sdk.api.session.room.model.message.MessageBeaconInfoContent
|
import org.matrix.android.sdk.api.session.room.model.message.MessageBeaconInfoContent
|
||||||
import org.matrix.android.sdk.internal.di.UserId
|
import org.matrix.android.sdk.internal.di.UserId
|
||||||
import org.matrix.android.sdk.internal.session.room.state.SendStateTask
|
import org.matrix.android.sdk.internal.session.room.state.SendStateTask
|
||||||
@ -29,23 +30,23 @@ import org.matrix.android.sdk.internal.session.room.state.StateEventDataSource
|
|||||||
import org.matrix.android.sdk.internal.task.Task
|
import org.matrix.android.sdk.internal.task.Task
|
||||||
import javax.inject.Inject
|
import javax.inject.Inject
|
||||||
|
|
||||||
internal interface StopLiveLocationShareTask : Task<StopLiveLocationShareTask.Params, Unit> {
|
internal interface StopLiveLocationShareTask : Task<StopLiveLocationShareTask.Params, UpdateLiveLocationShareResult> {
|
||||||
data class Params(
|
data class Params(
|
||||||
val roomId: String,
|
val roomId: String,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO update unit tests
|
||||||
internal class DefaultStopLiveLocationShareTask @Inject constructor(
|
internal class DefaultStopLiveLocationShareTask @Inject constructor(
|
||||||
@UserId private val userId: String,
|
@UserId private val userId: String,
|
||||||
private val sendStateTask: SendStateTask,
|
private val sendStateTask: SendStateTask,
|
||||||
private val stateEventDataSource: StateEventDataSource,
|
private val stateEventDataSource: StateEventDataSource,
|
||||||
) : StopLiveLocationShareTask {
|
) : StopLiveLocationShareTask {
|
||||||
|
|
||||||
@Throws
|
override suspend fun execute(params: StopLiveLocationShareTask.Params): UpdateLiveLocationShareResult {
|
||||||
override suspend fun execute(params: StopLiveLocationShareTask.Params) {
|
val beaconInfoStateEvent = getLiveLocationBeaconInfoForUser(userId, params.roomId) ?: return getResultForIncorrectBeaconInfoEvent()
|
||||||
val beaconInfoStateEvent = getLiveLocationBeaconInfoForUser(userId, params.roomId) ?: return
|
val stateKey = beaconInfoStateEvent.stateKey ?: return getResultForIncorrectBeaconInfoEvent()
|
||||||
val stateKey = beaconInfoStateEvent.stateKey ?: return
|
val content = beaconInfoStateEvent.getClearContent()?.toModel<MessageBeaconInfoContent>() ?: return getResultForIncorrectBeaconInfoEvent()
|
||||||
val content = beaconInfoStateEvent.getClearContent()?.toModel<MessageBeaconInfoContent>() ?: return
|
|
||||||
val updatedContent = content.copy(isLive = false).toContent()
|
val updatedContent = content.copy(isLive = false).toContent()
|
||||||
val sendStateTaskParams = SendStateTask.Params(
|
val sendStateTaskParams = SendStateTask.Params(
|
||||||
roomId = params.roomId,
|
roomId = params.roomId,
|
||||||
@ -53,9 +54,21 @@ internal class DefaultStopLiveLocationShareTask @Inject constructor(
|
|||||||
eventType = EventType.STATE_ROOM_BEACON_INFO.first(),
|
eventType = EventType.STATE_ROOM_BEACON_INFO.first(),
|
||||||
body = updatedContent
|
body = updatedContent
|
||||||
)
|
)
|
||||||
sendStateTask.executeRetry(sendStateTaskParams, 3)
|
return try {
|
||||||
|
val eventId = sendStateTask.executeRetry(sendStateTaskParams, 3)
|
||||||
|
if (eventId.isNotEmpty()) {
|
||||||
|
UpdateLiveLocationShareResult.Success(eventId)
|
||||||
|
} else {
|
||||||
|
UpdateLiveLocationShareResult.Failure(Exception("empty event id for new state event"))
|
||||||
|
}
|
||||||
|
} catch (error: Throwable) {
|
||||||
|
UpdateLiveLocationShareResult.Failure(error)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun getResultForIncorrectBeaconInfoEvent() =
|
||||||
|
UpdateLiveLocationShareResult.Failure(Exception("incorrect last beacon info event"))
|
||||||
|
|
||||||
private fun getLiveLocationBeaconInfoForUser(userId: String, roomId: String): Event? {
|
private fun getLiveLocationBeaconInfoForUser(userId: String, roomId: String): Event? {
|
||||||
return EventType.STATE_ROOM_BEACON_INFO
|
return EventType.STATE_ROOM_BEACON_INFO
|
||||||
.mapNotNull {
|
.mapNotNull {
|
||||||
|
@ -131,25 +131,28 @@ class LocationSharingService : VectorService(), LocationTracker.Callback {
|
|||||||
fun stopSharingLocation(roomId: String) {
|
fun stopSharingLocation(roomId: String) {
|
||||||
Timber.i("### LocationSharingService.stopSharingLocation for $roomId")
|
Timber.i("### LocationSharingService.stopSharingLocation for $roomId")
|
||||||
|
|
||||||
// Send a new beacon info state by setting live field as false
|
launchInIO { session ->
|
||||||
sendStoppedBeaconInfo(roomId)
|
// Send a new beacon info state by setting live field as false
|
||||||
|
when (sendStoppedBeaconInfo(session, roomId)) {
|
||||||
|
is UpdateLiveLocationShareResult.Success -> {
|
||||||
|
synchronized(roomArgsMap) {
|
||||||
|
val beaconIds = roomArgsMap
|
||||||
|
.filter { it.value.roomId == roomId }
|
||||||
|
.map { it.key }
|
||||||
|
beaconIds.forEach { roomArgsMap.remove(it) }
|
||||||
|
|
||||||
synchronized(roomArgsMap) {
|
tryToDestroyMe()
|
||||||
val beaconIds = roomArgsMap
|
}
|
||||||
.filter { it.value.roomId == roomId }
|
}
|
||||||
.map { it.key }
|
else -> Unit
|
||||||
beaconIds.forEach { roomArgsMap.remove(it) }
|
}
|
||||||
|
|
||||||
tryToDestroyMe()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun sendStoppedBeaconInfo(roomId: String) {
|
private suspend fun sendStoppedBeaconInfo(session: Session, roomId: String): UpdateLiveLocationShareResult? {
|
||||||
launchInIO { session ->
|
return session.getRoom(roomId)
|
||||||
session.getRoom(roomId)
|
?.locationSharingService()
|
||||||
?.locationSharingService()
|
?.stopLiveLocationShare()
|
||||||
?.stopLiveLocationShare()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onLocationUpdate(locationData: LocationData) {
|
override fun onLocationUpdate(locationData: LocationData) {
|
||||||
|
@ -28,6 +28,7 @@ class LocationSharingServiceConnection @Inject constructor(
|
|||||||
) : ServiceConnection {
|
) : ServiceConnection {
|
||||||
|
|
||||||
interface Callback {
|
interface Callback {
|
||||||
|
// TODO add onLocationServiceError()
|
||||||
fun onLocationServiceRunning()
|
fun onLocationServiceRunning()
|
||||||
fun onLocationServiceStopped()
|
fun onLocationServiceStopped()
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user