Query+Combine.swift 2.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. // Copyright 2021 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. #if canImport(Combine) && swift(>=5.0)
  15. import Combine
  16. import FirebaseFirestore
  17. @available(swift 5.0)
  18. @available(iOS 13.0, macOS 10.15, macCatalyst 13.0, tvOS 13.0, watchOS 6.0, *)
  19. public extension Query {
  20. // MARK: - Get Documents
  21. /// Reads the documents matching this query.
  22. ///
  23. /// - Parameter source: Indicates whether the results should be fetched from the cache only
  24. /// (`Source.cache`), the server only (`Source.server`), or to attempt the server and fall back
  25. /// to the cache (`Source.default`).
  26. /// - Returns: A publisher emitting a `QuerySnapshot` instance.
  27. func getDocuments(source: FirestoreSource = .default) -> Future<QuerySnapshot, Error> {
  28. Future { promise in
  29. self.getDocuments(source: source) { snapshot, error in
  30. if let error = error {
  31. promise(.failure(error))
  32. } else if let snapshot = snapshot {
  33. promise(.success(snapshot))
  34. }
  35. }
  36. }
  37. }
  38. // MARK: - Snapshot Publisher
  39. /// Registers a publisher that publishes query snapshot changes.
  40. ///
  41. /// - Parameter includeMetadataChanges: Whether metadata-only changes (i.e. only
  42. /// `QuerySnapshot.metadata` changed) should trigger snapshot events.
  43. /// - Returns: A publisher emitting `QuerySnapshot` instances.
  44. func snapshotPublisher(includeMetadataChanges: Bool = false)
  45. -> AnyPublisher<QuerySnapshot, Error> {
  46. let subject = PassthroughSubject<QuerySnapshot, Error>()
  47. let listenerHandle =
  48. addSnapshotListener(includeMetadataChanges: includeMetadataChanges) { snapshot, error in
  49. if let error = error {
  50. subject.send(completion: .failure(error))
  51. } else if let snapshot = snapshot {
  52. subject.send(snapshot)
  53. }
  54. }
  55. return subject
  56. .handleEvents(receiveCancel: listenerHandle.remove)
  57. .eraseToAnyPublisher()
  58. }
  59. }
  60. #endif