mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			95 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			95 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Swift
		
	
import SessionUtilitiesKit
 | 
						|
import PromiseKit
 | 
						|
 | 
						|
public final class MessageReceiveJob : NSObject, Job, NSCoding { // NSObject/NSCoding conformance is needed for YapDatabase compatibility
 | 
						|
    public let data: Data
 | 
						|
    public let openGroupMessageServerID: UInt64?
 | 
						|
    public let openGroupID: String?
 | 
						|
    public let isBackgroundPoll: Bool
 | 
						|
    public var delegate: JobDelegate?
 | 
						|
    public var id: String?
 | 
						|
    public var failureCount: UInt = 0
 | 
						|
 | 
						|
    // MARK: Settings
 | 
						|
    public class var collection: String { return "MessageReceiveJobCollection" }
 | 
						|
    public static let maxFailureCount: UInt = 10
 | 
						|
 | 
						|
    // MARK: Initialization
 | 
						|
    public init(data: Data, openGroupMessageServerID: UInt64? = nil, openGroupID: String? = nil, isBackgroundPoll: Bool) {
 | 
						|
        self.data = data
 | 
						|
        self.openGroupMessageServerID = openGroupMessageServerID
 | 
						|
        self.openGroupID = openGroupID
 | 
						|
        self.isBackgroundPoll = isBackgroundPoll
 | 
						|
        #if DEBUG
 | 
						|
        if openGroupMessageServerID != nil { assert(openGroupID != nil) }
 | 
						|
        if openGroupID != nil { assert(openGroupMessageServerID != nil) }
 | 
						|
        #endif
 | 
						|
    }
 | 
						|
 | 
						|
    // MARK: Coding
 | 
						|
    public init?(coder: NSCoder) {
 | 
						|
        guard let data = coder.decodeObject(forKey: "data") as! Data?,
 | 
						|
            let id = coder.decodeObject(forKey: "id") as! String?,
 | 
						|
            let isBackgroundPoll = coder.decodeObject(forKey: "isBackgroundPoll") as! Bool? else { return nil }
 | 
						|
        self.data = data
 | 
						|
        self.openGroupMessageServerID = coder.decodeObject(forKey: "openGroupMessageServerID") as! UInt64?
 | 
						|
        self.openGroupID = coder.decodeObject(forKey: "openGroupID") as! String?
 | 
						|
        self.isBackgroundPoll = isBackgroundPoll
 | 
						|
        self.id = id
 | 
						|
        self.failureCount = coder.decodeObject(forKey: "failureCount") as! UInt? ?? 0
 | 
						|
    }
 | 
						|
 | 
						|
    public func encode(with coder: NSCoder) {
 | 
						|
        coder.encode(data, forKey: "data")
 | 
						|
        coder.encode(openGroupMessageServerID, forKey: "openGroupMessageServerID")
 | 
						|
        coder.encode(openGroupID, forKey: "openGroupID")
 | 
						|
        coder.encode(isBackgroundPoll, forKey: "isBackgroundPoll")
 | 
						|
        coder.encode(id, forKey: "id")
 | 
						|
        coder.encode(failureCount, forKey: "failureCount")
 | 
						|
    }
 | 
						|
 | 
						|
    // MARK: Running
 | 
						|
    public func execute() {
 | 
						|
        let _: Promise<Void> = execute()
 | 
						|
    }
 | 
						|
    
 | 
						|
    public func execute() -> Promise<Void> {
 | 
						|
        if let id = id { // Can be nil (e.g. when background polling)
 | 
						|
            JobQueue.currentlyExecutingJobs.insert(id)
 | 
						|
        }
 | 
						|
        let (promise, seal) = Promise<Void>.pending()
 | 
						|
        SNMessagingKitConfiguration.shared.storage.write(with: { transaction in // Intentionally capture self
 | 
						|
            do {
 | 
						|
                let isRetry = (self.failureCount != 0)
 | 
						|
                let (message, proto) = try MessageReceiver.parse(self.data, openGroupMessageServerID: self.openGroupMessageServerID, isRetry: isRetry, using: transaction)
 | 
						|
                try MessageReceiver.handle(message, associatedWithProto: proto, openGroupID: self.openGroupID, isBackgroundPoll: self.isBackgroundPoll, using: transaction)
 | 
						|
                self.handleSuccess()
 | 
						|
                seal.fulfill(())
 | 
						|
            } catch {
 | 
						|
                if let error = error as? MessageReceiver.Error, !error.isRetryable {
 | 
						|
                    SNLog("Message receive job permanently failed due to error: \(error).")
 | 
						|
                    self.handlePermanentFailure(error: error)
 | 
						|
                } else {
 | 
						|
                    SNLog("Couldn't receive message due to error: \(error).")
 | 
						|
                    self.handleFailure(error: error)
 | 
						|
                }
 | 
						|
                seal.fulfill(()) // The promise is just used to keep track of when we're done
 | 
						|
            }
 | 
						|
        }, completion: { })
 | 
						|
        return promise
 | 
						|
    }
 | 
						|
 | 
						|
    private func handleSuccess() {
 | 
						|
        delegate?.handleJobSucceeded(self)
 | 
						|
    }
 | 
						|
 | 
						|
    private func handlePermanentFailure(error: Error) {
 | 
						|
        delegate?.handleJobFailedPermanently(self, with: error)
 | 
						|
    }
 | 
						|
 | 
						|
    private func handleFailure(error: Error) {
 | 
						|
        delegate?.handleJobFailed(self, with: error)
 | 
						|
    }
 | 
						|
}
 | 
						|
 |