FSTLocalSerializer.mm 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  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/FSTLocalSerializer.h"
  17. #include <cinttypes>
  18. #include <utility>
  19. #include <vector>
  20. #import "FIRTimestamp.h"
  21. #import "Firestore/Protos/objc/firestore/local/MaybeDocument.pbobjc.h"
  22. #import "Firestore/Protos/objc/firestore/local/Mutation.pbobjc.h"
  23. #import "Firestore/Protos/objc/firestore/local/Target.pbobjc.h"
  24. #import "Firestore/Protos/objc/google/firestore/v1/Document.pbobjc.h"
  25. #import "Firestore/Source/Local/FSTQueryData.h"
  26. #import "Firestore/Source/Model/FSTDocument.h"
  27. #import "Firestore/Source/Model/FSTMutationBatch.h"
  28. #import "Firestore/Source/Remote/FSTSerializerBeta.h"
  29. #include "Firestore/core/include/firebase/firestore/timestamp.h"
  30. #include "Firestore/core/src/firebase/firestore/core/query.h"
  31. #include "Firestore/core/src/firebase/firestore/model/document.h"
  32. #include "Firestore/core/src/firebase/firestore/model/document_key.h"
  33. #include "Firestore/core/src/firebase/firestore/model/snapshot_version.h"
  34. #include "Firestore/core/src/firebase/firestore/util/hard_assert.h"
  35. using firebase::Timestamp;
  36. using firebase::firestore::core::Query;
  37. using firebase::firestore::model::DocumentKey;
  38. using firebase::firestore::model::DocumentState;
  39. using firebase::firestore::model::ListenSequenceNumber;
  40. using firebase::firestore::model::ObjectValue;
  41. using firebase::firestore::model::SnapshotVersion;
  42. using firebase::firestore::model::TargetId;
  43. @interface FSTLocalSerializer ()
  44. @property(nonatomic, strong, readonly) FSTSerializerBeta *remoteSerializer;
  45. @end
  46. /** Serializer for values stored in the LocalStore. */
  47. @implementation FSTLocalSerializer
  48. - (instancetype)initWithRemoteSerializer:(FSTSerializerBeta *)remoteSerializer {
  49. self = [super init];
  50. if (self) {
  51. _remoteSerializer = remoteSerializer;
  52. }
  53. return self;
  54. }
  55. - (FSTPBMaybeDocument *)encodedMaybeDocument:(FSTMaybeDocument *)document {
  56. FSTPBMaybeDocument *proto = [FSTPBMaybeDocument message];
  57. if ([document isKindOfClass:[FSTDeletedDocument class]]) {
  58. FSTDeletedDocument *deletedDocument = (FSTDeletedDocument *)document;
  59. proto.noDocument = [self encodedDeletedDocument:deletedDocument];
  60. proto.hasCommittedMutations = deletedDocument.hasCommittedMutations;
  61. } else if ([document isKindOfClass:[FSTDocument class]]) {
  62. FSTDocument *existingDocument = (FSTDocument *)document;
  63. if (existingDocument.proto != nil) {
  64. proto.document = existingDocument.proto;
  65. } else {
  66. proto.document = [self encodedDocument:existingDocument];
  67. }
  68. proto.hasCommittedMutations = existingDocument.hasCommittedMutations;
  69. } else if ([document isKindOfClass:[FSTUnknownDocument class]]) {
  70. FSTUnknownDocument *unknownDocument = (FSTUnknownDocument *)document;
  71. proto.unknownDocument = [self encodedUnknownDocument:unknownDocument];
  72. proto.hasCommittedMutations = YES;
  73. } else {
  74. HARD_FAIL("Unknown document type %s", NSStringFromClass([document class]));
  75. }
  76. return proto;
  77. }
  78. - (FSTMaybeDocument *)decodedMaybeDocument:(FSTPBMaybeDocument *)proto {
  79. switch (proto.documentTypeOneOfCase) {
  80. case FSTPBMaybeDocument_DocumentType_OneOfCase_Document:
  81. return [self decodedDocument:proto.document
  82. withCommittedMutations:proto.hasCommittedMutations];
  83. case FSTPBMaybeDocument_DocumentType_OneOfCase_NoDocument:
  84. return [self decodedDeletedDocument:proto.noDocument
  85. withCommittedMutations:proto.hasCommittedMutations];
  86. case FSTPBMaybeDocument_DocumentType_OneOfCase_UnknownDocument:
  87. return [self decodedUnknownDocument:proto.unknownDocument];
  88. default:
  89. HARD_FAIL("Unknown MaybeDocument %s", proto);
  90. }
  91. }
  92. /**
  93. * Encodes a Document for local storage. This differs from the v1 RPC serializer for Documents in
  94. * that it preserves the updateTime, which is considered an output only value by the server.
  95. */
  96. - (GCFSDocument *)encodedDocument:(FSTDocument *)document {
  97. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  98. GCFSDocument *proto = [GCFSDocument message];
  99. proto.name = [remoteSerializer encodedDocumentKey:document.key];
  100. proto.fields = [remoteSerializer encodedFields:document.data];
  101. proto.updateTime = [remoteSerializer encodedVersion:document.version];
  102. return proto;
  103. }
  104. /** Decodes a Document proto to the equivalent model. */
  105. - (FSTDocument *)decodedDocument:(GCFSDocument *)document
  106. withCommittedMutations:(BOOL)committedMutations {
  107. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  108. ObjectValue data = [remoteSerializer decodedFields:document.fields];
  109. DocumentKey key = [remoteSerializer decodedDocumentKey:document.name];
  110. SnapshotVersion version = [remoteSerializer decodedVersion:document.updateTime];
  111. return [FSTDocument documentWithData:std::move(data)
  112. key:std::move(key)
  113. version:version
  114. state:committedMutations ? DocumentState::kCommittedMutations
  115. : DocumentState::kSynced];
  116. }
  117. /** Encodes a NoDocument value to the equivalent proto. */
  118. - (FSTPBNoDocument *)encodedDeletedDocument:(FSTDeletedDocument *)document {
  119. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  120. FSTPBNoDocument *proto = [FSTPBNoDocument message];
  121. proto.name = [remoteSerializer encodedDocumentKey:document.key];
  122. proto.readTime = [remoteSerializer encodedVersion:document.version];
  123. return proto;
  124. }
  125. /** Decodes a NoDocument proto to the equivalent model. */
  126. - (FSTDeletedDocument *)decodedDeletedDocument:(FSTPBNoDocument *)proto
  127. withCommittedMutations:(BOOL)committedMutations {
  128. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  129. DocumentKey key = [remoteSerializer decodedDocumentKey:proto.name];
  130. SnapshotVersion version = [remoteSerializer decodedVersion:proto.readTime];
  131. return [FSTDeletedDocument documentWithKey:key
  132. version:version
  133. hasCommittedMutations:committedMutations];
  134. }
  135. /** Encodes an UnknownDocument value to the equivalent proto. */
  136. - (FSTPBUnknownDocument *)encodedUnknownDocument:(FSTUnknownDocument *)document {
  137. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  138. FSTPBUnknownDocument *proto = [FSTPBUnknownDocument message];
  139. proto.name = [remoteSerializer encodedDocumentKey:document.key];
  140. proto.version = [remoteSerializer encodedVersion:document.version];
  141. return proto;
  142. }
  143. /** Decodes an UnknownDocument proto to the equivalent model. */
  144. - (FSTUnknownDocument *)decodedUnknownDocument:(FSTPBUnknownDocument *)proto {
  145. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  146. DocumentKey key = [remoteSerializer decodedDocumentKey:proto.name];
  147. SnapshotVersion version = [remoteSerializer decodedVersion:proto.version];
  148. return [FSTUnknownDocument documentWithKey:key version:version];
  149. }
  150. - (FSTPBWriteBatch *)encodedMutationBatch:(FSTMutationBatch *)batch {
  151. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  152. FSTPBWriteBatch *proto = [FSTPBWriteBatch message];
  153. proto.batchId = batch.batchID;
  154. proto.localWriteTime = [remoteSerializer encodedTimestamp:batch.localWriteTime];
  155. NSMutableArray<GCFSWrite *> *baseWrites = proto.baseWritesArray;
  156. for (FSTMutation *baseMutation : [batch baseMutations]) {
  157. [baseWrites addObject:[remoteSerializer encodedMutation:baseMutation]];
  158. }
  159. NSMutableArray<GCFSWrite *> *writes = proto.writesArray;
  160. for (FSTMutation *mutation : [batch mutations]) {
  161. [writes addObject:[remoteSerializer encodedMutation:mutation]];
  162. }
  163. return proto;
  164. }
  165. - (FSTMutationBatch *)decodedMutationBatch:(FSTPBWriteBatch *)batch {
  166. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  167. int batchID = batch.batchId;
  168. std::vector<FSTMutation *> baseMutations;
  169. for (GCFSWrite *write in batch.baseWritesArray) {
  170. baseMutations.push_back([remoteSerializer decodedMutation:write]);
  171. }
  172. std::vector<FSTMutation *> mutations;
  173. for (GCFSWrite *write in batch.writesArray) {
  174. mutations.push_back([remoteSerializer decodedMutation:write]);
  175. }
  176. Timestamp localWriteTime = [remoteSerializer decodedTimestamp:batch.localWriteTime];
  177. return [[FSTMutationBatch alloc] initWithBatchID:batchID
  178. localWriteTime:localWriteTime
  179. baseMutations:std::move(baseMutations)
  180. mutations:std::move(mutations)];
  181. }
  182. - (FSTPBTarget *)encodedQueryData:(FSTQueryData *)queryData {
  183. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  184. HARD_ASSERT(queryData.purpose == FSTQueryPurposeListen,
  185. "only queries with purpose %s may be stored, got %s", FSTQueryPurposeListen,
  186. queryData.purpose);
  187. FSTPBTarget *proto = [FSTPBTarget message];
  188. proto.targetId = queryData.targetID;
  189. proto.lastListenSequenceNumber = queryData.sequenceNumber;
  190. proto.snapshotVersion = [remoteSerializer encodedVersion:queryData.snapshotVersion];
  191. proto.resumeToken = queryData.resumeToken;
  192. const Query &query = queryData.query;
  193. if (query.IsDocumentQuery()) {
  194. proto.documents = [remoteSerializer encodedDocumentsTarget:query];
  195. } else {
  196. proto.query = [remoteSerializer encodedQueryTarget:query];
  197. }
  198. return proto;
  199. }
  200. - (FSTQueryData *)decodedQueryData:(FSTPBTarget *)target {
  201. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  202. TargetId targetID = target.targetId;
  203. ListenSequenceNumber sequenceNumber = target.lastListenSequenceNumber;
  204. SnapshotVersion version = [remoteSerializer decodedVersion:target.snapshotVersion];
  205. NSData *resumeToken = target.resumeToken;
  206. Query query;
  207. switch (target.targetTypeOneOfCase) {
  208. case FSTPBTarget_TargetType_OneOfCase_Documents:
  209. query = [remoteSerializer decodedQueryFromDocumentsTarget:target.documents];
  210. break;
  211. case FSTPBTarget_TargetType_OneOfCase_Query:
  212. query = [remoteSerializer decodedQueryFromQueryTarget:target.query];
  213. break;
  214. default:
  215. HARD_FAIL("Unknown Target.targetType %s", target.targetTypeOneOfCase);
  216. }
  217. return [[FSTQueryData alloc] initWithQuery:std::move(query)
  218. targetID:targetID
  219. listenSequenceNumber:sequenceNumber
  220. purpose:FSTQueryPurposeListen
  221. snapshotVersion:version
  222. resumeToken:resumeToken];
  223. }
  224. - (GPBTimestamp *)encodedVersion:(const SnapshotVersion &)version {
  225. return [self.remoteSerializer encodedVersion:version];
  226. }
  227. - (SnapshotVersion)decodedVersion:(GPBTimestamp *)version {
  228. return [self.remoteSerializer decodedVersion:version];
  229. }
  230. @end