FileProviderData.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  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 FileProvider
  24. class FileProviderData: NSObject {
  25. var fileManager = FileManager()
  26. var account = ""
  27. var accountUser = ""
  28. var accountUserID = ""
  29. var accountPassword = ""
  30. var accountUrl = ""
  31. var homeServerUrl = ""
  32. // Max item for page
  33. let itemForPage = 20
  34. // List of etag for serverUrl
  35. var listServerUrlEtag = [String:String]()
  36. // Anchor
  37. var currentAnchor: UInt64 = 0
  38. // Rank favorite
  39. var listFavoriteIdentifierRank = [String:NSNumber]()
  40. // Queue for trade-safe
  41. let queueTradeSafe = DispatchQueue(label: "com.nextcloud.fileproviderextension.tradesafe", attributes: .concurrent)
  42. // Item for signalEnumerator
  43. var fileProviderSignalDeleteContainerItemIdentifier = [NSFileProviderItemIdentifier:NSFileProviderItemIdentifier]()
  44. var fileProviderSignalUpdateContainerItem = [NSFileProviderItemIdentifier:FileProviderItem]()
  45. var fileProviderSignalDeleteWorkingSetItemIdentifier = [NSFileProviderItemIdentifier:NSFileProviderItemIdentifier]()
  46. var fileProviderSignalUpdateWorkingSetItem = [NSFileProviderItemIdentifier:FileProviderItem]()
  47. // UserDefaults
  48. var ncUserDefaults = UserDefaults(suiteName: NCBrandOptions.sharedInstance.capabilitiesGroups)
  49. // MARK: -
  50. func setupActiveAccount() -> Bool {
  51. if CCUtility.getDisableFilesApp() || NCBrandOptions.sharedInstance.disable_openin_file {
  52. return false
  53. }
  54. guard let activeAccount = NCManageDatabase.sharedInstance.getAccountActive() else {
  55. return false
  56. }
  57. if account == "" {
  58. queueTradeSafe.sync(flags: .barrier) {
  59. account = activeAccount.account
  60. accountUser = activeAccount.user
  61. accountUserID = activeAccount.userID
  62. accountPassword = CCUtility.getPassword(activeAccount.account)
  63. accountUrl = activeAccount.url
  64. homeServerUrl = CCUtility.getHomeServerUrlActiveUrl(activeAccount.url)
  65. }
  66. } else if account != activeAccount.account {
  67. assert(false, "change user")
  68. }
  69. return true
  70. }
  71. // MARK: -
  72. func getAccountFromItemIdentifier(_ itemIdentifier: NSFileProviderItemIdentifier) -> String? {
  73. let fileID = itemIdentifier.rawValue
  74. return NCManageDatabase.sharedInstance.getMetadata(predicate: NSPredicate(format: "fileID == %@", fileID))?.account
  75. }
  76. func getTableMetadataFromItemIdentifier(_ itemIdentifier: NSFileProviderItemIdentifier) -> tableMetadata? {
  77. let fileID = itemIdentifier.rawValue
  78. return NCManageDatabase.sharedInstance.getMetadata(predicate: NSPredicate(format: "fileID == %@", fileID))
  79. }
  80. func getItemIdentifier(metadata: tableMetadata) -> NSFileProviderItemIdentifier {
  81. return NSFileProviderItemIdentifier(metadata.fileID)
  82. }
  83. func createFileIdentifierOnFileSystem(metadata: tableMetadata) {
  84. let itemIdentifier = getItemIdentifier(metadata: metadata)
  85. if metadata.directory {
  86. CCUtility.getDirectoryProviderStorageFileID(itemIdentifier.rawValue)
  87. } else {
  88. CCUtility.getDirectoryProviderStorageFileID(itemIdentifier.rawValue, fileNameView: metadata.fileNameView)
  89. }
  90. }
  91. func getParentItemIdentifier(metadata: tableMetadata) -> NSFileProviderItemIdentifier? {
  92. if let directory = NCManageDatabase.sharedInstance.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", metadata.account, metadata.serverUrl)) {
  93. if directory.serverUrl == homeServerUrl {
  94. return NSFileProviderItemIdentifier(NSFileProviderItemIdentifier.rootContainer.rawValue)
  95. } else {
  96. // get the metadata.FileID of parent Directory
  97. if let metadata = NCManageDatabase.sharedInstance.getMetadata(predicate: NSPredicate(format: "fileID == %@", directory.fileID)) {
  98. let identifier = getItemIdentifier(metadata: metadata)
  99. return identifier
  100. }
  101. }
  102. }
  103. return nil
  104. }
  105. func getTableDirectoryFromParentItemIdentifier(_ parentItemIdentifier: NSFileProviderItemIdentifier) -> tableDirectory? {
  106. var predicate: NSPredicate
  107. if parentItemIdentifier == .rootContainer {
  108. predicate = NSPredicate(format: "account == %@ AND serverUrl == %@", account, homeServerUrl)
  109. } else {
  110. guard let metadata = getTableMetadataFromItemIdentifier(parentItemIdentifier) else {
  111. return nil
  112. }
  113. predicate = NSPredicate(format: "fileID == %@", metadata.fileID)
  114. }
  115. guard let directory = NCManageDatabase.sharedInstance.getTableDirectory(predicate: predicate) else {
  116. return nil
  117. }
  118. return directory
  119. }
  120. // MARK: -
  121. func updateFavoriteForWorkingSet() {
  122. var updateWorkingSet = false
  123. let oldListFavoriteIdentifierRank = listFavoriteIdentifierRank
  124. listFavoriteIdentifierRank = NCManageDatabase.sharedInstance.getTableMetadatasDirectoryFavoriteIdentifierRank(account: account)
  125. // (ADD)
  126. for (identifier, _) in listFavoriteIdentifierRank {
  127. if !oldListFavoriteIdentifierRank.keys.contains(identifier) {
  128. guard let metadata = NCManageDatabase.sharedInstance.getMetadata(predicate: NSPredicate(format: "fileID == %@", identifier)) else {
  129. continue
  130. }
  131. guard let parentItemIdentifier = getParentItemIdentifier(metadata: metadata) else {
  132. continue
  133. }
  134. let item = FileProviderItem(metadata: metadata, parentItemIdentifier: parentItemIdentifier, providerData: self)
  135. queueTradeSafe.sync(flags: .barrier) {
  136. fileProviderSignalUpdateWorkingSetItem[item.itemIdentifier] = item
  137. }
  138. updateWorkingSet = true
  139. }
  140. }
  141. // (REMOVE)
  142. for (identifier, _) in oldListFavoriteIdentifierRank {
  143. if !listFavoriteIdentifierRank.keys.contains(identifier) {
  144. guard let metadata = NCManageDatabase.sharedInstance.getMetadata(predicate: NSPredicate(format: "fileID == %@", identifier)) else {
  145. continue
  146. }
  147. let itemIdentifier = getItemIdentifier(metadata: metadata)
  148. queueTradeSafe.sync(flags: .barrier) {
  149. fileProviderSignalDeleteWorkingSetItemIdentifier[itemIdentifier] = itemIdentifier
  150. }
  151. updateWorkingSet = true
  152. }
  153. }
  154. if updateWorkingSet {
  155. signalEnumerator(for: [.workingSet])
  156. }
  157. }
  158. // MARK: -
  159. // Convinent method to signal the enumeration for containers.
  160. //
  161. func signalEnumerator(for containerItemIdentifiers: [NSFileProviderItemIdentifier]) {
  162. currentAnchor += 1
  163. for containerItemIdentifier in containerItemIdentifiers {
  164. NSFileProviderManager.default.signalEnumerator(for: containerItemIdentifier) { error in
  165. if let error = error {
  166. print("SignalEnumerator for \(containerItemIdentifier) returned error: \(error)")
  167. }
  168. }
  169. }
  170. }
  171. // MARK: -
  172. func copyFile(_ atPath: String, toPath: String) -> Error? {
  173. var errorResult: Error?
  174. if !fileManager.fileExists(atPath: atPath) {
  175. return NSError(domain: NSCocoaErrorDomain, code: NSFileNoSuchFileError, userInfo:[:])
  176. }
  177. do {
  178. try fileManager.removeItem(atPath: toPath)
  179. } catch let error {
  180. print("error: \(error)")
  181. }
  182. do {
  183. try fileManager.copyItem(atPath: atPath, toPath: toPath)
  184. } catch let error {
  185. errorResult = error
  186. }
  187. return errorResult
  188. }
  189. func moveFile(_ atPath: String, toPath: String) -> Error? {
  190. var errorResult: Error?
  191. if atPath == toPath {
  192. return nil
  193. }
  194. if !fileManager.fileExists(atPath: atPath) {
  195. return NSError(domain: NSCocoaErrorDomain, code: NSFileNoSuchFileError, userInfo:[:])
  196. }
  197. do {
  198. try fileManager.removeItem(atPath: toPath)
  199. } catch let error {
  200. print("error: \(error)")
  201. }
  202. do {
  203. try fileManager.moveItem(atPath: atPath, toPath: toPath)
  204. } catch let error {
  205. errorResult = error
  206. }
  207. return errorResult
  208. }
  209. func deleteFile(_ atPath: String) -> Error? {
  210. var errorResult: Error?
  211. do {
  212. try fileManager.removeItem(atPath: atPath)
  213. } catch let error {
  214. errorResult = error
  215. }
  216. return errorResult
  217. }
  218. func fileExists(atPath: String) -> Bool {
  219. return fileManager.fileExists(atPath: atPath)
  220. }
  221. }