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.
176 lines
7.8 KiB
Swift
176 lines
7.8 KiB
Swift
import Foundation
|
|
import SessionUtilitiesKit
|
|
import SessionSnodeKit
|
|
import SignalCoreKit
|
|
|
|
public final class AttachmentDownloadJob : NSObject, Job, NSCoding { // NSObject/NSCoding conformance is needed for YapDatabase compatibility
|
|
public let attachmentID: String
|
|
public let tsMessageID: String
|
|
public let threadID: String
|
|
public var delegate: JobDelegate?
|
|
public var id: String?
|
|
public var failureCount: UInt = 0
|
|
public var isDeferred = false
|
|
|
|
public enum Error : LocalizedError {
|
|
case noAttachment
|
|
case invalidURL
|
|
|
|
public var errorDescription: String? {
|
|
switch self {
|
|
case .noAttachment: return "No such attachment."
|
|
case .invalidURL: return "Invalid file URL."
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: Settings
|
|
public class var collection: String { return "AttachmentDownloadJobCollection" }
|
|
public static let maxFailureCount: UInt = 20
|
|
|
|
// MARK: Initialization
|
|
public init(attachmentID: String, tsMessageID: String, threadID: String) {
|
|
self.attachmentID = attachmentID
|
|
self.tsMessageID = tsMessageID
|
|
self.threadID = threadID
|
|
}
|
|
|
|
// MARK: Coding
|
|
public init?(coder: NSCoder) {
|
|
guard let attachmentID = coder.decodeObject(forKey: "attachmentID") as! String?,
|
|
let tsMessageID = coder.decodeObject(forKey: "tsIncomingMessageID") as! String?,
|
|
let threadID = coder.decodeObject(forKey: "threadID") as! String?,
|
|
let id = coder.decodeObject(forKey: "id") as! String? else { return nil }
|
|
self.attachmentID = attachmentID
|
|
self.tsMessageID = tsMessageID
|
|
self.threadID = threadID
|
|
self.id = id
|
|
self.failureCount = coder.decodeObject(forKey: "failureCount") as! UInt? ?? 0
|
|
self.isDeferred = coder.decodeBool(forKey: "isDeferred")
|
|
}
|
|
|
|
public func encode(with coder: NSCoder) {
|
|
coder.encode(attachmentID, forKey: "attachmentID")
|
|
coder.encode(tsMessageID, forKey: "tsIncomingMessageID")
|
|
coder.encode(threadID, forKey: "threadID")
|
|
coder.encode(id, forKey: "id")
|
|
coder.encode(failureCount, forKey: "failureCount")
|
|
coder.encode(isDeferred, forKey: "isDeferred")
|
|
}
|
|
|
|
// MARK: Running
|
|
public func execute() {
|
|
execute(completion: nil)
|
|
}
|
|
|
|
public func execute(completion: (() -> Void)?) {
|
|
if let id = id {
|
|
JobQueue.currentlyExecutingJobs.insert(id)
|
|
}
|
|
guard !isDeferred else { return }
|
|
if TSAttachment.fetch(uniqueId: attachmentID) is TSAttachmentStream {
|
|
// FIXME: It's not clear * how * this happens, but apparently we can get to this point
|
|
// from time to time with an already downloaded attachment.
|
|
completion?()
|
|
return handleSuccess()
|
|
}
|
|
guard let pointer = TSAttachment.fetch(uniqueId: attachmentID) as? TSAttachmentPointer else {
|
|
completion?()
|
|
return handleFailure(error: Error.noAttachment)
|
|
}
|
|
let storage = SNMessagingKitConfiguration.shared.storage
|
|
storage.write(with: { transaction in
|
|
storage.setAttachmentState(to: .downloading, for: pointer, associatedWith: self.tsMessageID, using: transaction)
|
|
}, completion: { })
|
|
let temporaryFilePath = URL(fileURLWithPath: OWSTemporaryDirectoryAccessibleAfterFirstAuth() + UUID().uuidString)
|
|
let handleFailure: (Swift.Error) -> Void = { error in // Intentionally capture self
|
|
OWSFileSystem.deleteFile(temporaryFilePath.absoluteString)
|
|
if let error = error as? Error, case .noAttachment = error {
|
|
storage.write(with: { transaction in
|
|
storage.setAttachmentState(to: .failed, for: pointer, associatedWith: self.tsMessageID, using: transaction)
|
|
}, completion: { })
|
|
self.handlePermanentFailure(error: error)
|
|
} else if let error = error as? OnionRequestAPI.Error, case .httpRequestFailedAtDestination(let statusCode, _, _) = error,
|
|
statusCode == 400 {
|
|
// Otherwise, the attachment will show a state of downloading forever,
|
|
// and the message won't be able to be marked as read.
|
|
storage.write(with: { transaction in
|
|
storage.setAttachmentState(to: .failed, for: pointer, associatedWith: self.tsMessageID, using: transaction)
|
|
}, completion: { })
|
|
// This usually indicates a file that has expired on the server, so there's no need to retry.
|
|
completion?()
|
|
self.handlePermanentFailure(error: error)
|
|
} else {
|
|
completion?()
|
|
self.handleFailure(error: error)
|
|
}
|
|
}
|
|
if let tsMessage = TSMessage.fetch(uniqueId: tsMessageID), let v2OpenGroup = storage.getV2OpenGroup(for: tsMessage.uniqueThreadId) {
|
|
guard let fileAsString = pointer.downloadURL.split(separator: "/").last, let file = UInt64(fileAsString) else {
|
|
return handleFailure(Error.invalidURL)
|
|
}
|
|
OpenGroupAPIV2.download(file, from: v2OpenGroup.room, on: v2OpenGroup.server).done(on: DispatchQueue.global(qos: .userInitiated)) { data in
|
|
self.handleDownloadedAttachment(data: data, temporaryFilePath: temporaryFilePath, pointer: pointer, failureHandler: handleFailure)
|
|
completion?()
|
|
}.catch(on: DispatchQueue.global()) { error in
|
|
handleFailure(error)
|
|
}
|
|
} else {
|
|
guard let fileAsString = pointer.downloadURL.split(separator: "/").last, let file = UInt64(fileAsString) else {
|
|
return handleFailure(Error.invalidURL)
|
|
}
|
|
let useOldServer = pointer.downloadURL.contains(FileServerAPIV2.oldServer)
|
|
FileServerAPIV2.download(file, useOldServer: useOldServer).done(on: DispatchQueue.global(qos: .userInitiated)) { data in
|
|
self.handleDownloadedAttachment(data: data, temporaryFilePath: temporaryFilePath, pointer: pointer, failureHandler: handleFailure)
|
|
completion?()
|
|
}.catch(on: DispatchQueue.global()) { error in
|
|
handleFailure(error)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func handleDownloadedAttachment(data: Data, temporaryFilePath: URL, pointer: TSAttachmentPointer, failureHandler: (Swift.Error) -> Void) {
|
|
let storage = SNMessagingKitConfiguration.shared.storage
|
|
do {
|
|
try data.write(to: temporaryFilePath, options: .atomic)
|
|
} catch {
|
|
return failureHandler(error)
|
|
}
|
|
let plaintext: Data
|
|
if let key = pointer.encryptionKey, let digest = pointer.digest, key.count > 0 && digest.count > 0 {
|
|
do {
|
|
plaintext = try Cryptography.decryptAttachment(data, withKey: key, digest: digest, unpaddedSize: pointer.byteCount)
|
|
} catch {
|
|
return failureHandler(error)
|
|
}
|
|
} else {
|
|
plaintext = data // Open group attachments are unencrypted
|
|
}
|
|
let stream = TSAttachmentStream(pointer: pointer)
|
|
do {
|
|
try stream.write(plaintext)
|
|
} catch {
|
|
return failureHandler(error)
|
|
}
|
|
OWSFileSystem.deleteFile(temporaryFilePath.absoluteString)
|
|
storage.write(with: { transaction in
|
|
storage.persist(stream, associatedWith: self.tsMessageID, using: transaction)
|
|
}, completion: {
|
|
self.handleSuccess()
|
|
})
|
|
}
|
|
|
|
private func handleSuccess() {
|
|
delegate?.handleJobSucceeded(self)
|
|
}
|
|
|
|
private func handlePermanentFailure(error: Swift.Error) {
|
|
delegate?.handleJobFailedPermanently(self, with: error)
|
|
}
|
|
|
|
private func handleFailure(error: Swift.Error) {
|
|
delegate?.handleJobFailed(self, with: error)
|
|
}
|
|
}
|
|
|