FSTLevelDBQueryCache.mm 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334
  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/FSTLevelDBQueryCache.h"
  17. #include <leveldb/db.h>
  18. #include <leveldb/write_batch.h>
  19. #include <string>
  20. #import "Firestore/Protos/objc/firestore/local/Target.pbobjc.h"
  21. #import "Firestore/Source/Core/FSTQuery.h"
  22. #import "Firestore/Source/Local/FSTLevelDB.h"
  23. #import "Firestore/Source/Local/FSTLevelDBKey.h"
  24. #import "Firestore/Source/Local/FSTLocalSerializer.h"
  25. #import "Firestore/Source/Local/FSTQueryData.h"
  26. #import "Firestore/Source/Local/FSTWriteGroup.h"
  27. #import "Firestore/Source/Model/FSTDocumentKey.h"
  28. #import "Firestore/Source/Util/FSTAssert.h"
  29. NS_ASSUME_NONNULL_BEGIN
  30. using Firestore::StringView;
  31. using leveldb::DB;
  32. using leveldb::Iterator;
  33. using leveldb::ReadOptions;
  34. using leveldb::Slice;
  35. using leveldb::Status;
  36. using leveldb::WriteOptions;
  37. @interface FSTLevelDBQueryCache ()
  38. /** A write-through cached copy of the metadata for the query cache. */
  39. @property(nonatomic, strong, nullable) FSTPBTargetGlobal *metadata;
  40. @property(nonatomic, strong, readonly) FSTLocalSerializer *serializer;
  41. @end
  42. @implementation FSTLevelDBQueryCache {
  43. // The DB pointer is shared with all cooperating LevelDB-related objects.
  44. std::shared_ptr<DB> _db;
  45. /**
  46. * The last received snapshot version. This is part of `metadata` but we store it separately to
  47. * avoid extra conversion to/from GPBTimestamp.
  48. */
  49. FSTSnapshotVersion *_lastRemoteSnapshotVersion;
  50. }
  51. + (nullable FSTPBTargetGlobal *)readTargetMetadataFromDB:(std::shared_ptr<DB>)db {
  52. std::string key = [FSTLevelDBTargetGlobalKey key];
  53. std::string value;
  54. Status status = db->Get([FSTLevelDB standardReadOptions], key, &value);
  55. if (status.IsNotFound()) {
  56. return nil;
  57. } else if (!status.ok()) {
  58. FSTFail(@"metadataForKey: failed loading key %s with status: %s", key.c_str(),
  59. status.ToString().c_str());
  60. }
  61. NSData *data =
  62. [[NSData alloc] initWithBytesNoCopy:(void *)value.data() length:value.size() freeWhenDone:NO];
  63. NSError *error;
  64. FSTPBTargetGlobal *proto = [FSTPBTargetGlobal parseFromData:data error:&error];
  65. if (!proto) {
  66. FSTFail(@"FSTPBTargetGlobal failed to parse: %@", error);
  67. }
  68. return proto;
  69. }
  70. - (instancetype)initWithDB:(std::shared_ptr<DB>)db serializer:(FSTLocalSerializer *)serializer {
  71. if (self = [super init]) {
  72. FSTAssert(db, @"db must not be NULL");
  73. _db = db;
  74. _serializer = serializer;
  75. }
  76. return self;
  77. }
  78. - (void)start {
  79. FSTPBTargetGlobal *metadata = [FSTLevelDBQueryCache readTargetMetadataFromDB:_db];
  80. FSTAssert(
  81. metadata != nil,
  82. @"Found nil metadata, expected schema to be at version 0 which ensures metadata existence");
  83. _lastRemoteSnapshotVersion = [self.serializer decodedVersion:metadata.lastRemoteSnapshotVersion];
  84. self.metadata = metadata;
  85. }
  86. #pragma mark - FSTQueryCache implementation
  87. - (FSTTargetID)highestTargetID {
  88. return self.metadata.highestTargetId;
  89. }
  90. - (FSTListenSequenceNumber)highestListenSequenceNumber {
  91. return self.metadata.highestListenSequenceNumber;
  92. }
  93. - (FSTSnapshotVersion *)lastRemoteSnapshotVersion {
  94. return _lastRemoteSnapshotVersion;
  95. }
  96. - (void)setLastRemoteSnapshotVersion:(FSTSnapshotVersion *)snapshotVersion
  97. group:(FSTWriteGroup *)group {
  98. _lastRemoteSnapshotVersion = snapshotVersion;
  99. self.metadata.lastRemoteSnapshotVersion = [self.serializer encodedVersion:snapshotVersion];
  100. [group setMessage:self.metadata forKey:[FSTLevelDBTargetGlobalKey key]];
  101. }
  102. - (void)shutdown {
  103. _db.reset();
  104. }
  105. - (void)addQueryData:(FSTQueryData *)queryData group:(FSTWriteGroup *)group {
  106. FSTTargetID targetID = queryData.targetID;
  107. std::string key = [FSTLevelDBTargetKey keyWithTargetID:targetID];
  108. [group setMessage:[self.serializer encodedQueryData:queryData] forKey:key];
  109. NSString *canonicalID = queryData.query.canonicalID;
  110. std::string indexKey =
  111. [FSTLevelDBQueryTargetKey keyWithCanonicalID:canonicalID targetID:targetID];
  112. std::string emptyBuffer;
  113. [group setData:emptyBuffer forKey:indexKey];
  114. BOOL saveMetadata = NO;
  115. FSTPBTargetGlobal *metadata = self.metadata;
  116. if (targetID > metadata.highestTargetId) {
  117. metadata.highestTargetId = targetID;
  118. saveMetadata = YES;
  119. }
  120. if (queryData.sequenceNumber > metadata.highestListenSequenceNumber) {
  121. metadata.highestListenSequenceNumber = queryData.sequenceNumber;
  122. saveMetadata = YES;
  123. }
  124. if (saveMetadata) {
  125. [group setMessage:metadata forKey:[FSTLevelDBTargetGlobalKey key]];
  126. }
  127. }
  128. - (void)removeQueryData:(FSTQueryData *)queryData group:(FSTWriteGroup *)group {
  129. FSTTargetID targetID = queryData.targetID;
  130. [self removeMatchingKeysForTargetID:targetID group:group];
  131. std::string key = [FSTLevelDBTargetKey keyWithTargetID:targetID];
  132. [group removeMessageForKey:key];
  133. std::string indexKey =
  134. [FSTLevelDBQueryTargetKey keyWithCanonicalID:queryData.query.canonicalID targetID:targetID];
  135. [group removeMessageForKey:indexKey];
  136. }
  137. /**
  138. * Parses the given bytes as an FSTPBTarget protocol buffer and then converts to the equivalent
  139. * query data.
  140. */
  141. - (FSTQueryData *)decodedTargetWithSlice:(Slice)slice {
  142. NSData *data =
  143. [[NSData alloc] initWithBytesNoCopy:(void *)slice.data() length:slice.size() freeWhenDone:NO];
  144. NSError *error;
  145. FSTPBTarget *proto = [FSTPBTarget parseFromData:data error:&error];
  146. if (!proto) {
  147. FSTFail(@"FSTPBTarget failed to parse: %@", error);
  148. }
  149. return [self.serializer decodedQueryData:proto];
  150. }
  151. - (nullable FSTQueryData *)queryDataForQuery:(FSTQuery *)query {
  152. // Scan the query-target index starting with a prefix starting with the given query's canonicalID.
  153. // Note that this is a scan rather than a get because canonicalIDs are not required to be unique
  154. // per target.
  155. Slice canonicalID = StringView(query.canonicalID);
  156. std::unique_ptr<Iterator> indexItererator(_db->NewIterator([FSTLevelDB standardReadOptions]));
  157. std::string indexPrefix = [FSTLevelDBQueryTargetKey keyPrefixWithCanonicalID:canonicalID];
  158. indexItererator->Seek(indexPrefix);
  159. // Simultaneously scan the targets table. This works because each (canonicalID, targetID) pair is
  160. // unique and ordered, so when scanning a table prefixed by exactly one canonicalID, all the
  161. // targetIDs will be unique and in order.
  162. std::string targetPrefix = [FSTLevelDBTargetKey keyPrefix];
  163. std::unique_ptr<Iterator> targetIterator(_db->NewIterator([FSTLevelDB standardReadOptions]));
  164. FSTLevelDBQueryTargetKey *rowKey = [[FSTLevelDBQueryTargetKey alloc] init];
  165. for (; indexItererator->Valid(); indexItererator->Next()) {
  166. Slice indexKey = indexItererator->key();
  167. // Only consider rows matching exactly the specific canonicalID of interest.
  168. if (!indexKey.starts_with(indexPrefix) || ![rowKey decodeKey:indexKey] ||
  169. canonicalID != rowKey.canonicalID) {
  170. // End of this canonicalID's possible targets.
  171. break;
  172. }
  173. // Each row is a unique combination of canonicalID and targetID, so this foreign key reference
  174. // can only occur once.
  175. std::string targetKey = [FSTLevelDBTargetKey keyWithTargetID:rowKey.targetID];
  176. targetIterator->Seek(targetKey);
  177. if (!targetIterator->Valid() || targetIterator->key() != targetKey) {
  178. NSString *foundKeyDescription = @"the end of the table";
  179. if (targetIterator->Valid()) {
  180. foundKeyDescription = [FSTLevelDBKey descriptionForKey:targetIterator->key()];
  181. }
  182. FSTFail(
  183. @"Dangling query-target reference found: "
  184. @"%@ points to %@; seeking there found %@",
  185. [FSTLevelDBKey descriptionForKey:indexKey], [FSTLevelDBKey descriptionForKey:targetKey],
  186. foundKeyDescription);
  187. }
  188. // Finally after finding a potential match, check that the query is actually equal to the
  189. // requested query.
  190. FSTQueryData *target = [self decodedTargetWithSlice:targetIterator->value()];
  191. if ([target.query isEqual:query]) {
  192. return target;
  193. }
  194. }
  195. return nil;
  196. }
  197. #pragma mark Matching Key tracking
  198. - (void)addMatchingKeys:(FSTDocumentKeySet *)keys
  199. forTargetID:(FSTTargetID)targetID
  200. group:(FSTWriteGroup *)group {
  201. // Store an empty value in the index which is equivalent to serializing a GPBEmpty message. In the
  202. // future if we wanted to store some other kind of value here, we can parse these empty values as
  203. // with some other protocol buffer (and the parser will see all default values).
  204. std::string emptyBuffer;
  205. [keys enumerateObjectsUsingBlock:^(FSTDocumentKey *documentKey, BOOL *stop) {
  206. [group setData:emptyBuffer
  207. forKey:[FSTLevelDBTargetDocumentKey keyWithTargetID:targetID documentKey:documentKey]];
  208. [group setData:emptyBuffer
  209. forKey:[FSTLevelDBDocumentTargetKey keyWithDocumentKey:documentKey targetID:targetID]];
  210. }];
  211. }
  212. - (void)removeMatchingKeys:(FSTDocumentKeySet *)keys
  213. forTargetID:(FSTTargetID)targetID
  214. group:(FSTWriteGroup *)group {
  215. [keys enumerateObjectsUsingBlock:^(FSTDocumentKey *key, BOOL *stop) {
  216. [group
  217. removeMessageForKey:[FSTLevelDBTargetDocumentKey keyWithTargetID:targetID documentKey:key]];
  218. [group
  219. removeMessageForKey:[FSTLevelDBDocumentTargetKey keyWithDocumentKey:key targetID:targetID]];
  220. [self.garbageCollector addPotentialGarbageKey:key];
  221. }];
  222. }
  223. - (void)removeMatchingKeysForTargetID:(FSTTargetID)targetID group:(FSTWriteGroup *)group {
  224. std::string indexPrefix = [FSTLevelDBTargetDocumentKey keyPrefixWithTargetID:targetID];
  225. std::unique_ptr<Iterator> indexIterator(_db->NewIterator([FSTLevelDB standardReadOptions]));
  226. indexIterator->Seek(indexPrefix);
  227. FSTLevelDBTargetDocumentKey *rowKey = [[FSTLevelDBTargetDocumentKey alloc] init];
  228. for (; indexIterator->Valid(); indexIterator->Next()) {
  229. Slice indexKey = indexIterator->key();
  230. // Only consider rows matching this specific targetID.
  231. if (![rowKey decodeKey:indexKey] || rowKey.targetID != targetID) {
  232. break;
  233. }
  234. FSTDocumentKey *documentKey = rowKey.documentKey;
  235. // Delete both index rows
  236. [group removeMessageForKey:indexKey];
  237. [group removeMessageForKey:[FSTLevelDBDocumentTargetKey keyWithDocumentKey:documentKey
  238. targetID:targetID]];
  239. [self.garbageCollector addPotentialGarbageKey:documentKey];
  240. }
  241. }
  242. - (FSTDocumentKeySet *)matchingKeysForTargetID:(FSTTargetID)targetID {
  243. std::string indexPrefix = [FSTLevelDBTargetDocumentKey keyPrefixWithTargetID:targetID];
  244. std::unique_ptr<Iterator> indexIterator(_db->NewIterator([FSTLevelDB standardReadOptions]));
  245. indexIterator->Seek(indexPrefix);
  246. FSTDocumentKeySet *result = [FSTDocumentKeySet keySet];
  247. FSTLevelDBTargetDocumentKey *rowKey = [[FSTLevelDBTargetDocumentKey alloc] init];
  248. for (; indexIterator->Valid(); indexIterator->Next()) {
  249. Slice indexKey = indexIterator->key();
  250. // Only consider rows matching this specific targetID.
  251. if (![rowKey decodeKey:indexKey] || rowKey.targetID != targetID) {
  252. break;
  253. }
  254. result = [result setByAddingObject:rowKey.documentKey];
  255. }
  256. return result;
  257. }
  258. #pragma mark - FSTGarbageSource implementation
  259. - (BOOL)containsKey:(FSTDocumentKey *)key {
  260. std::string indexPrefix = [FSTLevelDBDocumentTargetKey keyPrefixWithResourcePath:key.path];
  261. std::unique_ptr<Iterator> indexIterator(_db->NewIterator([FSTLevelDB standardReadOptions]));
  262. indexIterator->Seek(indexPrefix);
  263. if (indexIterator->Valid()) {
  264. FSTLevelDBDocumentTargetKey *rowKey = [[FSTLevelDBDocumentTargetKey alloc] init];
  265. if ([rowKey decodeKey:indexIterator->key()] && [rowKey.documentKey isEqualToKey:key]) {
  266. return YES;
  267. }
  268. }
  269. return NO;
  270. }
  271. @end
  272. NS_ASSUME_NONNULL_END