web-dev-qa-db-fra.com

Swift 3 et NSURLSession issue

Grâce à Apple, mon projet iOS 9 'Swift 2.3' est complètement inutilisable avec le 'Swift 3' d'iOS 10 ...

J'ai réparé presque tout sauf que j'ai un problème avec l'utilisation de NSURLSession, Xcode me dit qu'il a été renommé en URLSession, si je le renomme, Xcode me dira:

utilisation de type non déclaré URLSession

La fondation est importée.

Quelle est la solution?!

Par exemple, je l'utilise de cette façon ...

lazy var defaultSession: URLSession = {
    let configuration = URLSessionConfiguration.background(withIdentifier: "reCoded.BGDownload")
    configuration.sessionSendsLaunchEvents = true
    configuration.isDiscretionary = true
    let session = URLSession(configuration: configuration, delegate: self, delegateQueue, queue: nil)
    return session
}()

et même avec les méthodes de délégué le même problème.

8
Raffi

Essayez d’utiliser Foundation.URLSession partout où vous utilisez URLSession.

7
Jorn van Dijk

/ Je dois le faire fonctionner/Dans certains cas, essayez de copier votre code ailleurs, puis supprimez tout le contenu de votre classe qui utilise URLSession, saisissez à nouveau les méthodes de session et remettez en place le code copié.

2
Raffi

Mettez à jour vos fonctions URLSessin avec; 

func urlSession(_ session: URLSession, dataTask: URLSessionDataTask, didReceive data: Data) {
    self.data.append(data as Data)  
}

func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
    if error != nil {
        print("Failed to download data")
    }else {
        print("Data downloaded")
        self.parseJSON()
    }
}
1
Ali

Voici où je suis en ce moment. Ce n'est pas parfait mais ça marche peut-être la moitié du temps.

Tout d'abord, dans la classe où ma URLsession est définie:

import Foundation
class Central: NSObject, URLSessionDataDelegate, URLSessionDelegate, URLSessionTaskDelegate, URLSessionDownloadDelegate {

Je ne pense pas que tout cela soit nécessaire, mais le voilà. Ensuite, voici la fonction appelée par mon extraction en arrière-plan:

func getWebData() {
    var defaults: UserDefaults = UserDefaults.standard
    let backgroundConfigObject = URLSessionConfiguration.background(withIdentifier: "myBGconfig")
    let backgroundSession = URLSession(configuration: backgroundConfigObject, delegate: self, delegateQueue: nil)
    urlString = "https://www.powersmartpricing.org/psp/servlet?type=dayslider"
    if let url = URL(string: urlString) {
    let rateTask = backgroundSession.downloadTask(with: URL(string: urlString)!)
    rateTask.taskDescription = "rate"
    rateTask.resume()
}

Quand la tâche revient:

func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didFinishDownloadingTo location: URL ) {
    if downloadTask.taskDescription == "rate" {  // I run 2 web tasks during the session
        if let data = NSData(contentsOf: location) {
           var return1 = String(data: data as! Data, encoding: String.Encoding.utf8)!
             DispatchQueue.global(qos: .userInteractive).asyncAfter(deadline: .now() + 0.2){
                var defaults: UserDefaults = UserDefaults.standard
                defaults.set(myNumber, forKey: "electricRate") // myNumber is an extract of the text in returned web data
                defaults.set(Date(), forKey: "rateUpdate")
                defaults.synchronize()
                self.calcSetting()  //Calls another function defined in the same class.  That function sends the user a notification.
                let notificationName = Notification.Name("GotWebData")
                NotificationCenter.default.post(name: notificationName, object: nil)
            }   //  Closes the Dispatch
        }
      if session.configuration.identifier == "myBGconfig" {
          print("about to invalidate the session")
          session.invalidateAndCancel()
       }
}

Je n'ai pas encore compris comment tuer la session lorsque les DEUX tâches sont terminées. Je la supprime maintenant lorsque l'une ou l'autre est terminée, avec invalidateAndCancel comme ci-dessus.

Et enfin, pour attraper les erreurs:

func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didCompleteWithError: Error?) {
      if downloadTask.taskDescription == "rate" {
        print("rate download failed with error \(didCompleteWithError)")
    }
    if downloadTask.taskDescription == "other" {
        print("other download failed with error \(didCompleteWithError)")
    }
 downloadTask.resume()  //  I'm hoping this retries if a task fails?
}

func urlSession(_ session: URLSession, didBecomeInvalidWithError error: Error?) {
    if let error = error as? NSError {
        print("invalidate, error %@ / %d", error.domain, error.code)
    } else {
        print("invalidate, no error")
    }
}
0
Wayne Henderson

Je peux expliquer comment, mais en jouant avec le code, cela a fonctionné dans Swift 3 après deux jours de frustration. Je suppose que Swift 3 a supprimé beaucoup de mots inutiles.

let task = Foundation.URLSession.shared.dataTask(with: <#T##URL#>, completionHandler: <#T##(Data?, URLResponse?, Error?) -> Void#>)
0
user3748984