StorageUpdateMetadataTask.swift 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /// Copyright 2022 Google LLC
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. import Foundation
  15. #if COCOAPODS
  16. import GTMSessionFetcher
  17. #else
  18. import GTMSessionFetcherCore
  19. #endif
  20. /**
  21. * Task which provides the ability to delete an object in Firebase Storage.
  22. */
  23. class StorageUpdateMetadataTask: StorageTask, StorageTaskManagement {
  24. private var fetcher: GTMSessionFetcher?
  25. private var fetcherCompletion: ((Data?, NSError?) -> Void)?
  26. private var taskCompletion: ((_ metadata: StorageMetadata?, _: Error?) -> Void)?
  27. private var updateMetadata: StorageMetadata
  28. init(reference: StorageReference,
  29. fetcherService: GTMSessionFetcherService,
  30. queue: DispatchQueue,
  31. metadata: StorageMetadata,
  32. completion: ((_: StorageMetadata?, _: Error?) -> Void)?) {
  33. updateMetadata = metadata
  34. super.init(reference: reference, service: fetcherService, queue: queue)
  35. taskCompletion = completion
  36. }
  37. deinit {
  38. self.fetcher?.stopFetching()
  39. }
  40. /**
  41. * Prepares a task and begins execution.
  42. */
  43. func enqueue() {
  44. let completion = taskCompletion
  45. taskCompletion = { (metadata: StorageMetadata?, error: Error?) in
  46. completion?(metadata, error)
  47. // Reference self in completion handler in order to retain self until completion is called.
  48. self.taskCompletion = nil
  49. }
  50. dispatchQueue.async { [weak self] in
  51. guard let self = self else { return }
  52. var request = self.baseRequest
  53. let updateDictionary = self.updateMetadata.updatedMetadata()
  54. let updateData = try? JSONSerialization.data(withJSONObject: updateDictionary)
  55. request.httpMethod = "PATCH"
  56. request.timeoutInterval = self.reference.storage.maxOperationRetryTime
  57. request.httpBody = updateData
  58. request.setValue("application/json; charset=UTF-8", forHTTPHeaderField: "Content-Type")
  59. if let count = updateData?.count {
  60. request.setValue("\(count)", forHTTPHeaderField: "Content-Length")
  61. }
  62. let fetcher = self.fetcherService.fetcher(with: request)
  63. fetcher.comment = "GetMetadataTask"
  64. self.fetcher = fetcher
  65. self.fetcherCompletion = { [weak self] (data: Data?, error: NSError?) in
  66. guard let self = self else { return }
  67. var metadata: StorageMetadata?
  68. if let error {
  69. if self.error == nil {
  70. self.error = StorageErrorCode.error(withServerError: error, ref: self.reference)
  71. }
  72. } else {
  73. if let data,
  74. let responseDictionary = try? JSONSerialization
  75. .jsonObject(with: data) as? [String: AnyHashable] {
  76. metadata = StorageMetadata(dictionary: responseDictionary)
  77. metadata?.fileType = .file
  78. } else {
  79. self.error = StorageErrorCode.error(withInvalidRequest: data)
  80. }
  81. }
  82. self.taskCompletion?(metadata, self.error)
  83. self.fetcherCompletion = nil
  84. }
  85. fetcher.comment = "UpdateMetadataTask"
  86. self.fetcher?.beginFetch { [weak self] data, error in
  87. self?.fetcherCompletion?(data, error as? NSError)
  88. }
  89. }
  90. }
  91. }