// Copyright 2022 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. import Foundation #if COCOAPODS import GTMSessionFetcher #else import GTMSessionFetcherCore #endif /** * Task which provides the ability to get a download URL for an object in Firebase Storage. */ internal class StorageGetDownloadURLTask: StorageTask, StorageTaskManagement { private var fetcher: GTMSessionFetcher? private var fetcherCompletion: ((Data?, NSError?) -> Void)? private var taskCompletion: ((_ downloadURL: URL?, _: Error?) -> Void)? internal init(reference: StorageReference, fetcherService: GTMSessionFetcherService, queue: DispatchQueue, completion: ((_: URL?, _: Error?) -> Void)?) { super.init(reference: reference, service: fetcherService, queue: queue) taskCompletion = completion } deinit { self.fetcher?.stopFetching() } /** * Prepares a task and begins execution. */ internal func enqueue() { weak var weakSelf = self DispatchQueue.global(qos: .background).async { guard let strongSelf = weakSelf else { return } var request = strongSelf.baseRequest request.httpMethod = "GET" request.timeoutInterval = strongSelf.reference.storage.maxOperationRetryTime let callback = strongSelf.taskCompletion strongSelf.taskCompletion = nil let fetcher = strongSelf.fetcherService.fetcher(with: request) fetcher.comment = "GetDownloadURLTask" strongSelf.fetcher = fetcher strongSelf.fetcherCompletion = { (data: Data?, error: NSError?) in var downloadURL: URL? if let error = error { if self.error == nil { self.error = StorageErrorCode.error(withServerError: error, ref: self.reference) } } else { if let data = data, let responseDictionary = try? JSONSerialization .jsonObject(with: data) as? [String: Any] { downloadURL = strongSelf.downloadURLFromMetadataDictionary(responseDictionary) if downloadURL == nil { self.error = NSError(domain: StorageErrorDomain, code: StorageErrorCode.unknown.rawValue, userInfo: [NSLocalizedDescriptionKey: "Failed to retrieve a download URL."]) } } else { self.error = StorageErrorCode.error(withInvalidRequest: data) } } callback?(downloadURL, self.error) self.fetcherCompletion = nil } strongSelf.fetcher?.beginFetch { data, error in let strongSelf = weakSelf if let fetcherCompletion = strongSelf?.fetcherCompletion { fetcherCompletion(data, error as? NSError) } } } } internal func downloadURLFromMetadataDictionary(_ dictionary: [String: Any]) -> URL? { let downloadTokens = dictionary["downloadTokens"] guard let downloadTokens = downloadTokens as? String, downloadTokens.count > 0 else { return nil } let downloadTokenArray = downloadTokens.components(separatedBy: ",") let bucket = dictionary["bucket"] ?? "" let path = dictionary["name"] as? String ?? "" let fullPath = "/v0/b/\(bucket)/o/\(StorageUtils.GCSEscapedString(path))" var components = URLComponents() components.scheme = reference.storage.scheme components.host = reference.storage.host components.port = reference.storage.port components.percentEncodedPath = fullPath // The backend can return an arbitrary number of download tokens, but we only expose the first // token via the download URL. let altItem = URLQueryItem(name: "alt", value: "media") let tokenItem = URLQueryItem(name: "token", value: downloadTokenArray[0]) components.queryItems = [altItem, tokenItem] return components.url } }