@ -12,6 +12,7 @@ import SignalCoreKit
import SessionUtilitiesKit
public final class NotificationServiceExtension : UNNotificationServiceExtension {
private let dependencies : Dependencies = Dependencies ( )
private var didPerformSetup = false
private var contentHandler : ( ( UNNotificationContent ) -> Void ) ?
private var request : UNNotificationRequest ?
@ -28,7 +29,10 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
override public func didReceive ( _ request : UNNotificationRequest , withContentHandler contentHandler : @ escaping ( UNNotificationContent ) -> Void ) {
self . contentHandler = contentHandler
self . request = request
// I t ' s t e c h n i c a l l y p o s s i b l e f o r ' c o m p l e t e S i l e n t l y ' t o b e c a l l e d t w i c e d u e t o t h e N S E t i m e o u t s o
self . hasCompleted . mutate { $0 = false }
// A b o r t i f t h e m a i n a p p i s r u n n i n g
guard ! ( UserDefaults . sharedLokiProject ? [ . isMainAppActive ] ) . defaulting ( to : false ) else {
Log . info ( " didReceive called while main app running. " )
@ -47,180 +51,196 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
Singleton . setup ( appContext : NotificationServiceExtensionContext ( ) )
}
let isCallOngoing : Bool = ( UserDefaults . sharedLokiProject ? [ . isCallOngoing ] )
. defaulting ( to : false )
// P e r f o r m m a i n s e t u p
Storage . resumeDatabaseAccess ( )
DispatchQueue . main . sync { self . setUpIfNecessary ( ) { } }
// H a n d l e t h e p u s h n o t i f i c a t i o n
Singleton . appReadiness . runNowOrWhenAppDidBecomeReady {
let ( maybeData , metadata , result ) = PushNotificationAPI . processNotification (
notificationContent : notificationContent
)
guard
( result = = . success || result = = . legacySuccess ) ,
let data : Data = maybeData
else {
switch result {
// I f w e g o t a n e x p l i c i t f a i l u r e , o r w e g o t a s u c c e s s b u t n o c o n t e n t t h e n s h o w
// t h e f a l l b a c k n o t i f i c a t i o n
case . success , . legacySuccess , . failure , . legacyFailure :
return self . handleFailure ( for : notificationContent , error : . processing ( result ) )
// J u s t l o g i f t h e n o t i f i c a t i o n w a s t o o l o n g ( a ~ 2 k m e s s a g e s h o u l d b e a b l e t o f i t s o
// t h e s e w i l l m o s t c o m m o n l y b e c a l l o r c o n f i g m e s s a g e s )
case . successTooLong :
Log . info ( " Received too long notification for namespace: \( metadata . namespace ) . " )
return self . completeSilenty ( )
case . legacyForceSilent , . failureNoContent : return self . completeSilenty ( )
}
DispatchQueue . main . sync {
self . setUpIfNecessary ( ) { [ weak self ] in
self ? . handleNotification ( notificationContent , isPerformingResetup : false )
}
}
}
private func handleNotification ( _ notificationContent : UNMutableNotificationContent , isPerformingResetup : Bool ) {
let ( maybeData , metadata , result ) = PushNotificationAPI . processNotification (
notificationContent : notificationContent ,
using : dependencies
)
guard metadata . accountId = = getUserHexEncodedPublicKey ( using : dependencies ) else {
guard ! isPerformingResetup else {
Log . error ( " Received notification for an accountId that isn't the current user, resetup failed. " )
return self . completeSilenty ( )
}
// H A C K : I t i s i m p o r t a n t t o u s e w r i t e s y n c h r o n o u s l y h e r e t o a v o i d a r a c e c o n d i t i o n
// w h e r e t h e c o m p l e t e S i l e n t y ( ) i s c a l l e d b e f o r e t h e l o c a l n o t i f i c a t i o n r e q u e s t
// i s a d d e d t o n o t i f i c a t i o n c e n t e r
Storage . shared . write { [ weak self ] db in
do {
guard let processedMessage : ProcessedMessage = try Message . processRawReceivedMessageAsNotification ( db , data : data , metadata : metadata ) else {
throw NotificationError . messageProcessing
}
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
else {
switch result {
// I f w e g o t a n e x p l i c i t f a i l u r e , o r w e g o t a s u c c e s s b u t n o c o n t e n t t h e n s h o w
// t h e f a l l b a c k n o t i f i c a t i o n
case . success , . legacySuccess , . failure , . legacyFailure :
return self . handleFailure ( for : notificationContent , error : . processing ( result ) )
// J u s t l o g i f t h e n o t i f i c a t i o n w a s t o o l o n g ( a ~ 2 k m e s s a g e s h o u l d b e a b l e t o f i t s o
// t h e s e w i l l m o s t c o m m o n l y b e c a l l o r c o n f i g m e s s a g e s )
case . successTooLong :
Log . info ( " Received too long notification for namespace: \( metadata . namespace ) , dataLength: \( metadata . dataLength ) . " )
return self . completeSilenty ( )
switch processedMessage {
// / C u s t o m h a n d l e c o n f i g m e s s a g e s ( a s t h e y d o n ' t g e t h a n d l e d b y t h e n o r m a l ` M e s s a g e R e c e i v e r . h a n d l e ` c a l l
case . config ( let publicKey , let namespace , let serverHash , let serverTimestampMs , let data ) :
try LibSession . handleConfigMessages (
db ,
messages : [
ConfigMessageReceiveJob . Details . MessageInfo (
namespace : namespace ,
serverHash : serverHash ,
serverTimestampMs : serverTimestampMs ,
data : data
case . legacyForceSilent , . failureNoContent : return self . completeSilenty ( )
}
}
let isCallOngoing : Bool = ( UserDefaults . sharedLokiProject ? [ . isCallOngoing ] )
. defaulting ( to : false )
// H A C K : I t i s i m p o r t a n t t o u s e w r i t e s y n c h r o n o u s l y h e r e t o a v o i d a r a c e c o n d i t i o n
// w h e r e t h e c o m p l e t e S i l e n t y ( ) i s c a l l e d b e f o r e t h e l o c a l n o t i f i c a t i o n r e q u e s t
// i s a d d e d t o n o t i f i c a t i o n c e n t e r
dependencies . storage . write { [ weak self , dependencies ] db in
do {
guard let processedMessage : ProcessedMessage = try Message . processRawReceivedMessageAsNotification ( db , data : data , metadata : metadata , using : dependencies ) else {
throw NotificationError . messageProcessing
}
switch processedMessage {
// / C u s t o m h a n d l e c o n f i g m e s s a g e s ( a s t h e y d o n ' t g e t h a n d l e d b y t h e n o r m a l ` M e s s a g e R e c e i v e r . h a n d l e ` c a l l
case . config ( let publicKey , let namespace , let serverHash , let serverTimestampMs , let data ) :
try LibSession . handleConfigMessages (
db ,
messages : [
ConfigMessageReceiveJob . Details . MessageInfo (
namespace : namespace ,
serverHash : serverHash ,
serverTimestampMs : serverTimestampMs ,
data : data
)
] ,
publicKey : publicKey
)
// / D u e t o t h e w a y t h e ` C a l l M e s s a g e ` w o r k s w e n e e d t o c u s t o m h a n d l e i t ' s b e h a v i o u r w i t h i n t h e n o t i f i c a t i o n
// / e x t e n s i o n , f o r a l l o t h e r m e s s a g e t y p e s w e w a n t t o j u s t u s e t h e s t a n d a r d ` M e s s a g e R e c e i v e r . h a n d l e ` c a l l
case . standard ( let threadId , let threadVariant , _ , let messageInfo ) where messageInfo . message is CallMessage :
guard let callMessage = messageInfo . message as ? CallMessage else {
throw NotificationError . ignorableMessage
}
// T h r o w i f t h e m e s s a g e i s o u t d a t e d a n d s h o u l d n ' t b e p r o c e s s e d
try MessageReceiver . throwIfMessageOutdated (
db ,
message : messageInfo . message ,
threadId : threadId ,
threadVariant : threadVariant ,
using : dependencies
)
try MessageReceiver . handleCallMessage (
db ,
threadId : threadId ,
threadVariant : threadVariant ,
message : callMessage
)
guard case . preOffer = callMessage . kind else {
throw NotificationError . ignorableMessage
}
switch ( db [ . areCallsEnabled ] , isCallOngoing ) {
case ( false , _ ) :
if
let sender : String = callMessage . sender ,
let interaction : Interaction = try MessageReceiver . insertCallInfoMessage (
db ,
for : callMessage ,
state : . permissionDenied
)
] ,
publicKey : publicKey
)
// / D u e t o t h e w a y t h e ` C a l l M e s s a g e ` w o r k s w e n e e d t o c u s t o m h a n d l e i t ' s b e h a v i o u r w i t h i n t h e n o t i f i c a t i o n
// / e x t e n s i o n , f o r a l l o t h e r m e s s a g e t y p e s w e w a n t t o j u s t u s e t h e s t a n d a r d ` M e s s a g e R e c e i v e r . h a n d l e ` c a l l
case . standard ( let threadId , let threadVariant , _ , let messageInfo ) where messageInfo . message is CallMessage :
guard let callMessage = messageInfo . message as ? CallMessage else {
throw NotificationError . ignorableMessage
}
// T h r o w i f t h e m e s s a g e i s o u t d a t e d a n d s h o u l d n ' t b e p r o c e s s e d
try MessageReceiver . throwIfMessageOutdated (
db ,
message : messageInfo . message ,
threadId : threadId ,
threadVariant : threadVariant
)
try MessageReceiver . handleCallMessage (
db ,
threadId : threadId ,
threadVariant : threadVariant ,
message : callMessage
)
guard case . preOffer = callMessage . kind else {
throw NotificationError . ignorableMessage
}
switch ( db [ . areCallsEnabled ] , isCallOngoing ) {
case ( false , _ ) :
if
let sender : String = callMessage . sender ,
let interaction : Interaction = try MessageReceiver . insertCallInfoMessage (
{
let thread : SessionThread = try SessionThread
. fetchOrCreate (
db ,
for : callMessage ,
state : . permissionDenied
id : sender ,
variant : . contact ,
shouldBeVisible : nil
)
{
let thread : SessionThread = try SessionThread
. fetchOrCreate (
// N o t i f y t h e u s e r i f t h e c a l l m e s s a g e w a s n ' t a l r e a d y r e a d
if ! interaction . wasRead {
SessionEnvironment . shared ? . notificationsManager . wrappedValue ?
. notifyUser (
db ,
id : sender ,
variant : . contact ,
shouldBeVisible : nil
forIncomingCall : interaction ,
in : thread ,
applicationState : . background
)
// N o t i f y t h e u s e r i f t h e c a l l m e s s a g e w a s n ' t a l r e a d y r e a d
if ! interaction . wasRead {
SessionEnvironment . shared ? . notificationsManager . wrappedValue ?
. notifyUser (
db ,
forIncomingCall : interaction ,
in : thread ,
applicationState : . background
)
}
}
case ( true , true ) :
try MessageReceiver . handleIncomingCallOfferInBusyState (
db ,
message : callMessage
)
case ( true , false ) :
try MessageReceiver . insertCallInfoMessage ( db , for : callMessage )
return self ? . handleSuccessForIncomingCall ( db , for : callMessage )
}
// P e r f o r m a n y r e q u i r e d p o s t - h a n d l i n g l o g i c
try MessageReceiver . postHandleMessage (
db ,
threadId : threadId ,
threadVariant : threadVariant ,
message : messageInfo . message
)
case . standard ( let threadId , let threadVariant , let proto , let messageInfo ) :
try MessageReceiver . handle (
db ,
threadId : threadId ,
threadVariant : threadVariant ,
message : messageInfo . message ,
serverExpirationTimestamp : messageInfo . serverExpirationTimestamp ,
associatedWithProto : proto
)
}
db . afterNextTransaction (
onCommit : { _ in self ? . completeSilenty ( ) } ,
onRollback : { _ in self ? . completeSilenty ( ) }
)
}
catch {
// 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 . invalidGroupPublicKey , MessageReceiverError . noGroupKeyPair ,
MessageReceiverError . outdatedMessage , NotificationError . ignorableMessage :
self ? . completeSilenty ( )
case NotificationError . messageProcessing :
self ? . handleFailure ( for : notificationContent , error : . messageProcessing )
}
case let msgError as MessageReceiverError :
self ? . handleFailure ( for : notificationContent , error : . messageHandling ( msgError ) )
case ( true , true ) :
try MessageReceiver . handleIncomingCallOfferInBusyState (
db ,
message : callMessage
)
default : self ? . handleFailure ( for : notificationContent , error : . other ( error ) )
case ( true , false ) :
try MessageReceiver . insertCallInfoMessage ( db , for : callMessage )
return self ? . handleSuccessForIncomingCall ( db , for : callMessage )
}
// P e r f o r m a n y r e q u i r e d p o s t - h a n d l i n g l o g i c
try MessageReceiver . postHandleMessage (
db ,
threadId : threadId ,
threadVariant : threadVariant ,
message : messageInfo . message
)
case . standard ( let threadId , let threadVariant , let proto , let messageInfo ) :
try MessageReceiver . handle (
db ,
threadId : threadId ,
threadVariant : threadVariant ,
message : messageInfo . message ,
serverExpirationTimestamp : messageInfo . serverExpirationTimestamp ,
associatedWithProto : proto ,
using : dependencies
)
}
db . afterNextTransaction (
onCommit : { _ in self ? . completeSilenty ( ) } ,
onRollback : { _ in self ? . completeSilenty ( ) }
)
}
catch {
// 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 . invalidGroupPublicKey , MessageReceiverError . noGroupKeyPair ,
MessageReceiverError . outdatedMessage , NotificationError . ignorableMessage :
self ? . completeSilenty ( )
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 ) )
}
db . afterNextTransactionNested (
onCommit : { _ in handleError ( ) } ,
onRollback : { _ in handleError ( ) }
)
throw error
}
db . afterNextTransaction (
onCommit : { _ in handleError ( ) } ,
onRollback : { _ in handleError ( ) }
)
throw error
}
}
}
@ -233,7 +253,7 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
// 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 }
guard ! didPerformSetup else { return completion ( ) }
Log . info ( " Performing setup. " )
didPerformSetup = true
@ -257,7 +277,6 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
// S e t u p L i b S e s s i o n
LibSession . addLogger ( )
LibSession . createNetworkIfNeeded ( )
} ,
migrationsCompletion : { [ weak self ] result , needsConfigSync in
switch result {
@ -278,45 +297,64 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
}
DispatchQueue . main . async {
self ? . versionMigrationsDidComplete ( needsConfigSync : needsConfigSync )
self ? . versionMigrationsDidComplete ( needsConfigSync : needsConfigSync , completion : completion )
}
}
completion ( )
}
)
}
private func versionMigrationsDidComplete ( needsConfigSync : Bool ) {
private func versionMigrationsDidComplete ( needsConfigSync : Bool , completion : @ escaping ( ) -> Void ) {
AssertIsOnMainThread ( )
// 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 ) )
}
}
checkIsAppReady ( migrationsCompleted : true )
}
private func checkIsAppReady ( migrationsCompleted : Bool ) {
AssertIsOnMainThread ( )
// O n l y m a r k t h e a p p a s r e a d y o n c e .
guard ! Singleton . appReadiness . isAppReady else { return }
// 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 && migrationsCompleted else {
guard Storage . shared . isValid else {
Log . error ( " Storage invalid. " )
self . completeSilenty ( )
return
return self . completeSilenty ( )
}
// 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 ( )
dependencies . caches . mutate ( cache : . general ) { $0 . clearCachedUserPublicKey ( ) }
self . setUpIfNecessary ( ) { [ weak self ] 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
)
}
self ? . handleNotification ( notificationContent , isPerformingResetup : true )
}
SignalUtilitiesKit . Configuration . performMainSetup ( )
// 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 ( )
}
// MARK: H a n d l e c o m p l e t i o n
@ -434,5 +472,6 @@ public final class NotificationServiceExtension: UNNotificationServiceExtension
let userInfo : [ String : Any ] = [ NotificationServiceExtension . isFromRemoteKey : true ]
content . userInfo = userInfo
contentHandler ! ( content )
hasCompleted . mutate { $0 = true }
}
}