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.
		
		
		
		
		
			
		
			
				
	
	
		
			120 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			120 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Swift
		
	
| 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
 | |
| 
 | |
|         public var errorDescription: String? {
 | |
|             switch self {
 | |
|             case .noAttachment: return "No such attachment."
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     // 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 openGroup = SNMessagingKitConfiguration.shared.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)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     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: { })
 | |
|     }
 | |
| }
 | |
| 
 |