@ -81,53 +81,106 @@ describe('SwarmPolling', () => {
expect ( swarmPolling . getPollingTimeout ( fakeConvo ) ) . to . eq ( SWARM_POLLING_TIMEOUT . INACTIVE ) ;
} ) ;
it ( 'returns ACTIVE for convo with less than two days old activeAt' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) - 2 * 23 * 3600 * 1000 ) ; // 23 * 2 = 46 hours old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . ACTIVE
) ;
} ) ;
describe ( 'legacy groups' , ( ) = > {
it ( 'returns ACTIVE for convo with less than two days old activeAt' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) - 2 * 23 * 3600 * 1000 ) ; // 23 * 2 = 46 hours old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . ACTIVE
) ;
} ) ;
it ( 'returns INACTIVE for convo with undefined activeAt' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , undefined ) ;
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . INACTIVE
) ;
} ) ;
it ( 'returns INACTIVE for convo with undefined activeAt' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , undefined ) ;
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . INACTIVE
) ;
} ) ;
it ( 'returns MEDIUM_ACTIVE for convo with activeAt of more than 2 days but less than a week old' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 25 * 2 ) ; // 25 hours x 2 = 50 hours old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . MEDIUM_ACTIVE
) ;
it ( 'returns MEDIUM_ACTIVE for convo with activeAt of more than 2 days but less than a week old' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 25 * 2 ) ; // 25 hours x 2 = 50 hours old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . MEDIUM_ACTIVE
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 24 * 7 + 3600 ) ; // a week minus an hour old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . MEDIUM_ACTIVE
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 24 * 7 + 3600 ) ; // a week minus an hour old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . MEDIUM_ACTIVE
) ;
} ) ;
it ( 'returns INACTIVE for convo with activeAt of more than a week' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 24 * 8 ) ; // 8 days
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . INACTIVE
) ;
} ) ;
} ) ;
it ( 'returns INACTIVE for convo with activeAt of more than a week' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 24 * 8 ) ; // 8 days
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . INACTIVE
) ;
describe ( 'groupv3' , ( ) = > {
it ( 'returns ACTIVE for convo with less than two days old activeAt' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) - 2 * 23 * 3600 * 1000 ) ; // 23 * 2 = 46 hours old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . ACTIVE
) ;
} ) ;
it ( 'returns INACTIVE for convo with undefined activeAt' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , undefined ) ;
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . INACTIVE
) ;
} ) ;
it ( 'returns MEDIUM_ACTIVE for convo with activeAt of more than 2 days but less than a week old' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 25 * 2 ) ; // 25 hours x 2 = 50 hours old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . MEDIUM_ACTIVE
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 24 * 7 + 3600 ) ; // a week minus an hour old
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . MEDIUM_ACTIVE
) ;
} ) ;
it ( 'returns INACTIVE for convo with activeAt of more than a week' , ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) - 1000 * 3600 * 24 * 8 ) ; // 8 days
expect ( swarmPolling . getPollingTimeout ( PubKey . cast ( convo . id as string ) ) ) . to . eq (
SWARM_POLLING_TIMEOUT . INACTIVE
) ;
} ) ;
} ) ;
} ) ;
@ -174,210 +227,453 @@ describe('SwarmPolling', () => {
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
} ) ;
it ( 'does run for group pubkey on start no matter the recent timestamp ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
describe ( 'legacy group' , ( ) = > {
it ( 'does run for group pubkey on start no matter the recent timestamp ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does only poll from -10 for closed groups if HF >= 19.1 ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
getItemByIdStub . restore ( ) ;
getItemByIdStub = TestUtils . stubData ( 'getItemById' ) ;
getItemByIdStub
. withArgs ( 'hasSeenHardfork190' )
. resolves ( { id : 'hasSeenHardfork190' , value : true } )
. withArgs ( 'hasSeenHardfork191' )
. resolves ( { id : 'hasSeenHardfork191' , value : true } ) ;
convo . set ( 'active_at' , 1 ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
it ( 'does run for groupv3 pubkey on start no matter the recent timestamp ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
await swarmPolling . start ( true ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
getItemByIdStub . restore ( ) ;
getItemByIdStub = TestUtils . stubData ( 'getItemById' ) ;
it ( 'does only poll from -10 for closed groups if HF >= 19.1 ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
getItemByIdStub . restore ( ) ;
getItemByIdStub = TestUtils . stubData ( 'getItemById' ) ;
getItemByIdStub
. withArgs ( 'hasSeenHardfork190' )
. resolves ( { id : 'hasSeenHardfork190' , value : true } )
. withArgs ( 'hasSeenHardfork191' )
. resolves ( { id : 'hasSeenHardfork191' , value : true } ) ;
convo . set ( 'active_at' , 1 ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
getItemByIdStub . resolves ( ) ;
} ) ;
await swarmPolling . start ( true ) ;
it ( 'does run for group pubkey on start but not another time if activeAt is old ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
getItemByIdStub . restore ( ) ;
getItemByIdStub = TestUtils . stubData ( 'getItemById' ) ;
convo . set ( 'active_at' , 1 ) ; // really old
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
getItemByIdStub . resolves ( ) ;
} ) ;
// this calls the stub 2 times, one for our direct pubkey and one for the group
await swarmPolling . start ( true ) ;
it ( 'does run for group pubkey on start but not another time if activeAt is old ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
// this should only call the stub one more time: for our direct pubkey but not for the group pubkey
await swarmPolling . pollForAllKeys ( ) ;
convo . set ( 'active_at' , 1 ) ; // really old
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 3 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
} ) ;
// this calls the stub 2 times, one for our direct pubkey and one for the group
await swarmPolling . start ( true ) ;
it ( 'does run twice if activeAt less than one hour ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
// this should only call the stub one more time: for our direct pubkey but not for the group pubkey
await swarmPolling . pollForAllKeys ( ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
clock . tick ( 9000 ) ;
// no need to do that as the tick will trigger a call in all cases after 5 secs await swarmPolling.pollForAllKeys();
/ * * t h i s i s n o t e a s y t o e x p l a i n , b u t
* - during the swarmPolling . start , we get two calls to pollOnceForKeySpy ( one for our id and one for group od )
* - the clock ticks 9 sec , and another call of pollOnceForKeySpy get started , but as we do not await them , this test fails .
* the only fix is to restore the clock and force the a small sleep to let the thing run in bg
* /
clock . restore ( ) ;
await sleepFor ( 10 ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCall ( 3 ) . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 3 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt is inactive and we tick longer than 2 minutes ', async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
it ( 'does run twice if activeAt less than one hour ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
// this call the stub two times already, one for our direct pubkey and one for the group
await swarmPolling . start ( true ) ;
const timeToTick = 3 * 60 * 1000 ;
swarmPolling . forcePolledTimestamp ( groupConvoPubkey , Date . now ( ) - timeToTick ) ;
// more than week old, so inactive group but we have to tick after more than 2 min
convo . set ( 'active_at' , Date . now ( ) - 7 * 25 * 3600 * 1000 ) ;
clock . tick ( timeToTick ) ;
/ * * t h i s i s n o t e a s y t o e x p l a i n , b u t
* - during the swarmPolling . start , we get two calls to pollOnceForKeySpy ( one for our id and one for group od )
* - the clock ticks 9 sec , and another call of pollOnceForKeySpy get started , but as we do not await them , this test fails .
* the only fix is to restore the clock and force the a small sleep to let the thing run in bg
* /
clock . restore ( ) ;
await sleepFor ( 10 ) ;
// we should have two more calls here, so 4 total.
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
clock . tick ( 9000 ) ;
// no need to do that as the tick will trigger a call in all cases after 5 secs await swarmPolling.pollForAllKeys();
/ * * t h i s i s n o t e a s y t o e x p l a i n , b u t
* - during the swarmPolling . start , we get two calls to pollOnceForKeySpy ( one for our id and one for group od )
* - the clock ticks 9 sec , and another call of pollOnceForKeySpy get started , but as we do not await them , this test fails .
* the only fix is to restore the clock and force the a small sleep to let the thing run in bg
* /
clock . restore ( ) ;
await sleepFor ( 10 ) ;
it ( 'does run once only if group is inactive and we tick less than 2 minutes ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils. generateFakePubKeyStr ( ) ,
ConversationTypeEnum. GROUP
) ;
pollOnceForKeySpy . resetHistory ( ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCall ( 3 ) . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
it ( 'does run twice if activeAt is inactive and we tick longer than 2 minutes' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
// this call the stub two times already, one for our direct pubkey and one for the group
await swarmPolling . start ( true ) ;
const timeToTick = 3 * 60 * 1000 ;
swarmPolling . forcePolledTimestamp ( groupConvoPubkey , Date . now ( ) - timeToTick ) ;
// more than week old, so inactive group but we have to tick after more than 2 min
convo . set ( 'active_at' , Date . now ( ) - 7 * 25 * 3600 * 1000 ) ;
clock . tick ( timeToTick ) ;
/ * * t h i s i s n o t e a s y t o e x p l a i n , b u t
* - during the swarmPolling . start , we get two calls to pollOnceForKeySpy ( one for our id and one for group od )
* - the clock ticks 9 sec , and another call of pollOnceForKeySpy get started , but as we do not await them , this test fails .
* the only fix is to restore the clock and force the a small sleep to let the thing run in bg
* /
clock . restore ( ) ;
await sleepFor ( 10 ) ;
// we should have two more calls here, so 4 total.
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run once only if group is inactive and we tick less than 2 minutes ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
// more than a week old, we should not tick after just 5 seconds
convo . set ( 'active_at' , Date . now ( ) - 7 * 24 * 3600 * 1000 - 3600 * 1000 ) ;
// more than a week old, we should not tick after just 5 seconds
convo . set ( 'active_at' , Date . now ( ) - 7 * 24 * 3600 * 1000 - 3600 * 1000 ) ;
clock . tick ( 1 * 60 * 1000 ) ;
clock . tick ( 1 * 60 * 1000 ) ;
// we should have only one more call here, the one for our direct pubkey fetch
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 3 ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ; // this one comes from the swarmPolling.start
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
} ) ;
// we should have only one more call here, the one for our direct pubkey fetch
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 3 ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ; // this one comes from the swarmPolling.start
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
} ) ;
describe ( 'multiple runs' , ( ) = > {
let convo : ConversationModel ;
let groupConvoPubkey : PubKey ;
describe ( 'multiple runs' , ( ) = > {
let convo : ConversationModel ;
let groupConvoPubkey : PubKey ;
beforeEach ( async ( ) = > {
convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
} ) ;
it ( 'does run twice if activeAt is less than 2 days' , async ( ) = > {
pollOnceForKeySpy . resetHistory ( ) ;
// less than 2 days old, this is an active group
convo . set ( 'active_at' , Date . now ( ) - 2 * 24 * 3600 * 1000 - 3600 * 1000 ) ;
const timeToTick = 6 * 1000 ;
swarmPolling . forcePolledTimestamp ( convo . id , timeToTick ) ;
// we tick more than 5 sec
clock . tick ( timeToTick ) ;
await swarmPolling . pollForAllKeys ( ) ;
// we have 4 calls total. 2 for our direct promises run each 5 seconds, and 2 for the group pubkey active (so run every 5 sec too)
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
// first two calls are our pubkey
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt is more than 2 days old and we tick more than one minute ' , async ( ) = > {
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) - 2 * 25 * 3600 * 1000 ) ; // medium active
const timeToTick = 65 * 1000 ;
swarmPolling . forcePolledTimestamp ( convo . id , timeToTick ) ;
clock . tick ( timeToTick ) ; // should tick twice more (one more our direct pubkey and one for the group)
await swarmPolling . pollForAllKeys ( ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
// first two calls are our pubkey
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
} ) ;
} ) ;
beforeEach ( async ( ) = > {
convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakePubKeyStr ( ) ,
ConversationTypeEnum . GROUP
describe ( 'group v3' , ( ) = > {
it ( 'does run for group pubkey on start no matter the recent timestamp ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run for groupv3 pubkey on start no matter the recent timestamp ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt is less than 2 days' , async ( ) = > {
pollOnceForKeySpy . resetHistory ( ) ;
// less than 2 days old, this is an active group
convo . set ( 'active_at' , Date . now ( ) - 2 * 24 * 3600 * 1000 - 3600 * 1000 ) ;
it ( 'does only poll from -10 for closed groups if HF >= 19.1 ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
getItemByIdStub . restore ( ) ;
getItemByIdStub = TestUtils . stubData ( 'getItemById' ) ;
getItemByIdStub
. withArgs ( 'hasSeenHardfork190' )
. resolves ( { id : 'hasSeenHardfork190' , value : true } )
. withArgs ( 'hasSeenHardfork191' )
. resolves ( { id : 'hasSeenHardfork191' , value : true } ) ;
convo . set ( 'active_at' , 1 ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
const timeToTick = 6 * 1000 ;
await swarmPolling . start ( true ) ;
swarmPolling . forcePolledTimestamp ( convo . id , timeToTick ) ;
// we tick more than 5 sec
clock . tick ( timeToTick ) ;
// our pubkey will be polled for, hence the 2
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 2 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
getItemByIdStub . restore ( ) ;
getItemByIdStub = TestUtils . stubData ( 'getItemById' ) ;
getItemByIdStub . resolves ( ) ;
} ) ;
it ( 'does run for group pubkey on start but not another time if activeAt is old ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , 1 ) ; // really old
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
// this calls the stub 2 times, one for our direct pubkey and one for the group
await swarmPolling . start ( true ) ;
// this should only call the stub one more time: for our direct pubkey but not for the group pubkey
await swarmPolling . pollForAllKeys ( ) ;
// we have 4 calls total. 2 for our direct promises run each 5 seconds, and 2 for the group pubkey active (so run every 5 sec too)
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 3 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt less than one hour ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
clock . tick ( 9000 ) ;
// no need to do that as the tick will trigger a call in all cases after 5 secs await swarmPolling.pollForAllKeys();
/ * * t h i s i s n o t e a s y t o e x p l a i n , b u t
* - during the swarmPolling . start , we get two calls to pollOnceForKeySpy ( one for our id and one for group od )
* - the clock ticks 9 sec , and another call of pollOnceForKeySpy get started , but as we do not await them , this test fails .
* the only fix is to restore the clock and force the a small sleep to let the thing run in bg
* /
clock . restore ( ) ;
await sleepFor ( 10 ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
// first two calls are our pubkey
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCall ( 3 ) . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt is inactive and we tick longer than 2 minutes' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
// this call the stub two times already, one for our direct pubkey and one for the group
await swarmPolling . start ( true ) ;
const timeToTick = 3 * 60 * 1000 ;
swarmPolling . forcePolledTimestamp ( groupConvoPubkey , Date . now ( ) - timeToTick ) ;
// more than week old, so inactive group but we have to tick after more than 2 min
convo . set ( 'active_at' , Date . now ( ) - 7 * 25 * 3600 * 1000 ) ;
clock . tick ( timeToTick ) ;
/ * * t h i s i s n o t e a s y t o e x p l a i n , b u t
* - during the swarmPolling . start , we get two calls to pollOnceForKeySpy ( one for our id and one for group od )
* - the clock ticks 9 sec , and another call of pollOnceForKeySpy get started , but as we do not await them , this test fails .
* the only fix is to restore the clock and force the a small sleep to let the thing run in bg
* /
clock . restore ( ) ;
await sleepFor ( 10 ) ;
// we should have two more calls here, so 4 total.
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt is more than 2 days old and we tick more than one minute ' , async ( ) = > {
it ( 'does run once only if group is inactive and we tick less than 2 minutes ' , async ( ) = > {
const convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) - 2 * 25 * 3600 * 1000 ) ; // medium active
const timeToTick = 65 * 1000 ;
swarmPolling . forcePolledTimestamp ( convo . id , timeToTick ) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
const groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
clock . tick ( timeToTick ) ; // should tick twice more (one more our direct pubkey and one for the group)
// more than a week old, we should not tick after just 5 seconds
convo . set ( 'active_at' , Date . now ( ) - 7 * 24 * 3600 * 1000 - 3600 * 1000 ) ;
await swarmPolling . pollForAllKeys ( ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
clock . tick ( 1 * 60 * 1000 ) ;
// first two calls are our pubkey
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
// we should have only one more call here, the one for our direct pubkey fetch
expect ( pollOnceForKeySpy . callCount) . to . eq ( 3 ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ; // this one comes from the swarmPolling.start
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
describe ( 'multiple runs' , ( ) = > {
let convo : ConversationModel ;
let groupConvoPubkey : PubKey ;
beforeEach ( async ( ) = > {
convo = getConversationController ( ) . getOrCreate (
TestUtils . generateFakeClosedGroupV3PkStr ( ) ,
ConversationTypeEnum . GROUPV3
) ;
convo . set ( 'active_at' , Date . now ( ) ) ;
groupConvoPubkey = PubKey . cast ( convo . id as string ) ;
swarmPolling . addGroupId ( groupConvoPubkey ) ;
await swarmPolling . start ( true ) ;
} ) ;
it ( 'does run twice if activeAt is less than 2 days' , async ( ) = > {
pollOnceForKeySpy . resetHistory ( ) ;
// less than 2 days old, this is an active group
convo . set ( 'active_at' , Date . now ( ) - 2 * 24 * 3600 * 1000 - 3600 * 1000 ) ;
const timeToTick = 6 * 1000 ;
swarmPolling . forcePolledTimestamp ( convo . id , timeToTick ) ;
// we tick more than 5 sec
clock . tick ( timeToTick ) ;
await swarmPolling . pollForAllKeys ( ) ;
// we have 4 calls total. 2 for our direct promises run each 5 seconds, and 2 for the group pubkey active (so run every 5 sec too)
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
// first two calls are our pubkey
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
it ( 'does run twice if activeAt is more than 2 days old and we tick more than one minute ' , async ( ) = > {
pollOnceForKeySpy . resetHistory ( ) ;
convo . set ( 'active_at' , Date . now ( ) - 2 * 25 * 3600 * 1000 ) ; // medium active
const timeToTick = 65 * 1000 ;
swarmPolling . forcePolledTimestamp ( convo . id , timeToTick ) ;
clock . tick ( timeToTick ) ; // should tick twice more (one more our direct pubkey and one for the group)
await swarmPolling . pollForAllKeys ( ) ;
expect ( pollOnceForKeySpy . callCount ) . to . eq ( 4 ) ;
// first two calls are our pubkey
expect ( pollOnceForKeySpy . firstCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . secondCall . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
expect ( pollOnceForKeySpy . thirdCall . args ) . to . deep . eq ( [ ourPubkey , false , [ 0 ] ] ) ;
expect ( pollOnceForKeySpy . getCalls ( ) [ 3 ] . args ) . to . deep . eq ( [ groupConvoPubkey , true , [ - 10 ] ] ) ;
} ) ;
} ) ;
} ) ;
} ) ;