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.
168 lines
7.7 KiB
Swift
168 lines
7.7 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() {
|
|
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 if Features.useV2FileServer && storage.getOpenGroup(for: threadID) == nil {
|
|
AttachmentUploadJob.upload(stream, using: FileServerAPIV2.upload, encrypt: true, onSuccess: handleSuccess, onFailure: handleFailure)
|
|
} else { // Legacy
|
|
let openGroup = storage.getOpenGroup(for: threadID)
|
|
let server = openGroup?.server ?? FileServerAPI.server
|
|
// FIXME: A lot of what's currently happening in FileServerAPI should really be happening here
|
|
FileServerAPI.uploadAttachment(stream, with: attachmentID, to: server).done(on: DispatchQueue.global(qos: .userInitiated)) { // Intentionally capture self
|
|
self.handleSuccess()
|
|
}.catch(on: DispatchQueue.global(qos: .userInitiated)) { error in
|
|
if let error = error as? Error, case .noAttachment = error {
|
|
self.handlePermanentFailure(error: error)
|
|
} else if let error = error as? DotNetAPI.Error, !error.isRetryable {
|
|
self.handlePermanentFailure(error: error)
|
|
} else {
|
|
self.handleFailure(error: error)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
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: false, 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(FileServerAPI.maxFileSize) / FileServerAPI.fileSizeORMultiplier {
|
|
onFailure?(FileServerAPI.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 interaction: TSInteraction?
|
|
let transaction = transaction as! YapDatabaseReadWriteTransaction
|
|
TSDatabaseSecondaryIndexes.enumerateMessages(withTimestamp: self.message.sentTimestamp!, with: { _, key, _ in
|
|
interaction = TSInteraction.fetch(uniqueId: key, transaction: transaction)
|
|
}, using: transaction)
|
|
interaction?.touch(with: transaction) // To refresh the associated message cell and hide the loader
|
|
}, 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: { })
|
|
}
|
|
}
|
|
|