NCNetworking.swift 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. //
  2. // NCNetworking.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 23/10/19.
  6. // Copyright © 2019 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 OpenSSL
  25. import NextcloudKit
  26. import Alamofire
  27. import Queuer
  28. @objc public protocol NCNetworkingDelegate {
  29. @objc optional func downloadProgress(_ progress: Float, totalBytes: Int64, totalBytesExpected: Int64, fileName: String, serverUrl: String, session: URLSession, task: URLSessionTask)
  30. @objc optional func uploadProgress(_ progress: Float, totalBytes: Int64, totalBytesExpected: Int64, fileName: String, serverUrl: String, session: URLSession, task: URLSessionTask)
  31. @objc optional func downloadComplete(fileName: String, serverUrl: String, etag: String?, date: NSDate?, dateLastModified: NSDate?, length: Int64, fileNameLocalPath: String?, task: URLSessionTask, error: NKError)
  32. @objc optional func uploadComplete(fileName: String, serverUrl: String, ocId: String?, etag: String?, date: NSDate?, size: Int64, fileNameLocalPath: String?, task: URLSessionTask, error: NKError)
  33. }
  34. #if EXTENSION_FILE_PROVIDER_EXTENSION || EXTENSION_WIDGET
  35. @objc protocol uploadE2EEDelegate: AnyObject { }
  36. #endif
  37. @objcMembers
  38. class NCNetworking: NSObject, NKCommonDelegate {
  39. public static let shared: NCNetworking = {
  40. let instance = NCNetworking()
  41. return instance
  42. }()
  43. public struct TransferInForegorund {
  44. var ocId: String
  45. var progress: Float
  46. }
  47. struct FileNameServerUrl: Hashable {
  48. var fileName: String
  49. var serverUrl: String
  50. }
  51. weak var delegate: NCNetworkingDelegate?
  52. let utilityFileSystem = NCUtilityFileSystem()
  53. let utility = NCUtility()
  54. var lastReachability: Bool = true
  55. var networkReachability: NKCommon.TypeReachability?
  56. let downloadRequest = ThreadSafeDictionary<String, DownloadRequest>()
  57. let uploadRequest = ThreadSafeDictionary<String, UploadRequest>()
  58. let uploadMetadataInBackground = ThreadSafeDictionary<FileNameServerUrl, tableMetadata>()
  59. let downloadMetadataInBackground = ThreadSafeDictionary<FileNameServerUrl, tableMetadata>()
  60. var transferInForegorund: TransferInForegorund?
  61. lazy var nkBackground: NKBackground = {
  62. let nckb = NKBackground(nkCommonInstance: NextcloudKit.shared.nkCommonInstance)
  63. return nckb
  64. }()
  65. public let sessionMaximumConnectionsPerHost = 5
  66. public let sessionDownloadBackground: String = "com.nextcloud.session.download.background"
  67. public let sessionUploadBackground: String = "com.nextcloud.session.upload.background"
  68. public let sessionUploadBackgroundWWan: String = "com.nextcloud.session.upload.backgroundWWan"
  69. public let sessionUploadBackgroundExtension: String = "com.nextcloud.session.upload.extension"
  70. public lazy var sessionManagerDownloadBackground: URLSession = {
  71. let configuration = URLSessionConfiguration.background(withIdentifier: sessionDownloadBackground)
  72. configuration.allowsCellularAccess = true
  73. configuration.sessionSendsLaunchEvents = true
  74. configuration.isDiscretionary = false
  75. configuration.httpMaximumConnectionsPerHost = sessionMaximumConnectionsPerHost
  76. configuration.requestCachePolicy = NSURLRequest.CachePolicy.reloadIgnoringLocalCacheData
  77. let session = URLSession(configuration: configuration, delegate: nkBackground, delegateQueue: OperationQueue.main)
  78. return session
  79. }()
  80. public lazy var sessionManagerUploadBackground: URLSession = {
  81. let configuration = URLSessionConfiguration.background(withIdentifier: sessionUploadBackground)
  82. configuration.allowsCellularAccess = true
  83. configuration.sessionSendsLaunchEvents = true
  84. configuration.isDiscretionary = false
  85. configuration.httpMaximumConnectionsPerHost = sessionMaximumConnectionsPerHost
  86. configuration.requestCachePolicy = NSURLRequest.CachePolicy.reloadIgnoringLocalCacheData
  87. let session = URLSession(configuration: configuration, delegate: nkBackground, delegateQueue: OperationQueue.main)
  88. return session
  89. }()
  90. public lazy var sessionManagerUploadBackgroundWWan: URLSession = {
  91. let configuration = URLSessionConfiguration.background(withIdentifier: sessionUploadBackgroundWWan)
  92. configuration.allowsCellularAccess = false
  93. configuration.sessionSendsLaunchEvents = true
  94. configuration.isDiscretionary = false
  95. configuration.httpMaximumConnectionsPerHost = sessionMaximumConnectionsPerHost
  96. configuration.requestCachePolicy = NSURLRequest.CachePolicy.reloadIgnoringLocalCacheData
  97. let session = URLSession(configuration: configuration, delegate: nkBackground, delegateQueue: OperationQueue.main)
  98. return session
  99. }()
  100. #if EXTENSION
  101. public lazy var sessionManagerUploadBackgroundExtension: URLSession = {
  102. let configuration = URLSessionConfiguration.background(withIdentifier: sessionUploadBackgroundExtension)
  103. configuration.allowsCellularAccess = true
  104. configuration.sessionSendsLaunchEvents = true
  105. configuration.isDiscretionary = false
  106. configuration.httpMaximumConnectionsPerHost = sessionMaximumConnectionsPerHost
  107. configuration.requestCachePolicy = NSURLRequest.CachePolicy.reloadIgnoringLocalCacheData
  108. configuration.sharedContainerIdentifier = NCBrandOptions.shared.capabilitiesGroups
  109. let session = URLSession(configuration: configuration, delegate: nkBackground, delegateQueue: OperationQueue.main)
  110. return session
  111. }()
  112. #endif
  113. // REQUESTS
  114. var requestsUnifiedSearch: [DataRequest] = []
  115. // OPERATIONQUEUE
  116. let downloadThumbnailQueue = Queuer(name: "downloadThumbnailQueue", maxConcurrentOperationCount: 10, qualityOfService: .default)
  117. let downloadThumbnailActivityQueue = Queuer(name: "downloadThumbnailActivityQueue", maxConcurrentOperationCount: 10, qualityOfService: .default)
  118. let unifiedSearchQueue = Queuer(name: "unifiedSearchQueue", maxConcurrentOperationCount: 1, qualityOfService: .default)
  119. let saveLivePhotoQueue = Queuer(name: "saveLivePhotoQueue", maxConcurrentOperationCount: 1, qualityOfService: .default)
  120. let downloadQueue = Queuer(name: "downloadQueue", maxConcurrentOperationCount: NCBrandOptions.shared.maxConcurrentOperationDownload, qualityOfService: .default)
  121. let downloadAvatarQueue = Queuer(name: "downloadAvatarQueue", maxConcurrentOperationCount: 10, qualityOfService: .default)
  122. let convertLivePhotoQueue = Queuer(name: "convertLivePhotoQueue", maxConcurrentOperationCount: 10, qualityOfService: .default)
  123. // MARK: - init
  124. override init() {
  125. super.init()
  126. #if EXTENSION
  127. print("Start Background Upload Extension: ", sessionUploadBackgroundExtension)
  128. #else
  129. print("Start Background Download: ", sessionManagerDownloadBackground)
  130. print("Start Background Upload: ", sessionManagerUploadBackground)
  131. print("Start Background Upload WWan: ", sessionManagerUploadBackgroundWWan)
  132. #endif
  133. }
  134. // MARK: - Communication Delegate
  135. func networkReachabilityObserver(_ typeReachability: NKCommon.TypeReachability) {
  136. if typeReachability == NKCommon.TypeReachability.reachableCellular || typeReachability == NKCommon.TypeReachability.reachableEthernetOrWiFi {
  137. if !lastReachability {
  138. #if !EXTENSION
  139. NCService().startRequestServicesServer()
  140. #endif
  141. }
  142. lastReachability = true
  143. } else {
  144. if lastReachability {
  145. let error = NKError(errorCode: NCGlobal.shared.errorNetworkNotAvailable, errorDescription: "")
  146. NCContentPresenter().messageNotification("_network_not_available_", error: error, delay: NCGlobal.shared.dismissAfterSecond, type: NCContentPresenter.messageType.info)
  147. }
  148. lastReachability = false
  149. }
  150. networkReachability = typeReachability
  151. }
  152. func authenticationChallenge(_ session: URLSession,
  153. didReceive challenge: URLAuthenticationChallenge,
  154. completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) {
  155. DispatchQueue.global().async {
  156. self.checkTrustedChallenge(session, didReceive: challenge, completionHandler: completionHandler)
  157. }
  158. }
  159. func urlSessionDidFinishEvents(forBackgroundURLSession session: URLSession) {
  160. #if !EXTENSION
  161. if let appDelegate = UIApplication.shared.delegate as? AppDelegate, let completionHandler = appDelegate.backgroundSessionCompletionHandler {
  162. NextcloudKit.shared.nkCommonInstance.writeLog("[INFO] Called urlSessionDidFinishEvents for Background URLSession")
  163. appDelegate.backgroundSessionCompletionHandler = nil
  164. completionHandler()
  165. }
  166. #endif
  167. }
  168. // MARK: -
  169. func cancelAllQueue() {
  170. downloadQueue.cancelAll()
  171. downloadThumbnailQueue.cancelAll()
  172. downloadThumbnailActivityQueue.cancelAll()
  173. downloadAvatarQueue.cancelAll()
  174. unifiedSearchQueue.cancelAll()
  175. saveLivePhotoQueue.cancelAll()
  176. convertLivePhotoQueue.cancelAll()
  177. }
  178. func cancelAllTask() {
  179. cancelAllQueue()
  180. cancelDataTask()
  181. cancelDownloadTasks()
  182. cancelUploadTasks()
  183. cancelDownloadBackgroundTask()
  184. cancelUploadBackgroundTask()
  185. }
  186. // MARK: - Pinning check
  187. public func checkTrustedChallenge(_ session: URLSession,
  188. didReceive challenge: URLAuthenticationChallenge,
  189. completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) {
  190. let protectionSpace: URLProtectionSpace = challenge.protectionSpace
  191. let directoryCertificate = utilityFileSystem.directoryCertificates
  192. let host = challenge.protectionSpace.host
  193. let certificateSavedPath = directoryCertificate + "/" + host + ".der"
  194. var isTrusted: Bool
  195. if let serverTrust: SecTrust = protectionSpace.serverTrust, let certificate = SecTrustGetCertificateAtIndex(serverTrust, 0) {
  196. // extarct certificate txt
  197. saveX509Certificate(certificate, host: host, directoryCertificate: directoryCertificate)
  198. let isServerTrusted = SecTrustEvaluateWithError(serverTrust, nil)
  199. let certificateCopyData = SecCertificateCopyData(certificate)
  200. let data = CFDataGetBytePtr(certificateCopyData)
  201. let size = CFDataGetLength(certificateCopyData)
  202. let certificateData = NSData(bytes: data, length: size)
  203. certificateData.write(toFile: directoryCertificate + "/" + host + ".tmp", atomically: true)
  204. if isServerTrusted {
  205. isTrusted = true
  206. } else if let certificateDataSaved = NSData(contentsOfFile: certificateSavedPath), certificateData.isEqual(to: certificateDataSaved as Data) {
  207. isTrusted = true
  208. } else {
  209. isTrusted = false
  210. }
  211. } else {
  212. isTrusted = false
  213. }
  214. if isTrusted {
  215. completionHandler(URLSession.AuthChallengeDisposition.useCredential, URLCredential(trust: challenge.protectionSpace.serverTrust!))
  216. } else {
  217. #if !EXTENSION
  218. DispatchQueue.main.async { (UIApplication.shared.delegate as? AppDelegate)?.trustCertificateError(host: host) }
  219. #endif
  220. completionHandler(URLSession.AuthChallengeDisposition.performDefaultHandling, nil)
  221. }
  222. }
  223. func writeCertificate(host: String) {
  224. let directoryCertificate = utilityFileSystem.directoryCertificates
  225. let certificateAtPath = directoryCertificate + "/" + host + ".tmp"
  226. let certificateToPath = directoryCertificate + "/" + host + ".der"
  227. if !utilityFileSystem.copyFile(atPath: certificateAtPath, toPath: certificateToPath) {
  228. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] Write certificare error")
  229. }
  230. }
  231. func saveX509Certificate(_ certificate: SecCertificate, host: String, directoryCertificate: String) {
  232. let certNamePathTXT = directoryCertificate + "/" + host + ".txt"
  233. let data: CFData = SecCertificateCopyData(certificate)
  234. let mem = BIO_new_mem_buf(CFDataGetBytePtr(data), Int32(CFDataGetLength(data)))
  235. let x509cert = d2i_X509_bio(mem, nil)
  236. if x509cert == nil {
  237. NextcloudKit.shared.nkCommonInstance.writeLog("[ERROR] OpenSSL couldn't parse X509 Certificate")
  238. } else {
  239. // save details
  240. if FileManager.default.fileExists(atPath: certNamePathTXT) {
  241. do {
  242. try FileManager.default.removeItem(atPath: certNamePathTXT)
  243. } catch { }
  244. }
  245. let fileCertInfo = fopen(certNamePathTXT, "w")
  246. if fileCertInfo != nil {
  247. let output = BIO_new_fp(fileCertInfo, BIO_NOCLOSE)
  248. X509_print_ex(output, x509cert, UInt(XN_FLAG_COMPAT), UInt(X509_FLAG_COMPAT))
  249. BIO_free(output)
  250. }
  251. fclose(fileCertInfo)
  252. X509_free(x509cert)
  253. }
  254. BIO_free(mem)
  255. }
  256. func checkPushNotificationServerProxyCertificateUntrusted(viewController: UIViewController?,
  257. completion: @escaping (_ error: NKError) -> Void) {
  258. guard let host = URL(string: NCBrandOptions.shared.pushNotificationServerProxy)?.host else { return }
  259. NextcloudKit.shared.checkServer(serverUrl: NCBrandOptions.shared.pushNotificationServerProxy) { error in
  260. guard error == .success else {
  261. completion(.success)
  262. return
  263. }
  264. if error == .success {
  265. NCNetworking.shared.writeCertificate(host: host)
  266. completion(error)
  267. } else if error.errorCode == NSURLErrorServerCertificateUntrusted {
  268. let alertController = UIAlertController(title: NSLocalizedString("_ssl_certificate_untrusted_", comment: ""), message: NSLocalizedString("_connect_server_anyway_", comment: ""), preferredStyle: .alert)
  269. alertController.addAction(UIAlertAction(title: NSLocalizedString("_yes_", comment: ""), style: .default, handler: { _ in
  270. NCNetworking.shared.writeCertificate(host: host)
  271. completion(.success)
  272. }))
  273. alertController.addAction(UIAlertAction(title: NSLocalizedString("_no_", comment: ""), style: .default, handler: { _ in
  274. completion(error)
  275. }))
  276. alertController.addAction(UIAlertAction(title: NSLocalizedString("_certificate_details_", comment: ""), style: .default, handler: { _ in
  277. if let navigationController = UIStoryboard(name: "NCViewCertificateDetails", bundle: nil).instantiateInitialViewController() as? UINavigationController,
  278. let vcCertificateDetails = navigationController.topViewController as? NCViewCertificateDetails {
  279. vcCertificateDetails.host = host
  280. viewController?.present(navigationController, animated: true)
  281. }
  282. }))
  283. viewController?.present(alertController, animated: true)
  284. }
  285. }
  286. }
  287. }