session-ios/SessionMessagingKit/Jobs/AttachmentUploadJob.swift

158 lines
6.8 KiB
Swift

import PromiseKit
import SessionUtilitiesKit
public final class AttachmentUploadJob : NSObject, Job, NSCoding { // NSObject/NSCoding conformance is needed for YapDatabase compatibility
public let attachmentID: String
public let threadID: String
public let message: Message
public let messageSendJobID: String
public var delegate: JobDelegate?
public var id: String?
public var failureCount: UInt = 0
public enum Error : LocalizedError {
case noAttachment
case encryptionFailed
public var errorDescription: String? {
switch self {
case .noAttachment: return "No such attachment."
case .encryptionFailed: return "Couldn't encrypt file."
}
}
}
// MARK: Settings
public class var collection: String { return "AttachmentUploadJobCollection" }
public static let maxFailureCount: UInt = 20
// MARK: Initialization
public init(attachmentID: String, threadID: String, message: Message, messageSendJobID: String) {
self.attachmentID = attachmentID
self.threadID = threadID
self.message = message
self.messageSendJobID = messageSendJobID
}
// MARK: Coding
public init?(coder: NSCoder) {
guard let attachmentID = coder.decodeObject(forKey: "attachmentID") as! String?,
let threadID = coder.decodeObject(forKey: "threadID") as! String?,
let message = coder.decodeObject(forKey: "message") as! Message?,
let messageSendJobID = coder.decodeObject(forKey: "messageSendJobID") as! String?,
let id = coder.decodeObject(forKey: "id") as! String? else { return nil }
self.attachmentID = attachmentID
self.threadID = threadID
self.message = message
self.messageSendJobID = messageSendJobID
self.id = id
self.failureCount = coder.decodeObject(forKey: "failureCount") as! UInt? ?? 0
}
public func encode(with coder: NSCoder) {
coder.encode(attachmentID, forKey: "attachmentID")
coder.encode(threadID, forKey: "threadID")
coder.encode(message, forKey: "message")
coder.encode(messageSendJobID, forKey: "messageSendJobID")
coder.encode(id, forKey: "id")
coder.encode(failureCount, forKey: "failureCount")
}
// MARK: Running
public func execute() {
if let id = id {
JobQueue.currentlyExecutingJobs.insert(id)
}
guard let stream = TSAttachment.fetch(uniqueId: attachmentID) as? TSAttachmentStream else {
return handleFailure(error: Error.noAttachment)
}
guard !stream.isUploaded else { return handleSuccess() } // Should never occur
let storage = SNMessagingKitConfiguration.shared.storage
if let v2OpenGroup = storage.getV2OpenGroup(for: threadID) {
AttachmentUploadJob.upload(stream, using: { data in return OpenGroupAPIV2.upload(data, to: v2OpenGroup.room, on: v2OpenGroup.server) }, encrypt: false, onSuccess: handleSuccess, onFailure: handleFailure)
} else {
AttachmentUploadJob.upload(stream, using: FileServerAPIV2.upload, encrypt: true, onSuccess: handleSuccess, onFailure: handleFailure)
}
}
public static func upload(_ stream: TSAttachmentStream, using upload: (Data) -> Promise<UInt64>, encrypt: Bool, onSuccess: (() -> Void)?, onFailure: ((Swift.Error) -> Void)?) {
// Get the attachment
guard var data = try? stream.readDataFromFile() else {
SNLog("Couldn't read attachment from disk.")
onFailure?(Error.noAttachment); return
}
// Encrypt the attachment if needed
if encrypt {
var encryptionKey = NSData()
var digest = NSData()
guard let ciphertext = Cryptography.encryptAttachmentData(data, shouldPad: true, outKey: &encryptionKey, outDigest: &digest) else {
SNLog("Couldn't encrypt attachment.")
onFailure?(Error.encryptionFailed); return
}
stream.encryptionKey = encryptionKey as Data
stream.digest = digest as Data
data = ciphertext
}
// Check the file size
SNLog("File size: \(data.count) bytes.")
if Double(data.count) > Double(FileServerAPIV2.maxFileSize) / FileServerAPIV2.fileSizeORMultiplier {
onFailure?(FileServerAPIV2.Error.maxFileSizeExceeded); return
}
// Send the request
stream.isUploaded = false
stream.save()
upload(data).done(on: DispatchQueue.global(qos: .userInitiated)) { fileID in
let downloadURL = "\(FileServerAPIV2.server)/files/\(fileID)"
stream.serverId = fileID
stream.isUploaded = true
stream.downloadURL = downloadURL
stream.save()
onSuccess?()
}.catch { error in
onFailure?(error)
}
}
private func handleSuccess() {
SNLog("Attachment uploaded successfully.")
delegate?.handleJobSucceeded(self)
SNMessagingKitConfiguration.shared.storage.resumeMessageSendJobIfNeeded(messageSendJobID)
Storage.shared.write(with: { transaction in
var message: TSMessage?
let transaction = transaction as! YapDatabaseReadWriteTransaction
TSDatabaseSecondaryIndexes.enumerateMessages(withTimestamp: self.message.sentTimestamp!, with: { _, key, _ in
message = TSMessage.fetch(uniqueId: key, transaction: transaction)
}, using: transaction)
if let message = message {
MessageInvalidator.invalidate(message, with: transaction)
}
}, completion: { })
}
private func handlePermanentFailure(error: Swift.Error) {
SNLog("Attachment upload failed permanently due to error: \(error).")
delegate?.handleJobFailedPermanently(self, with: error)
failAssociatedMessageSendJob(with: error)
}
private func handleFailure(error: Swift.Error) {
SNLog("Attachment upload failed due to error: \(error).")
delegate?.handleJobFailed(self, with: error)
if failureCount + 1 == AttachmentUploadJob.maxFailureCount {
failAssociatedMessageSendJob(with: error)
}
}
private func failAssociatedMessageSendJob(with error: Swift.Error) {
let storage = SNMessagingKitConfiguration.shared.storage
let messageSendJob = storage.getMessageSendJob(for: messageSendJobID)
storage.write(with: { transaction in // Intentionally capture self
MessageSender.handleFailedMessageSend(self.message, with: error, using: transaction)
if let messageSendJob = messageSendJob {
storage.markJobAsFailed(messageSendJob, using: transaction)
}
}, completion: { })
}
}