[SES-2970] - Add expired groups handling (#962)

pull/1710/head
SessionHero01 1 month ago committed by GitHub
parent b3623f2874
commit 097dc273b0
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -78,12 +78,13 @@ import org.thoughtcrime.securesms.dependencies.AppComponent;
import org.thoughtcrime.securesms.dependencies.ConfigFactory;
import org.thoughtcrime.securesms.dependencies.DatabaseComponent;
import org.thoughtcrime.securesms.dependencies.DatabaseModule;
import org.thoughtcrime.securesms.dependencies.PollerFactory;
import org.thoughtcrime.securesms.emoji.EmojiSource;
import org.thoughtcrime.securesms.groups.ExpiredGroupManager;
import org.thoughtcrime.securesms.groups.OpenGroupManager;
import org.thoughtcrime.securesms.groups.handler.AdminStateSync;
import org.thoughtcrime.securesms.groups.handler.CleanupInvitationHandler;
import org.thoughtcrime.securesms.groups.handler.DestroyedGroupSync;
import org.thoughtcrime.securesms.groups.GroupPollerManager;
import org.thoughtcrime.securesms.groups.handler.RemoveGroupMemberHandler;
import org.thoughtcrime.securesms.home.HomeActivity;
import org.thoughtcrime.securesms.jobmanager.impl.NetworkConstraint;
@ -98,6 +99,7 @@ import org.thoughtcrime.securesms.service.ExpiringMessageManager;
import org.thoughtcrime.securesms.service.KeyCachingService;
import org.thoughtcrime.securesms.sskenvironment.ReadReceiptManager;
import org.thoughtcrime.securesms.sskenvironment.TypingStatusRepository;
import org.thoughtcrime.securesms.util.AppVisibilityManager;
import org.thoughtcrime.securesms.util.Broadcaster;
import org.thoughtcrime.securesms.util.VersionDataFetcher;
import org.thoughtcrime.securesms.webrtc.CallMessageProcessor;
@ -151,7 +153,6 @@ public class ApplicationContext extends Application implements DefaultLifecycleO
@Inject MessageDataProvider messageDataProvider;
@Inject TextSecurePreferences textSecurePreferences;
@Inject ConfigFactory configFactory;
@Inject PollerFactory pollerFactory;
@Inject LastSentTimestampCache lastSentTimestampCache;
@Inject VersionDataFetcher versionDataFetcher;
@Inject PushRegistrationHandler pushRegistrationHandler;
@ -175,6 +176,9 @@ public class ApplicationContext extends Application implements DefaultLifecycleO
@Inject LegacyClosedGroupPollerV2 legacyClosedGroupPollerV2;
@Inject LegacyGroupDeprecationManager legacyGroupDeprecationManager;
@Inject CleanupInvitationHandler cleanupInvitationHandler;
@Inject AppVisibilityManager appVisibilityManager; // Exists here only to start upon app starts
@Inject GroupPollerManager groupPollerManager; // Exists here only to start upon app starts
@Inject ExpiredGroupManager expiredGroupManager; // Exists here only to start upon app starts
public volatile boolean isAppVisible;
public String KEYGUARD_LOCK_TAG = NonTranslatableStringConstants.APP_NAME + ":KeyguardLock";
@ -350,7 +354,6 @@ public class ApplicationContext extends Application implements DefaultLifecycleO
if (poller != null) {
poller.stopIfNeeded();
}
pollerFactory.stopAll();
legacyClosedGroupPollerV2.stopAll();
versionDataFetcher.stopTimedVersionCheck();
}
@ -359,7 +362,6 @@ public class ApplicationContext extends Application implements DefaultLifecycleO
public void onTerminate() {
stopKovenant(); // Loki
OpenGroupManager.INSTANCE.stopPolling();
pollerFactory.stopAll();
versionDataFetcher.stopTimedVersionCheck();
super.onTerminate();
}
@ -462,7 +464,6 @@ public class ApplicationContext extends Application implements DefaultLifecycleO
if (poller != null) {
poller.startIfNeeded();
}
pollerFactory.startAll();
legacyClosedGroupPollerV2.start();
}

