123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302 |
- //
- // NCCommunication.swift
- // Nextcloud
- //
- // Created by Marino Faggiana on 12/10/19.
- // Copyright © 2018 Marino Faggiana. All rights reserved.
- //
- // Author Marino Faggiana <marino.faggiana@nextcloud.com>
- //
- // This program is free software: you can redistribute it and/or modify
- // it under the terms of the GNU General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // This program is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- // GNU General Public License for more details.
- //
- // You should have received a copy of the GNU General Public License
- // along with this program. If not, see <http://www.gnu.org/licenses/>.
- //
- import Foundation
- import Alamofire
- import SwiftyXMLParser
- class NCCommunication: SessionDelegate {
- @objc static let sharedInstance: NCCommunication = {
- let instance = NCCommunication()
- return instance
- }()
-
- var username = ""
- var password = ""
- var userAgent: String?
-
- //MARK: - Settings
- @objc func settingAccount(username: String, password: String) {
- self.username = username
- self.password = password
- }
-
- @objc func settingUserAgent(_ userAgent: String) {
- self.userAgent = userAgent
- }
-
- //MARK: - webDAV
- @objc func createFolder(serverUrl: String, fileName: String, completionHandler: @escaping (_ error: Error?) -> Void) {
-
- // url
- var serverUrl = serverUrl
- var url: URLConvertible
- do {
- if serverUrl.last == "/" {
- serverUrl = serverUrl + fileName
- } else {
- serverUrl = serverUrl + "/" + fileName
- }
- try url = serverUrl.asURL()
- } catch let error {
- completionHandler(error)
- return
- }
-
- // method
- let method = HTTPMethod(rawValue: "MKCOL")
-
- // headers
- var headers: HTTPHeaders = [.authorization(username: self.username, password: self.password)]
- if let userAgent = self.userAgent { headers.update(.userAgent(userAgent)) }
-
- AF.request(url, method: method, parameters:nil, encoding: URLEncoding.default, headers: headers, interceptor: nil).validate(statusCode: 200..<300).response { (response) in
- switch response.result {
- case.failure(let error):
- completionHandler(error)
- case .success( _):
- completionHandler(nil)
- }
- }
- }
-
- @objc func deleteFileOrFolder(serverUrl: String, fileName: String, completionHandler: @escaping (_ error: Error?) -> Void) {
-
- // url
- var serverUrl = serverUrl
- var url: URLConvertible
- do {
- if serverUrl.last == "/" {
- serverUrl = serverUrl + fileName
- } else {
- serverUrl = serverUrl + "/" + fileName
- }
- try url = serverUrl.asURL()
- } catch let error {
- completionHandler(error)
- return
- }
-
- // method
- let method = HTTPMethod(rawValue: "DELETE")
-
- // headers
- var headers: HTTPHeaders = [.authorization(username: self.username, password: self.password)]
- if let userAgent = self.userAgent { headers.update(.userAgent(userAgent)) }
-
- AF.request(url, method: method, parameters:nil, encoding: URLEncoding.default, headers: headers, interceptor: nil).validate(statusCode: 200..<300).response { (response) in
- switch response.result {
- case.failure(let error):
- completionHandler(error)
- case .success( _):
- completionHandler(nil)
- }
- }
- }
-
- @objc func moveFileOrFolder(fileNamePath: String, fileNamePathDestination: String,completionHandler: @escaping (_ error: Error?) -> Void) {
-
- // url
- var url: URLConvertible
- do {
- try url = fileNamePath.asURL()
- } catch let error {
- completionHandler(error)
- return
- }
-
- // method
- let method = HTTPMethod(rawValue: "MOVE")
-
- // headers
- var headers: HTTPHeaders = [.authorization(username: self.username, password: self.password)]
- if let userAgent = self.userAgent { headers.update(.userAgent(userAgent)) }
- headers.update(name: "Destination", value: fileNamePathDestination.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) ?? "")
- headers.update(name: "Overwrite", value: "T")
-
- AF.request(url, method: method, parameters:nil, encoding: URLEncoding.default, headers: headers, interceptor: nil).validate(statusCode: 200..<300).response { (response) in
- switch response.result {
- case.failure(let error):
- completionHandler(error)
- case .success( _):
- completionHandler(nil)
- }
- }
- }
-
- @objc func readFileOrFolder(serverUrl: String, depth: String, completionHandler: @escaping (_ result: [NCFile], _ error: Error?) -> Void) {
-
- var files = [NCFile]()
- let dataFile =
- """
- <?xml version=\"1.0\" encoding=\"UTF-8\"?>
- <d:propfind xmlns:d=\"DAV:\" xmlns:oc=\"http://owncloud.org/ns\" xmlns:nc=\"http://nextcloud.org/ns\">
- <d:prop>"
- <d:getlastmodified />
- <d:getetag />
- <d:getcontenttype />
- <d:resourcetype />
- <d:getcontentlength />
- <permissions xmlns=\"http://owncloud.org/ns\"/>
- <id xmlns=\"http://owncloud.org/ns\"/>
- <fileid xmlns=\"http://owncloud.org/ns\"/>
- <size xmlns=\"http://owncloud.org/ns\"/>
- <favorite xmlns=\"http://owncloud.org/ns\"/>
- <is-encrypted xmlns=\"http://nextcloud.org/ns\"/>
- <mount-type xmlns=\"http://nextcloud.org/ns\"/>
- <owner-id xmlns=\"http://owncloud.org/ns\"/>
- <owner-display-name xmlns=\"http://owncloud.org/ns\"/>
- <comments-unread xmlns=\"http://owncloud.org/ns\"/>
- <has-preview xmlns=\"http://nextcloud.org/ns\"/>
- </d:prop>
- </d:propfind>
- """
- // url
- var url: URLConvertible
- do {
- try url = serverUrl.asURL()
- } catch let error {
- completionHandler(files, error)
- return
- }
-
- // method
- let method = HTTPMethod(rawValue: "PROPFIND")
-
- // headers
- var headers: HTTPHeaders = [.authorization(username: self.username, password: self.password)]
- if let userAgent = self.userAgent { headers.update(.userAgent(userAgent)) }
- headers.update(.contentType("application/xml"))
- headers.update(name: "Depth", value: depth)
- // request
- var urlRequest: URLRequest
- do {
- try urlRequest = URLRequest(url: url, method: method, headers: headers)
- urlRequest.httpBody = dataFile.data(using: .utf8)
- } catch let error {
- completionHandler(files, error)
- return
- }
-
- AF.request(urlRequest).validate(statusCode: 200..<300).responseData { (response) in
- switch response.result {
- case.failure(let error):
- completionHandler(files, error)
- case .success( _):
- if let data = response.data {
- let xml = XML.parse(data)
- let elements = xml["d:multistatus", "d:response"]
- for element in elements {
- let file = NCFile()
- if let href = element["d:href"].text {
- file.path = href.removingPercentEncoding ?? ""
- }
- let propstat = element["d:propstat"][0]
-
- if let getlastmodified = propstat["d:prop", "d:getlastmodified"].text {
- if let date = NCCommunicationCommon.sharedInstance.convertDate(getlastmodified, format: "EEE, dd MMM y HH:mm:ss zzz") {
- file.date = date
- }
- }
- if let getetag = propstat["d:prop", "d:getetag"].text {
- file.etag = getetag.replacingOccurrences(of: "\"", with: "")
- }
-
- if let quotaavailablebytes = propstat["d:prop", "d:quota-available-bytes"].text {
- file.quotaAvailableBytes = Double(quotaavailablebytes) ?? 0
- }
- if let quotausedbytes = propstat["d:prop", "d:quota-used-bytes"].text {
- file.quotaUsedBytes = Double(quotausedbytes) ?? 0
- }
-
- files.append(file)
- }
- }
- completionHandler(files, nil)
- }
- }
- }
-
- //MARK: - Download
-
- @objc func download(serverUrl: String, fileName: String, fileNamePathDestination: String, completionHandler: @escaping (_ error: Error?) -> Void) {
-
- // url
- var serverUrl = serverUrl
- var url: URLConvertible
- do {
- if serverUrl.last == "/" {
- serverUrl = serverUrl + fileName
- } else {
- serverUrl = serverUrl + "/" + fileName
- }
- try url = serverUrl.asURL()
- } catch let error {
- completionHandler(error)
- return
- }
-
- // destination
- var destination: Alamofire.DownloadRequest.Destination?
- if let fileNamePathDestinationURL = URL(string: fileNamePathDestination) {
- let destinationFile: DownloadRequest.Destination = { _, _ in
- return (fileNamePathDestinationURL, [.removePreviousFile, .createIntermediateDirectories])
- }
- destination = destinationFile
- }
-
- // headers
- var headers: HTTPHeaders = [.authorization(username: self.username, password: self.password)]
- if let userAgent = self.userAgent { headers.update(.userAgent(userAgent)) }
-
- AF.download(url, method: .get, parameters: nil, encoding: URLEncoding.default, headers: headers, interceptor: nil, to: destination).downloadProgress { progress in
- //self.postProgress(progress: progress)
- }.responseData { response in
- switch response.result {
- case.failure(let error):
- completionHandler(error)
- case .success( _):
- completionHandler(nil)
- }
- }
- }
-
- //MARK: - SessionDelegate
- func urlSession(_ session: URLSession, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) {
-
- if CCCertificate.sharedManager().checkTrustedChallenge(challenge) {
- completionHandler(URLSession.AuthChallengeDisposition.performDefaultHandling, URLCredential.init(trust: challenge.protectionSpace.serverTrust!))
- } else {
- completionHandler(URLSession.AuthChallengeDisposition.performDefaultHandling, nil)
- }
-
- }
- }
|