session-ios/SessionMessagingKit/Jobs/MessageReceiveJob.swift

90 lines
3.6 KiB
Swift
Raw Normal View History

2020-11-09 00:58:47 +01:00
import SessionUtilitiesKit
2020-12-07 01:21:24 +01:00
import PromiseKit
2020-11-17 06:23:13 +01:00
public final class MessageReceiveJob : NSObject, Job, NSCoding { // NSObject/NSCoding conformance is needed for YapDatabase compatibility
2020-11-24 10:09:23 +01:00
public let data: Data
2020-11-30 01:00:28 +01:00
public let openGroupMessageServerID: UInt64?
public let openGroupID: String?
2020-12-07 01:21:24 +01:00
public let isBackgroundPoll: Bool
2020-11-08 03:12:38 +01:00
public var delegate: JobDelegate?
2020-11-17 06:23:13 +01:00
public var id: String?
2020-11-08 03:12:38 +01:00
public var failureCount: UInt = 0
// MARK: Settings
public class var collection: String { return "MessageReceiveJobCollection" }
2020-11-08 03:54:40 +01:00
public static let maxFailureCount: UInt = 10
// MARK: Initialization
2020-12-07 01:21:24 +01:00
public init(data: Data, openGroupMessageServerID: UInt64? = nil, openGroupID: String? = nil, isBackgroundPoll: Bool) {
self.data = data
2020-11-30 01:00:28 +01:00
self.openGroupMessageServerID = openGroupMessageServerID
self.openGroupID = openGroupID
2020-12-07 01:21:24 +01:00
self.isBackgroundPoll = isBackgroundPoll
2020-11-30 01:00:28 +01:00
#if DEBUG
if openGroupMessageServerID != nil { assert(openGroupID != nil) }
if openGroupID != nil { assert(openGroupMessageServerID != nil) }
#endif
}
// MARK: Coding
public init?(coder: NSCoder) {
2020-11-17 06:23:13 +01:00
guard let data = coder.decodeObject(forKey: "data") as! Data?,
2020-12-07 01:21:24 +01:00
let id = coder.decodeObject(forKey: "id") as! String?,
let isBackgroundPoll = coder.decodeObject(forKey: "isBackgroundPoll") as! Bool? else { return nil }
self.data = data
2020-11-30 01:00:28 +01:00
self.openGroupMessageServerID = coder.decodeObject(forKey: "openGroupMessageServerID") as! UInt64?
self.openGroupID = coder.decodeObject(forKey: "openGroupID") as! String?
2020-12-07 01:21:24 +01:00
self.isBackgroundPoll = isBackgroundPoll
2020-11-24 10:09:23 +01:00
self.id = id
self.failureCount = coder.decodeObject(forKey: "failureCount") as! UInt? ?? 0
}
public func encode(with coder: NSCoder) {
coder.encode(data, forKey: "data")
2020-11-30 01:00:28 +01:00
coder.encode(openGroupMessageServerID, forKey: "openGroupMessageServerID")
coder.encode(openGroupID, forKey: "openGroupID")
2020-12-07 01:21:24 +01:00
coder.encode(isBackgroundPoll, forKey: "isBackgroundPoll")
2020-11-24 10:09:23 +01:00
coder.encode(id, forKey: "id")
coder.encode(failureCount, forKey: "failureCount")
}
// MARK: Running
public func execute() {
2020-12-07 01:21:24 +01:00
let _: Promise<Void> = execute()
}
public func execute() -> Promise<Void> {
let (promise, seal) = Promise<Void>.pending()
2020-12-07 06:00:21 +01:00
SNMessagingKitConfiguration.shared.storage.write(with: { transaction in // Intentionally capture self
2020-11-24 10:09:23 +01:00
do {
2020-11-30 01:00:28 +01:00
let (message, proto) = try MessageReceiver.parse(self.data, openGroupMessageServerID: self.openGroupMessageServerID, using: transaction)
2020-12-07 01:21:24 +01:00
try MessageReceiver.handle(message, associatedWithProto: proto, openGroupID: self.openGroupID, isBackgroundPoll: self.isBackgroundPoll, using: transaction)
2020-11-24 10:09:23 +01:00
self.handleSuccess()
2020-12-07 01:21:24 +01:00
seal.fulfill(())
2020-11-24 10:09:23 +01:00
} catch {
2020-11-30 01:00:28 +01:00
SNLog("Couldn't receive message due to error: \(error).")
2020-11-24 10:09:23 +01:00
if let error = error as? MessageReceiver.Error, !error.isRetryable {
self.handlePermanentFailure(error: error)
} else {
self.handleFailure(error: error)
}
2020-12-07 01:21:24 +01:00
seal.fulfill(()) // The promise is just used to keep track of when we're done
}
2020-11-17 06:23:13 +01:00
}, completion: { })
2020-12-07 01:21:24 +01:00
return promise
}
private func handleSuccess() {
2020-11-08 03:12:38 +01:00
delegate?.handleJobSucceeded(self)
}
2020-11-18 05:36:51 +01:00
private func handlePermanentFailure(error: Error) {
delegate?.handleJobFailedPermanently(self, with: error)
}
private func handleFailure(error: Error) {
2020-11-08 03:12:38 +01:00
delegate?.handleJobFailed(self, with: error)
}
}