// // NCOperationQueue.swift // Nextcloud // // Created by Marino Faggiana on 03/06/2020. // Copyright © 2020 Marino Faggiana. All rights reserved. // // Author Marino Faggiana // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // import UIKit import Queuer import NCCommunication @objc class NCOperationQueue: NSObject { @objc public static let shared: NCOperationQueue = { let instance = NCOperationQueue() return instance }() private var downloadQueue = Queuer(name: "downloadQueue", maxConcurrentOperationCount: 5, qualityOfService: .default) private let deleteQueue = Queuer(name: "deleteQueue", maxConcurrentOperationCount: 1, qualityOfService: .default) private let copyMoveQueue = Queuer(name: "copyMoveQueue", maxConcurrentOperationCount: 1, qualityOfService: .default) private let synchronizationQueue = Queuer(name: "synchronizationQueue", maxConcurrentOperationCount: 1, qualityOfService: .default) private let downloadThumbnailQueue = Queuer(name: "downloadThumbnailQueue", maxConcurrentOperationCount: 10, qualityOfService: .default) private var timerReadFileForMediaQueue: Timer? @objc func cancelAllQueue() { downloadCancelAll() deleteCancelAll() copyMoveCancelAll() synchronizationCancelAll() downloadThumbnailCancelAll() } // Download file func download(metadata: tableMetadata, selector: String) { for operation in downloadQueue.operations as! [NCOperationDownload] { if operation.metadata.ocId == metadata.ocId { return } } downloadQueue.addOperation(NCOperationDownload.init(metadata: metadata, selector: selector)) } @objc func downloadCancelAll() { downloadQueue.cancelAll() } @objc func downloadCount() -> Int { return downloadQueue.operationCount } @objc func downloadExists(metadata: tableMetadata) -> Bool { for operation in downloadQueue.operations as! [NCOperationDownload] { if operation.metadata.ocId == metadata.ocId { return true } } return false } // Delete file @objc func delete(metadata: tableMetadata, onlyLocal: Bool) { for operation in deleteQueue.operations as! [NCOperationDelete] { if operation.metadata.ocId == metadata.ocId { return } } deleteQueue.addOperation(NCOperationDelete.init(metadata: metadata, onlyLocal: onlyLocal)) } @objc func deleteCancelAll() { deleteQueue.cancelAll() } @objc func deleteCount() -> Int { return deleteQueue.operationCount } // Copy Move file @objc func copyMove(metadata: tableMetadata, serverUrl: String, overwrite: Bool, move: Bool) { for operation in copyMoveQueue.operations as! [NCOperationCopyMove] { if operation.metadata.ocId == metadata.ocId { return } } copyMoveQueue.addOperation(NCOperationCopyMove.init(metadata: metadata, serverUrlTo: serverUrl, overwrite: overwrite, move: move)) } @objc func copyMoveCancelAll() { copyMoveQueue.cancelAll() } @objc func copyMoveCount() -> Int { return copyMoveQueue.operationCount } // Synchronization @objc func synchronizationMetadata(_ metadata: tableMetadata, selector: String) { for operation in synchronizationQueue.operations as! [NCOperationSynchronization] { if operation.metadata.ocId == metadata.ocId { return } } synchronizationQueue.addOperation(NCOperationSynchronization.init(metadata: metadata, selector: selector)) } @objc func synchronizationCancelAll() { synchronizationQueue.cancelAll() } // Download Thumbnail @objc func downloadThumbnail(metadata: tableMetadata, urlBase: String, view: Any, indexPath: IndexPath) { if metadata.hasPreview && metadata.status == NCGlobal.shared.metadataStatusNormal && (!CCUtility.fileProviderStoragePreviewIconExists(metadata.ocId, etag: metadata.etag)) { for operation in downloadThumbnailQueue.operations as! [NCOperationDownloadThumbnail] { if operation.metadata.ocId == metadata.ocId { return } } downloadThumbnailQueue.addOperation(NCOperationDownloadThumbnail.init(metadata: metadata, urlBase: urlBase, view: view, indexPath: indexPath)) } } func cancelDownloadThumbnail(metadata: tableMetadata) { for operation in downloadThumbnailQueue.operations as! [NCOperationDownloadThumbnail] { if operation.metadata.ocId == metadata.ocId { operation.cancel() } } } @objc func downloadThumbnailCancelAll() { downloadThumbnailQueue.cancelAll() } } //MARK: - class NCOperationDownload: ConcurrentOperation { var metadata: tableMetadata var selector: String init(metadata: tableMetadata, selector: String) { self.metadata = tableMetadata.init(value: metadata) self.selector = selector } override func start() { if isCancelled { self.finish() } else { NCNetworking.shared.download(metadata: metadata, selector: self.selector) { (_) in self.finish() } } } } //MARK: - class NCOperationDelete: ConcurrentOperation { var metadata: tableMetadata var onlyLocal: Bool init(metadata: tableMetadata, onlyLocal: Bool) { self.metadata = tableMetadata.init(value: metadata) self.onlyLocal = onlyLocal } override func start() { if isCancelled { self.finish() } else { NCNetworking.shared.deleteMetadata(metadata, account: metadata.account, urlBase: metadata.urlBase, onlyLocal: onlyLocal) { (errorCode, errorDescription) in if errorCode != 0 { NCContentPresenter.shared.messageNotification("_error_", description: errorDescription, delay: NCGlobal.shared.dismissAfterSecond, type: NCContentPresenter.messageType.error, errorCode: errorCode) } self.finish() } } } } //MARK: - class NCOperationCopyMove: ConcurrentOperation { var metadata: tableMetadata var serverUrlTo: String var overwrite: Bool var move: Bool init(metadata: tableMetadata, serverUrlTo: String, overwrite: Bool, move: Bool) { self.metadata = tableMetadata.init(value: metadata) self.serverUrlTo = serverUrlTo self.overwrite = overwrite self.move = move } override func start() { if isCancelled { self.finish() } else { if move { NCNetworking.shared.moveMetadata(metadata, serverUrlTo: serverUrlTo, overwrite: overwrite) { (errorCode, errorDescription) in if errorCode != 0 { NCContentPresenter.shared.messageNotification("_error_", description: errorDescription, delay: NCGlobal.shared.dismissAfterSecond, type: NCContentPresenter.messageType.error, errorCode: errorCode) } self.finish() } } else { NCNetworking.shared.copyMetadata(metadata, serverUrlTo: serverUrlTo, overwrite: overwrite) { (errorCode, errorDescription) in if errorCode != 0 { NCContentPresenter.shared.messageNotification("_error_", description: errorDescription, delay: NCGlobal.shared.dismissAfterSecond, type: NCContentPresenter.messageType.error, errorCode: errorCode) } self.finish() } } } } } //MARK: - class NCOperationSynchronization: ConcurrentOperation { var metadata: tableMetadata var selector: String var download: Bool init(metadata: tableMetadata, selector: String) { self.metadata = tableMetadata.init(value: metadata) self.selector = selector if selector == NCGlobal.shared.selectorDownloadFile || selector == NCGlobal.shared.selectorDownloadAllFile { self.download = true } else { self.download = false } } override func start() { if isCancelled { self.finish() } else { if metadata.directory { let serverUrl = metadata.serverUrl + "/" + metadata.fileName let directory = NCManageDatabase.shared.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", metadata.account, serverUrl)) NCCommunication.shared.readFileOrFolder(serverUrlFileName: serverUrl, depth: "0", showHiddenFiles: CCUtility.getShowHiddenFiles()) { (account, files, responseData, errorCode, errorDescription) in if (errorCode == 0) && (directory?.etag != files.first?.etag || self.selector == NCGlobal.shared.selectorDownloadAllFile) { NCCommunication.shared.readFileOrFolder(serverUrlFileName: serverUrl, depth: "1", showHiddenFiles: CCUtility.getShowHiddenFiles()) { (account, files, responseData, errorCode, errorDescription) in if errorCode == 0 { NCManageDatabase.shared.convertNCCommunicationFilesToMetadatas(files, useMetadataFolder: true, account: account) { (metadataFolder, metadatasFolder, metadatas) in let metadatasResult = NCManageDatabase.shared.getMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@ AND status == %d", account, serverUrl, NCGlobal.shared.metadataStatusNormal)) if self.selector == NCGlobal.shared.selectorDownloadAllFile { NCManageDatabase.shared.updateMetadatas(metadatas, metadatasResult: metadatasResult) for metadata in metadatas { if metadata.directory { NCOperationQueue.shared.synchronizationMetadata(metadata, selector: self.selector) } else { if NCManageDatabase.shared.isDownloadMetadata(metadata, download: true) { NCOperationQueue.shared.download(metadata: metadata, selector: self.selector) } } } } else { let metadatasChanged = NCManageDatabase.shared.updateMetadatas(metadatas, metadatasResult: metadatasResult, addExistsInLocal: self.download, addCompareEtagLocal: true, addDirectorySynchronized: true) for metadata in metadatasChanged.metadatasUpdate { if metadata.directory { NCOperationQueue.shared.synchronizationMetadata(metadata, selector: self.selector) } } for metadata in metadatasChanged.metadatasLocalUpdate { NCOperationQueue.shared.download(metadata: metadata, selector: self.selector) } } // Update etag directory NCManageDatabase.shared.addDirectory(encrypted: metadataFolder.e2eEncrypted, favorite: metadataFolder.favorite, ocId: metadataFolder.ocId, fileId: metadataFolder.fileId, etag: metadataFolder.etag, permissions: metadataFolder.permissions, serverUrl: serverUrl, richWorkspace: metadataFolder.richWorkspace, account: metadataFolder.account) } } else if errorCode == NCGlobal.shared.errorResourceNotFound && self.metadata.directory { NCManageDatabase.shared.deleteDirectoryAndSubDirectory(serverUrl: self.metadata.serverUrl, account: self.metadata.account) } self.finish() } } else { let metadatas = NCManageDatabase.shared.getMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", account, serverUrl)) for metadata in metadatas { if metadata.directory { NCOperationQueue.shared.synchronizationMetadata(metadata, selector: self.selector) } else { if NCManageDatabase.shared.isDownloadMetadata(metadata, download: self.download) { NCOperationQueue.shared.download(metadata: metadata, selector: self.selector) } } } self.finish() } } } else { if NCManageDatabase.shared.isDownloadMetadata(metadata, download: self.download) { NCOperationQueue.shared.download(metadata: metadata, selector: self.selector) } self.finish() } } } } //MARK: - class NCOperationDownloadThumbnail: ConcurrentOperation { var metadata: tableMetadata var urlBase: String var view: Any var indexPath: IndexPath init(metadata: tableMetadata, urlBase: String, view: Any, indexPath: IndexPath) { self.metadata = tableMetadata.init(value: metadata) self.urlBase = urlBase self.view = view self.indexPath = indexPath } override func start() { if isCancelled { self.finish() } else { let fileNamePath = CCUtility.returnFileNamePath(fromFileName: metadata.fileName, serverUrl: metadata.serverUrl, urlBase: urlBase, account: metadata.account)! let fileNamePreviewLocalPath = CCUtility.getDirectoryProviderStoragePreviewOcId(metadata.ocId, etag: metadata.etag)! let fileNameIconLocalPath = CCUtility.getDirectoryProviderStorageIconOcId(metadata.ocId, etag: metadata.etag)! NCCommunication.shared.downloadPreview(fileNamePathOrFileId: fileNamePath, fileNamePreviewLocalPath: fileNamePreviewLocalPath , widthPreview: NCGlobal.shared.sizePreview, heightPreview: NCGlobal.shared.sizePreview, fileNameIconLocalPath: fileNameIconLocalPath, sizeIcon: NCGlobal.shared.sizeIcon) { (account, imagePreview, imageIcon, errorCode, errorDescription) in var cell: NCImageCellProtocol? if self.view is UICollectionView { if self.indexPath.section < (self.view as! UICollectionView).numberOfSections && self.indexPath.row < (self.view as! UICollectionView).numberOfItems(inSection: self.indexPath.section) { cell = (self.view as! UICollectionView).cellForItem(at: self.indexPath) as? NCImageCellProtocol } } else { if self.indexPath.section < (self.view as! UITableView).numberOfSections && self.indexPath.row < (self.view as! UITableView).numberOfRows(inSection: self.indexPath.section) { cell = (self.view as! UITableView).cellForRow(at: self.indexPath) as? NCImageCellProtocol } } if (cell != nil) { var previewImage: UIImage! if errorCode == 0 && imageIcon != nil { previewImage = imageIcon } else { if self.metadata.iconName.count > 0 { previewImage = UIImage(named: self.metadata.iconName) } else { previewImage = UIImage(named: "file") } } cell!.filePreviewImageView.backgroundColor = nil UIView.transition(with: cell!.filePreviewImageView, duration: 0.75, options: .transitionCrossDissolve, animations: { cell!.filePreviewImageView.image = previewImage! }, completion: nil) } self.finish() } } } }