FileProviderEnumerator.swift 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. //
  2. // FileProviderEnumerator.swift
  3. // Files
  4. //
  5. // Created by Marino Faggiana on 26/03/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 FileProvider
  25. import NCCommunication
  26. class FileProviderEnumerator: NSObject, NSFileProviderEnumerator {
  27. var enumeratedItemIdentifier: NSFileProviderItemIdentifier
  28. var serverUrl: String?
  29. init(enumeratedItemIdentifier: NSFileProviderItemIdentifier) {
  30. self.enumeratedItemIdentifier = enumeratedItemIdentifier
  31. // Select ServerUrl
  32. if (enumeratedItemIdentifier == .rootContainer) {
  33. serverUrl = fileProviderData.shared.homeServerUrl
  34. } else {
  35. let metadata = fileProviderUtility.shared.getTableMetadataFromItemIdentifier(enumeratedItemIdentifier)
  36. if metadata != nil {
  37. if let directorySource = NCManageDatabase.shared.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", metadata!.account, metadata!.serverUrl)) {
  38. serverUrl = directorySource.serverUrl + "/" + metadata!.fileName
  39. }
  40. }
  41. }
  42. super.init()
  43. }
  44. func invalidate() {
  45. }
  46. func enumerateItems(for observer: NSFileProviderEnumerationObserver, startingAt page: NSFileProviderPage) {
  47. var items: [NSFileProviderItemProtocol] = []
  48. /*** WorkingSet ***/
  49. if enumeratedItemIdentifier == .workingSet {
  50. var itemIdentifierMetadata: [NSFileProviderItemIdentifier: tableMetadata] = [:]
  51. // ***** Tags *****
  52. let tags = NCManageDatabase.shared.getTags(predicate: NSPredicate(format: "account == %@", fileProviderData.shared.account))
  53. for tag in tags {
  54. guard let metadata = NCManageDatabase.shared.getMetadataFromOcId(tag.ocId) else { continue }
  55. fileProviderUtility.shared.createocIdentifierOnFileSystem(metadata: metadata)
  56. itemIdentifierMetadata[fileProviderUtility.shared.getItemIdentifier(metadata: metadata)] = metadata
  57. }
  58. // ***** Favorite *****
  59. fileProviderData.shared.listFavoriteIdentifierRank = NCManageDatabase.shared.getTableMetadatasDirectoryFavoriteIdentifierRank(account: fileProviderData.shared.account)
  60. for (identifier, _) in fileProviderData.shared.listFavoriteIdentifierRank {
  61. guard let metadata = NCManageDatabase.shared.getMetadataFromOcId(identifier) else { continue }
  62. itemIdentifierMetadata[fileProviderUtility.shared.getItemIdentifier(metadata: metadata)] = metadata
  63. }
  64. // create items
  65. for (_, metadata) in itemIdentifierMetadata {
  66. let parentItemIdentifier = fileProviderUtility.shared.getParentItemIdentifier(metadata: metadata)
  67. if parentItemIdentifier != nil {
  68. let item = FileProviderItem(metadata: metadata, parentItemIdentifier: parentItemIdentifier!)
  69. items.append(item)
  70. }
  71. }
  72. observer.didEnumerate(items)
  73. observer.finishEnumerating(upTo: nil)
  74. } else {
  75. /*** ServerUrl ***/
  76. let paginationEndpoint = NCManageDatabase.shared.getCapabilitiesServerString(account: fileProviderData.shared.account, elements: NCElementsJSON.shared.capabilitiesPaginationEndpoint)
  77. guard let serverUrl = serverUrl else {
  78. observer.finishEnumerating(upTo: nil)
  79. return
  80. }
  81. if (page == NSFileProviderPage.initialPageSortedByDate as NSFileProviderPage || page == NSFileProviderPage.initialPageSortedByName as NSFileProviderPage) {
  82. if paginationEndpoint != nil {
  83. self.getPagination(endpoint: paginationEndpoint!, serverUrl: serverUrl, page: 1, limit: fileProviderData.shared.itemForPage) { (metadatas) in
  84. self.completeObserver(observer, numPage: 1, metadatas: metadatas)
  85. }
  86. } else {
  87. self.readFileOrFolder(serverUrl: serverUrl) { (metadatas) in
  88. self.completeObserver(observer, numPage: 1, metadatas: metadatas)
  89. }
  90. }
  91. } else {
  92. let numPage = Int(String(data: page.rawValue, encoding: .utf8)!)!
  93. if paginationEndpoint != nil {
  94. self.getPagination(endpoint: paginationEndpoint!, serverUrl: serverUrl, page: numPage, limit: fileProviderData.shared.itemForPage) { (metadatas) in
  95. self.completeObserver(observer, numPage: numPage, metadatas: metadatas)
  96. }
  97. } else {
  98. completeObserver(observer, numPage: numPage, metadatas: nil)
  99. }
  100. }
  101. }
  102. }
  103. func enumerateChanges(for observer: NSFileProviderChangeObserver, from anchor: NSFileProviderSyncAnchor) {
  104. var itemsDelete: [NSFileProviderItemIdentifier] = []
  105. var itemsUpdate: [FileProviderItem] = []
  106. // Report the deleted items
  107. //
  108. if self.enumeratedItemIdentifier == .workingSet {
  109. for (itemIdentifier, _) in fileProviderData.shared.fileProviderSignalDeleteWorkingSetItemIdentifier {
  110. itemsDelete.append(itemIdentifier)
  111. }
  112. fileProviderData.shared.fileProviderSignalDeleteWorkingSetItemIdentifier.removeAll()
  113. } else {
  114. for (itemIdentifier, _) in fileProviderData.shared.fileProviderSignalDeleteContainerItemIdentifier {
  115. itemsDelete.append(itemIdentifier)
  116. }
  117. fileProviderData.shared.fileProviderSignalDeleteContainerItemIdentifier.removeAll()
  118. }
  119. // Report the updated items
  120. //
  121. if self.enumeratedItemIdentifier == .workingSet {
  122. for (_, item) in fileProviderData.shared.fileProviderSignalUpdateWorkingSetItem {
  123. itemsUpdate.append(item)
  124. }
  125. fileProviderData.shared.fileProviderSignalUpdateWorkingSetItem.removeAll()
  126. } else {
  127. for (_, item) in fileProviderData.shared.fileProviderSignalUpdateContainerItem {
  128. itemsUpdate.append(item)
  129. }
  130. fileProviderData.shared.fileProviderSignalUpdateContainerItem.removeAll()
  131. }
  132. observer.didDeleteItems(withIdentifiers: itemsDelete)
  133. observer.didUpdate(itemsUpdate)
  134. let data = "\(fileProviderData.shared.currentAnchor)".data(using: .utf8)
  135. observer.finishEnumeratingChanges(upTo: NSFileProviderSyncAnchor(data!), moreComing: false)
  136. }
  137. func currentSyncAnchor(completionHandler: @escaping (NSFileProviderSyncAnchor?) -> Void) {
  138. let data = "\(fileProviderData.shared.currentAnchor)".data(using: .utf8)
  139. completionHandler(NSFileProviderSyncAnchor(data!))
  140. }
  141. // --------------------------------------------------------------------------------------------
  142. // MARK: - User Function + Network
  143. // --------------------------------------------------------------------------------------------
  144. func completeObserver(_ observer: NSFileProviderEnumerationObserver, numPage: Int, metadatas: [tableMetadata]?) {
  145. var numPage = numPage
  146. var items: [NSFileProviderItemProtocol] = []
  147. if (metadatas != nil) {
  148. for metadata in metadatas! {
  149. if metadata.e2eEncrypted || (metadata.session != "" && metadata.session != NCNetworking.shared.sessionIdentifierBackgroundExtension) { continue }
  150. fileProviderUtility.shared.createocIdentifierOnFileSystem(metadata: metadata)
  151. let parentItemIdentifier = fileProviderUtility.shared.getParentItemIdentifier(metadata: metadata)
  152. if parentItemIdentifier != nil {
  153. let item = FileProviderItem(metadata: metadata, parentItemIdentifier: parentItemIdentifier!)
  154. items.append(item)
  155. }
  156. }
  157. observer.didEnumerate(items)
  158. }
  159. if (items.count == fileProviderData.shared.itemForPage) {
  160. numPage += 1
  161. let providerPage = NSFileProviderPage("\(numPage)".data(using: .utf8)!)
  162. observer.finishEnumerating(upTo: providerPage)
  163. } else {
  164. observer.finishEnumerating(upTo: nil)
  165. }
  166. }
  167. func readFileOrFolder(serverUrl: String, completionHandler: @escaping (_ metadatas: [tableMetadata]?) -> Void) {
  168. var directoryEtag: String?
  169. if let tableDirectory = NCManageDatabase.shared.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl)) {
  170. directoryEtag = tableDirectory.etag
  171. }
  172. NCCommunication.shared.readFileOrFolder(serverUrlFileName: serverUrl, depth: "0", showHiddenFiles: CCUtility.getShowHiddenFiles()) { (account, files, responseData, errorCode, errorDescription) in
  173. if directoryEtag != files.first?.etag {
  174. NCCommunication.shared.readFileOrFolder(serverUrlFileName: serverUrl, depth: "1", showHiddenFiles: CCUtility.getShowHiddenFiles()) { (account, files, responseData, errorCode, errorDescription) in
  175. if errorCode == 0 {
  176. DispatchQueue.global().async {
  177. NCManageDatabase.shared.convertNCCommunicationFilesToMetadatas(files, useMetadataFolder: true, account: account) { (metadataFolder, metadatasFolder, metadatas) in
  178. let metadatasResult = NCManageDatabase.shared.getMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@ AND status == %d", account, serverUrl, NCGlobal.shared.metadataStatusNormal))
  179. NCManageDatabase.shared.updateMetadatas(metadatas, metadatasResult: metadatasResult)
  180. for metadata in metadatasFolder {
  181. let serverUrl = metadata.serverUrl + "/" + metadata.fileNameView
  182. NCManageDatabase.shared.addDirectory(encrypted: metadata.e2eEncrypted, favorite: metadata.favorite, ocId: metadata.ocId, fileId: metadata.fileId, etag: nil, permissions: metadata.permissions, serverUrl: serverUrl, richWorkspace: metadata.richWorkspace, account: metadata.account)
  183. }
  184. let metadatas = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl), sorted: "fileName", ascending: true)
  185. completionHandler(metadatas)
  186. }
  187. }
  188. } else {
  189. let metadatas = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl), sorted: "fileName", ascending: true)
  190. completionHandler(metadatas)
  191. }
  192. }
  193. } else {
  194. let metadatas = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl), sorted: "fileName", ascending: true)
  195. completionHandler(metadatas)
  196. }
  197. }
  198. }
  199. func getPagination(endpoint:String, serverUrl: String, page: Int, limit: Int, completionHandler: @escaping (_ metadatas: [tableMetadata]?) -> Void) {
  200. let offset = (page - 1) * limit
  201. var fileNamePath = CCUtility.returnPathfromServerUrl(serverUrl, urlBase: fileProviderData.shared.accountUrlBase, account: fileProviderData.shared.account)!
  202. if fileNamePath == "" {
  203. fileNamePath = "/"
  204. }
  205. var directoryEtag: String?
  206. if let tableDirectory = NCManageDatabase.shared.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl)) {
  207. if page == 1 {
  208. directoryEtag = tableDirectory.etag
  209. }
  210. }
  211. NCCommunication.shared.readFileOrFolder(serverUrlFileName: serverUrl, depth: "0", showHiddenFiles: CCUtility.getShowHiddenFiles()) { (account, files, responseData, errorCode, errorDescription) in
  212. if errorCode == 0 && files.count == 1 && directoryEtag != files.first?.etag {
  213. if page == 1 {
  214. let metadataFolder = NCManageDatabase.shared.convertNCFileToMetadata(files[0], isEncrypted: false, account: account)
  215. NCManageDatabase.shared.addMetadata(metadataFolder)
  216. 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)
  217. }
  218. NCCommunication.shared.iosHelper(fileNamePath: fileNamePath, serverUrl: serverUrl, offset: offset, limit: limit) { (account, files, errorCode, errorDescription) in
  219. if errorCode == 0 {
  220. DispatchQueue.global().async {
  221. NCManageDatabase.shared.convertNCCommunicationFilesToMetadatas(files, useMetadataFolder: false, account: account) { (metadataFolder, metadatasFolder, metadatas) in
  222. let metadatasResult = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@ AND status == %d", fileProviderData.shared.account, serverUrl, NCGlobal.shared.metadataStatusNormal), page: page, limit: fileProviderData.shared.itemForPage, sorted: "fileName", ascending: true)
  223. NCManageDatabase.shared.updateMetadatas(metadatas, metadatasResult: metadatasResult)
  224. for metadata in metadatasFolder {
  225. let serverUrl = metadata.serverUrl + "/" + metadata.fileNameView
  226. NCManageDatabase.shared.addDirectory(encrypted: metadata.e2eEncrypted, favorite: metadata.favorite, ocId: metadata.ocId, fileId: metadata.fileId, etag: nil, permissions: metadata.permissions, serverUrl: serverUrl, richWorkspace: nil, account: metadata.account)
  227. }
  228. let metadatas = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl), page: page, limit: fileProviderData.shared.itemForPage, sorted: "fileName", ascending: true)
  229. completionHandler(metadatas)
  230. }
  231. }
  232. } else {
  233. let metadatas = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl), page: page, limit: fileProviderData.shared.itemForPage, sorted: "fileName", ascending: true)
  234. completionHandler(metadatas)
  235. }
  236. }
  237. } else {
  238. let metadatas = NCManageDatabase.shared.getAdvancedMetadatas(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", fileProviderData.shared.account, serverUrl), sorted: "fileName", ascending: true)
  239. completionHandler(metadatas)
  240. }
  241. }
  242. }
  243. }