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, 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: { }) } }