@ -12,8 +12,8 @@ import SignalUtilitiesKit
import SessionUtilitiesKit
public final class NotificationServiceExtension : UNNotificationServiceExtension {
private let dependencies : Dependencies = Dependencies ( )
private var didPerformSetup = false
private var dependencies : Dependencies = Dependencies ( )
private var startTime: CFTimeInterval = 0
private var contentHandler : ( ( UNNotificationContent ) -> Void ) ?
private var request : UNNotificationRequest ?
private var hasCompleted : Atomic < Bool > = Atomic ( false )
@ -27,6 +27,7 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
// MARK: D i d r e c e i v e a r e m o t e p u s h n o t i f i c a t i o n r e q u e s t
override public func didReceive ( _ request : UNNotificationRequest , withContentHandler contentHandler : @ escaping ( UNNotificationContent ) -> Void ) {
self . startTime = CACurrentMediaTime ( )
self . contentHandler = contentHandler
self . request = request
@ -51,37 +52,23 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
Singleton . setup ( appContext : NotificationServiceExtensionContext ( ) )
}
// P e r f o r m m a i n s e t u p
Storage . resumeDatabaseAccess ( using : dependencies )
// / P e r f o r m m a i n s e t u p ( c r e a t e a n e w ` D e p e n d e n c i e s ` i n s t a n c e e a c h t i m e s o w e d o n ' t n e e d t o w o r r y a b o u t s t a t e f r o m p r e v i o u s
// / n o t i f i c a t i o n s c a u s i n g i s s u e s w i t h n e w n o t i f i c a t i o n s
self . dependencies = Dependencies ( )
DispatchQueue . main . sync {
self . setUpIfNecessary ( ) { [ weak self ] in
self . performSetup { [ weak self ] in
self ? . handleNotification ( notificationContent , isPerformingResetup : false )
}
}
}
private func handleNotification ( _ notificationContent : UNMutableNotificationContent , isPerformingResetup : Bool ) {
let userSessionId : String = getUserHexEncodedPublicKey ( using : dependencies )
let ( maybeData , metadata , result ) = PushNotificationAPI . processNotification (
notificationContent : notificationContent ,
using : dependencies
)
// / T h e r e i s a n a n n o y i n g i s s u e w h e r e c l e a r i n g a c c o u n t d a t a a n d c r e a t i n g a n e w a c c o u n t c a n r e s u l t i n t h e u s e r r e c e i v i n g p u s h n o t i f i c a t i o n s
// / f o r t h e n e w a c c o u n t b u t t h e N o t i f i c a t i o n S e r v i c e E x t e n s i o n h a v i n g c a c h e d s t a t e b a s e d o n t h e o l d a c c o u n t
// /
// / I n o r d e r t o a v o i d t h i s w e c h e c k i f t h e a c c o u n t t h e n o t i f i c a t i o n w a s s e n t t o m a t c h e s t h e c u r r e n t u s e r s s e s s i o n I d a n d i f i t d o e s n ' t ( a n d t h e
// / n o t i f i c a t i o n i s f o r a m e s s a g e s t o r e d i n o n e o f t h e u s e r s n a m e s p a c e s ) t h e n t r y t o r e - s e t u p t h e n o t i f i c a t i o n e x t e n s i o n
guard ! metadata . namespace . isCurrentUserNamespace || metadata . accountId = = userSessionId else {
guard ! isPerformingResetup else {
Log . error ( " Received notification for an accountId that isn't the current user, resetup failed. " )
return self . completeSilenty ( handledNotification : false )
}
Log . warn ( " Received notification for an accountId that isn't the current user, attempting to resetup. " )
return self . forceResetup ( notificationContent )
}
guard
( result = = . success || result = = . legacySuccess ) ,
let data : Data = maybeData
@ -246,31 +233,35 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
// I f a n e r r o r o c c u r r e d w e w a n t t o r o l l b a c k t h e t r a n s a c t i o n ( b y t h r o w i n g ) a n d t h e n h a n d l e
// t h e e r r o r o u t s i d e o f t h e d a t a b a s e
let handleError = {
switch error {
case MessageReceiverError . noGroupKeyPair :
Log . warn ( " Failed due to having no legacy group decryption keys. " )
self ? . completeSilenty ( handledNotification : false )
case MessageReceiverError . outdatedMessage :
Log . info ( " Ignoring notification for already seen message. " )
self ? . completeSilenty ( handledNotification : false )
case NotificationError . ignorableMessage :
Log . info ( " Ignoring message which requires no notification. " )
self ? . completeSilenty ( handledNotification : false )
case MessageReceiverError . duplicateMessage , MessageReceiverError . duplicateControlMessage ,
MessageReceiverError . duplicateMessageNewSnode :
Log . info ( " Ignoring duplicate message (probably received it just before going to the background). " )
self ? . completeSilenty ( handledNotification : false )
case NotificationError . messageProcessing :
self ? . handleFailure ( for : notificationContent , error : . messageProcessing )
case let msgError as MessageReceiverError :
self ? . handleFailure ( for : notificationContent , error : . messageHandling ( msgError ) )
default : self ? . handleFailure ( for : notificationContent , error : . other ( error ) )
// D i s p a t c h t o t h e n e x t r u n l o o p t o e n s u r e w e a r e o u t o f t h e d a t a b a s e w r i t e t h r e a d b e f o r e
// h a n d l i n g t h e r e s u l t ( a n d s u s p e n d i n g t h e d a t a b a s e )
DispatchQueue . main . async {
switch error {
case MessageReceiverError . noGroupKeyPair :
Log . warn ( " Failed due to having no legacy group decryption keys. " )
self ? . completeSilenty ( handledNotification : false )
case MessageReceiverError . outdatedMessage :
Log . info ( " Ignoring notification for already seen message. " )
self ? . completeSilenty ( handledNotification : false )
case NotificationError . ignorableMessage :
Log . info ( " Ignoring message which requires no notification. " )
self ? . completeSilenty ( handledNotification : false )
case MessageReceiverError . duplicateMessage , MessageReceiverError . duplicateControlMessage ,
MessageReceiverError . duplicateMessageNewSnode :
Log . info ( " Ignoring duplicate message (probably received it just before going to the background). " )
self ? . completeSilenty ( handledNotification : false )
case NotificationError . messageProcessing :
self ? . handleFailure ( for : notificationContent , error : . messageProcessing )
case let msgError as MessageReceiverError :
self ? . handleFailure ( for : notificationContent , error : . messageHandling ( msgError ) )
default : self ? . handleFailure ( for : notificationContent , error : . other ( error ) )
}
}
}
@ -285,18 +276,29 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
// MARK: S e t u p
private func setUpIfNecessary ( completion : @ escaping ( ) -> Void ) {
Log . assertOnMainThread ( )
// T h e N S E w i l l o f t e n r e - u s e t h e s a m e p r o c e s s , s o i f w e ' r e
// a l r e a d y s e t u p w e w a n t t o d o n o t h i n g ; w e ' r e a l r e a d y r e a d y
// t o p r o c e s s n e w m e s s a g e s .
guard ! didPerformSetup else { return completion ( ) }
private func performSetup ( completion : @ escaping ( ) -> Void ) {
Log . info ( " Performing setup. " )
didPerformSetup = true
_ = AppVersion . shared
// FIXME: R e m o v e t h e s e o n c e t h e d a t a b a s e i n s t a n c e i s f u l l y m a n a g e d v i a ` D e p e n d e n c i e s `
if AppSetup . hasRun {
dependencies . storage . resumeDatabaseAccess ( )
dependencies . storage . reconfigureDatabase ( )
dependencies . caches . mutate ( cache : . general ) { $0 . clearCachedUserPublicKey ( ) }
// I f w e h a d a l r e a d y d o n e a s e t u p t h e n ` l i b S e s s i o n ` w o n ' t h a v e b e e n r e - s e t u p s o
// w e n e e d t o d o s o n o w ( t h i s e n s u r e s i t h a s t h e c o r r e c t u s e r k e y s a s w e l l )
LibSession . clearMemoryState ( using : dependencies )
dependencies . storage . read { [ dependencies ] db in
LibSession . loadState (
db ,
userPublicKey : getUserHexEncodedPublicKey ( db , using : dependencies ) ,
ed25519SecretKey : Identity . fetchUserEd25519KeyPair ( db ) ? . secretKey ,
using : dependencies
)
}
}
AppSetup . setupEnvironment (
retrySetupIfDatabaseInvalid : true ,
@ -315,26 +317,39 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
// S e t u p L i b S e s s i o n
LibSession . addLogger ( )
} ,
migrationsCompletion : { [ weak self ] result , needsConfigSync in
migrationsCompletion : { [ weak self , dependencies ] result , _ in
switch result {
case . failure ( let error ) :
Log . error ( " Failed to complete migrations: \( error ) . " )
self ? . completeSilenty ( handledNotification : false )
case . success :
// W e s h o u l d n e v e r r e c e i v e a n o n - v o i p n o t i f i c a t i o n o n a n a p p t h a t d o e s n ' t s u p p o r t
// a p p e x t e n s i o n s s i n c e w e h a v e t o i n f o r m t h e s e r v i c e w e w a n t e d t h e s e , s o i n t h e o r y
// t h i s p a t h s h o u l d n e v e r o c c u r . H o w e v e r , t h e s e r v i c e d o e s h a v e o u r p u s h t o k e n
// s o i t i s p o s s i b l e t h a t c o u l d c h a n g e i n t h e f u t u r e . I f i t d o e s , d o n o t h i n g
// a n d d o n ' t d i s t u r b t h e u s e r . M e s s a g e s w i l l b e p r o c e s s e d w h e n t h e y o p e n t h e a p p .
guard Storage . shared [ . isReadyForAppExtensions ] else {
Log . error ( " Not ready for extensions. " )
self ? . completeSilenty ( handledNotification : false )
return
}
DispatchQueue . main . async {
self ? . versionMigrationsDidComplete ( needsConfigSync : needsConfigSync , completion : completion )
// E n s u r e s t o r a g e i s a c t u a l l y v a l i d
guard dependencies . storage . isValid else {
Log . error ( " Storage invalid. " )
self ? . completeSilenty ( handledNotification : false )
return
}
// W e s h o u l d n e v e r r e c e i v e a n o n - v o i p n o t i f i c a t i o n o n a n a p p t h a t d o e s n ' t s u p p o r t
// a p p e x t e n s i o n s s i n c e w e h a v e t o i n f o r m t h e s e r v i c e w e w a n t e d t h e s e , s o i n t h e o r y
// t h i s p a t h s h o u l d n e v e r o c c u r . H o w e v e r , t h e s e r v i c e d o e s h a v e o u r p u s h t o k e n
// s o i t i s p o s s i b l e t h a t c o u l d c h a n g e i n t h e f u t u r e . I f i t d o e s , d o n o t h i n g
// a n d d o n ' t d i s t u r b t h e u s e r . M e s s a g e s w i l l b e p r o c e s s e d w h e n t h e y o p e n t h e a p p .
guard dependencies . storage [ . isReadyForAppExtensions ] else {
Log . error ( " Not ready for extensions. " )
self ? . completeSilenty ( handledNotification : false )
return
}
// I f t h e a p p w a s n ' t r e a d y t h e n m a r k i t a s r e a d y n o w
if ! Singleton . appReadiness . isAppReady {
// N o t e t h a t t h i s d o e s m u c h m o r e t h a n s e t a f l a g ; i t w i l l a l s o r u n a l l d e f e r r e d b l o c k s .
Singleton . appReadiness . setAppReady ( )
}
completion ( )
}
}
} ,
@ -342,60 +357,6 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
)
}
private func versionMigrationsDidComplete ( needsConfigSync : Bool , completion : @ escaping ( ) -> Void ) {
Log . assertOnMainThread ( )
// I f w e n e e d a c o n f i g s y n c t h e n t r i g g e r i t n o w
if needsConfigSync {
Storage . shared . write { db in
ConfigurationSyncJob . enqueue ( db , publicKey : getUserHexEncodedPublicKey ( db ) )
}
}
// A p p i s n ' t r e a d y u n t i l s t o r a g e i s r e a d y A N D a l l v e r s i o n m i g r a t i o n s a r e c o m p l e t e .
guard Storage . shared . isValid else {
Log . error ( " Storage invalid. " )
return self . completeSilenty ( handledNotification : false )
}
// I f t h e a p p w a s n ' t r e a d y t h e n m a r k i t a s r e a d y n o w
if ! Singleton . appReadiness . isAppReady {
// N o t e t h a t t h i s d o e s m u c h m o r e t h a n s e t a f l a g ; i t w i l l a l s o r u n a l l d e f e r r e d b l o c k s .
Singleton . appReadiness . setAppReady ( )
}
completion ( )
}
// / I t ' s p o s s i b l e f o r t h e N o t i f i c a t i o n E x t e n s i o n t o s t i l l h a v e s o m e k i n d o f c a c h e d d a t a f r o m t h e o l d d a t a b a s e a f t e r i t ' s b e e n d e l e t e d
// / w h e n a n e w a c c o u n t i s c r e a t e d s h o r t l y a f t e r , t h i s r e s u l t s i n w e i r d e r r o r s w h e n r e c e i v i n g P N s f o r t h e n e w a c c o u n t
// /
// / I n o r d e r t o a v o i d t h i s s i t u a t i o n w e c h e c k t o s e e w h e t h e r t h e r e c e i v e d P N i s t a r g e t t i n g t h e c u r r e n t u s e r a n d , i f n o t , w e c a l l t h i s
// / m e t h o d t o f o r c e a r e s e t u p o f t h e n o t i f i c a t i o n e x t e n s i o n
// /
// / * * N o t e : * * W e n e e d t o r e c o n f i g u r e t h e d a t a b a s e h e r e b e c a u s e i f t h e d a t a b a s e w a s d e l e t e d i t ' s p o s s i b l e f o r t h e N o t i f i c a t i o n E x t e n s i o n
// / t o s o m e h o w s t i l l h a v e s o m e f o r m o f a c c e s s t o t h e o l d o n e
private func forceResetup ( _ notificationContent : UNMutableNotificationContent ) {
Storage . reconfigureDatabase ( )
LibSession . clearMemoryState ( using : dependencies )
dependencies . caches . mutate ( cache : . general ) { $0 . clearCachedUserPublicKey ( ) }
self . setUpIfNecessary ( ) { [ weak self , dependencies ] in
// I f w e h a d a l r e a d y d o n e a s e t u p t h e n ` l i b S e s s i o n ` w o n ' t h a v e b e e n r e - s e t u p s o
// w e n e e d t o d o s o n o w ( t h i s e n s u r e s i t h a s t h e c o r r e c t u s e r k e y s a s w e l l )
Storage . shared . read { db in
LibSession . loadState (
db ,
userPublicKey : getUserHexEncodedPublicKey ( db ) ,
ed25519SecretKey : Identity . fetchUserEd25519KeyPair ( db ) ? . secretKey ,
using : dependencies
)
}
self ? . handleNotification ( notificationContent , isPerformingResetup : true )
}
}
// MARK: H a n d l e c o m p l e t i o n
override public func serviceExtensionTimeWillExpire ( ) {
@ -416,15 +377,17 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
else { return }
let silentContent : UNMutableNotificationContent = UNMutableNotificationContent ( )
silentContent . badge = Storage . shared
. read { db in try Interaction . fetchUnreadCount ( db ) }
. map { NSNumber ( value : $0 ) }
. defaulting ( to : NSNumber ( value : 0 ) )
Log . info ( handledNotification ? " Completed after handling notification. " : " Completed silently. " )
if ! isMainAppAndActive {
Storage . suspendDatabaseAccess ( using : dependencies )
silentContent . badge = dependencies . storage
. read { db in try Interaction . fetchUnreadCount ( db ) }
. map { NSNumber ( value : $0 ) }
. defaulting ( to : NSNumber ( value : 0 ) )
dependencies . storage . suspendDatabaseAccess ( )
}
let duration : CFTimeInterval = ( CACurrentMediaTime ( ) - startTime )
Log . info ( handledNotification ? " Completed after handling notification in \( . seconds ( duration ) , unit : . ms ) . " : " Completed silently after \( . seconds ( duration ) , unit : . ms ) . " )
Log . flush ( )
self . contentHandler ! ( silentContent )
@ -502,8 +465,10 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
}
private func handleFailure ( for content : UNMutableNotificationContent , error : NotificationError ) {
Log . error ( " Show generic failure message due to error: \( error ) . " )
Storage . suspendDatabaseAccess ( using : dependencies )
dependencies . storage . suspendDatabaseAccess ( )
let duration : CFTimeInterval = ( CACurrentMediaTime ( ) - startTime )
Log . error ( " Show generic failure message after \( . seconds ( duration ) , unit : . ms ) due to error: \( error ) . " )
Log . flush ( )
content . title = " Session "