@ -42,7 +42,6 @@ import org.thoughtcrime.securesms.database.MmsSmsDatabase
import org.thoughtcrime.securesms.database.RecipientDatabase
import org.thoughtcrime.securesms.database.ThreadDatabase
import org.thoughtcrime.securesms.database.model.MmsMessageRecord
import org.thoughtcrime.securesms.dependencies.PollerFactory
import org.thoughtcrime.securesms.groups.ClosedGroupManager
import org.thoughtcrime.securesms.groups.OpenGroupManager
import org.thoughtcrime.securesms.repository.ConversationRepository
@ -63,7 +62,6 @@ class ConfigToDatabaseSync @Inject constructor(
private val storage: StorageProtocol,
private val threadDatabase: ThreadDatabase,
private val recipientDatabase: RecipientDatabase,
private val pollerFactory: PollerFactory,
private val clock: SnodeClock,
private val profileManager: ProfileManager,
private val preferences: TextSecurePreferences,
@ -321,9 +319,6 @@ class ConfigToDatabaseSync @Inject constructor(
groupThreadsToKeep[closedGroup.groupAccountId] = threadId
storage.setPinned(threadId, closedGroup.priority == PRIORITY_PINNED)
if (!closedGroup.invited && !closedGroup.kicked) {
pollerFactory.pollerFor(closedGroup.groupAccountId)?.start()
}
if (closedGroup.destroyed) {
handleDestroyedGroup(threadId = threadId)
@ -332,8 +327,7 @@ class ConfigToDatabaseSync @Inject constructor(
val toRemove = existingClosedGroupThreads - groupThreadsToKeep.keys
Log.d(TAG, "Removing ${toRemove.size} closed groups")
toRemove.forEach { (groupId, threadId) ->
pollerFactory.pollerFor(groupId)?.stop()
toRemove.forEach { (_, threadId) ->
storage.removeClosedGroupThread(threadId)
}

@ -481,6 +481,7 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
updateUnreadCountIndicator()
updatePlaceholder()
setUpBlockedBanner()
setUpExpiredGroupBanner()
binding.searchBottomBar.setEventListener(this)
updateSendAfterApprovalText()
setUpMessageRequests()
@ -822,9 +823,18 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
// called from onCreate
private fun setUpBlockedBanner() {
val recipient = viewModel.recipient?.takeUnless { it.isGroupOrCommunityRecipient } ?: return
binding.blockedBannerTextView.text = applicationContext.getString(R.string.blockBlockedDescription)
binding.blockedBanner.isVisible = recipient.isBlocked
binding.blockedBanner.setOnClickListener { unblock() }
binding.conversationHeader.blockedBannerTextView.text = applicationContext.getString(R.string.blockBlockedDescription)
binding.conversationHeader.blockedBanner.isVisible = recipient.isBlocked
binding.conversationHeader.blockedBanner.setOnClickListener { unblock() }
}
private fun setUpExpiredGroupBanner() {
lifecycleScope.launch {
viewModel.showExpiredGroupBanner
.collectLatest {
binding.conversationHeader.groupExpiredBanner.isVisible = it
}
}
}
private fun setUpOutdatedClientBanner() {
@ -833,13 +843,13 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
val shouldShowLegacy = ExpirationConfiguration.isNewConfigEnabled &&
legacyRecipient != null
binding.outdatedDisappearingBanner.isVisible = shouldShowLegacy
binding.conversationHeader.outdatedDisappearingBanner.isVisible = shouldShowLegacy
if (shouldShowLegacy) {
val txt = Phrase.from(this, R.string.disappearingMessagesLegacy)
.put(NAME_KEY, legacyRecipient!!.toShortString())
.format()
binding.outdatedDisappearingBannerTextView.text = txt
binding.conversationHeader.outdatedDisappearingBannerTextView.text = txt
}
}
@ -848,11 +858,11 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
viewModel.legacyGroupBanner
.collectLatest { banner ->
if (banner == null) {
binding.outdatedGroupBanner.isVisible = false
binding.outdatedGroupBanner.text = null
binding.conversationHeader.outdatedGroupBanner.isVisible = false
binding.conversationHeader.outdatedGroupBanner.text = null
} else {
binding.outdatedGroupBanner.isVisible = true
binding.outdatedGroupBanner.text = SpannableStringBuilder(banner)
binding.conversationHeader.outdatedGroupBanner.isVisible = true
binding.conversationHeader.outdatedGroupBanner.text = SpannableStringBuilder(banner)
.apply {
// Append a space as a placeholder
append(" ")
@ -875,7 +885,7 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
)
}
binding.outdatedGroupBanner.setOnClickListener {
binding.conversationHeader.outdatedGroupBanner.setOnClickListener {
showOpenUrlDialog("https://getsession.org/groups")
}
}
@ -1018,7 +1028,7 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
runOnUiThread {
val threadRecipient = viewModel.recipient ?: return@runOnUiThread
if (threadRecipient.isContactRecipient) {
binding.blockedBanner.isVisible = threadRecipient.isBlocked
binding.conversationHeader.blockedBanner.isVisible = threadRecipient.isBlocked
}
invalidateOptionsMenu()
updateSendAfterApprovalText()
@ -1035,15 +1045,15 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
}
private fun setUpMessageRequests() {
binding.acceptMessageRequestButton.setOnClickListener {
binding.messageRequestBar.acceptMessageRequestButton.setOnClickListener {
viewModel.acceptMessageRequest()
}
binding.messageRequestBlock.setOnClickListener {
binding.messageRequestBar.messageRequestBlock.setOnClickListener {
block(deleteThread = true)
}
binding.declineMessageRequestButton.setOnClickListener {
binding.messageRequestBar.declineMessageRequestButton.setOnClickListener {
fun doDecline() {
viewModel.declineMessageRequest()
finish()
@ -1063,12 +1073,12 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
.map { it.messageRequestState }
.distinctUntilChanged()
.collectLatest { state ->
binding.messageRequestBar.isVisible = state is MessageRequestUiState.Visible
binding.messageRequestBar.root.isVisible = state is MessageRequestUiState.Visible
if (state is MessageRequestUiState.Visible) {
binding.sendAcceptsTextView.setText(state.acceptButtonText)
binding.messageRequestBlock.isVisible = state.blockButtonText != null
binding.messageRequestBlock.text = state.blockButtonText
binding.messageRequestBar.sendAcceptsTextView.setText(state.acceptButtonText)
binding.messageRequestBar.messageRequestBlock.isVisible = state.blockButtonText != null
binding.messageRequestBar.messageRequestBlock.text = state.blockButtonText
}
}
}
@ -1076,7 +1086,7 @@ class ConversationActivityV2 : PassphraseRequiredActionBarActivity(), InputBarDe
}
private fun acceptMessageRequest() {
binding.messageRequestBar.isVisible = false
binding.messageRequestBar.root.isVisible = false
viewModel.acceptMessageRequest()
}

@ -16,6 +16,7 @@ import dagger.hilt.android.qualifiers.ApplicationContext
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.channels.consumeEach
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.SharedFlow
@ -23,6 +24,7 @@ import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.filterNotNull
import kotlinx.coroutines.flow.flowOf
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.stateIn
import kotlinx.coroutines.flow.update
@ -60,6 +62,7 @@ import org.thoughtcrime.securesms.database.model.MessageId
import org.thoughtcrime.securesms.database.model.MessageRecord
import org.thoughtcrime.securesms.database.model.MmsMessageRecord
import org.thoughtcrime.securesms.dependencies.ConfigFactory
import org.thoughtcrime.securesms.groups.ExpiredGroupManager
import org.thoughtcrime.securesms.groups.OpenGroupManager
import org.thoughtcrime.securesms.mms.AudioSlide
import org.thoughtcrime.securesms.repository.ConversationRepository
@ -82,6 +85,7 @@ class ConversationViewModel(
private val configFactory: ConfigFactory,
private val groupManagerV2: GroupManagerV2,
val legacyGroupDeprecationManager: LegacyGroupDeprecationManager,
private val expiredGroupManager: ExpiredGroupManager,
) : ViewModel() {
val showSendAfterApprovalText: Boolean
@ -245,6 +249,13 @@ class ConversationViewModel(
&& state != LegacyGroupDeprecationManager.DeprecationState.NOT_DEPRECATING
}.stateIn(viewModelScope, SharingStarted.Eagerly, false)
val showExpiredGroupBanner: Flow<Boolean> = if (recipient?.isGroupV2Recipient != true) {
flowOf(false)
} else {
val groupId = AccountId(recipient!!.address.toString())
expiredGroupManager.expiredGroups.map { groupId in it }
}
private val attachmentDownloadHandler = AttachmentDownloadHandler(
storage = storage,
messageDataProvider = messageDataProvider,
@ -1089,6 +1100,7 @@ class ConversationViewModel(
private val configFactory: ConfigFactory,
private val groupManagerV2: GroupManagerV2,
private val legacyGroupDeprecationManager: LegacyGroupDeprecationManager,
private val expiredGroupManager: ExpiredGroupManager,
) : ViewModelProvider.Factory {
override fun <T : ViewModel> create(modelClass: Class<T>): T {
@ -1107,6 +1119,7 @@ class ConversationViewModel(
configFactory = configFactory,
groupManagerV2 = groupManagerV2,
legacyGroupDeprecationManager = legacyGroupDeprecationManager,
expiredGroupManager = expiredGroupManager,
) as T
}
}

@ -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()
}
}

@ -1,38 +1,32 @@
package org.thoughtcrime.securesms.dependencies
import android.content.Context
import dagger.Lazy
import dagger.Module
import dagger.Provides
import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent
import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.DelicateCoroutinesApi
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.GlobalScope
import org.session.libsession.database.StorageProtocol
import org.session.libsession.messaging.groups.GroupManagerV2
import org.session.libsession.messaging.groups.GroupScope
import org.session.libsession.messaging.groups.LegacyGroupDeprecationManager
import org.session.libsession.messaging.sending_receiving.pollers.ClosedGroupPoller
import org.session.libsession.messaging.sending_receiving.pollers.LegacyClosedGroupPollerV2
import org.session.libsession.snode.SnodeClock
import org.session.libsession.utilities.ConfigFactoryProtocol
import org.session.libsession.utilities.TextSecurePreferences
import org.session.libsignal.database.LokiAPIDatabaseProtocol
import org.thoughtcrime.securesms.database.ConfigDatabase
import org.thoughtcrime.securesms.database.ThreadDatabase
import javax.inject.Named
import javax.inject.Singleton
@Suppress("OPT_IN_USAGE")
const val POLLER_SCOPE = "poller_coroutine_scope"
@Module
@InstallIn(SingletonComponent::class)
object SessionUtilModule {
private const val POLLER_SCOPE = "poller_coroutine_scope"
@OptIn(DelicateCoroutinesApi::class)
@Provides
@Named(POLLER_SCOPE)
fun providePollerScope(): CoroutineScope = GlobalScope
@ -42,24 +36,6 @@ object SessionUtilModule {
@Named(POLLER_SCOPE)
fun provideExecutor(): CoroutineDispatcher = Dispatchers.IO.limitedParallelism(1)
@Provides
@Singleton
fun providePollerFactory(@Named(POLLER_SCOPE) coroutineScope: CoroutineScope,
@Named(POLLER_SCOPE) dispatcher: CoroutineDispatcher,
configFactory: ConfigFactory,
storage: Lazy<StorageProtocol>,
groupManagerV2: Lazy<GroupManagerV2>,
lokiApiDatabase: LokiAPIDatabaseProtocol,
clock: SnodeClock) = PollerFactory(
scope = coroutineScope,
executor = dispatcher,
configFactory = configFactory,
groupManagerV2 = groupManagerV2,
storage = storage,
lokiApiDatabase = lokiApiDatabase,
clock = clock,
)
@Provides
@Singleton
fun provideSnodeClock() = SnodeClock()
@ -77,6 +53,7 @@ object SessionUtilModule {
return LegacyClosedGroupPollerV2(storage, deprecationManager)
}
@Provides
@Singleton
fun provideLegacyGroupDeprecationManager(prefs: TextSecurePreferences): LegacyGroupDeprecationManager {

@ -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())
}

@ -63,7 +63,6 @@ import org.thoughtcrime.securesms.database.LokiMessageDatabase
import org.thoughtcrime.securesms.database.MmsSmsDatabase
import org.thoughtcrime.securesms.database.ThreadDatabase
import org.thoughtcrime.securesms.dependencies.ConfigFactory
import org.thoughtcrime.securesms.dependencies.PollerFactory
import org.thoughtcrime.securesms.util.SessionMetaProtocol
import java.util.concurrent.TimeUnit
import javax.inject.Inject
@ -78,7 +77,6 @@ class GroupManagerV2Impl @Inject constructor(
private val mmsSmsDatabase: MmsSmsDatabase,
private val lokiDatabase: LokiMessageDatabase,
private val threadDatabase: ThreadDatabase,
private val pollerFactory: PollerFactory,
private val profileManager: SSKEnvironment.ProfileManagerProtocol,
@ApplicationContext val application: Context,
private val clock: SnodeClock,
@ -86,6 +84,7 @@ class GroupManagerV2Impl @Inject constructor(
private val lokiAPIDatabase: LokiAPIDatabase,
private val configUploader: ConfigUploader,
private val scope: GroupScope,
private val groupPollerManager: GroupPollerManager,
) : GroupManagerV2 {
private val dispatcher = Dispatchers.Default
@ -196,7 +195,6 @@ class GroupManagerV2Impl @Inject constructor(
profileManager.setName(application, recipient, groupName)
storage.setRecipientApprovedMe(recipient, true)
storage.setRecipientApproved(recipient, true)
pollerFactory.updatePollers()
// Invite members
JobQueue.shared.add(
@ -494,8 +492,6 @@ class GroupManagerV2Impl @Inject constructor(
}
}
pollerFactory.pollerFor(groupId)?.stop()
// Delete conversation and group configs
storage.getThreadId(Address.fromSerialized(groupId.hexString))
?.let(storage::deleteConversation)
@ -678,14 +674,12 @@ class GroupManagerV2Impl @Inject constructor(
))
}
val poller = checkNotNull(pollerFactory.pollerFor(group.groupAccountId)) { "Unable to start a poller for groups " }
poller.start()
// We need to wait until we have the first data polled from the poller, otherwise
// we won't have the necessary configs to send invite response/or do anything else.
// We can't hang on here forever if things don't work out, bail out if it's the camse
withTimeout(20_000L) {
poller.state.filterIsInstance<ClosedGroupPoller.StartedState>()
groupPollerManager.watchGroupPollingState(group.groupAccountId)
.filterIsInstance<ClosedGroupPoller.StartedState>()
.filter { it.hadAtLeastOneSuccessfulPoll }
.first()
}
@ -905,9 +899,6 @@ class GroupManagerV2Impl @Inject constructor(
override suspend fun handleKicked(groupId: AccountId): Unit = scope.launchAndWait(groupId, "Handle kicked") {
Log.d(TAG, "We were kicked from the group, delete and stop polling")
// Stop polling the group immediately
pollerFactory.pollerFor(groupId)?.stop()
val userId = requireNotNull(storage.getUserPublicKey()) { "No current user available" }
val group = configFactory.getGroup(groupId) ?: return@launchAndWait

@ -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
}

@ -37,10 +37,11 @@
android:id="@+id/conversationRecyclerView"
android:layout_width="match_parent"
android:layout_height="0dp"
tools:visibility="gone"
app:layout_constraintVertical_weight="1"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintBottom_toTopOf="@+id/typingIndicatorViewContainer"
app:layout_constraintTop_toBottomOf="@id/outdatedGroupBanner" />
app:layout_constraintTop_toBottomOf="@id/conversation_header" />
<org.thoughtcrime.securesms.conversation.v2.components.TypingIndicatorViewContainer
android:focusable="false"
@ -62,6 +63,7 @@
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/recreateGroupButtonContainer"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintVertical_bias="1"
/>
<org.thoughtcrime.securesms.conversation.v2.search.SearchBottomBar
@ -209,7 +211,7 @@
android:layout_centerInParent="true"
android:textSize="@dimen/very_small_font_size"
android:textColor="?android:textColorPrimary"
android:text="8" />
tools:text="8" />
</RelativeLayout>
@ -223,70 +225,12 @@
android:visibility="gone"
app:layout_constraintBottom_toBottomOf="parent" />
<FrameLayout
android:id="@+id/blockedBanner"
android:contentDescription="@string/AccessibilityId_blockedBanner"
<include
android:id="@+id/conversation_header"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:layout_constraintTop_toBottomOf="@+id/toolbar"
android:background="?danger"
android:visibility="gone"
>
<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"
app:layout_constraintTop_toBottomOf="@+id/blockedBanner"
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"
app:layout_constraintTop_toBottomOf="@+id/outdatedDisappearingBanner"
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" />
layout="@layout/view_conversation_header" />
<TextView
android:padding="@dimen/medium_spacing"
@ -297,74 +241,21 @@
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginHorizontal="@dimen/large_spacing"
app:layout_constraintTop_toBottomOf="@+id/outdatedGroupBanner"
app:layout_constraintTop_toBottomOf="@+id/conversation_header"
android:contentDescription="@string/AccessibilityId_empty_conversation"
tools:text="Some Control Message Text"
/>
<LinearLayout
<include
android:id="@+id/messageRequestBar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:layout_constraintBottom_toTopOf="@+id/recreateGroupButtonContainer"
app:layout_constraintTop_toBottomOf="@+id/textSendAfterApproval"
layout="@layout/view_conversation_message_request_bar"
android:layout_marginBottom="@dimen/large_spacing"
android:orientation="vertical"
android:visibility="gone"
tools:visibility="visible">
<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>
tools:visibility="visible"
android:visibility="gone" />
<FrameLayout
android:id="@+id/recreateGroupButtonContainer"

@ -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>

@ -305,4 +305,12 @@ Java_network_loki_messenger_libsession_1util_GroupKeysConfig_admin(JNIEnv *env,
std::lock_guard lock{util::util_mutex_};
auto ptr = ptrToKeys(env, thiz);
return ptr->admin();
}
extern "C"
JNIEXPORT jint JNICALL
Java_network_loki_messenger_libsession_1util_GroupKeysConfig_size(JNIEnv *env, jobject thiz) {
std::lock_guard lock{util::util_mutex_};
auto ptr = ptrToKeys(env, thiz);
return ptr->size();
}

@ -458,6 +458,7 @@ interface ReadableGroupKeysConfig {
fun subAccountSign(message: ByteArray, signingValue: ByteArray): GroupKeysConfig.SwarmAuth
fun getSubAccountToken(sessionId: AccountId, canWrite: Boolean = true, canDelete: Boolean = false): ByteArray
fun currentGeneration(): Int
fun size(): Int
}
interface MutableGroupKeysConfig : ReadableGroupKeysConfig {
@ -539,6 +540,7 @@ class GroupKeysConfig private constructor(
external override fun currentGeneration(): Int
external fun admin(): Boolean
external override fun size(): Int
data class SwarmAuth(
val subAccount: String,

@ -29,6 +29,9 @@ sealed class GroupInfo {
fun hasAdminKey() = adminKey != null
val shouldPoll: Boolean
get() = !invited && !kicked && !destroyed
companion object {
/**
* Generate the group's admin key(64 bytes) from seed (32 bytes, normally used

@ -8,6 +8,7 @@ import kotlinx.coroutines.async
import kotlinx.coroutines.delay
import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.update
import kotlinx.coroutines.isActive
import kotlinx.coroutines.launch
import kotlinx.coroutines.supervisorScope
@ -54,7 +55,11 @@ class ClosedGroupPoller(
sealed interface State
data object IdleState : State
data class StartedState(internal val job: Job, val hadAtLeastOneSuccessfulPoll: Boolean = false) : State
data class StartedState(
internal val job: Job,
val expired: Boolean? = null,
val hadAtLeastOneSuccessfulPoll: Boolean = false,
) : State
private val mutableState = MutableStateFlow<State>(IdleState)
val state: StateFlow<State> get() = mutableState
@ -120,7 +125,7 @@ class ClosedGroupPoller(
}
fun stop() {
Log.d(TAG, "Stopping closed group poller for ${closedGroupSessionId.hexString.take(4)}")
Log.d(TAG, "Stopping closed group poller for $closedGroupSessionId")
(state.value as? StartedState)?.job?.cancel()
}
@ -238,11 +243,17 @@ class ClosedGroupPoller(
saveLastMessageHash(snode, infoMessage, Namespace.CLOSED_GROUP_INFO())
saveLastMessageHash(snode, membersMessage, Namespace.CLOSED_GROUP_MEMBERS())
val isGroupExpired = configFactoryProtocol.withGroupConfigs(closedGroupSessionId) {
it.groupKeys.size() == 0
}
// As soon as we have handled config messages, the polling count as successful,
// as normally the outside world really only cares about configs.
val currentState = state.value as? StartedState
if (currentState != null && !currentState.hadAtLeastOneSuccessfulPoll) {
mutableState.value = currentState.copy(hadAtLeastOneSuccessfulPoll = true)
mutableState.update {
(it as? StartedState)?.copy(
hadAtLeastOneSuccessfulPoll = true,
expired = isGroupExpired,
) ?: it
}
val regularMessages = groupMessageRetrieval.await()

@ -26,6 +26,13 @@ data class AccountId(
Hex.fromStringCondensed(hexString.drop(2))
}
override fun toString(): String {
return StringBuilder(8)
.append(hexString, 0, 5)
.append("...")
.toString()
}
override fun compareTo(other: AccountId): Int {
return hexString.compareTo(other.hexString)
}

Loading…
Cancel
Save