-
Notifications
You must be signed in to change notification settings - Fork 749
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fixes myroomnick changing Display Name #5618
Changes from 22 commits
bbfefbf
4c1e651
927eeb3
0200f48
6c1f6dc
7f64c62
25ca684
c650843
981d157
c1b1386
6c3c309
304e2f1
669af4d
4465525
de8cc07
b88eaf1
e80386b
f103d46
5d5a06b
2337b18
8964b55
5f949e9
dd988d0
8b80cf0
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
Fixes display name being changed when using /myroomnick |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -33,23 +33,49 @@ internal class RoomMemberEventHandler @Inject constructor( | |
@UserId private val myUserId: String | ||
) { | ||
|
||
fun handle(realm: Realm, roomId: String, event: Event, aggregator: SyncResponsePostTreatmentAggregator? = null): Boolean { | ||
fun handle(realm: Realm, | ||
roomId: String, | ||
event: Event, | ||
isInitialSync: Boolean, | ||
aggregator: SyncResponsePostTreatmentAggregator? = null): Boolean { | ||
if (event.type != EventType.STATE_ROOM_MEMBER) { | ||
return false | ||
} | ||
val userId = event.stateKey ?: return false | ||
val roomMember = event.getFixedRoomMemberContent() | ||
return handle(realm, roomId, userId, roomMember, aggregator) | ||
val eventUserId = event.stateKey ?: return false | ||
val roomMember = event.getFixedRoomMemberContent() ?: return false | ||
|
||
return if (isInitialSync) { | ||
handleInitialSync(realm, roomId, myUserId, eventUserId, roomMember, aggregator) | ||
} else { | ||
handleIncrementalSync( | ||
realm, | ||
roomId, | ||
eventUserId, | ||
roomMember, | ||
event.resolvedPrevContent(), | ||
aggregator | ||
) | ||
} | ||
} | ||
|
||
fun handle(realm: Realm, | ||
roomId: String, | ||
userId: String, | ||
roomMember: RoomMemberContent?, | ||
aggregator: SyncResponsePostTreatmentAggregator? = null): Boolean { | ||
if (roomMember == null) { | ||
return false | ||
private fun handleInitialSync(realm: Realm, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Having two methods with the same name, one public, one private is not ideal to me. |
||
roomId: String, | ||
currentUserId: String, | ||
eventUserId: String, | ||
roomMember: RoomMemberContent, | ||
aggregator: SyncResponsePostTreatmentAggregator?): Boolean { | ||
if (currentUserId != eventUserId) { | ||
saveUserEntityLocallyIfNecessary(realm, eventUserId, roomMember) | ||
} | ||
saveRoomMemberEntityLocally(realm, roomId, eventUserId, roomMember) | ||
updateDirectChatsIfNecessary(roomId, roomMember, aggregator) | ||
return true | ||
} | ||
|
||
private fun saveRoomMemberEntityLocally(realm: Realm, | ||
roomId: String, | ||
userId: String, | ||
roomMember: RoomMemberContent) { | ||
val roomMemberEntity = RoomMemberEntityFactory.create( | ||
roomId, | ||
userId, | ||
|
@@ -58,26 +84,62 @@ internal class RoomMemberEventHandler @Inject constructor( | |
// but we want to preserve presence record value and not replace it with null | ||
getExistingPresenceState(realm, roomId, userId)) | ||
realm.insertOrUpdate(roomMemberEntity) | ||
} | ||
|
||
/** | ||
* Get the already existing presence state for a specific user & room in order NOT to be replaced in RoomMemberSummaryEntity | ||
* by NULL value. | ||
*/ | ||
private fun getExistingPresenceState(realm: Realm, roomId: String, userId: String): UserPresenceEntity? { | ||
return RoomMemberSummaryEntity.where(realm, roomId, userId).findFirst()?.userPresenceEntity | ||
} | ||
|
||
private fun saveUserEntityLocallyIfNecessary(realm: Realm, | ||
userId: String, | ||
roomMember: RoomMemberContent) { | ||
if (roomMember.membership.isActive()) { | ||
val userEntity = UserEntityFactory.create(userId, roomMember) | ||
realm.insertOrUpdate(userEntity) | ||
saveUserLocally(realm, userId, roomMember) | ||
} | ||
} | ||
|
||
private fun saveUserLocally(realm: Realm, userId: String, roomMember: RoomMemberContent) { | ||
val userEntity = UserEntityFactory.create(userId, roomMember) | ||
realm.insertOrUpdate(userEntity) | ||
} | ||
|
||
private fun updateDirectChatsIfNecessary(roomId: String, | ||
roomMember: RoomMemberContent, | ||
aggregator: SyncResponsePostTreatmentAggregator?) { | ||
// check whether this new room member event may be used to update the directs dictionary in account data | ||
// this is required to handle correctly invite by email in DM | ||
val mxId = roomMember.thirdPartyInvite?.signed?.mxid | ||
if (mxId != null && mxId != myUserId) { | ||
aggregator?.directChatsToCheck?.put(roomId, mxId) | ||
} | ||
return true | ||
} | ||
|
||
/** | ||
* Get the already existing presence state for a specific user & room in order NOT to be replaced in RoomMemberSummaryEntity | ||
* by NULL value. | ||
*/ | ||
private fun handleIncrementalSync(realm: Realm, | ||
roomId: String, | ||
eventUserId: String, | ||
roomMember: RoomMemberContent, | ||
prevContent: Map<String, Any>?, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could be of type |
||
aggregator: SyncResponsePostTreatmentAggregator?): Boolean { | ||
val previousDisplayName = prevContent?.get("displayname") | ||
val previousAvatar = prevContent?.get("avatar_url") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe add |
||
|
||
private fun getExistingPresenceState(realm: Realm, roomId: String, userId: String): UserPresenceEntity? { | ||
return RoomMemberSummaryEntity.where(realm, roomId, userId).findFirst()?.userPresenceEntity | ||
if (previousDisplayName.isDifferentFrom(roomMember.displayName) || | ||
previousAvatar.isDifferentFrom(roomMember.avatarUrl)) { | ||
aggregator?.userIdsToFetch?.add(eventUserId) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If aggregator is null, there is no need to do all this treatment, maybe replace by: if (aggregator != null) {
val previousDisplayName = prevContent?.get("displayname") as? String
val previousAvatar = prevContent?.get("avatar_url") as? String
if (previousDisplayName != roomMember.displayName ||
previousAvatar != roomMember.avatarUrl) {
aggregator.userIdsToFetch.add(eventUserId)
}
} There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I tested this and it does work! |
||
} | ||
|
||
saveRoomMemberEntityLocally(realm, roomId, eventUserId, roomMember) | ||
// At the end of the sync, fetch all the profiles from the aggregator | ||
updateDirectChatsIfNecessary(roomId, roomMember, aggregator) | ||
return true | ||
} | ||
|
||
private fun Any?.isDifferentFrom(value: Any?) = when { | ||
this == null || this == value -> false | ||
else -> true | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can be removed |
||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -22,4 +22,7 @@ internal class SyncResponsePostTreatmentAggregator { | |
|
||
// Map of roomId to directUserId | ||
val directChatsToCheck = mutableMapOf<String, String>() | ||
|
||
// List of userIds to fetch and update at the end of incremental syncs | ||
val userIdsToFetch = mutableListOf<String>() | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks for the renaming :) |
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -16,22 +16,31 @@ | |
|
||
package org.matrix.android.sdk.internal.session.sync.handler | ||
|
||
import com.zhuinden.monarchy.Monarchy | ||
import org.matrix.android.sdk.api.MatrixPatterns | ||
import org.matrix.android.sdk.api.extensions.tryOrNull | ||
import org.matrix.android.sdk.api.session.user.model.User | ||
import org.matrix.android.sdk.internal.di.SessionDatabase | ||
import org.matrix.android.sdk.internal.session.profile.GetProfileInfoTask | ||
import org.matrix.android.sdk.internal.session.sync.RoomSyncEphemeralTemporaryStore | ||
import org.matrix.android.sdk.internal.session.sync.SyncResponsePostTreatmentAggregator | ||
import org.matrix.android.sdk.internal.session.sync.model.accountdata.toMutable | ||
import org.matrix.android.sdk.internal.session.user.UserEntityFactory | ||
import org.matrix.android.sdk.internal.session.user.accountdata.DirectChatsHelper | ||
import org.matrix.android.sdk.internal.session.user.accountdata.UpdateUserAccountDataTask | ||
import javax.inject.Inject | ||
|
||
internal class SyncResponsePostTreatmentAggregatorHandler @Inject constructor( | ||
private val directChatsHelper: DirectChatsHelper, | ||
private val ephemeralTemporaryStore: RoomSyncEphemeralTemporaryStore, | ||
private val updateUserAccountDataTask: UpdateUserAccountDataTask | ||
private val updateUserAccountDataTask: UpdateUserAccountDataTask, | ||
private val getProfileInfoTask: GetProfileInfoTask, | ||
@SessionDatabase private val monarchy: Monarchy, | ||
) { | ||
suspend fun handle(synResHaResponsePostTreatmentAggregator: SyncResponsePostTreatmentAggregator) { | ||
cleanupEphemeralFiles(synResHaResponsePostTreatmentAggregator.ephemeralFilesToDelete) | ||
updateDirectUserIds(synResHaResponsePostTreatmentAggregator.directChatsToCheck) | ||
suspend fun handle(aggregator: SyncResponsePostTreatmentAggregator) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks for having renamed this parameter which was quite typoed :) |
||
cleanupEphemeralFiles(aggregator.ephemeralFilesToDelete) | ||
updateDirectUserIds(aggregator.directChatsToCheck) | ||
fetchAndUpdateUsers(aggregator.userIdsToFetch) | ||
} | ||
|
||
private fun cleanupEphemeralFiles(ephemeralFilesToDelete: List<String>) { | ||
|
@@ -59,13 +68,33 @@ internal class SyncResponsePostTreatmentAggregatorHandler @Inject constructor( | |
} | ||
|
||
// remove roomId from currentDirectUserId entry | ||
hasUpdate = hasUpdate or(directChats[currentDirectUserId]?.remove(roomId) == true) | ||
hasUpdate = hasUpdate or (directChats[currentDirectUserId]?.remove(roomId) == true) | ||
// remove currentDirectUserId entry if there is no attached room anymore | ||
hasUpdate = hasUpdate or(directChats.takeIf { it[currentDirectUserId].isNullOrEmpty() }?.remove(currentDirectUserId) != null) | ||
hasUpdate = hasUpdate or (directChats.takeIf { it[currentDirectUserId].isNullOrEmpty() }?.remove(currentDirectUserId) != null) | ||
} | ||
} | ||
if (hasUpdate) { | ||
updateUserAccountDataTask.execute(UpdateUserAccountDataTask.DirectChatParams(directMessages = directChats)) | ||
} | ||
} | ||
|
||
private suspend fun fetchAndUpdateUsers(userIdsToFetch: List<String>) { | ||
fetchUsers(userIdsToFetch) | ||
.takeIf { it.isNotEmpty() } | ||
?.saveLocally() | ||
} | ||
|
||
private suspend fun fetchUsers(userIdsToFetch: List<String>) = userIdsToFetch.mapNotNull { | ||
tryOrNull { | ||
val profileJson = getProfileInfoTask.execute(GetProfileInfoTask.Params(it)) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would have put the |
||
User.fromJson(it, profileJson) | ||
} | ||
} | ||
|
||
private fun List<User>.saveLocally() { | ||
val userEntities = map { user -> UserEntityFactory.create(user) } | ||
monarchy.doWithRealm { | ||
it.insertOrUpdate(userEntities) | ||
} | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. It's good to split code into several smaller methods, but it's maybe harder to read in this particular case. What about replacing those 3 methods with a single one: private suspend fun fetchAndUpdateUsers(usersToFetch: List<String>) {
usersToFetch
.map { userId ->
val profileJson = profileService.getProfile(userId)
User.fromJson(userId, profileJson)
}
.map { user -> UserEntityFactory.create(user) }
.let { userEntities ->
monarchy.doWithRealm {
it.insertOrUpdate(userEntities)
}
}
} Also has the advantage to do on single Realm transaction instead of N transactions with your code. WDYT? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I personally prefer it split. The advantage is you can get the whole context of the action from the first method, and only if you're concerned about the details of the implementation do you need to read through the bottom methods. I agree with making it a single Realm transaction though. I'll push changes to make this happen There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Your latest implementation is better, thanks. Maybe just skip the transaction if the list is empty, which will be the case nearly all the time, or in case of error when fetching the profile. |
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -207,6 +207,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
syncLocalTimestampMillis: Long, | ||
aggregator: SyncResponsePostTreatmentAggregator): RoomEntity { | ||
Timber.v("Handle join sync for room $roomId") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This comment is useful when debugging sync request. Can you restore it please? |
||
val isInitialSync = insertType == EventInsertType.INITIAL_SYNC | ||
|
||
val ephemeralResult = (roomSync.ephemeral as? LazyRoomSyncEphemeral.Parsed) | ||
?._roomSyncEphemeral | ||
|
@@ -241,7 +242,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
} | ||
// Give info to crypto module | ||
cryptoService.onStateEvent(roomId, event) | ||
roomMemberEventHandler.handle(realm, roomId, event, aggregator) | ||
roomMemberEventHandler.handle(realm, roomId, event, isInitialSync, aggregator) | ||
} | ||
} | ||
if (roomSync.timeline?.events?.isNotEmpty() == true) { | ||
|
@@ -282,7 +283,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
roomSync: InvitedRoomSync, | ||
insertType: EventInsertType, | ||
syncLocalTimestampMillis: Long): RoomEntity { | ||
Timber.v("Handle invited sync for room $roomId") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ah I miss that at the previous review. Same remark: |
||
val isInitialSync = insertType == EventInsertType.INITIAL_SYNC | ||
val roomEntity = RoomEntity.getOrCreate(realm, roomId) | ||
roomEntity.membership = Membership.INVITE | ||
if (roomSync.inviteState != null && roomSync.inviteState.events.isNotEmpty()) { | ||
|
@@ -296,7 +297,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
eventId = eventEntity.eventId | ||
root = eventEntity | ||
} | ||
roomMemberEventHandler.handle(realm, roomId, event) | ||
roomMemberEventHandler.handle(realm, roomId, event, isInitialSync) | ||
} | ||
} | ||
val inviterEvent = roomSync.inviteState?.events?.lastOrNull { | ||
|
@@ -312,6 +313,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
roomSync: RoomSync, | ||
insertType: EventInsertType, | ||
syncLocalTimestampMillis: Long): RoomEntity { | ||
val isInitialSync = insertType == EventInsertType.INITIAL_SYNC | ||
val roomEntity = RoomEntity.getOrCreate(realm, roomId) | ||
for (event in roomSync.state?.events.orEmpty()) { | ||
if (event.eventId == null || event.stateKey == null || event.type == null) { | ||
|
@@ -323,7 +325,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
eventId = event.eventId | ||
root = eventEntity | ||
} | ||
roomMemberEventHandler.handle(realm, roomId, event) | ||
roomMemberEventHandler.handle(realm, roomId, event, isInitialSync) | ||
} | ||
for (event in roomSync.timeline?.events.orEmpty()) { | ||
if (event.eventId == null || event.senderId == null || event.type == null) { | ||
|
@@ -337,7 +339,7 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
root = eventEntity | ||
} | ||
if (event.type == EventType.STATE_ROOM_MEMBER) { | ||
roomMemberEventHandler.handle(realm, roomEntity.roomId, event) | ||
roomMemberEventHandler.handle(realm, roomEntity.roomId, event, isInitialSync) | ||
} | ||
} | ||
} | ||
|
@@ -381,11 +383,11 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
continue | ||
} | ||
|
||
eventIds.add(event.eventId) | ||
liveEventService.get().dispatchLiveEventReceived(event, roomId, insertType == EventInsertType.INITIAL_SYNC) | ||
|
||
val isInitialSync = insertType == EventInsertType.INITIAL_SYNC | ||
|
||
eventIds.add(event.eventId) | ||
liveEventService.get().dispatchLiveEventReceived(event, roomId, isInitialSync) | ||
|
||
if (event.isEncrypted() && !isInitialSync) { | ||
runBlocking { | ||
decryptIfNeeded(event, roomId) | ||
|
@@ -404,9 +406,8 @@ internal class RoomSyncHandler @Inject constructor(private val readReceiptHandle | |
root = eventEntity | ||
} | ||
if (event.type == EventType.STATE_ROOM_MEMBER) { | ||
val fixedContent = event.getFixedRoomMemberContent() | ||
roomMemberContentsByUser[event.stateKey] = fixedContent | ||
roomMemberEventHandler.handle(realm, roomEntity.roomId, event.stateKey, fixedContent, aggregator) | ||
roomMemberContentsByUser[event.stateKey] = event.getFixedRoomMemberContent() | ||
roomMemberEventHandler.handle(realm, roomEntity.roomId, event, isInitialSync, aggregator) | ||
} | ||
} | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why those 2 lines of code have been reordered? I think it's better to restore the original order since it's faster to first check for the
stateKey
.