FileProviderData.swift 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. //
  2. // FileProviderData.swift
  3. // Files
  4. //
  5. // Created by Marino Faggiana on 27/05/18.
  6. // Copyright © 2018 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. import UIKit
  24. import NextcloudKit
  25. class fileProviderData: NSObject {
  26. static let shared: fileProviderData = {
  27. let instance = fileProviderData()
  28. return instance
  29. }()
  30. var domain: NSFileProviderDomain?
  31. var fileProviderManager: NSFileProviderManager = NSFileProviderManager.default
  32. let utilityFileSystem = NCUtilityFileSystem()
  33. var account = ""
  34. var user = ""
  35. var userId = ""
  36. var accountUrlBase = ""
  37. var homeServerUrl = ""
  38. var listFavoriteIdentifierRank: [String: NSNumber] = [:]
  39. var fileProviderSignalDeleteContainerItemIdentifier: [NSFileProviderItemIdentifier: NSFileProviderItemIdentifier] = [:]
  40. var fileProviderSignalUpdateContainerItem: [NSFileProviderItemIdentifier: FileProviderItem] = [:]
  41. var fileProviderSignalDeleteWorkingSetItemIdentifier: [NSFileProviderItemIdentifier: NSFileProviderItemIdentifier] = [:]
  42. var fileProviderSignalUpdateWorkingSetItem: [NSFileProviderItemIdentifier: FileProviderItem] = [:]
  43. enum FileProviderError: Error {
  44. case downloadError
  45. case uploadError
  46. }
  47. enum TypeSignal: String {
  48. case delete
  49. case update
  50. case workingSet
  51. }
  52. struct UploadMetadata {
  53. var id: String
  54. var metadata: tableMetadata
  55. var task: URLSessionUploadTask?
  56. }
  57. var uploadMetadata: [UploadMetadata] = []
  58. // MARK: -
  59. func setupAccount(domain: NSFileProviderDomain?, providerExtension: NSFileProviderExtension) -> tableAccount? {
  60. self.domain = domain
  61. if let domain, let fileProviderManager = NSFileProviderManager(for: domain) {
  62. self.fileProviderManager = fileProviderManager
  63. }
  64. // LOG
  65. NextcloudKit.shared.nkCommonInstance.pathLog = utilityFileSystem.directoryGroup
  66. let levelLog = NCKeychain().logLevel
  67. NextcloudKit.shared.nkCommonInstance.levelLog = levelLog
  68. let version = NSString(format: NCBrandOptions.shared.textCopyrightNextcloudiOS as NSString, NCUtility().getVersionApp()) as String
  69. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Start File Provider session with level \(levelLog) " + version + " (File Provider Extension)")
  70. // NO DOMAIN -> Set default account
  71. if domain == nil {
  72. guard let activeAccount = NCManageDatabase.shared.getActiveAccount() else { return nil }
  73. account = activeAccount.account
  74. user = activeAccount.user
  75. userId = activeAccount.userId
  76. accountUrlBase = activeAccount.urlBase
  77. homeServerUrl = utilityFileSystem.getHomeServer(urlBase: activeAccount.urlBase, userId: activeAccount.userId)
  78. NCManageDatabase.shared.setCapabilities(account: account)
  79. NextcloudKit.shared.setup(account: activeAccount.account, user: activeAccount.user, userId: activeAccount.userId, password: NCKeychain().getPassword(account: activeAccount.account), urlBase: activeAccount.urlBase, userAgent: userAgent, nextcloudVersion: NCGlobal.shared.capabilityServerVersionMajor, groupIdentifier: NCBrandOptions.shared.capabilitiesGroup, delegate: NCNetworking.shared)
  80. NCNetworking.shared.delegate = providerExtension as? NCNetworkingDelegate
  81. return tableAccount.init(value: activeAccount)
  82. }
  83. // DOMAIN
  84. let accounts = NCManageDatabase.shared.getAllAccount()
  85. if accounts.isEmpty { return nil }
  86. for activeAccount in accounts {
  87. guard let url = NSURL(string: activeAccount.urlBase) else { continue }
  88. guard let host = url.host else { continue }
  89. let accountDomain = activeAccount.userId + " (" + host + ")"
  90. if accountDomain == domain!.identifier.rawValue {
  91. account = activeAccount.account
  92. user = activeAccount.user
  93. userId = activeAccount.userId
  94. accountUrlBase = activeAccount.urlBase
  95. homeServerUrl = utilityFileSystem.getHomeServer(urlBase: activeAccount.urlBase, userId: activeAccount.userId)
  96. NCManageDatabase.shared.setCapabilities(account: account)
  97. NextcloudKit.shared.setup(account: activeAccount.account, user: activeAccount.user, userId: activeAccount.userId, password: NCKeychain().getPassword(account: activeAccount.account), urlBase: activeAccount.urlBase, userAgent: userAgent, nextcloudVersion: NCGlobal.shared.capabilityServerVersionMajor, groupIdentifier: NCBrandOptions.shared.capabilitiesGroup, delegate: NCNetworking.shared)
  98. NCNetworking.shared.delegate = providerExtension as? NCNetworkingDelegate
  99. return tableAccount.init(value: activeAccount)
  100. }
  101. }
  102. return nil
  103. }
  104. // MARK: -
  105. @discardableResult
  106. func signalEnumerator(ocId: String, type: TypeSignal) -> FileProviderItem? {
  107. guard let metadata = NCManageDatabase.shared.getMetadataFromOcId(ocId),
  108. let parentItemIdentifier = fileProviderUtility().getParentItemIdentifier(metadata: metadata) else { return nil }
  109. let item = FileProviderItem(metadata: metadata, parentItemIdentifier: parentItemIdentifier)
  110. if type == .delete {
  111. fileProviderData.shared.fileProviderSignalDeleteContainerItemIdentifier[item.itemIdentifier] = item.itemIdentifier
  112. fileProviderData.shared.fileProviderSignalDeleteWorkingSetItemIdentifier[item.itemIdentifier] = item.itemIdentifier
  113. }
  114. if type == .update {
  115. fileProviderData.shared.fileProviderSignalUpdateContainerItem[item.itemIdentifier] = item
  116. fileProviderData.shared.fileProviderSignalUpdateWorkingSetItem[item.itemIdentifier] = item
  117. }
  118. if type == .workingSet {
  119. fileProviderData.shared.fileProviderSignalUpdateWorkingSetItem[item.itemIdentifier] = item
  120. }
  121. if type == .delete || type == .update {
  122. fileProviderManager.signalEnumerator(for: parentItemIdentifier) { _ in }
  123. }
  124. fileProviderManager.signalEnumerator(for: .workingSet) { _ in }
  125. return item
  126. }
  127. // MARK: -
  128. func appendUploadMetadata(id: String, metadata: tableMetadata, task: URLSessionUploadTask?) {
  129. if let index = uploadMetadata.firstIndex(where: { $0.id == id }) {
  130. uploadMetadata.remove(at: index)
  131. }
  132. uploadMetadata.append(UploadMetadata(id: id, metadata: metadata, task: task))
  133. }
  134. func getUploadMetadata(id: String) -> UploadMetadata? {
  135. return uploadMetadata.filter({ $0.id == id }).first
  136. }
  137. }