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
You have to use a session that configured for running in background.