FSTLocalSerializer.mm 11 KB

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