NCUtilityFileSystem.swift 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. //
  2. // NCUtilityFileSystem.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 28/05/2020.
  6. // Copyright © 2020 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 PhotosUI
  25. class NCUtilityFileSystem: NSObject {
  26. @objc static let shared: NCUtilityFileSystem = {
  27. let instance = NCUtilityFileSystem()
  28. return instance
  29. }()
  30. let fileManager = FileManager.default
  31. @objc func getFileSize(asset: PHAsset) -> Int64 {
  32. let resources = PHAssetResource.assetResources(for: asset)
  33. if let resource = resources.first {
  34. if resource.responds(to: #selector(NSDictionary.fileSize)) {
  35. let unsignedInt64 = resource.value(forKey: "fileSize") as! CLong
  36. return Int64(bitPattern: UInt64(unsignedInt64))
  37. }
  38. }
  39. return 0
  40. }
  41. @objc func getFileSize(filePath: String) -> Int64 {
  42. do {
  43. let attributes = try fileManager.attributesOfItem(atPath: filePath)
  44. return attributes[FileAttributeKey.size] as? Int64 ?? 0
  45. } catch {
  46. print(error)
  47. }
  48. return 0
  49. }
  50. @objc func getFileModificationDate(filePath: String) -> NSDate? {
  51. do {
  52. let attributes = try fileManager.attributesOfItem(atPath: filePath)
  53. return attributes[FileAttributeKey.modificationDate] as? NSDate
  54. } catch {
  55. print(error)
  56. }
  57. return nil
  58. }
  59. @objc func getFileCreationDate(filePath: String) -> NSDate? {
  60. do {
  61. let attributes = try fileManager.attributesOfItem(atPath: filePath)
  62. return attributes[FileAttributeKey.creationDate] as? NSDate
  63. } catch {
  64. print(error)
  65. }
  66. return nil
  67. }
  68. @objc func writeFile(fileURL: URL, text: String) -> Bool {
  69. do {
  70. try FileManager.default.removeItem(at: fileURL)
  71. } catch {
  72. print(error)
  73. }
  74. do {
  75. try text.write(to: fileURL, atomically: true, encoding: .utf8)
  76. return true
  77. } catch {
  78. print(error)
  79. return false
  80. }
  81. }
  82. @objc func deleteFile(filePath: String) {
  83. do {
  84. try FileManager.default.removeItem(atPath: filePath)
  85. } catch {
  86. print(error)
  87. }
  88. }
  89. @discardableResult
  90. @objc func moveFile(atPath: String, toPath: String) -> Bool {
  91. if atPath == toPath { return true }
  92. do {
  93. try FileManager.default.removeItem(atPath: toPath)
  94. } catch {
  95. print(error)
  96. }
  97. do {
  98. try FileManager.default.copyItem(atPath: atPath, toPath: toPath)
  99. try FileManager.default.removeItem(atPath: atPath)
  100. return true
  101. } catch {
  102. print(error)
  103. return false
  104. }
  105. }
  106. @discardableResult
  107. @objc func copyFile(atPath: String, toPath: String) -> Bool {
  108. if atPath == toPath { return true }
  109. do {
  110. try FileManager.default.removeItem(atPath: toPath)
  111. } catch {
  112. print(error)
  113. }
  114. do {
  115. try FileManager.default.copyItem(atPath: atPath, toPath: toPath)
  116. return true
  117. } catch {
  118. print(error)
  119. return false
  120. }
  121. }
  122. @objc func moveFileInBackground(atPath: String, toPath: String) {
  123. if atPath == toPath { return }
  124. DispatchQueue.global().async {
  125. try? FileManager.default.removeItem(atPath: toPath)
  126. try? FileManager.default.copyItem(atPath: atPath, toPath: toPath)
  127. try? FileManager.default.removeItem(atPath: atPath)
  128. }
  129. }
  130. @objc func linkItem(atPath: String, toPath: String) {
  131. try? FileManager.default.removeItem(atPath: toPath)
  132. try? FileManager.default.linkItem(atPath: atPath, toPath: toPath)
  133. }
  134. // MARK: -
  135. @objc func getWebDAV(account: String) -> String {
  136. // return NCManageDatabase.shared.getCapabilitiesServerString(account: account, elements: NCElementsJSON.shared.capabilitiesWebDavRoot) ?? "remote.php/webdav"
  137. return "remote.php/dav"
  138. }
  139. @objc func getHomeServer(account: String) -> String {
  140. var home = self.getWebDAV(account: account)
  141. if let tableAccount = NCManageDatabase.shared.getAccount(predicate: NSPredicate(format: "account == %@", account)) {
  142. home = tableAccount.urlBase + "/" + self.getWebDAV(account: account) + "/files/" + tableAccount.userId
  143. }
  144. return home
  145. }
  146. @objc func getPath(metadata: tableMetadata) -> String {
  147. return metadata.path.replacingOccurrences(of: "/remote.php/dav/files/"+metadata.user, with: "") + metadata.fileName
  148. }
  149. @objc func deletingLastPathComponent(account: String, serverUrl: String) -> String {
  150. if getHomeServer(account: account) == serverUrl { return serverUrl }
  151. let fileName = (serverUrl as NSString).lastPathComponent
  152. let serverUrl = serverUrl.replacingOccurrences(of: "/"+fileName, with: "", options: String.CompareOptions.backwards, range: nil)
  153. return serverUrl
  154. }
  155. @objc func createFileName(_ fileName: String, serverUrl: String, account: String) -> String {
  156. var resultFileName = fileName
  157. var exitLoop = false
  158. while exitLoop == false {
  159. if NCManageDatabase.shared.getMetadata(predicate: NSPredicate(format: "fileNameView == %@ AND serverUrl == %@ AND account == %@", resultFileName, serverUrl, account)) != nil {
  160. var name = NSString(string: resultFileName).deletingPathExtension
  161. let ext = NSString(string: resultFileName).pathExtension
  162. let characters = Array(name)
  163. if characters.count < 2 {
  164. if ext == "" {
  165. resultFileName = name + " " + "1"
  166. } else {
  167. resultFileName = name + " " + "1" + "." + ext
  168. }
  169. } else {
  170. let space = characters[characters.count-2]
  171. let numChar = characters[characters.count-1]
  172. var num = Int(String(numChar))
  173. if space == " " && num != nil {
  174. name = String(name.dropLast())
  175. num = num! + 1
  176. if ext == "" {
  177. resultFileName = name + "\(num!)"
  178. } else {
  179. resultFileName = name + "\(num!)" + "." + ext
  180. }
  181. } else {
  182. if ext == "" {
  183. resultFileName = name + " " + "1"
  184. } else {
  185. resultFileName = name + " " + "1" + "." + ext
  186. }
  187. }
  188. }
  189. } else {
  190. exitLoop = true
  191. }
  192. }
  193. return resultFileName
  194. }
  195. @objc func getDirectorySize(directory: String) -> Int64 {
  196. let url = URL(fileURLWithPath: directory)
  197. let manager = FileManager.default
  198. var totalSize: Int64 = 0
  199. if let enumerator = manager.enumerator(at: url, includingPropertiesForKeys: [.isRegularFileKey], options: []) {
  200. for case let fileURL as URL in enumerator {
  201. if let attributes = try? manager.attributesOfItem(atPath: fileURL.path) {
  202. if let size = attributes[.size] as? Int64 {
  203. totalSize += size
  204. }
  205. }
  206. }
  207. }
  208. return totalSize
  209. }
  210. func cleanUp(directory: String, days: TimeInterval) {
  211. if days == 0 { return}
  212. let minimumDate = Date().addingTimeInterval(-days*24*60*60)
  213. let url = URL(fileURLWithPath: directory)
  214. var offlineDir: [String] = []
  215. var offlineFiles: [String] = []
  216. if let directories = NCManageDatabase.shared.getTablesDirectory(predicate: NSPredicate(format: "offline == true"), sorted: "serverUrl", ascending: true) {
  217. for directory: tableDirectory in directories {
  218. offlineDir.append(CCUtility.getDirectoryProviderStorageOcId(directory.ocId))
  219. }
  220. }
  221. let files = NCManageDatabase.shared.getTableLocalFiles(predicate: NSPredicate(format: "offline == true"), sorted: "fileName", ascending: true)
  222. for file: tableLocalFile in files {
  223. offlineFiles.append(CCUtility.getDirectoryProviderStorageOcId(file.ocId, fileNameView: file.fileName))
  224. }
  225. func meetsRequirement(date: Date) -> Bool {
  226. return date < minimumDate
  227. }
  228. let manager = FileManager.default
  229. if let enumerator = manager.enumerator(at: url, includingPropertiesForKeys: [.isRegularFileKey], options: []) {
  230. for case let fileURL as URL in enumerator {
  231. if let attributes = try? manager.attributesOfItem(atPath: fileURL.path) {
  232. if let date = CCUtility.getATime(fileURL.path) {
  233. if attributes[.size] as? Double == 0 { continue }
  234. if attributes[.type] as? FileAttributeType == FileAttributeType.typeDirectory { continue }
  235. if fileURL.pathExtension == NCGlobal.shared.extensionPreview { continue }
  236. // check offline
  237. if offlineFiles.contains(fileURL.path) { continue }
  238. let filter = offlineDir.filter({ fileURL.path.hasPrefix($0)})
  239. if filter.count > 0 { continue }
  240. // check date
  241. if meetsRequirement(date: date) {
  242. let folderURL = fileURL.deletingLastPathComponent()
  243. let ocId = folderURL.lastPathComponent
  244. do {
  245. try manager.removeItem(atPath: fileURL.path)
  246. } catch { }
  247. manager.createFile(atPath: fileURL.path, contents: nil, attributes: nil)
  248. NCManageDatabase.shared.deleteLocalFile(predicate: NSPredicate(format: "ocId == %@", ocId))
  249. }
  250. }
  251. }
  252. }
  253. }
  254. }
  255. }