[SES-2970] - Add expired groups handling (#962)
parent
b3623f2874
commit
097dc273b0
@ -1,71 +0,0 @@
|
||||
package org.thoughtcrime.securesms.dependencies
|
||||
|
||||
import dagger.Lazy
|
||||
import kotlinx.coroutines.CoroutineDispatcher
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import network.loki.messenger.libsession_util.util.GroupInfo
|
||||
import org.session.libsession.database.StorageProtocol
|
||||
import org.session.libsession.messaging.groups.GroupManagerV2
|
||||
import org.session.libsession.messaging.sending_receiving.pollers.ClosedGroupPoller
|
||||
import org.session.libsession.snode.SnodeClock
|
||||
import org.session.libsession.utilities.getGroup
|
||||
import org.session.libsignal.database.LokiAPIDatabaseProtocol
|
||||
import org.session.libsignal.utilities.AccountId
|
||||
import java.util.concurrent.ConcurrentHashMap
|
||||
|
||||
class PollerFactory(
|
||||
private val scope: CoroutineScope,
|
||||
private val executor: CoroutineDispatcher,
|
||||
private val configFactory: ConfigFactory,
|
||||
private val groupManagerV2: Lazy<GroupManagerV2>,
|
||||
private val storage: Lazy<StorageProtocol>,
|
||||
private val lokiApiDatabase: LokiAPIDatabaseProtocol,
|
||||
private val clock: SnodeClock,
|
||||
) {
|
||||
|
||||
private val pollers = ConcurrentHashMap<AccountId, ClosedGroupPoller>()
|
||||
|
||||
fun pollerFor(sessionId: AccountId): ClosedGroupPoller? {
|
||||
// Check if the group is currently in our config and approved, don't start if it isn't
|
||||
val invited = configFactory.getGroup(sessionId)?.invited
|
||||
|
||||
if (invited != false) return null
|
||||
|
||||
return pollers.getOrPut(sessionId) {
|
||||
ClosedGroupPoller(
|
||||
scope = scope,
|
||||
executor = executor,
|
||||
closedGroupSessionId = sessionId,
|
||||
configFactoryProtocol = configFactory,
|
||||
groupManagerV2 = groupManagerV2.get(),
|
||||
storage = storage.get(),
|
||||
lokiApiDatabase = lokiApiDatabase,
|
||||
clock = clock,
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
fun startAll() {
|
||||
configFactory
|
||||
.withUserConfigs { it.userGroups.allClosedGroupInfo() }
|
||||
.filterNot(GroupInfo.ClosedGroupInfo::invited)
|
||||
.forEach { pollerFor(it.groupAccountId)?.start() }
|
||||
}
|
||||
|
||||
fun stopAll() {
|
||||
pollers.forEach { (_, poller) ->
|
||||
poller.stop()
|
||||
}
|
||||
}
|
||||
|
||||
fun updatePollers() {
|
||||
val currentGroups = configFactory
|
||||
.withUserConfigs { it.userGroups.allClosedGroupInfo() }.filterNot(GroupInfo.ClosedGroupInfo::invited)
|
||||
val toRemove = pollers.filter { (id, _) -> id !in currentGroups.map { it.groupAccountId } }
|
||||
toRemove.forEach { (id, _) ->
|
||||
pollers.remove(id)?.stop()
|
||||
}
|
||||
startAll()
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,57 @@
|
||||
package org.thoughtcrime.securesms.groups
|
||||
|
||||
import kotlinx.coroutines.GlobalScope
|
||||
import kotlinx.coroutines.flow.SharingStarted
|
||||
import kotlinx.coroutines.flow.StateFlow
|
||||
import kotlinx.coroutines.flow.mapNotNull
|
||||
import kotlinx.coroutines.flow.scan
|
||||
import kotlinx.coroutines.flow.stateIn
|
||||
import org.session.libsession.messaging.sending_receiving.pollers.ClosedGroupPoller
|
||||
import org.session.libsignal.utilities.AccountId
|
||||
import org.session.libsignal.utilities.Log
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
/**
|
||||
* This class keeps track of what groups are expired.
|
||||
*
|
||||
* This is done by listening to the state of all group pollers and keeping track of last
|
||||
* known expired state of each group.
|
||||
*
|
||||
* The end result is not persisted and is only available in memory.
|
||||
*/
|
||||
@Singleton
|
||||
class ExpiredGroupManager @Inject constructor(
|
||||
pollerManager: GroupPollerManager,
|
||||
) {
|
||||
@Suppress("OPT_IN_USAGE")
|
||||
val expiredGroups: StateFlow<Set<AccountId>> = pollerManager.watchAllGroupPollingState()
|
||||
.mapNotNull { (groupId, state) ->
|
||||
val expired = (state as? ClosedGroupPoller.StartedState)?.expired
|
||||
|
||||
if (expired == null) {
|
||||
// Poller doesn't know about the expiration state yet, so we skip
|
||||
// the this update. It's important we do this as we want to use the
|
||||
// "last known state" if the poller doesn't know about the expiration state yet.
|
||||
return@mapNotNull null
|
||||
}
|
||||
|
||||
groupId to expired
|
||||
}
|
||||
|
||||
// This scan keep track of all expired groups. Whenever there is a new state for a group
|
||||
// poller, we compare the state with the previous state and update the set of expired groups.
|
||||
.scan(emptySet<AccountId>()) { previous, (groupId, expired) ->
|
||||
if (expired && groupId !in previous) {
|
||||
Log.d("ExpiredGroupManager", "Marking group $groupId expired.")
|
||||
previous + groupId
|
||||
} else if (!expired && groupId in previous) {
|
||||
Log.d("ExpiredGroupManager", "Unmarking group $groupId expired.")
|
||||
previous - groupId
|
||||
} else {
|
||||
previous
|
||||
}
|
||||
}
|
||||
|
||||
.stateIn(GlobalScope, SharingStarted.Eagerly, emptySet())
|
||||
}
|
@ -0,0 +1,149 @@
|
||||
package org.thoughtcrime.securesms.groups
|
||||
|
||||
import dagger.Lazy
|
||||
import kotlinx.coroutines.CoroutineDispatcher
|
||||
import kotlinx.coroutines.CoroutineScope
|
||||
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
||||
import kotlinx.coroutines.GlobalScope
|
||||
import kotlinx.coroutines.flow.Flow
|
||||
import kotlinx.coroutines.flow.SharingStarted
|
||||
import kotlinx.coroutines.flow.StateFlow
|
||||
import kotlinx.coroutines.flow.combine
|
||||
import kotlinx.coroutines.flow.distinctUntilChanged
|
||||
import kotlinx.coroutines.flow.filter
|
||||
import kotlinx.coroutines.flow.flatMapLatest
|
||||
import kotlinx.coroutines.flow.flowOf
|
||||
import kotlinx.coroutines.flow.map
|
||||
import kotlinx.coroutines.flow.merge
|
||||
import kotlinx.coroutines.flow.onStart
|
||||
import kotlinx.coroutines.flow.scan
|
||||
import kotlinx.coroutines.flow.stateIn
|
||||
import org.session.libsession.database.StorageProtocol
|
||||
import org.session.libsession.messaging.groups.GroupManagerV2
|
||||
import org.session.libsession.messaging.sending_receiving.pollers.ClosedGroupPoller
|
||||
import org.session.libsession.snode.SnodeClock
|
||||
import org.session.libsession.utilities.ConfigUpdateNotification
|
||||
import org.session.libsession.utilities.TextSecurePreferences
|
||||
import org.session.libsignal.database.LokiAPIDatabaseProtocol
|
||||
import org.session.libsignal.utilities.AccountId
|
||||
import org.session.libsignal.utilities.Log
|
||||
import org.thoughtcrime.securesms.dependencies.ConfigFactory
|
||||
import org.thoughtcrime.securesms.dependencies.POLLER_SCOPE
|
||||
import org.thoughtcrime.securesms.util.AppVisibilityManager
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Named
|
||||
import javax.inject.Singleton
|
||||
|
||||
/**
|
||||
* This class manages the lifecycle of group pollers.
|
||||
*
|
||||
* It listens for changes in the user's groups config and starts or stops pollers as needed. It
|
||||
* also considers the app's visibility state to decide whether to start or stop pollers.
|
||||
*
|
||||
* All the processes here are automatic and you don't need to do anything to start or stop pollers.
|
||||
*
|
||||
* This class also provide state monitoring facilities to check the state of a group poller.
|
||||
*/
|
||||
@Singleton
|
||||
class GroupPollerManager @Inject constructor(
|
||||
@Named(POLLER_SCOPE) scope: CoroutineScope,
|
||||
@Named(POLLER_SCOPE) executor: CoroutineDispatcher,
|
||||
configFactory: ConfigFactory,
|
||||
groupManagerV2: Lazy<GroupManagerV2>,
|
||||
storage: StorageProtocol,
|
||||
lokiApiDatabase: LokiAPIDatabaseProtocol,
|
||||
clock: SnodeClock,
|
||||
preferences: TextSecurePreferences,
|
||||
appVisibilityManager: AppVisibilityManager,
|
||||
) {
|
||||
@Suppress("OPT_IN_USAGE")
|
||||
private val activeGroupPollers: StateFlow<Map<AccountId, ClosedGroupPoller>> =
|
||||
combine(
|
||||
preferences.watchLocalNumber(),
|
||||
appVisibilityManager.isAppVisible
|
||||
) { localNumber, visible -> localNumber != null && visible }
|
||||
.distinctUntilChanged()
|
||||
|
||||
// This flatMap produces a flow of groups that should be polled now
|
||||
.flatMapLatest { shouldPoll ->
|
||||
if (shouldPoll) {
|
||||
(configFactory.configUpdateNotifications
|
||||
.filter { it is ConfigUpdateNotification.UserConfigsMerged || it == ConfigUpdateNotification.UserConfigsModified } as Flow<*>)
|
||||
.onStart { emit(Unit) }
|
||||
.map {
|
||||
configFactory.withUserConfigs { it.userGroups.allClosedGroupInfo() }
|
||||
.mapNotNullTo(hashSetOf()) { group ->
|
||||
group.groupAccountId.takeIf { group.shouldPoll }
|
||||
}
|
||||
}
|
||||
.distinctUntilChanged()
|
||||
} else {
|
||||
// There shouldn't be any group polling at this stage
|
||||
flowOf(emptySet())
|
||||
}
|
||||
}
|
||||
|
||||
// This scan compares the previous active group pollers with the incoming set of groups
|
||||
// that should be polled now, to work out which pollers should be started or stopped,
|
||||
// and finally emits the new state
|
||||
.scan(emptyMap<AccountId, ClosedGroupPoller>()) { previous, newActiveGroupIDs ->
|
||||
// Go through previous pollers and stop those that are not in the new set
|
||||
for ((groupId, poller) in previous) {
|
||||
if (groupId !in newActiveGroupIDs) {
|
||||
Log.d(TAG, "Stopping poller for $groupId")
|
||||
poller.stop()
|
||||
}
|
||||
}
|
||||
|
||||
// Go through new set, pick the existing pollers and create/start those that are
|
||||
// not in the previous map
|
||||
newActiveGroupIDs.associateWith { groupId ->
|
||||
var poller = previous[groupId]
|
||||
|
||||
if (poller == null) {
|
||||
Log.d(TAG, "Starting poller for $groupId")
|
||||
poller = ClosedGroupPoller(
|
||||
scope = scope,
|
||||
executor = executor,
|
||||
closedGroupSessionId = groupId,
|
||||
configFactoryProtocol = configFactory,
|
||||
groupManagerV2 = groupManagerV2.get(),
|
||||
storage = storage,
|
||||
lokiApiDatabase = lokiApiDatabase,
|
||||
clock = clock,
|
||||
).also { it.start() }
|
||||
}
|
||||
|
||||
poller
|
||||
}
|
||||
}
|
||||
|
||||
.stateIn(GlobalScope, SharingStarted.Eagerly, emptyMap())
|
||||
|
||||
|
||||
@Suppress("OPT_IN_USAGE")
|
||||
fun watchGroupPollingState(groupId: AccountId): Flow<ClosedGroupPoller.State> {
|
||||
return activeGroupPollers
|
||||
.flatMapLatest { pollers ->
|
||||
pollers[groupId]?.state ?: flowOf(ClosedGroupPoller.IdleState)
|
||||
}
|
||||
.distinctUntilChanged()
|
||||
}
|
||||
|
||||
@OptIn(ExperimentalCoroutinesApi::class)
|
||||
fun watchAllGroupPollingState(): Flow<Pair<AccountId, ClosedGroupPoller.State>> {
|
||||
return activeGroupPollers
|
||||
.flatMapLatest { pollers ->
|
||||
// Merge all poller states into a single flow of (groupId, state) pairs
|
||||
merge(
|
||||
*pollers
|
||||
.map { (id, poller) -> poller.state.map { state -> id to state } }
|
||||
.toTypedArray()
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
companion object {
|
||||
private const val TAG = "GroupPollerHandler"
|
||||
}
|
||||
}
|
@ -0,0 +1,28 @@
|
||||
package org.thoughtcrime.securesms.util
|
||||
|
||||
import androidx.lifecycle.DefaultLifecycleObserver
|
||||
import androidx.lifecycle.LifecycleOwner
|
||||
import androidx.lifecycle.ProcessLifecycleOwner
|
||||
import kotlinx.coroutines.flow.MutableStateFlow
|
||||
import kotlinx.coroutines.flow.StateFlow
|
||||
import javax.inject.Inject
|
||||
import javax.inject.Singleton
|
||||
|
||||
@Singleton
|
||||
class AppVisibilityManager @Inject constructor() {
|
||||
private val mutableIsAppVisible = MutableStateFlow(false)
|
||||
|
||||
init {
|
||||
ProcessLifecycleOwner.get().lifecycle.addObserver(object : DefaultLifecycleObserver {
|
||||
override fun onStart(owner: LifecycleOwner) {
|
||||
mutableIsAppVisible.value = true
|
||||
}
|
||||
|
||||
override fun onStop(owner: LifecycleOwner) {
|
||||
mutableIsAppVisible.value = false
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
val isAppVisible: StateFlow<Boolean> get() = mutableIsAppVisible
|
||||
}
|
@ -0,0 +1,84 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:orientation="vertical"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content">
|
||||
|
||||
<FrameLayout
|
||||
android:id="@+id/blockedBanner"
|
||||
android:contentDescription="@string/AccessibilityId_blockedBanner"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:background="?danger"
|
||||
android:visibility="gone"
|
||||
tools:visibility="visible"
|
||||
>
|
||||
|
||||
<TextView
|
||||
android:id="@+id/blockedBannerTextView"
|
||||
android:contentDescription="@string/AccessibilityId_blockedBannerText"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_gravity="center"
|
||||
android:layout_margin="@dimen/medium_spacing"
|
||||
android:textColor="@color/white"
|
||||
android:textSize="@dimen/small_font_size"
|
||||
android:textStyle="bold"
|
||||
tools:text="Elon is blocked. Unblock them?" />
|
||||
|
||||
</FrameLayout>
|
||||
|
||||
<FrameLayout
|
||||
android:id="@+id/outdatedDisappearingBanner"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:background="?colorAccent"
|
||||
android:contentDescription="@string/AccessibilityId_outdated_group_banner"
|
||||
android:visibility="gone"
|
||||
tools:visibility="visible">
|
||||
|
||||
<TextView
|
||||
android:id="@+id/outdatedDisappearingBannerTextView"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:gravity="center_horizontal"
|
||||
android:layout_gravity="center"
|
||||
android:layout_marginVertical="@dimen/very_small_spacing"
|
||||
android:layout_marginHorizontal="@dimen/medium_spacing"
|
||||
android:textColor="@color/black"
|
||||
android:textSize="@dimen/small_font_size"
|
||||
tools:text="This user's client is outdated, things may not work as expected" />
|
||||
|
||||
</FrameLayout>
|
||||
|
||||
<TextView
|
||||
android:id="@+id/outdatedGroupBanner"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_gravity="center"
|
||||
android:background="?colorAccent"
|
||||
android:paddingVertical="@dimen/very_small_spacing"
|
||||
android:gravity="center_horizontal"
|
||||
android:textColor="?message_sent_text_color"
|
||||
android:paddingHorizontal="@dimen/medium_spacing"
|
||||
android:contentDescription="@string/AccessibilityId_outdated_group_banner"
|
||||
android:textSize="@dimen/small_font_size"
|
||||
android:visibility="gone"
|
||||
tools:text="@string/groupLegacyBanner"
|
||||
tools:visibility="visible" />
|
||||
|
||||
<TextView
|
||||
android:id="@+id/groupExpiredBanner"
|
||||
android:text="@string/groupNotUpdatedWarning"
|
||||
android:gravity="center"
|
||||
android:background="@color/accent_orange"
|
||||
android:textColor="@color/black"
|
||||
android:paddingHorizontal="@dimen/medium_spacing"
|
||||
android:paddingVertical="@dimen/very_small_spacing"
|
||||
android:visibility="gone"
|
||||
tools:visibility="visible"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content" />
|
||||
|
||||
</LinearLayout>
|
@ -0,0 +1,59 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools"
|
||||
android:orientation="vertical"
|
||||
android:id="@+id/messageRequestBar"
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content">
|
||||
|
||||
<TextView
|
||||
android:id="@+id/messageRequestBlock"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_gravity="center"
|
||||
android:contentDescription="@string/AccessibilityId_messageRequestsBlock"
|
||||
android:textColor="?danger"
|
||||
android:paddingHorizontal="@dimen/massive_spacing"
|
||||
android:paddingVertical="@dimen/small_spacing"
|
||||
android:textSize="@dimen/text_size"
|
||||
tools:text="@string/deleteAfterGroupPR1BlockUser"/>
|
||||
|
||||
<TextView
|
||||
android:id="@+id/sendAcceptsTextView"
|
||||
android:layout_width="wrap_content"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_margin="@dimen/medium_spacing"
|
||||
android:alpha="0.6"
|
||||
android:gravity="center_horizontal"
|
||||
android:text="@string/messageRequestsAcceptDescription"
|
||||
android:textColor="?android:textColorPrimary"
|
||||
android:textSize="@dimen/small_font_size" />
|
||||
|
||||
<LinearLayout
|
||||
android:layout_width="match_parent"
|
||||
android:layout_height="wrap_content"
|
||||
android:layout_marginHorizontal="@dimen/medium_spacing"
|
||||
android:orientation="horizontal">
|
||||
|
||||
<Button
|
||||
android:id="@+id/acceptMessageRequestButton"
|
||||
style="@style/Widget.Session.Button.Common.ProminentOutline"
|
||||
android:contentDescription="@string/AccessibilityId_messageRequestsAccept"
|
||||
android:layout_width="0dp"
|
||||
android:layout_height="@dimen/medium_button_height"
|
||||
android:layout_weight="1"
|
||||
android:text="@string/accept" />
|
||||
|
||||
<Button
|
||||
android:id="@+id/declineMessageRequestButton"
|
||||
style="@style/Widget.Session.Button.Common.DangerOutline"
|
||||
android:contentDescription="@string/AccessibilityId_deleteRequest"
|
||||
android:layout_width="0dp"
|
||||
android:layout_height="@dimen/medium_button_height"
|
||||
android:layout_marginStart="@dimen/medium_spacing"
|
||||
android:layout_weight="1"
|
||||
android:text="@string/delete" />
|
||||
|
||||
</LinearLayout>
|
||||
|
||||
</LinearLayout>
|
Loading…
Reference in New Issue