FSTMemoryQueryCache.mm 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. /*
  2. * Copyright 2017 Google
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. #import "Firestore/Source/Local/FSTMemoryQueryCache.h"
  17. #import <Protobuf/GPBProtocolBuffers.h>
  18. #include <utility>
  19. #import "Firestore/Protos/objc/firestore/local/Target.pbobjc.h"
  20. #import "Firestore/Source/Core/FSTQuery.h"
  21. #import "Firestore/Source/Local/FSTMemoryPersistence.h"
  22. #import "Firestore/Source/Local/FSTQueryData.h"
  23. #import "Firestore/Source/Local/FSTReferenceSet.h"
  24. #include "Firestore/core/src/firebase/firestore/model/document_key.h"
  25. #include "Firestore/core/src/firebase/firestore/model/snapshot_version.h"
  26. using firebase::firestore::model::DocumentKey;
  27. using firebase::firestore::model::DocumentKeySet;
  28. using firebase::firestore::model::ListenSequenceNumber;
  29. using firebase::firestore::model::SnapshotVersion;
  30. using firebase::firestore::model::TargetId;
  31. NS_ASSUME_NONNULL_BEGIN
  32. @interface FSTMemoryQueryCache ()
  33. /** Maps a query to the data about that query. */
  34. @property(nonatomic, strong, readonly) NSMutableDictionary<FSTQuery *, FSTQueryData *> *queries;
  35. /** A ordered bidirectional mapping between documents and the remote target IDs. */
  36. @property(nonatomic, strong, readonly) FSTReferenceSet *references;
  37. /** The highest numbered target ID encountered. */
  38. @property(nonatomic, assign) TargetId highestTargetID;
  39. @property(nonatomic, assign) ListenSequenceNumber highestListenSequenceNumber;
  40. @end
  41. @implementation FSTMemoryQueryCache {
  42. FSTMemoryPersistence *_persistence;
  43. /** The last received snapshot version. */
  44. SnapshotVersion _lastRemoteSnapshotVersion;
  45. }
  46. - (instancetype)initWithPersistence:(FSTMemoryPersistence *)persistence {
  47. if (self = [super init]) {
  48. _persistence = persistence;
  49. _queries = [NSMutableDictionary dictionary];
  50. _references = [[FSTReferenceSet alloc] init];
  51. _lastRemoteSnapshotVersion = SnapshotVersion::None();
  52. }
  53. return self;
  54. }
  55. #pragma mark - FSTQueryCache implementation
  56. #pragma mark Query tracking
  57. - (TargetId)highestTargetID {
  58. return _highestTargetID;
  59. }
  60. - (ListenSequenceNumber)highestListenSequenceNumber {
  61. return _highestListenSequenceNumber;
  62. }
  63. - (const SnapshotVersion &)lastRemoteSnapshotVersion {
  64. return _lastRemoteSnapshotVersion;
  65. }
  66. - (void)setLastRemoteSnapshotVersion:(SnapshotVersion)snapshotVersion {
  67. _lastRemoteSnapshotVersion = std::move(snapshotVersion);
  68. }
  69. - (void)addQueryData:(FSTQueryData *)queryData {
  70. self.queries[queryData.query] = queryData;
  71. if (queryData.targetID > self.highestTargetID) {
  72. self.highestTargetID = queryData.targetID;
  73. }
  74. if (queryData.sequenceNumber > self.highestListenSequenceNumber) {
  75. self.highestListenSequenceNumber = queryData.sequenceNumber;
  76. }
  77. }
  78. - (void)updateQueryData:(FSTQueryData *)queryData {
  79. self.queries[queryData.query] = queryData;
  80. if (queryData.targetID > self.highestTargetID) {
  81. self.highestTargetID = queryData.targetID;
  82. }
  83. if (queryData.sequenceNumber > self.highestListenSequenceNumber) {
  84. self.highestListenSequenceNumber = queryData.sequenceNumber;
  85. }
  86. }
  87. - (int32_t)count {
  88. return (int32_t)[self.queries count];
  89. }
  90. - (void)removeQueryData:(FSTQueryData *)queryData {
  91. [self.queries removeObjectForKey:queryData.query];
  92. [self.references removeReferencesForID:queryData.targetID];
  93. }
  94. - (nullable FSTQueryData *)queryDataForQuery:(FSTQuery *)query {
  95. return self.queries[query];
  96. }
  97. - (void)enumerateTargetsUsingBlock:(void (^)(FSTQueryData *queryData, BOOL *stop))block {
  98. [self.queries
  99. enumerateKeysAndObjectsUsingBlock:^(FSTQuery *key, FSTQueryData *queryData, BOOL *stop) {
  100. block(queryData, stop);
  101. }];
  102. }
  103. - (int)removeQueriesThroughSequenceNumber:(ListenSequenceNumber)sequenceNumber
  104. liveQueries:(NSDictionary<NSNumber *, FSTQueryData *> *)liveQueries {
  105. NSMutableArray<FSTQuery *> *toRemove = [NSMutableArray array];
  106. [self.queries
  107. enumerateKeysAndObjectsUsingBlock:^(FSTQuery *query, FSTQueryData *queryData, BOOL *stop) {
  108. if (queryData.sequenceNumber <= sequenceNumber) {
  109. if (liveQueries[@(queryData.targetID)] == nil) {
  110. [toRemove addObject:query];
  111. [self.references removeReferencesForID:queryData.targetID];
  112. }
  113. }
  114. }];
  115. [self.queries removeObjectsForKeys:toRemove];
  116. return (int)[toRemove count];
  117. }
  118. #pragma mark Reference tracking
  119. - (void)addMatchingKeys:(const DocumentKeySet &)keys forTargetID:(TargetId)targetID {
  120. [self.references addReferencesToKeys:keys forID:targetID];
  121. for (const DocumentKey &key : keys) {
  122. [_persistence.referenceDelegate addReference:key];
  123. }
  124. }
  125. - (void)removeMatchingKeys:(const DocumentKeySet &)keys forTargetID:(TargetId)targetID {
  126. [self.references removeReferencesToKeys:keys forID:targetID];
  127. for (const DocumentKey &key : keys) {
  128. [_persistence.referenceDelegate removeReference:key];
  129. }
  130. }
  131. - (void)removeMatchingKeysForTargetID:(TargetId)targetID {
  132. [self.references removeReferencesForID:targetID];
  133. }
  134. - (DocumentKeySet)matchingKeysForTargetID:(TargetId)targetID {
  135. return [self.references referencedKeysForID:targetID];
  136. }
  137. - (BOOL)containsKey:(const firebase::firestore::model::DocumentKey &)key {
  138. return [self.references containsKey:key];
  139. }
  140. - (size_t)byteSizeWithSerializer:(FSTLocalSerializer *)serializer {
  141. __block size_t count = 0;
  142. [self.queries
  143. enumerateKeysAndObjectsUsingBlock:^(FSTQuery *key, FSTQueryData *queryData, BOOL *stop) {
  144. count += [[serializer encodedQueryData:queryData] serializedSize];
  145. }];
  146. return count;
  147. }
  148. @end
  149. NS_ASSUME_NONNULL_END