@ -2,6 +2,7 @@ package org.session.libsession.messaging.sending_receiving.pollers
import android.util.SparseArray
import android.util.SparseArray
import androidx.core.util.valueIterator
import androidx.core.util.valueIterator
import kotlinx.coroutines.CoroutineScope
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.runBlocking
import kotlinx.coroutines.runBlocking
import network.loki.messenger.libsession_util.ConfigBase
import network.loki.messenger.libsession_util.ConfigBase
@ -13,6 +14,11 @@ import java.util.Timer
import java.util.TimerTask
import java.util.TimerTask
import kotlin.time.Duration.Companion.days
import kotlin.time.Duration.Companion.days
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.Job
import kotlinx.coroutines.cancel
import kotlinx.coroutines.delay
import kotlinx.coroutines.isActive
import kotlinx.coroutines.launch
import nl.komponents.kovenant.Deferred
import nl.komponents.kovenant.Deferred
import nl.komponents.kovenant.Promise
import nl.komponents.kovenant.Promise
import nl.komponents.kovenant.deferred
import nl.komponents.kovenant.deferred
@ -26,8 +32,12 @@ import org.session.libsession.messaging.jobs.JobQueue
import org.session.libsession.messaging.jobs.MessageReceiveParameters
import org.session.libsession.messaging.jobs.MessageReceiveParameters
import org.session.libsession.snode.RawResponse
import org.session.libsession.snode.RawResponse
import org.session.libsession.snode.SnodeAPI
import org.session.libsession.snode.SnodeAPI
import org.session.libsession.snode.SnodeAPI.KEY_BODY
import org.session.libsession.snode.SnodeAPI.KEY_CODE
import org.session.libsession.snode.SnodeAPI.KEY_RESULTS
import org.session.libsession.snode.SnodeModule
import org.session.libsession.snode.SnodeModule
import org.session.libsession.snode.utilities.asyncPromise
import org.session.libsession.snode.utilities.asyncPromise
import org.session.libsession.snode.utilities.await
import org.session.libsession.utilities.ConfigFactoryProtocol
import org.session.libsession.utilities.ConfigFactoryProtocol
import org.session.libsession.utilities.ConfigMessage
import org.session.libsession.utilities.ConfigMessage
import org.session.libsession.utilities.UserConfigType
import org.session.libsession.utilities.UserConfigType
@ -40,8 +50,6 @@ import org.session.libsignal.utilities.Util.SECURE_RANDOM
private const val TAG = " Poller "
private const val TAG = " Poller "
private class PromiseCanceledException : Exception ( " Promise canceled. " )
class Poller (
class Poller (
private val configFactory : ConfigFactoryProtocol ,
private val configFactory : ConfigFactoryProtocol ,
private val storage : StorageProtocol ,
private val storage : StorageProtocol ,
@ -50,9 +58,9 @@ class Poller(
private val userPublicKey : String
private val userPublicKey : String
get ( ) = storage . getUserPublicKey ( ) . orEmpty ( )
get ( ) = storage . getUserPublicKey ( ) . orEmpty ( )
private var hasStarted : Boolean = false
var scope : CoroutineScope ? = null
private val usedSnodes : MutableSet < Snode > = mutableSetOf ( )
var is CaughtUp = false
var is Polling: Boolean = false
// region Settings
// region Settings
companion object {
companion object {
@ -64,25 +72,26 @@ class Poller(
// region Public API
// region Public API
fun startIfNeeded ( ) {
fun startIfNeeded ( ) {
if ( hasStarted ) { return }
if ( scope != null ) { return }
Log . d ( TAG , " Started polling. " )
Log . d ( TAG , " Started polling. " )
hasStarted = true
scope = CoroutineScope ( Dispatchers . Default )
setUpPolling ( RETRY _INTERVAL _MS )
scope ?. launch {
setUpPolling ( )
}
}
}
fun stopIfNeeded ( ) {
fun stopIfNeeded ( ) {
Log . d ( TAG , " Stopped polling. " )
Log . d ( TAG , " Stopped polling. " )
hasStarted = false
scope ?. cancel ( )
usedSnodes . clear ( )
scope = null
isPolling = false
}
}
fun retrieveUserProfile ( ) {
fun retrieveUserProfile ( ) {
Log . d ( TAG , " Retrieving user profile. for key = $userPublicKey " )
Log . d ( TAG , " Retrieving user profile. for key = $userPublicKey " )
SnodeAPI . getSwarm ( userPublicKey ) . bind {
SnodeAPI . getSwarm ( userPublicKey ) . success {
usedSnodes . clear ( )
pollUserProfile ( it . random ( ) )
deferred < Unit , Exception > ( ) . also { exception ->
pollNextSnode ( userProfileOnly = true , exception )
} . promise
} . fail { exception ->
} . fail { exception ->
Log . e ( TAG , " Failed to retrieve user profile. " , exception )
Log . e ( TAG , " Failed to retrieve user profile. " , exception )
}
}
@ -90,51 +99,41 @@ class Poller(
// endregion
// endregion
// region Private API
// region Private API
private fun setUpPolling ( delay : Long ) {
private suspend fun setUpPolling ( ) {
if ( ! hasStarted ) { return ; }
val pollPool = hashSetOf < Snode > ( ) // pollPool is the list of snodes we can use while rotating snodes from our swarm
val thread = Thread . currentThread ( )
var retryScalingFactor = 1.0f // We increment the retry interval by NEXT_RETRY_MULTIPLIER times this value, which we bump on each failure
SnodeAPI . getSwarm ( userPublicKey ) . bind {
usedSnodes . clear ( )
while ( true ) {
val deferred = deferred < Unit , Exception > ( )
Log . d ( TAG , " Polling... " )
pollNextSnode ( deferred = deferred )
deferred . promise
isPolling = true
} . success {
val nextDelay = if ( isCaughtUp ) RETRY _INTERVAL _MS else 0
// check if the polling pool is empty
Timer ( ) . schedule ( object : TimerTask ( ) {
if ( pollPool . isEmpty ( ) ) {
override fun run ( ) {
// if it is empty, fill it with the snodes from our swarm
thread . run { setUpPolling ( RETRY _INTERVAL _MS ) }
pollPool . addAll ( SnodeAPI . getSwarm ( userPublicKey ) . await ( ) )
}
} , nextDelay )
} . fail {
val nextDelay = minOf ( MAX _RETRY _INTERVAL _MS , ( delay * NEXT _RETRY _MULTIPLIER ) . toLong ( ) )
Timer ( ) . schedule ( object : TimerTask ( ) {
override fun run ( ) {
thread . run { setUpPolling ( nextDelay ) }
}
} , nextDelay )
}
}
private fun pollNextSnode ( userProfileOnly : Boolean = false , deferred : Deferred < Unit , Exception > ) {
val swarm = SnodeModule . shared . storage . getSwarm ( userPublicKey ) ?: setOf ( )
val unusedSnodes = swarm . subtract ( usedSnodes )
if ( unusedSnodes . isNotEmpty ( ) ) {
val index = SECURE_RANDOM . nextInt ( unusedSnodes . size )
val nextSnode = unusedSnodes . elementAt ( index )
usedSnodes . add ( nextSnode )
Log . d ( TAG , " Polling $nextSnode . " )
poll ( userProfileOnly , nextSnode , deferred ) . fail { exception ->
if ( exception is PromiseCanceledException ) {
Log . d ( TAG , " Polling $nextSnode canceled. " )
} else {
Log . d ( TAG , " Polling $nextSnode failed; dropping it and switching to next snode. " )
SnodeAPI . dropSnodeFromSwarmIfNeeded ( nextSnode , userPublicKey )
pollNextSnode ( userProfileOnly , deferred )
}
}
// randomly get a snode from the pool
val currentNode = pollPool . random ( )
// remove that snode from the pool
pollPool . remove ( currentNode )
var pollDelay = RETRY _INTERVAL _MS
try {
poll ( currentNode )
retryScalingFactor = 1f
} catch ( e : Exception ) {
Log . e ( TAG , " Error while polling: " , e )
pollDelay = minOf ( MAX _RETRY _INTERVAL _MS , ( RETRY _INTERVAL _MS * ( NEXT _RETRY _MULTIPLIER * retryScalingFactor ) ) . toLong ( ) )
retryScalingFactor ++
} finally {
isPolling = false
}
}
} else {
isCaughtUp = true
// wait before polling again
deferred . resolve ( )
delay ( pollDelay )
}
}
}
}
@ -184,14 +183,8 @@ class Poller(
}
}
}
}
private fun poll ( userProfileOnly : Boolean , snode : Snode , deferred : Deferred < Unit , Exception > ) : Promise < Unit , Exception > {
//todo we will need to modify this further to fit within the new coroutine setup (currently used by ApplicationContext which is a java class)
if ( userProfileOnly ) {
private fun pollUserProfile ( snode : Snode ) {
return pollUserProfile ( snode , deferred )
}
return poll ( snode , deferred )
}
private fun pollUserProfile ( snode : Snode , deferred : Deferred < Unit , Exception > ) : Promise < Unit , Exception > = GlobalScope . asyncPromise {
val requests = mutableListOf < SnodeAPI . SnodeBatchRequestInfo > ( )
val requests = mutableListOf < SnodeAPI . SnodeBatchRequestInfo > ( )
val hashesToExtend = mutableSetOf < String > ( )
val hashesToExtend = mutableSetOf < String > ( )
val userAuth = requireNotNull ( MessagingModuleConfiguration . shared . storage . userAuth )
val userAuth = requireNotNull ( MessagingModuleConfiguration . shared . storage . userAuth )
@ -224,14 +217,13 @@ class Poller(
if ( requests . isNotEmpty ( ) ) {
if ( requests . isNotEmpty ( ) ) {
SnodeAPI . getRawBatchResponse ( snode , userPublicKey , requests ) . bind { rawResponses ->
SnodeAPI . getRawBatchResponse ( snode , userPublicKey , requests ) . bind { rawResponses ->
isCaughtUp = true
if ( ! deferred . promise . isDone ( ) ) {
val responseList = ( rawResponses [ KEY _RESULTS ] as List < RawResponse > )
val responseList = ( rawResponses [ " results " ] as List < RawResponse > )
responseList . getOrNull ( 0 ) ?. let { rawResponse ->
responseList . getOrNull ( 0 ) ?. let { rawResponse ->
if ( rawResponse [ " code " ] as ? Int != 200 ) {
if ( rawResponse [ KEY _CODE ] as ? Int != 200 ) {
Log . e ( TAG , " Batch sub-request had non-200 response code, returned code ${(rawResponse[ "code" ] as? Int) ?: "[unknown]"}" )
Log . e ( TAG , " Batch sub-request had non-200 response code, returned code ${(rawResponse[ KEY_CODE ] as? Int) ?: "[unknown]"}" )
} else {
} else {
val body = rawResponse [ " body " ] as ? RawResponse
val body = rawResponse [ KEY _BODY ] as ? RawResponse
if ( body == null ) {
if ( body == null ) {
Log . e ( TAG , " Batch sub-request didn't contain a body " )
Log . e ( TAG , " Batch sub-request didn't contain a body " )
} else {
} else {
@ -239,7 +231,7 @@ class Poller(
}
}
}
}
}
}
}
Promise . ofSuccess ( Unit )
Promise . ofSuccess ( Unit )
} . fail {
} . fail {
Log . e ( TAG , " Failed to get raw batch response " , it )
Log . e ( TAG , " Failed to get raw batch response " , it )
@ -247,10 +239,7 @@ class Poller(
}
}
}
}
private suspend fun poll ( snode : Snode ) {
private fun poll ( snode : Snode , deferred : Deferred < Unit , Exception > ) : Promise < Unit , Exception > {
if ( ! hasStarted ) { return Promise . ofFail ( PromiseCanceledException ( ) ) }
return GlobalScope . asyncPromise {
val userAuth = requireNotNull ( MessagingModuleConfiguration . shared . storage . userAuth )
val userAuth = requireNotNull ( MessagingModuleConfiguration . shared . storage . userAuth )
val requestSparseArray = SparseArray < SnodeAPI . SnodeBatchRequestInfo > ( )
val requestSparseArray = SparseArray < SnodeAPI . SnodeBatchRequestInfo > ( )
// get messages
// get messages
@ -290,8 +279,7 @@ class Poller(
requestSparseArray [ namespace ] = request
requestSparseArray [ namespace ] = request
}
}
val requests =
val requests = requestSparseArray . valueIterator ( ) . asSequence ( ) . toMutableList ( )
requestSparseArray . valueIterator ( ) . asSequence ( ) . toMutableList ( )
if ( hashesToExtend . isNotEmpty ( ) ) {
if ( hashesToExtend . isNotEmpty ( ) ) {
SnodeAPI . buildAuthenticatedAlterTtlBatchRequest (
SnodeAPI . buildAuthenticatedAlterTtlBatchRequest (
@ -305,12 +293,8 @@ class Poller(
}
}
if ( requests . isNotEmpty ( ) ) {
if ( requests . isNotEmpty ( ) ) {
SnodeAPI . getRawBatchResponse ( snode , userPublicKey , requests ) . bind { rawResponses ->
val rawResponses = SnodeAPI . getRawBatchResponse ( snode , userPublicKey , requests ) . await ( )
isCaughtUp = true
val responseList = ( rawResponses [ KEY _RESULTS ] as List < RawResponse > )
if ( deferred . promise . isDone ( ) ) {
return @bind Promise . ofSuccess ( Unit )
} else {
val responseList = ( rawResponses [ " results " ] as List < RawResponse > )
// in case we had null configs, the array won't be fully populated
// in case we had null configs, the array won't be fully populated
// index of the sparse array key iterator should be the request index, with the key being the namespace
// index of the sparse array key iterator should be the request index, with the key being the namespace
UserConfigType . entries
UserConfigType . entries
@ -318,11 +302,11 @@ class Poller(
. filter { ( _ , i ) -> i >= 0 }
. filter { ( _ , i ) -> i >= 0 }
. forEach { ( configType , requestIndex ) ->
. forEach { ( configType , requestIndex ) ->
responseList . getOrNull ( requestIndex ) ?. let { rawResponse ->
responseList . getOrNull ( requestIndex ) ?. let { rawResponse ->
if ( rawResponse [ " code " ] as ? Int != 200 ) {
if ( rawResponse [ KEY _CODE ] as ? Int != 200 ) {
Log . e ( TAG , " Batch sub-request had non-200 response code, returned code ${(rawResponse[ "code" ] as? Int) ?: "[unknown]"}" )
Log . e ( TAG , " Batch sub-request had non-200 response code, returned code ${(rawResponse[ KEY_CODE ] as? Int) ?: "[unknown]"}" )
return @forEach
return @forEach
}
}
val body = rawResponse [ " body " ] as ? RawResponse
val body = rawResponse [ KEY _BODY ] as ? RawResponse
if ( body == null ) {
if ( body == null ) {
Log . e ( TAG , " Batch sub-request didn't contain a body " )
Log . e ( TAG , " Batch sub-request didn't contain a body " )
return @forEach
return @forEach
@ -336,31 +320,21 @@ class Poller(
val personalResponseIndex = requestSparseArray . indexOfKey ( Namespace . DEFAULT ( ) )
val personalResponseIndex = requestSparseArray . indexOfKey ( Namespace . DEFAULT ( ) )
if ( personalResponseIndex >= 0 ) {
if ( personalResponseIndex >= 0 ) {
responseList . getOrNull ( personalResponseIndex ) ?. let { rawResponse ->
responseList . getOrNull ( personalResponseIndex ) ?. let { rawResponse ->
if ( rawResponse [ " code " ] as ? Int != 200 ) {
if ( rawResponse [ KEY _CODE ] as ? Int != 200 ) {
Log . e ( TAG , " Batch sub-request for personal messages had non-200 response code, returned code ${(rawResponse["code"] as? Int) ?: "[unknown]"} " )
// If we got a non-success response then the snode might be bad
// If we got a non-success response then the snode might be bad so we should try rotate
throw ( RuntimeException ( " Batch sub-request for personal messages had non-200 response code, returned code ${(rawResponse[KEY_CODE] as? Int) ?: "[unknown]"} " ) )
// to a different one just in case
pollNextSnode ( deferred = deferred )
return @bind Promise . ofSuccess ( Unit )
} else {
} else {
val body = rawResponse [ " body " ] as ? RawResponse
val body = rawResponse [ KEY _BODY ] as ? RawResponse
if ( body == null ) {
if ( body == null ) {
Log . e ( TAG , " Batch sub-request for personal messages didn't contain a body " )
throw ( RuntimeException ( " Batch sub-request for personal messages didn't contain a body " ) )
} else {
} else {
processPersonalMessages ( snode , body )
processPersonalMessages ( snode , body )
}
}
}
}
}
}
}
} else {
throw ( SnodeAPI . Error . Generic )
poll ( snode , deferred )
}
} . fail {
Log . e ( TAG , " Failed to get raw batch response " , it )
poll ( snode , deferred )
}
}
}
}
}
}
}
// endregion
}
}