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

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

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

@ -16,6 +16,7 @@ import dagger.hilt.android.qualifiers.ApplicationContext
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.channels.consumeEach import kotlinx.coroutines.channels.consumeEach
import kotlinx.coroutines.flow.Flow
import kotlinx.coroutines.flow.MutableSharedFlow import kotlinx.coroutines.flow.MutableSharedFlow
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.SharedFlow import kotlinx.coroutines.flow.SharedFlow
@ -23,6 +24,7 @@ import kotlinx.coroutines.flow.SharingStarted
import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.combine import kotlinx.coroutines.flow.combine
import kotlinx.coroutines.flow.filterNotNull import kotlinx.coroutines.flow.filterNotNull
import kotlinx.coroutines.flow.flowOf
import kotlinx.coroutines.flow.map import kotlinx.coroutines.flow.map
import kotlinx.coroutines.flow.stateIn import kotlinx.coroutines.flow.stateIn
import kotlinx.coroutines.flow.update 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.MessageRecord
import org.thoughtcrime.securesms.database.model.MmsMessageRecord import org.thoughtcrime.securesms.database.model.MmsMessageRecord
import org.thoughtcrime.securesms.dependencies.ConfigFactory import org.thoughtcrime.securesms.dependencies.ConfigFactory
import org.thoughtcrime.securesms.groups.ExpiredGroupManager
import org.thoughtcrime.securesms.groups.OpenGroupManager import org.thoughtcrime.securesms.groups.OpenGroupManager
import org.thoughtcrime.securesms.mms.AudioSlide import org.thoughtcrime.securesms.mms.AudioSlide
import org.thoughtcrime.securesms.repository.ConversationRepository import org.thoughtcrime.securesms.repository.ConversationRepository
@ -82,6 +85,7 @@ class ConversationViewModel(
private val configFactory: ConfigFactory, private val configFactory: ConfigFactory,
private val groupManagerV2: GroupManagerV2, private val groupManagerV2: GroupManagerV2,
val legacyGroupDeprecationManager: LegacyGroupDeprecationManager, val legacyGroupDeprecationManager: LegacyGroupDeprecationManager,
private val expiredGroupManager: ExpiredGroupManager,
) : ViewModel() { ) : ViewModel() {
val showSendAfterApprovalText: Boolean val showSendAfterApprovalText: Boolean
@ -245,6 +249,13 @@ class ConversationViewModel(
&& state != LegacyGroupDeprecationManager.DeprecationState.NOT_DEPRECATING && state != LegacyGroupDeprecationManager.DeprecationState.NOT_DEPRECATING
}.stateIn(viewModelScope, SharingStarted.Eagerly, false) }.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( private val attachmentDownloadHandler = AttachmentDownloadHandler(
storage = storage, storage = storage,
messageDataProvider = messageDataProvider, messageDataProvider = messageDataProvider,
@ -1089,6 +1100,7 @@ class ConversationViewModel(
private val configFactory: ConfigFactory, private val configFactory: ConfigFactory,
private val groupManagerV2: GroupManagerV2, private val groupManagerV2: GroupManagerV2,
private val legacyGroupDeprecationManager: LegacyGroupDeprecationManager, private val legacyGroupDeprecationManager: LegacyGroupDeprecationManager,
private val expiredGroupManager: ExpiredGroupManager,
) : ViewModelProvider.Factory { ) : ViewModelProvider.Factory {
override fun <T : ViewModel> create(modelClass: Class<T>): T { override fun <T : ViewModel> create(modelClass: Class<T>): T {
@ -1107,6 +1119,7 @@ class ConversationViewModel(
configFactory = configFactory, configFactory = configFactory,
groupManagerV2 = groupManagerV2, groupManagerV2 = groupManagerV2,
legacyGroupDeprecationManager = legacyGroupDeprecationManager, legacyGroupDeprecationManager = legacyGroupDeprecationManager,
expiredGroupManager = expiredGroupManager,
) as T ) 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 package org.thoughtcrime.securesms.dependencies
import android.content.Context
import dagger.Lazy
import dagger.Module import dagger.Module
import dagger.Provides import dagger.Provides
import dagger.hilt.InstallIn import dagger.hilt.InstallIn
import dagger.hilt.android.qualifiers.ApplicationContext
import dagger.hilt.components.SingletonComponent import dagger.hilt.components.SingletonComponent
import kotlinx.coroutines.CoroutineDispatcher import kotlinx.coroutines.CoroutineDispatcher
import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.DelicateCoroutinesApi
import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.ExperimentalCoroutinesApi
import kotlinx.coroutines.GlobalScope import kotlinx.coroutines.GlobalScope
import org.session.libsession.database.StorageProtocol 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.GroupScope
import org.session.libsession.messaging.groups.LegacyGroupDeprecationManager 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.messaging.sending_receiving.pollers.LegacyClosedGroupPollerV2
import org.session.libsession.snode.SnodeClock import org.session.libsession.snode.SnodeClock
import org.session.libsession.utilities.ConfigFactoryProtocol
import org.session.libsession.utilities.TextSecurePreferences 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.Named
import javax.inject.Singleton import javax.inject.Singleton
@Suppress("OPT_IN_USAGE") const val POLLER_SCOPE = "poller_coroutine_scope"
@Module @Module
@InstallIn(SingletonComponent::class) @InstallIn(SingletonComponent::class)
object SessionUtilModule { object SessionUtilModule {
private const val POLLER_SCOPE = "poller_coroutine_scope" @OptIn(DelicateCoroutinesApi::class)
@Provides @Provides
@Named(POLLER_SCOPE) @Named(POLLER_SCOPE)
fun providePollerScope(): CoroutineScope = GlobalScope fun providePollerScope(): CoroutineScope = GlobalScope
@ -42,24 +36,6 @@ object SessionUtilModule {
@Named(POLLER_SCOPE) @Named(POLLER_SCOPE)
fun provideExecutor(): CoroutineDispatcher = Dispatchers.IO.limitedParallelism(1) 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 @Provides
@Singleton @Singleton
fun provideSnodeClock() = SnodeClock() fun provideSnodeClock() = SnodeClock()
@ -77,6 +53,7 @@ object SessionUtilModule {
return LegacyClosedGroupPollerV2(storage, deprecationManager) return LegacyClosedGroupPollerV2(storage, deprecationManager)
} }
@Provides @Provides
@Singleton @Singleton
fun provideLegacyGroupDeprecationManager(prefs: TextSecurePreferences): LegacyGroupDeprecationManager { 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.MmsSmsDatabase
import org.thoughtcrime.securesms.database.ThreadDatabase import org.thoughtcrime.securesms.database.ThreadDatabase
import org.thoughtcrime.securesms.dependencies.ConfigFactory import org.thoughtcrime.securesms.dependencies.ConfigFactory
import org.thoughtcrime.securesms.dependencies.PollerFactory
import org.thoughtcrime.securesms.util.SessionMetaProtocol import org.thoughtcrime.securesms.util.SessionMetaProtocol
import java.util.concurrent.TimeUnit import java.util.concurrent.TimeUnit
import javax.inject.Inject import javax.inject.Inject
@ -78,7 +77,6 @@ class GroupManagerV2Impl @Inject constructor(
private val mmsSmsDatabase: MmsSmsDatabase, private val mmsSmsDatabase: MmsSmsDatabase,
private val lokiDatabase: LokiMessageDatabase, private val lokiDatabase: LokiMessageDatabase,
private val threadDatabase: ThreadDatabase, private val threadDatabase: ThreadDatabase,
private val pollerFactory: PollerFactory,
private val profileManager: SSKEnvironment.ProfileManagerProtocol, private val profileManager: SSKEnvironment.ProfileManagerProtocol,
@ApplicationContext val application: Context, @ApplicationContext val application: Context,
private val clock: SnodeClock, private val clock: SnodeClock,
@ -86,6 +84,7 @@ class GroupManagerV2Impl @Inject constructor(
private val lokiAPIDatabase: LokiAPIDatabase, private val lokiAPIDatabase: LokiAPIDatabase,
private val configUploader: ConfigUploader, private val configUploader: ConfigUploader,
private val scope: GroupScope, private val scope: GroupScope,
private val groupPollerManager: GroupPollerManager,
) : GroupManagerV2 { ) : GroupManagerV2 {
private val dispatcher = Dispatchers.Default private val dispatcher = Dispatchers.Default
@ -196,7 +195,6 @@ class GroupManagerV2Impl @Inject constructor(
profileManager.setName(application, recipient, groupName) profileManager.setName(application, recipient, groupName)
storage.setRecipientApprovedMe(recipient, true) storage.setRecipientApprovedMe(recipient, true)
storage.setRecipientApproved(recipient, true) storage.setRecipientApproved(recipient, true)
pollerFactory.updatePollers()
// Invite members // Invite members
JobQueue.shared.add( JobQueue.shared.add(
@ -494,8 +492,6 @@ class GroupManagerV2Impl @Inject constructor(
} }
} }
pollerFactory.pollerFor(groupId)?.stop()
// Delete conversation and group configs // Delete conversation and group configs
storage.getThreadId(Address.fromSerialized(groupId.hexString)) storage.getThreadId(Address.fromSerialized(groupId.hexString))
?.let(storage::deleteConversation) ?.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 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 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 // We can't hang on here forever if things don't work out, bail out if it's the camse
withTimeout(20_000L) { withTimeout(20_000L) {
poller.state.filterIsInstance<ClosedGroupPoller.StartedState>() groupPollerManager.watchGroupPollingState(group.groupAccountId)
.filterIsInstance<ClosedGroupPoller.StartedState>()
.filter { it.hadAtLeastOneSuccessfulPoll } .filter { it.hadAtLeastOneSuccessfulPoll }
.first() .first()
} }
@ -905,9 +899,6 @@ class GroupManagerV2Impl @Inject constructor(
override suspend fun handleKicked(groupId: AccountId): Unit = scope.launchAndWait(groupId, "Handle kicked") { 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") 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 userId = requireNotNull(storage.getUserPublicKey()) { "No current user available" }
val group = configFactory.getGroup(groupId) ?: return@launchAndWait 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:id="@+id/conversationRecyclerView"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="0dp" android:layout_height="0dp"
tools:visibility="gone"
app:layout_constraintVertical_weight="1" app:layout_constraintVertical_weight="1"
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintBottom_toTopOf="@+id/typingIndicatorViewContainer" 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 <org.thoughtcrime.securesms.conversation.v2.components.TypingIndicatorViewContainer
android:focusable="false" android:focusable="false"
@ -62,6 +63,7 @@
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/recreateGroupButtonContainer" app:layout_constraintTop_toBottomOf="@+id/recreateGroupButtonContainer"
app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintVertical_bias="1"
/> />
<org.thoughtcrime.securesms.conversation.v2.search.SearchBottomBar <org.thoughtcrime.securesms.conversation.v2.search.SearchBottomBar
@ -209,7 +211,7 @@
android:layout_centerInParent="true" android:layout_centerInParent="true"
android:textSize="@dimen/very_small_font_size" android:textSize="@dimen/very_small_font_size"
android:textColor="?android:textColorPrimary" android:textColor="?android:textColorPrimary"
android:text="8" /> tools:text="8" />
</RelativeLayout> </RelativeLayout>
@ -223,70 +225,12 @@
android:visibility="gone" android:visibility="gone"
app:layout_constraintBottom_toBottomOf="parent" /> app:layout_constraintBottom_toBottomOf="parent" />
<FrameLayout <include
android:id="@+id/blockedBanner" android:id="@+id/conversation_header"
android:contentDescription="@string/AccessibilityId_blockedBanner"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
app:layout_constraintTop_toBottomOf="@+id/toolbar" app:layout_constraintTop_toBottomOf="@+id/toolbar"
android:background="?danger" layout="@layout/view_conversation_header" />
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" />
<TextView <TextView
android:padding="@dimen/medium_spacing" android:padding="@dimen/medium_spacing"
@ -297,74 +241,21 @@
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_marginHorizontal="@dimen/large_spacing" 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" android:contentDescription="@string/AccessibilityId_empty_conversation"
tools:text="Some Control Message Text" tools:text="Some Control Message Text"
/> />
<LinearLayout <include
android:id="@+id/messageRequestBar" android:id="@+id/messageRequestBar"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
app:layout_constraintBottom_toTopOf="@+id/recreateGroupButtonContainer" app:layout_constraintBottom_toTopOf="@+id/recreateGroupButtonContainer"
app:layout_constraintTop_toBottomOf="@+id/textSendAfterApproval" app:layout_constraintTop_toBottomOf="@+id/textSendAfterApproval"
layout="@layout/view_conversation_message_request_bar"
android:layout_marginBottom="@dimen/large_spacing" android:layout_marginBottom="@dimen/large_spacing"
android:orientation="vertical" tools:visibility="visible"
android:visibility="gone" 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>
<FrameLayout <FrameLayout
android:id="@+id/recreateGroupButtonContainer" 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>

@ -306,3 +306,11 @@ Java_network_loki_messenger_libsession_1util_GroupKeysConfig_admin(JNIEnv *env,
auto ptr = ptrToKeys(env, thiz); auto ptr = ptrToKeys(env, thiz);
return ptr->admin(); 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 subAccountSign(message: ByteArray, signingValue: ByteArray): GroupKeysConfig.SwarmAuth
fun getSubAccountToken(sessionId: AccountId, canWrite: Boolean = true, canDelete: Boolean = false): ByteArray fun getSubAccountToken(sessionId: AccountId, canWrite: Boolean = true, canDelete: Boolean = false): ByteArray
fun currentGeneration(): Int fun currentGeneration(): Int
fun size(): Int
} }
interface MutableGroupKeysConfig : ReadableGroupKeysConfig { interface MutableGroupKeysConfig : ReadableGroupKeysConfig {
@ -539,6 +540,7 @@ class GroupKeysConfig private constructor(
external override fun currentGeneration(): Int external override fun currentGeneration(): Int
external fun admin(): Boolean external fun admin(): Boolean
external override fun size(): Int
data class SwarmAuth( data class SwarmAuth(
val subAccount: String, val subAccount: String,

@ -29,6 +29,9 @@ sealed class GroupInfo {
fun hasAdminKey() = adminKey != null fun hasAdminKey() = adminKey != null
val shouldPoll: Boolean
get() = !invited && !kicked && !destroyed
companion object { companion object {
/** /**
* Generate the group's admin key(64 bytes) from seed (32 bytes, normally used * 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.delay
import kotlinx.coroutines.flow.MutableStateFlow import kotlinx.coroutines.flow.MutableStateFlow
import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.flow.StateFlow
import kotlinx.coroutines.flow.update
import kotlinx.coroutines.isActive import kotlinx.coroutines.isActive
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import kotlinx.coroutines.supervisorScope import kotlinx.coroutines.supervisorScope
@ -54,7 +55,11 @@ class ClosedGroupPoller(
sealed interface State sealed interface State
data object IdleState : 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) private val mutableState = MutableStateFlow<State>(IdleState)
val state: StateFlow<State> get() = mutableState val state: StateFlow<State> get() = mutableState
@ -120,7 +125,7 @@ class ClosedGroupPoller(
} }
fun stop() { 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() (state.value as? StartedState)?.job?.cancel()
} }
@ -238,11 +243,17 @@ class ClosedGroupPoller(
saveLastMessageHash(snode, infoMessage, Namespace.CLOSED_GROUP_INFO()) saveLastMessageHash(snode, infoMessage, Namespace.CLOSED_GROUP_INFO())
saveLastMessageHash(snode, membersMessage, Namespace.CLOSED_GROUP_MEMBERS()) 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 soon as we have handled config messages, the polling count as successful,
// as normally the outside world really only cares about configs. // as normally the outside world really only cares about configs.
val currentState = state.value as? StartedState mutableState.update {
if (currentState != null && !currentState.hadAtLeastOneSuccessfulPoll) { (it as? StartedState)?.copy(
mutableState.value = currentState.copy(hadAtLeastOneSuccessfulPoll = true) hadAtLeastOneSuccessfulPoll = true,
expired = isGroupExpired,
) ?: it
} }
val regularMessages = groupMessageRetrieval.await() val regularMessages = groupMessageRetrieval.await()

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

Loading…
Cancel
Save