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.
		
		
		
		
		
			
		
			
				
	
	
		
			165 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			165 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Swift
		
	
import Foundation
 | 
						|
import SessionUtilitiesKit
 | 
						|
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() {
 | 
						|
        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.
 | 
						|
            return handleSuccess()
 | 
						|
        }
 | 
						|
        guard let pointer = TSAttachment.fetch(uniqueId: attachmentID) as? TSAttachmentPointer else {
 | 
						|
            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? DotNetAPI.Error, case .parsingFailed = error {
 | 
						|
                // No need to retry if the response is invalid. Most likely this means we (incorrectly)
 | 
						|
                // got a "Cannot GET ..." error from the file server.
 | 
						|
                storage.write(with: { transaction in
 | 
						|
                    storage.setAttachmentState(to: .failed, for: pointer, associatedWith: self.tsMessageID, using: transaction)
 | 
						|
                }, completion: { })
 | 
						|
                self.handlePermanentFailure(error: error)
 | 
						|
            } else {
 | 
						|
                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)
 | 
						|
            }.catch(on: DispatchQueue.global()) { error in
 | 
						|
                handleFailure(error)
 | 
						|
            }
 | 
						|
        } else if pointer.downloadURL.contains(FileServerAPIV2.server) {
 | 
						|
            guard let fileAsString = pointer.downloadURL.split(separator: "/").last, let file = UInt64(fileAsString) else {
 | 
						|
                return handleFailure(Error.invalidURL)
 | 
						|
            }
 | 
						|
            FileServerAPIV2.download(file).done(on: DispatchQueue.global(qos: .userInitiated)) { data in
 | 
						|
                self.handleDownloadedAttachment(data: data, temporaryFilePath: temporaryFilePath, pointer: pointer, failureHandler: handleFailure)
 | 
						|
            }.catch(on: DispatchQueue.global()) { error in
 | 
						|
                handleFailure(error)
 | 
						|
            }
 | 
						|
        } else { // Legacy
 | 
						|
            FileServerAPI.downloadAttachment(from: pointer.downloadURL).done(on: DispatchQueue.global(qos: .userInitiated)) { data in
 | 
						|
                self.handleDownloadedAttachment(data: data, temporaryFilePath: temporaryFilePath, pointer: pointer, failureHandler: handleFailure)
 | 
						|
            }.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)
 | 
						|
    }
 | 
						|
}
 | 
						|
 |