iOS App crash when app in background and downloading multiple files using UrlSessionDownloadTask

597 Views Asked by At

I am working on a iOS app which need to download multiple file (mostly images) from server and store them in Document Directory. I am using UrlSessionDownloadTask with background URLSession to download file using Urls. It work fine when app is in foreground but if i go to background and comes back to foreground app freezes and crash after some time. I try many thing but nothing works.

func downloadAllImages(retryCount: Int = 0,completion: @escaping((Bool)->Void)){
        
        var success: Bool = true
        var count = 0
        print("Image Downloading Start")
        //SVProgressHUD.setStatus("Downloading Images...")
        //CommonClass.showBanner(message: "Image Downloading Start")
        
        for (localName, severPath) in images {
            if DataManager.isInternetAvailable(){
                self.dispatchGroup.enter()
                self.dispatchGroupEnterCount += 1
                let path = severPath
                count += 1
                self.DownloadFile(urlPath: path, localName: localName,itemCount: count)
            }else {
                success = false
                CommonClass.showBanner(message: "No Internet")
            }
        }
        dispatchGroup.notify(queue: .main) {
            //retry If some Images failed to download
            completion(success)
        
        }
    }
    
    
    func DownloadFile(urlPath: String, localName: String, itemCount: Int){
        guard let url = URL(string: urlPath) else {
            DispatchQueue.main.sync {
                self.leaveDispatchGroup()
            }
            return
        }
       
        let urlRequest = URLRequest(url: url, cachePolicy: .reloadIgnoringLocalAndRemoteCacheData, timeoutInterval: 300)
        let downloadTask = session.downloadTask(with: urlRequest)
        self.localNameDict[downloadTask.taskIdentifier] = localName
        taskArray.append(downloadTask)
        
        downloadTask.resume()
        
    }

}

extension DataManager: URLSessionDownloadDelegate {
    
    func urlSession(_ session: URLSession, downloadTask: URLSessionDownloadTask, didFinishDownloadingTo location: URL) {
            do {
                let filemanager = FileManager()
                let documentsURL = filemanager.urls(for: .documentDirectory, in: .userDomainMask)[0]
                let fileURL = documentsURL.appendingPathComponent(self.localNameDict[downloadTask.taskIdentifier] ?? "img.jpeg")
                if filemanager.fileExists(atPath: fileURL.path){
                    try? filemanager.removeItem(at: fileURL)
                }
                try filemanager.copyItem(at: location, to: fileURL)
            }catch(let error) {
                print("ERROR In Storing: \(error.localizedDescription)")
            }
        
    }
    
    func urlSessionDidFinishEvents(forBackgroundURLSession session: URLSession) {
        DispatchQueue.main.async {
            guard let appDelegate = UIApplication.shared.delegate as? AppDelegate, let completionHandler = appDelegate.bgSessionCompletionHandler else {
                return
            }
            appDelegate.bgSessionCompletionHandler = nil
            completionHandler()
        }
    }

    func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
            if let error = error {
                print(error.localizedDescription)
            }
            self.leaveDispatchGroup()
        
    }

    func leaveDispatchGroup(){
        if dispatchGroupEnterCount > 0 {
                self.dispatchGroupEnterCount -= 1
                self.dispatchGroup.leave()
        }
    }
 }

running video of App: https://youtu.be/lNkMpZRNhGY

1

There are 1 best solutions below

0
On
extension URLSession {
    
    ///Background sessions let you perform uploads and downloads of content in the background while your app isn’t running. You can create a background session configuration by calling the backgroundSessionConfiguration(_:) method on the URLSessionConfiguration class.
    static let background = URLSession(
        configuration: .background(
            withIdentifier: "foo"))
}

You have to use a session that configured for running in background.