FSTLocalSerializer.mm 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  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. #import "Firestore/Protos/objc/firestore/local/MaybeDocument.pbobjc.h"
  19. #import "Firestore/Protos/objc/firestore/local/Mutation.pbobjc.h"
  20. #import "Firestore/Protos/objc/firestore/local/Target.pbobjc.h"
  21. #import "Firestore/Protos/objc/google/firestore/v1beta1/Document.pbobjc.h"
  22. #import "Firestore/Source/Core/FSTQuery.h"
  23. #import "Firestore/Source/Local/FSTQueryData.h"
  24. #import "Firestore/Source/Model/FSTDocument.h"
  25. #import "Firestore/Source/Model/FSTFieldValue.h"
  26. #import "Firestore/Source/Model/FSTMutationBatch.h"
  27. #import "Firestore/Source/Remote/FSTSerializerBeta.h"
  28. #import "Firestore/Source/Util/FSTAssert.h"
  29. #include "Firestore/core/src/firebase/firestore/model/document_key.h"
  30. using firebase::firestore::model::DocumentKey;
  31. @interface FSTLocalSerializer ()
  32. @property(nonatomic, strong, readonly) FSTSerializerBeta *remoteSerializer;
  33. @end
  34. /** Serializer for values stored in the LocalStore. */
  35. @implementation FSTLocalSerializer
  36. - (instancetype)initWithRemoteSerializer:(FSTSerializerBeta *)remoteSerializer {
  37. self = [super init];
  38. if (self) {
  39. _remoteSerializer = remoteSerializer;
  40. }
  41. return self;
  42. }
  43. - (FSTPBMaybeDocument *)encodedMaybeDocument:(FSTMaybeDocument *)document {
  44. FSTPBMaybeDocument *proto = [FSTPBMaybeDocument message];
  45. if ([document isKindOfClass:[FSTDeletedDocument class]]) {
  46. proto.noDocument = [self encodedDeletedDocument:(FSTDeletedDocument *)document];
  47. } else if ([document isKindOfClass:[FSTDocument class]]) {
  48. proto.document = [self encodedDocument:(FSTDocument *)document];
  49. } else {
  50. FSTFail(@"Unknown document type %@", NSStringFromClass([document class]));
  51. }
  52. return proto;
  53. }
  54. - (FSTMaybeDocument *)decodedMaybeDocument:(FSTPBMaybeDocument *)proto {
  55. switch (proto.documentTypeOneOfCase) {
  56. case FSTPBMaybeDocument_DocumentType_OneOfCase_Document:
  57. return [self decodedDocument:proto.document];
  58. case FSTPBMaybeDocument_DocumentType_OneOfCase_NoDocument:
  59. return [self decodedDeletedDocument:proto.noDocument];
  60. default:
  61. FSTFail(@"Unknown MaybeDocument %@", proto);
  62. }
  63. }
  64. /**
  65. * Encodes a Document for local storage. This differs from the v1beta1 RPC serializer for
  66. * Documents in that it preserves the updateTime, which is considered an output only value by the
  67. * server.
  68. */
  69. - (GCFSDocument *)encodedDocument:(FSTDocument *)document {
  70. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  71. GCFSDocument *proto = [GCFSDocument message];
  72. proto.name = [remoteSerializer encodedDocumentKey:document.key];
  73. proto.fields = [remoteSerializer encodedFields:document.data];
  74. proto.updateTime = [remoteSerializer encodedVersion:document.version];
  75. return proto;
  76. }
  77. /** Decodes a Document proto to the equivalent model. */
  78. - (FSTDocument *)decodedDocument:(GCFSDocument *)document {
  79. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  80. FSTObjectValue *data = [remoteSerializer decodedFields:document.fields];
  81. const DocumentKey key = [remoteSerializer decodedDocumentKey:document.name];
  82. FSTSnapshotVersion *version = [remoteSerializer decodedVersion:document.updateTime];
  83. return [FSTDocument documentWithData:data key:key version:version hasLocalMutations:NO];
  84. }
  85. /** Encodes a NoDocument value to the equivalent proto. */
  86. - (FSTPBNoDocument *)encodedDeletedDocument:(FSTDeletedDocument *)document {
  87. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  88. FSTPBNoDocument *proto = [FSTPBNoDocument message];
  89. proto.name = [remoteSerializer encodedDocumentKey:document.key];
  90. proto.readTime = [remoteSerializer encodedVersion:document.version];
  91. return proto;
  92. }
  93. /** Decodes a NoDocument proto to the equivalent model. */
  94. - (FSTDeletedDocument *)decodedDeletedDocument:(FSTPBNoDocument *)proto {
  95. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  96. const DocumentKey key = [remoteSerializer decodedDocumentKey:proto.name];
  97. FSTSnapshotVersion *version = [remoteSerializer decodedVersion:proto.readTime];
  98. return [FSTDeletedDocument documentWithKey:key version:version];
  99. }
  100. - (FSTPBWriteBatch *)encodedMutationBatch:(FSTMutationBatch *)batch {
  101. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  102. FSTPBWriteBatch *proto = [FSTPBWriteBatch message];
  103. proto.batchId = batch.batchID;
  104. proto.localWriteTime = [remoteSerializer encodedTimestamp:batch.localWriteTime];
  105. NSMutableArray<GCFSWrite *> *writes = proto.writesArray;
  106. for (FSTMutation *mutation in batch.mutations) {
  107. [writes addObject:[remoteSerializer encodedMutation:mutation]];
  108. }
  109. return proto;
  110. }
  111. - (FSTMutationBatch *)decodedMutationBatch:(FSTPBWriteBatch *)batch {
  112. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  113. int batchID = batch.batchId;
  114. NSMutableArray<FSTMutation *> *mutations = [NSMutableArray array];
  115. for (GCFSWrite *write in batch.writesArray) {
  116. [mutations addObject:[remoteSerializer decodedMutation:write]];
  117. }
  118. FIRTimestamp *localWriteTime = [remoteSerializer decodedTimestamp:batch.localWriteTime];
  119. return [[FSTMutationBatch alloc] initWithBatchID:batchID
  120. localWriteTime:localWriteTime
  121. mutations:mutations];
  122. }
  123. - (FSTPBTarget *)encodedQueryData:(FSTQueryData *)queryData {
  124. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  125. FSTAssert(queryData.purpose == FSTQueryPurposeListen,
  126. @"only queries with purpose %lu may be stored, got %lu",
  127. (unsigned long)FSTQueryPurposeListen, (unsigned long)queryData.purpose);
  128. FSTPBTarget *proto = [FSTPBTarget message];
  129. proto.targetId = queryData.targetID;
  130. proto.lastListenSequenceNumber = queryData.sequenceNumber;
  131. proto.snapshotVersion = [remoteSerializer encodedVersion:queryData.snapshotVersion];
  132. proto.resumeToken = queryData.resumeToken;
  133. FSTQuery *query = queryData.query;
  134. if ([query isDocumentQuery]) {
  135. proto.documents = [remoteSerializer encodedDocumentsTarget:query];
  136. } else {
  137. proto.query = [remoteSerializer encodedQueryTarget:query];
  138. }
  139. return proto;
  140. }
  141. - (FSTQueryData *)decodedQueryData:(FSTPBTarget *)target {
  142. FSTSerializerBeta *remoteSerializer = self.remoteSerializer;
  143. FSTTargetID targetID = target.targetId;
  144. FSTListenSequenceNumber sequenceNumber = target.lastListenSequenceNumber;
  145. FSTSnapshotVersion *version = [remoteSerializer decodedVersion:target.snapshotVersion];
  146. NSData *resumeToken = target.resumeToken;
  147. FSTQuery *query;
  148. switch (target.targetTypeOneOfCase) {
  149. case FSTPBTarget_TargetType_OneOfCase_Documents:
  150. query = [remoteSerializer decodedQueryFromDocumentsTarget:target.documents];
  151. break;
  152. case FSTPBTarget_TargetType_OneOfCase_Query:
  153. query = [remoteSerializer decodedQueryFromQueryTarget:target.query];
  154. break;
  155. default:
  156. FSTFail(@"Unknown Target.targetType %" PRId32, target.targetTypeOneOfCase);
  157. }
  158. return [[FSTQueryData alloc] initWithQuery:query
  159. targetID:targetID
  160. listenSequenceNumber:sequenceNumber
  161. purpose:FSTQueryPurposeListen
  162. snapshotVersion:version
  163. resumeToken:resumeToken];
  164. }
  165. - (GPBTimestamp *)encodedVersion:(FSTSnapshotVersion *)version {
  166. return [self.remoteSerializer encodedVersion:version];
  167. }
  168. - (FSTSnapshotVersion *)decodedVersion:(GPBTimestamp *)version {
  169. return [self.remoteSerializer decodedVersion:version];
  170. }
  171. @end