| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257 |
- /*
- * Copyright 2017 Google
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- #import "Firestore/Source/Remote/FSTSerializerBeta.h"
- #include <algorithm>
- #include <cinttypes>
- #include <set>
- #include <string>
- #include <utility>
- #include <vector>
- #import "Firestore/Protos/objc/google/firestore/v1/Common.pbobjc.h"
- #import "Firestore/Protos/objc/google/firestore/v1/Document.pbobjc.h"
- #import "Firestore/Protos/objc/google/firestore/v1/Firestore.pbobjc.h"
- #import "Firestore/Protos/objc/google/firestore/v1/Query.pbobjc.h"
- #import "Firestore/Protos/objc/google/firestore/v1/Write.pbobjc.h"
- #import "Firestore/Protos/objc/google/rpc/Status.pbobjc.h"
- #import "Firestore/Protos/objc/google/type/Latlng.pbobjc.h"
- #import "FIRFirestoreErrors.h"
- #import "FIRGeoPoint.h"
- #import "FIRTimestamp.h"
- #include "Firestore/core/include/firebase/firestore/firestore_errors.h"
- #include "Firestore/core/include/firebase/firestore/geo_point.h"
- #include "Firestore/core/src/firebase/firestore/core/bound.h"
- #include "Firestore/core/src/firebase/firestore/core/field_filter.h"
- #include "Firestore/core/src/firebase/firestore/core/filter.h"
- #include "Firestore/core/src/firebase/firestore/core/query.h"
- #include "Firestore/core/src/firebase/firestore/local/query_data.h"
- #include "Firestore/core/src/firebase/firestore/model/database_id.h"
- #include "Firestore/core/src/firebase/firestore/model/delete_mutation.h"
- #include "Firestore/core/src/firebase/firestore/model/document.h"
- #include "Firestore/core/src/firebase/firestore/model/document_key.h"
- #include "Firestore/core/src/firebase/firestore/model/field_mask.h"
- #include "Firestore/core/src/firebase/firestore/model/field_path.h"
- #include "Firestore/core/src/firebase/firestore/model/field_transform.h"
- #include "Firestore/core/src/firebase/firestore/model/field_value.h"
- #include "Firestore/core/src/firebase/firestore/model/maybe_document.h"
- #include "Firestore/core/src/firebase/firestore/model/no_document.h"
- #include "Firestore/core/src/firebase/firestore/model/patch_mutation.h"
- #include "Firestore/core/src/firebase/firestore/model/precondition.h"
- #include "Firestore/core/src/firebase/firestore/model/resource_path.h"
- #include "Firestore/core/src/firebase/firestore/model/set_mutation.h"
- #include "Firestore/core/src/firebase/firestore/model/transform_mutation.h"
- #include "Firestore/core/src/firebase/firestore/model/transform_operation.h"
- #include "Firestore/core/src/firebase/firestore/model/unknown_document.h"
- #include "Firestore/core/src/firebase/firestore/nanopb/byte_string.h"
- #include "Firestore/core/src/firebase/firestore/nanopb/nanopb_util.h"
- #include "Firestore/core/src/firebase/firestore/remote/existence_filter.h"
- #include "Firestore/core/src/firebase/firestore/remote/watch_change.h"
- #include "Firestore/core/src/firebase/firestore/util/hard_assert.h"
- #include "Firestore/core/src/firebase/firestore/util/status.h"
- #include "Firestore/core/src/firebase/firestore/util/string_apple.h"
- #include "absl/memory/memory.h"
- #include "absl/types/optional.h"
- namespace util = firebase::firestore::util;
- using firebase::Timestamp;
- using firebase::firestore::Error;
- using firebase::firestore::GeoPoint;
- using firebase::firestore::core::Bound;
- using firebase::firestore::core::CollectionGroupId;
- using firebase::firestore::core::Direction;
- using firebase::firestore::core::FieldFilter;
- using firebase::firestore::core::Filter;
- using firebase::firestore::core::FilterList;
- using firebase::firestore::core::OrderBy;
- using firebase::firestore::core::OrderByList;
- using firebase::firestore::core::Query;
- using firebase::firestore::local::QueryData;
- using firebase::firestore::local::QueryPurpose;
- using firebase::firestore::model::ArrayTransform;
- using firebase::firestore::model::DatabaseId;
- using firebase::firestore::model::DeleteMutation;
- using firebase::firestore::model::Document;
- using firebase::firestore::model::DocumentKey;
- using firebase::firestore::model::DocumentState;
- using firebase::firestore::model::FieldMask;
- using firebase::firestore::model::FieldPath;
- using firebase::firestore::model::FieldTransform;
- using firebase::firestore::model::FieldValue;
- using firebase::firestore::model::MaybeDocument;
- using firebase::firestore::model::Mutation;
- using firebase::firestore::model::MutationResult;
- using firebase::firestore::model::NoDocument;
- using firebase::firestore::model::NumericIncrementTransform;
- using firebase::firestore::model::ObjectValue;
- using firebase::firestore::model::PatchMutation;
- using firebase::firestore::model::Precondition;
- using firebase::firestore::model::ResourcePath;
- using firebase::firestore::model::ServerTimestampTransform;
- using firebase::firestore::model::SetMutation;
- using firebase::firestore::model::SnapshotVersion;
- using firebase::firestore::model::TargetId;
- using firebase::firestore::model::TransformMutation;
- using firebase::firestore::model::TransformOperation;
- using firebase::firestore::model::UnknownDocument;
- using firebase::firestore::nanopb::ByteString;
- using firebase::firestore::nanopb::MakeByteString;
- using firebase::firestore::nanopb::MakeNSData;
- using firebase::firestore::remote::DocumentWatchChange;
- using firebase::firestore::remote::ExistenceFilter;
- using firebase::firestore::remote::ExistenceFilterWatchChange;
- using firebase::firestore::remote::WatchChange;
- using firebase::firestore::remote::WatchTargetChange;
- using firebase::firestore::remote::WatchTargetChangeState;
- using Operator = Filter::Operator;
- NS_ASSUME_NONNULL_BEGIN
- namespace {
- absl::any Wrap(GCFSDocument *doc) {
- return absl::make_any<GCFSDocument *>(doc);
- }
- } // namespace
- @implementation FSTSerializerBeta {
- DatabaseId _databaseID;
- }
- - (instancetype)initWithDatabaseID:(DatabaseId)databaseID {
- self = [super init];
- if (self) {
- _databaseID = std::move(databaseID);
- }
- return self;
- }
- #pragma mark - SnapshotVersion <=> GPBTimestamp
- - (GPBTimestamp *)encodedTimestamp:(const Timestamp &)timestamp {
- GPBTimestamp *result = [GPBTimestamp message];
- result.seconds = timestamp.seconds();
- result.nanos = timestamp.nanoseconds();
- return result;
- }
- - (Timestamp)decodedTimestamp:(GPBTimestamp *)timestamp {
- return Timestamp{timestamp.seconds, timestamp.nanos};
- }
- - (GPBTimestamp *)encodedVersion:(const SnapshotVersion &)version {
- return [self encodedTimestamp:version.timestamp()];
- }
- - (SnapshotVersion)decodedVersion:(GPBTimestamp *)version {
- return SnapshotVersion{[self decodedTimestamp:version]};
- }
- #pragma mark - FIRGeoPoint <=> GTPLatLng
- - (GTPLatLng *)encodedGeoPoint:(const GeoPoint &)geoPoint {
- GTPLatLng *latLng = [GTPLatLng message];
- latLng.latitude = geoPoint.latitude();
- latLng.longitude = geoPoint.longitude();
- return latLng;
- }
- - (GeoPoint)decodedGeoPoint:(GTPLatLng *)latLng {
- return GeoPoint(latLng.latitude, latLng.longitude);
- }
- #pragma mark - DocumentKey <=> Key proto
- - (NSString *)encodedDocumentKey:(const DocumentKey &)key {
- return [self encodedResourcePathForDatabaseID:_databaseID path:key.path()];
- }
- - (DocumentKey)decodedDocumentKey:(NSString *)name {
- const ResourcePath path = [self decodedResourcePathWithDatabaseID:name];
- HARD_ASSERT(path[1] == _databaseID.project_id(),
- "Tried to deserialize key from different project.");
- HARD_ASSERT(path[3] == _databaseID.database_id(),
- "Tried to deserialize key from different datbase.");
- return DocumentKey{[self localResourcePathForQualifiedResourcePath:path]};
- }
- - (NSString *)encodedResourcePathForDatabaseID:(const DatabaseId &)databaseID
- path:(const ResourcePath &)path {
- return util::MakeNSString([self encodedResourcePathForDatabaseID:databaseID]
- .Append("documents")
- .Append(path)
- .CanonicalString());
- }
- - (ResourcePath)decodedResourcePathWithDatabaseID:(NSString *)name {
- const ResourcePath path = ResourcePath::FromString(util::MakeString(name));
- HARD_ASSERT([self validQualifiedResourcePath:path], "Tried to deserialize invalid key %s",
- path.CanonicalString());
- return path;
- }
- - (NSString *)encodedQueryPath:(const ResourcePath &)path {
- return [self encodedResourcePathForDatabaseID:_databaseID path:path];
- }
- - (ResourcePath)decodedQueryPath:(NSString *)name {
- const ResourcePath resource = [self decodedResourcePathWithDatabaseID:name];
- if (resource.size() == 4) {
- // In v1beta1 queries for collections at the root did not have a trailing "/documents". In v1
- // all resource paths contain "/documents". Preserve the ability to read the v1beta1 form for
- // compatibility with queries persisted in the local query cache.
- return ResourcePath{};
- } else {
- return [self localResourcePathForQualifiedResourcePath:resource];
- }
- }
- - (ResourcePath)encodedResourcePathForDatabaseID:(const DatabaseId &)databaseID {
- return ResourcePath{"projects", databaseID.project_id(), "databases", databaseID.database_id()};
- }
- - (ResourcePath)localResourcePathForQualifiedResourcePath:(const ResourcePath &)resourceName {
- HARD_ASSERT(resourceName.size() > 4 && resourceName[4] == "documents",
- "Tried to deserialize invalid key %s", resourceName.CanonicalString());
- return resourceName.PopFirst(5);
- }
- - (BOOL)validQualifiedResourcePath:(const ResourcePath &)path {
- return path.size() >= 4 && path[0] == "projects" && path[2] == "databases";
- }
- - (NSString *)encodedDatabaseID {
- return util::MakeNSString([self encodedResourcePathForDatabaseID:_databaseID].CanonicalString());
- }
- #pragma mark - FieldValue <=> Value proto
- - (GCFSValue *)encodedFieldValue:(const FieldValue &)fieldValue {
- switch (fieldValue.type()) {
- case FieldValue::Type::Null:
- return [self encodedNull];
- case FieldValue::Type::Boolean:
- return [self encodedBool:fieldValue.boolean_value()];
- case FieldValue::Type::Integer:
- return [self encodedInteger:fieldValue.integer_value()];
- case FieldValue::Type::Double:
- return [self encodedDouble:fieldValue.double_value()];
- case FieldValue::Type::Timestamp:
- return [self encodedTimestampValue:fieldValue.timestamp_value()];
- case FieldValue::Type::String:
- return [self encodedString:fieldValue.string_value()];
- case FieldValue::Type::Blob:
- return [self encodedBlobValue:fieldValue.blob_value()];
- case FieldValue::Type::Reference: {
- const auto &ref = fieldValue.reference_value();
- return [self encodedReferenceValueForDatabaseID:ref.database_id() key:ref.key()];
- }
- case FieldValue::Type::GeoPoint:
- return [self encodedGeoPointValue:fieldValue.geo_point_value()];
- case FieldValue::Type::Array: {
- GCFSValue *result = [GCFSValue message];
- result.arrayValue = [self encodedArrayValue:fieldValue.array_value()];
- return result;
- }
- case FieldValue::Type::Object: {
- GCFSValue *result = [GCFSValue message];
- result.mapValue = [self encodedMapValue:fieldValue.object_value()];
- return result;
- }
- case FieldValue::Type::ServerTimestamp:
- HARD_FAIL("Unhandled type %s on %s", fieldValue.type(), fieldValue.ToString());
- }
- UNREACHABLE();
- }
- - (FieldValue)decodedFieldValue:(GCFSValue *)valueProto {
- switch (valueProto.valueTypeOneOfCase) {
- case GCFSValue_ValueType_OneOfCase_NullValue:
- return FieldValue::Null();
- case GCFSValue_ValueType_OneOfCase_BooleanValue:
- return FieldValue::FromBoolean(valueProto.booleanValue);
- case GCFSValue_ValueType_OneOfCase_IntegerValue:
- return FieldValue::FromInteger(valueProto.integerValue);
- case GCFSValue_ValueType_OneOfCase_DoubleValue:
- return FieldValue::FromDouble(valueProto.doubleValue);
- case GCFSValue_ValueType_OneOfCase_StringValue:
- return FieldValue::FromString(util::MakeString(valueProto.stringValue));
- case GCFSValue_ValueType_OneOfCase_TimestampValue: {
- Timestamp value = [self decodedTimestamp:valueProto.timestampValue];
- return FieldValue::FromTimestamp(value);
- }
- case GCFSValue_ValueType_OneOfCase_GeoPointValue:
- return FieldValue::FromGeoPoint([self decodedGeoPoint:valueProto.geoPointValue]);
- case GCFSValue_ValueType_OneOfCase_BytesValue:
- return FieldValue::FromBlob(MakeByteString(valueProto.bytesValue));
- case GCFSValue_ValueType_OneOfCase_ReferenceValue:
- return [self decodedReferenceValue:valueProto.referenceValue];
- case GCFSValue_ValueType_OneOfCase_ArrayValue:
- return [self decodedArrayValue:valueProto.arrayValue];
- case GCFSValue_ValueType_OneOfCase_MapValue:
- return [self decodedMapValue:valueProto.mapValue];
- default:
- HARD_FAIL("Unhandled type %s on %s", valueProto.valueTypeOneOfCase, valueProto);
- }
- }
- - (GCFSValue *)encodedNull {
- GCFSValue *result = [GCFSValue message];
- result.nullValue = GPBNullValue_NullValue;
- return result;
- }
- - (GCFSValue *)encodedBool:(bool)value {
- GCFSValue *result = [GCFSValue message];
- result.booleanValue = value;
- return result;
- }
- - (GCFSValue *)encodedDouble:(double)value {
- GCFSValue *result = [GCFSValue message];
- result.doubleValue = value;
- return result;
- }
- - (GCFSValue *)encodedInteger:(int64_t)value {
- GCFSValue *result = [GCFSValue message];
- result.integerValue = value;
- return result;
- }
- - (GCFSValue *)encodedString:(absl::string_view)value {
- GCFSValue *result = [GCFSValue message];
- result.stringValue = util::MakeNSString(value);
- return result;
- }
- - (GCFSValue *)encodedTimestampValue:(const Timestamp &)value {
- GCFSValue *result = [GCFSValue message];
- result.timestampValue = [self encodedTimestamp:value];
- return result;
- }
- - (GCFSValue *)encodedGeoPointValue:(const GeoPoint &)value {
- GCFSValue *result = [GCFSValue message];
- result.geoPointValue = [self encodedGeoPoint:value];
- return result;
- }
- - (GCFSValue *)encodedBlobValue:(const ByteString &)value {
- GCFSValue *result = [GCFSValue message];
- result.bytesValue = MakeNSData(value);
- return result;
- }
- - (GCFSValue *)encodedReferenceValueForDatabaseID:(const DatabaseId &)databaseID
- key:(const DocumentKey &)key {
- HARD_ASSERT(databaseID == _databaseID, "Database %s:%s cannot encode reference from %s:%s",
- _databaseID.project_id(), _databaseID.database_id(), databaseID.project_id(),
- databaseID.database_id());
- GCFSValue *result = [GCFSValue message];
- result.referenceValue = [self encodedResourcePathForDatabaseID:databaseID path:key.path()];
- return result;
- }
- - (FieldValue)decodedReferenceValue:(NSString *)resourceName {
- const ResourcePath path = [self decodedResourcePathWithDatabaseID:resourceName];
- const std::string &project = path[1];
- const std::string &database = path[3];
- const DocumentKey key{[self localResourcePathForQualifiedResourcePath:path]};
- const DatabaseId database_id(project, database);
- HARD_ASSERT(database_id == _databaseID, "Database %s:%s cannot encode reference from %s:%s",
- _databaseID.project_id(), _databaseID.database_id(), database_id.project_id(),
- database_id.database_id());
- return FieldValue::FromReference(_databaseID, key);
- }
- - (GCFSArrayValue *)encodedArrayValue:(const FieldValue::Array &)arrayValue {
- GCFSArrayValue *proto = [GCFSArrayValue message];
- NSMutableArray<GCFSValue *> *protoContents = [proto valuesArray];
- for (const FieldValue &value : arrayValue) {
- GCFSValue *converted = [self encodedFieldValue:value];
- [protoContents addObject:converted];
- }
- return proto;
- }
- - (FieldValue)decodedArrayValue:(GCFSArrayValue *)arrayValue {
- FieldValue::Array contents;
- contents.reserve(arrayValue.valuesArray_Count);
- for (GCFSValue *value in arrayValue.valuesArray) {
- contents.push_back([self decodedFieldValue:value]);
- }
- return FieldValue::FromArray(std::move(contents));
- }
- - (GCFSMapValue *)encodedMapValue:(const FieldValue::Map &)value {
- GCFSMapValue *result = [GCFSMapValue message];
- result.fields = [self encodedMapFields:value];
- return result;
- }
- - (ObjectValue)decodedMapValue:(GCFSMapValue *)map {
- return [self decodedFields:map.fields];
- }
- /**
- * Encodes an ObjectValue into a dictionary.
- * @return a new dictionary that can be assigned to a field in another proto.
- */
- - (NSMutableDictionary<NSString *, GCFSValue *> *)encodedFields:(const ObjectValue &)value {
- return [self encodedMapFields:value.GetInternalValue()];
- }
- - (NSMutableDictionary<NSString *, GCFSValue *> *)encodedMapFields:(const FieldValue::Map &)value {
- NSMutableDictionary<NSString *, GCFSValue *> *result = [NSMutableDictionary dictionary];
- for (const auto &kv : value) {
- NSString *key = util::MakeNSString(kv.first);
- GCFSValue *converted = [self encodedFieldValue:kv.second];
- result[key] = converted;
- }
- return result;
- }
- - (ObjectValue)decodedFields:(NSDictionary<NSString *, GCFSValue *> *)fields {
- __block ObjectValue result = ObjectValue::Empty();
- [fields enumerateKeysAndObjectsUsingBlock:^(NSString *_Nonnull key, GCFSValue *_Nonnull obj,
- BOOL *_Nonnull stop) {
- FieldPath path{util::MakeString(key)};
- FieldValue value = [self decodedFieldValue:obj];
- result = result.Set(path, std::move(value));
- }];
- return result;
- }
- #pragma mark - ObjectValue <=> Document proto
- - (GCFSDocument *)encodedDocumentWithFields:(const ObjectValue &)objectValue
- key:(const DocumentKey &)key {
- GCFSDocument *proto = [GCFSDocument message];
- proto.name = [self encodedDocumentKey:key];
- proto.fields = [self encodedFields:objectValue];
- return proto;
- }
- #pragma mark - MaybeDocument <= BatchGetDocumentsResponse proto
- - (MaybeDocument)decodedMaybeDocumentFromBatch:(GCFSBatchGetDocumentsResponse *)response {
- switch (response.resultOneOfCase) {
- case GCFSBatchGetDocumentsResponse_Result_OneOfCase_Found:
- return [self decodedFoundDocument:response];
- case GCFSBatchGetDocumentsResponse_Result_OneOfCase_Missing:
- return [self decodedDeletedDocument:response];
- default:
- HARD_FAIL("Unknown document type: %s", response);
- }
- }
- - (Document)decodedFoundDocument:(GCFSBatchGetDocumentsResponse *)response {
- HARD_ASSERT(!!response.found, "Tried to deserialize a found document from a deleted document.");
- DocumentKey key = [self decodedDocumentKey:response.found.name];
- ObjectValue value = [self decodedFields:response.found.fields];
- SnapshotVersion version = [self decodedVersion:response.found.updateTime];
- HARD_ASSERT(version != SnapshotVersion::None(),
- "Got a document response with no snapshot version");
- return Document(std::move(value), std::move(key), version, DocumentState::kSynced,
- Wrap(response.found));
- }
- - (NoDocument)decodedDeletedDocument:(GCFSBatchGetDocumentsResponse *)response {
- HARD_ASSERT(!!response.missing, "Tried to deserialize a deleted document from a found document.");
- DocumentKey key = [self decodedDocumentKey:response.missing];
- SnapshotVersion version = [self decodedVersion:response.readTime];
- HARD_ASSERT(version != SnapshotVersion::None(),
- "Got a no document response with no snapshot version");
- return NoDocument(std::move(key), version, /* has_commited_mutations= */ false);
- }
- #pragma mark - Mutation => GCFSWrite proto
- - (GCFSWrite *)encodedMutation:(const Mutation &)mutation {
- using Type = Mutation::Type;
- GCFSWrite *proto = [GCFSWrite message];
- if (!mutation.precondition().is_none()) {
- proto.currentDocument = [self encodedPrecondition:mutation.precondition()];
- }
- switch (mutation.type()) {
- case Type::Set: {
- SetMutation set(mutation);
- proto.update = [self encodedDocumentWithFields:set.value() key:set.key()];
- return proto;
- }
- case Type::Patch: {
- PatchMutation patch(mutation);
- proto.update = [self encodedDocumentWithFields:patch.value() key:patch.key()];
- proto.updateMask = [self encodedFieldMask:patch.mask()];
- return proto;
- }
- case Type::Transform: {
- TransformMutation transform(mutation);
- proto.transform = [GCFSDocumentTransform message];
- proto.transform.document = [self encodedDocumentKey:transform.key()];
- proto.transform.fieldTransformsArray =
- [self encodedFieldTransforms:transform.field_transforms()];
- // NOTE: We set a precondition of exists: true as a safety-check, since we always combine
- // TransformMutations with a SetMutation or PatchMutation which (if successful) should
- // end up with an existing document.
- proto.currentDocument.exists = YES;
- return proto;
- }
- case Type::Delete: {
- DeleteMutation deleteMutation(mutation);
- proto.delete_p = [self encodedDocumentKey:deleteMutation.key()];
- return proto;
- }
- }
- UNREACHABLE();
- }
- - (Mutation)decodedMutation:(GCFSWrite *)mutation {
- Precondition precondition = [mutation hasCurrentDocument]
- ? [self decodedPrecondition:mutation.currentDocument]
- : Precondition::None();
- switch (mutation.operationOneOfCase) {
- case GCFSWrite_Operation_OneOfCase_Update:
- if (mutation.hasUpdateMask) {
- return PatchMutation([self decodedDocumentKey:mutation.update.name],
- [self decodedFields:mutation.update.fields],
- [self decodedFieldMask:mutation.updateMask], precondition);
- } else {
- return SetMutation([self decodedDocumentKey:mutation.update.name],
- [self decodedFields:mutation.update.fields], precondition);
- }
- case GCFSWrite_Operation_OneOfCase_Delete_p:
- return DeleteMutation([self decodedDocumentKey:mutation.delete_p], precondition);
- case GCFSWrite_Operation_OneOfCase_Transform: {
- HARD_ASSERT(precondition == Precondition::Exists(true),
- "Transforms must have precondition \"exists == true\"");
- return TransformMutation(
- [self decodedDocumentKey:mutation.transform.document],
- [self decodedFieldTransforms:mutation.transform.fieldTransformsArray]);
- }
- default:
- // Note that insert is intentionally unhandled, since we don't ever deal in them.
- HARD_FAIL("Unknown mutation operation: %s", mutation.operationOneOfCase);
- }
- }
- - (GCFSPrecondition *)encodedPrecondition:(const Precondition &)precondition {
- HARD_ASSERT(!precondition.is_none(), "Can't serialize an empty precondition");
- GCFSPrecondition *message = [GCFSPrecondition message];
- if (precondition.type() == Precondition::Type::UpdateTime) {
- message.updateTime = [self encodedVersion:precondition.update_time()];
- } else if (precondition.type() == Precondition::Type::Exists) {
- message.exists = precondition == Precondition::Exists(true);
- } else {
- HARD_FAIL("Unknown precondition: %s", precondition.ToString());
- }
- return message;
- }
- - (Precondition)decodedPrecondition:(GCFSPrecondition *)precondition {
- switch (precondition.conditionTypeOneOfCase) {
- case GCFSPrecondition_ConditionType_OneOfCase_GPBUnsetOneOfCase:
- return Precondition::None();
- case GCFSPrecondition_ConditionType_OneOfCase_Exists:
- return Precondition::Exists(precondition.exists);
- case GCFSPrecondition_ConditionType_OneOfCase_UpdateTime:
- return Precondition::UpdateTime([self decodedVersion:precondition.updateTime]);
- default:
- HARD_FAIL("Unrecognized Precondition one-of case %s", precondition);
- }
- }
- - (GCFSDocumentMask *)encodedFieldMask:(const FieldMask &)fieldMask {
- GCFSDocumentMask *mask = [GCFSDocumentMask message];
- for (const FieldPath &field : fieldMask) {
- [mask.fieldPathsArray addObject:util::MakeNSString(field.CanonicalString())];
- }
- return mask;
- }
- - (FieldMask)decodedFieldMask:(GCFSDocumentMask *)fieldMask {
- std::set<FieldPath> fields;
- for (NSString *path in fieldMask.fieldPathsArray) {
- fields.insert(FieldPath::FromServerFormat(util::MakeString(path)));
- }
- return FieldMask(std::move(fields));
- }
- - (NSMutableArray<GCFSDocumentTransform_FieldTransform *> *)encodedFieldTransforms:
- (const std::vector<FieldTransform> &)fieldTransforms {
- NSMutableArray *protos = [NSMutableArray array];
- for (const FieldTransform &fieldTransform : fieldTransforms) {
- GCFSDocumentTransform_FieldTransform *proto = [self encodedFieldTransform:fieldTransform];
- [protos addObject:proto];
- }
- return protos;
- }
- - (GCFSDocumentTransform_FieldTransform *)encodedFieldTransform:
- (const FieldTransform &)fieldTransform {
- GCFSDocumentTransform_FieldTransform *proto = [GCFSDocumentTransform_FieldTransform message];
- proto.fieldPath = util::MakeNSString(fieldTransform.path().CanonicalString());
- if (fieldTransform.transformation().type() == TransformOperation::Type::ServerTimestamp) {
- proto.setToServerValue = GCFSDocumentTransform_FieldTransform_ServerValue_RequestTime;
- } else if (fieldTransform.transformation().type() == TransformOperation::Type::ArrayUnion) {
- proto.appendMissingElements = [self
- encodedArrayTransformElements:ArrayTransform(fieldTransform.transformation()).elements()];
- } else if (fieldTransform.transformation().type() == TransformOperation::Type::ArrayRemove) {
- proto.removeAllFromArray_p = [self
- encodedArrayTransformElements:ArrayTransform(fieldTransform.transformation()).elements()];
- } else if (fieldTransform.transformation().type() == TransformOperation::Type::Increment) {
- const NumericIncrementTransform &incrementTransform =
- static_cast<const NumericIncrementTransform &>(fieldTransform.transformation());
- proto.increment = [self encodedFieldValue:incrementTransform.operand()];
- } else {
- HARD_FAIL("Unknown transform: %s type", fieldTransform.transformation().type());
- }
- return proto;
- }
- - (GCFSArrayValue *)encodedArrayTransformElements:(const std::vector<FieldValue> &)elements {
- GCFSArrayValue *proto = [GCFSArrayValue message];
- NSMutableArray<GCFSValue *> *protoContents = [proto valuesArray];
- for (const FieldValue &element : elements) {
- GCFSValue *converted = [self encodedFieldValue:element];
- [protoContents addObject:converted];
- }
- return proto;
- }
- - (std::vector<FieldTransform>)decodedFieldTransforms:
- (NSArray<GCFSDocumentTransform_FieldTransform *> *)protos {
- std::vector<FieldTransform> fieldTransforms;
- fieldTransforms.reserve(protos.count);
- for (GCFSDocumentTransform_FieldTransform *proto in protos) {
- switch (proto.transformTypeOneOfCase) {
- case GCFSDocumentTransform_FieldTransform_TransformType_OneOfCase_SetToServerValue: {
- HARD_ASSERT(
- proto.setToServerValue == GCFSDocumentTransform_FieldTransform_ServerValue_RequestTime,
- "Unknown transform setToServerValue: %s", proto.setToServerValue);
- fieldTransforms.emplace_back(FieldPath::FromServerFormat(util::MakeString(proto.fieldPath)),
- ServerTimestampTransform());
- break;
- }
- case GCFSDocumentTransform_FieldTransform_TransformType_OneOfCase_AppendMissingElements: {
- std::vector<FieldValue> elements =
- [self decodedArrayTransformElements:proto.appendMissingElements];
- fieldTransforms.emplace_back(
- FieldPath::FromServerFormat(util::MakeString(proto.fieldPath)),
- ArrayTransform(TransformOperation::Type::ArrayUnion, std::move(elements)));
- break;
- }
- case GCFSDocumentTransform_FieldTransform_TransformType_OneOfCase_RemoveAllFromArray_p: {
- std::vector<FieldValue> elements =
- [self decodedArrayTransformElements:proto.removeAllFromArray_p];
- fieldTransforms.emplace_back(
- FieldPath::FromServerFormat(util::MakeString(proto.fieldPath)),
- ArrayTransform(TransformOperation::Type::ArrayRemove, std::move(elements)));
- break;
- }
- case GCFSDocumentTransform_FieldTransform_TransformType_OneOfCase_Increment: {
- FieldValue operand = [self decodedFieldValue:proto.increment];
- fieldTransforms.emplace_back(FieldPath::FromServerFormat(util::MakeString(proto.fieldPath)),
- NumericIncrementTransform(std::move(operand)));
- break;
- }
- default:
- HARD_FAIL("Unknown transform: %s", proto);
- }
- }
- return fieldTransforms;
- }
- - (std::vector<FieldValue>)decodedArrayTransformElements:(GCFSArrayValue *)proto {
- __block std::vector<FieldValue> elements;
- [proto.valuesArray enumerateObjectsUsingBlock:^(GCFSValue *value, NSUInteger idx, BOOL *stop) {
- elements.push_back([self decodedFieldValue:value]);
- }];
- return elements;
- }
- #pragma mark - MutationResult <= GCFSWriteResult proto
- - (MutationResult)decodedMutationResult:(GCFSWriteResult *)mutation
- commitVersion:(const SnapshotVersion &)commitVersion {
- // NOTE: Deletes don't have an updateTime. Use commitVersion instead.
- SnapshotVersion version =
- mutation.hasUpdateTime ? [self decodedVersion:mutation.updateTime] : commitVersion;
- absl::optional<std::vector<FieldValue>> transformResults;
- if (mutation.transformResultsArray.count > 0) {
- transformResults = std::vector<FieldValue>{};
- for (GCFSValue *result in mutation.transformResultsArray) {
- transformResults->push_back([self decodedFieldValue:result]);
- }
- }
- return MutationResult(std::move(version), std::move(transformResults));
- }
- #pragma mark - QueryData => GCFSTarget proto
- - (nullable NSMutableDictionary<NSString *, NSString *> *)encodedListenRequestLabelsForQueryData:
- (const QueryData &)queryData {
- NSString *value = [self encodedLabelForPurpose:queryData.purpose()];
- if (!value) {
- return nil;
- }
- NSMutableDictionary<NSString *, NSString *> *result =
- [NSMutableDictionary dictionaryWithCapacity:1];
- [result setObject:value forKey:@"goog-listen-tags"];
- return result;
- }
- - (nullable NSString *)encodedLabelForPurpose:(QueryPurpose)purpose {
- switch (purpose) {
- case QueryPurpose::Listen:
- return nil;
- case QueryPurpose::ExistenceFilterMismatch:
- return @"existence-filter-mismatch";
- case QueryPurpose::LimboResolution:
- return @"limbo-document";
- default:
- HARD_FAIL("Unrecognized query purpose: %s", purpose);
- }
- }
- - (GCFSTarget *)encodedTarget:(const QueryData &)queryData {
- GCFSTarget *result = [GCFSTarget message];
- const Query &query = queryData.query();
- if (query.IsDocumentQuery()) {
- result.documents = [self encodedDocumentsTarget:query];
- } else {
- result.query = [self encodedQueryTarget:query];
- }
- result.targetId = queryData.target_id();
- result.resumeToken = MakeNullableNSData(queryData.resume_token());
- return result;
- }
- - (GCFSTarget_DocumentsTarget *)encodedDocumentsTarget:(const Query &)query {
- GCFSTarget_DocumentsTarget *result = [GCFSTarget_DocumentsTarget message];
- NSMutableArray<NSString *> *docs = result.documentsArray;
- [docs addObject:[self encodedQueryPath:query.path()]];
- return result;
- }
- - (Query)decodedQueryFromDocumentsTarget:(GCFSTarget_DocumentsTarget *)target {
- NSArray<NSString *> *documents = target.documentsArray;
- HARD_ASSERT(documents.count == 1, "DocumentsTarget contained other than 1 document %s",
- (unsigned long)documents.count);
- NSString *name = documents[0];
- return Query([self decodedQueryPath:name]);
- }
- - (GCFSTarget_QueryTarget *)encodedQueryTarget:(const Query &)query {
- // Dissect the path into parent, collectionId, and optional key filter.
- GCFSTarget_QueryTarget *queryTarget = [GCFSTarget_QueryTarget message];
- const ResourcePath &path = query.path();
- if (query.collection_group()) {
- HARD_ASSERT(path.size() % 2 == 0,
- "Collection group queries should be within a document path or root.");
- queryTarget.parent = [self encodedQueryPath:path];
- GCFSStructuredQuery_CollectionSelector *from = [GCFSStructuredQuery_CollectionSelector message];
- from.collectionId = util::MakeNSString(query.collection_group());
- from.allDescendants = YES;
- [queryTarget.structuredQuery.fromArray addObject:from];
- } else {
- HARD_ASSERT(path.size() % 2 != 0, "Document queries with filters are not supported.");
- queryTarget.parent = [self encodedQueryPath:path.PopLast()];
- GCFSStructuredQuery_CollectionSelector *from = [GCFSStructuredQuery_CollectionSelector message];
- from.collectionId = util::MakeNSString(path.last_segment());
- [queryTarget.structuredQuery.fromArray addObject:from];
- }
- // Encode the filters.
- GCFSStructuredQuery_Filter *_Nullable where = [self encodedFilters:query.filters()];
- if (where) {
- queryTarget.structuredQuery.where = where;
- }
- NSArray<GCFSStructuredQuery_Order *> *orders = [self encodedSortOrders:query.order_bys()];
- if (orders.count) {
- [queryTarget.structuredQuery.orderByArray addObjectsFromArray:orders];
- }
- if (query.limit() != Query::kNoLimit) {
- queryTarget.structuredQuery.limit.value = (int32_t)query.limit();
- }
- if (query.start_at()) {
- queryTarget.structuredQuery.startAt = [self encodedBound:*query.start_at()];
- }
- if (query.end_at()) {
- queryTarget.structuredQuery.endAt = [self encodedBound:*query.end_at()];
- }
- return queryTarget;
- }
- - (Query)decodedQueryFromQueryTarget:(GCFSTarget_QueryTarget *)target {
- ResourcePath path = [self decodedQueryPath:target.parent];
- GCFSStructuredQuery *query = target.structuredQuery;
- CollectionGroupId collectionGroup;
- NSUInteger fromCount = query.fromArray_Count;
- if (fromCount > 0) {
- HARD_ASSERT(fromCount == 1,
- "StructuredQuery.from with more than one collection is not supported.");
- GCFSStructuredQuery_CollectionSelector *from = query.fromArray[0];
- if (from.allDescendants) {
- collectionGroup = util::MakeStringPtr(from.collectionId);
- } else {
- path = path.Append(util::MakeString(from.collectionId));
- }
- }
- FilterList filterBy;
- if (query.hasWhere) {
- filterBy = [self decodedFilters:query.where];
- }
- OrderByList orderBy;
- if (query.orderByArray_Count > 0) {
- orderBy = [self decodedSortOrders:query.orderByArray];
- }
- int32_t limit = Query::kNoLimit;
- if (query.hasLimit) {
- limit = query.limit.value;
- }
- std::shared_ptr<Bound> startAt;
- if (query.hasStartAt) {
- startAt = [self decodedBound:query.startAt];
- }
- std::shared_ptr<Bound> endAt;
- if (query.hasEndAt) {
- endAt = [self decodedBound:query.endAt];
- }
- return Query(std::move(path), std::move(collectionGroup), std::move(filterBy), std::move(orderBy),
- limit, std::move(startAt), std::move(endAt));
- }
- #pragma mark Filters
- - (GCFSStructuredQuery_Filter *_Nullable)encodedFilters:(const FilterList &)filters {
- if (filters.empty()) {
- return nil;
- }
- NSMutableArray<GCFSStructuredQuery_Filter *> *protos = [NSMutableArray array];
- for (const auto &filter : filters) {
- if (filter.IsAFieldFilter()) {
- [protos addObject:[self encodedUnaryOrFieldFilter:FieldFilter(filter)]];
- }
- }
- if (protos.count == 1) {
- // Special case: no existing filters and we only need to add one filter. This can be made the
- // single root filter without a composite filter.
- return protos[0];
- }
- GCFSStructuredQuery_Filter *composite = [GCFSStructuredQuery_Filter message];
- composite.compositeFilter.op = GCFSStructuredQuery_CompositeFilter_Operator_And;
- composite.compositeFilter.filtersArray = protos;
- return composite;
- }
- - (FilterList)decodedFilters:(GCFSStructuredQuery_Filter *)proto {
- FilterList result;
- NSArray<GCFSStructuredQuery_Filter *> *filters;
- if (proto.filterTypeOneOfCase ==
- GCFSStructuredQuery_Filter_FilterType_OneOfCase_CompositeFilter) {
- HARD_ASSERT(proto.compositeFilter.op == GCFSStructuredQuery_CompositeFilter_Operator_And,
- "Only AND-type composite filters are supported, got %s", proto.compositeFilter.op);
- filters = proto.compositeFilter.filtersArray;
- } else {
- filters = @[ proto ];
- }
- result = result.reserve(filters.count);
- for (GCFSStructuredQuery_Filter *filter in filters) {
- switch (filter.filterTypeOneOfCase) {
- case GCFSStructuredQuery_Filter_FilterType_OneOfCase_CompositeFilter:
- HARD_FAIL("Nested composite filters are not supported");
- case GCFSStructuredQuery_Filter_FilterType_OneOfCase_FieldFilter:
- result = result.push_back([self decodedFieldFilter:filter.fieldFilter]);
- break;
- case GCFSStructuredQuery_Filter_FilterType_OneOfCase_UnaryFilter:
- result = result.push_back([self decodedUnaryFilter:filter.unaryFilter]);
- break;
- default:
- HARD_FAIL("Unrecognized Filter.filterType %s", filter.filterTypeOneOfCase);
- }
- }
- return result;
- }
- - (GCFSStructuredQuery_Filter *)encodedUnaryOrFieldFilter:(const FieldFilter &)filter {
- GCFSStructuredQuery_Filter *proto = [GCFSStructuredQuery_Filter message];
- if (filter.op() == Operator::Equal) {
- if (filter.value().is_null()) {
- proto.unaryFilter.field = [self encodedFieldPath:filter.field()];
- proto.unaryFilter.op = GCFSStructuredQuery_UnaryFilter_Operator_IsNull;
- return proto;
- }
- if (filter.value().is_nan()) {
- proto.unaryFilter.field = [self encodedFieldPath:filter.field()];
- proto.unaryFilter.op = GCFSStructuredQuery_UnaryFilter_Operator_IsNan;
- return proto;
- }
- }
- GCFSStructuredQuery_FieldFilter *fieldFilter = proto.fieldFilter;
- fieldFilter.field = [self encodedFieldPath:filter.field()];
- fieldFilter.op = [self encodedFieldFilterOperator:filter.op()];
- fieldFilter.value = [self encodedFieldValue:filter.value()];
- return proto;
- }
- - (const FieldFilter)decodedFieldFilter:(GCFSStructuredQuery_FieldFilter *)proto {
- FieldPath fieldPath = FieldPath::FromServerFormat(util::MakeString(proto.field.fieldPath));
- Filter::Operator op = [self decodedFieldFilterOperator:proto.op];
- FieldValue value = [self decodedFieldValue:proto.value];
- return FieldFilter::Create(std::move(fieldPath), op, std::move(value));
- }
- - (FieldFilter)decodedUnaryFilter:(GCFSStructuredQuery_UnaryFilter *)proto {
- FieldPath field = FieldPath::FromServerFormat(util::MakeString(proto.field.fieldPath));
- switch (proto.op) {
- case GCFSStructuredQuery_UnaryFilter_Operator_IsNull:
- return FieldFilter::Create(std::move(field), Operator::Equal, FieldValue::Null());
- case GCFSStructuredQuery_UnaryFilter_Operator_IsNan:
- return FieldFilter::Create(std::move(field), Operator::Equal, FieldValue::Nan());
- default:
- HARD_FAIL("Unrecognized UnaryFilter.operator %s", proto.op);
- }
- }
- - (GCFSStructuredQuery_FieldReference *)encodedFieldPath:(const FieldPath &)fieldPath {
- GCFSStructuredQuery_FieldReference *ref = [GCFSStructuredQuery_FieldReference message];
- ref.fieldPath = util::MakeNSString(fieldPath.CanonicalString());
- return ref;
- }
- - (GCFSStructuredQuery_FieldFilter_Operator)encodedFieldFilterOperator:
- (Filter::Operator)filterOperator {
- switch (filterOperator) {
- case Filter::Operator::LessThan:
- return GCFSStructuredQuery_FieldFilter_Operator_LessThan;
- case Filter::Operator::LessThanOrEqual:
- return GCFSStructuredQuery_FieldFilter_Operator_LessThanOrEqual;
- case Filter::Operator::Equal:
- return GCFSStructuredQuery_FieldFilter_Operator_Equal;
- case Filter::Operator::GreaterThanOrEqual:
- return GCFSStructuredQuery_FieldFilter_Operator_GreaterThanOrEqual;
- case Filter::Operator::GreaterThan:
- return GCFSStructuredQuery_FieldFilter_Operator_GreaterThan;
- case Filter::Operator::ArrayContains:
- return GCFSStructuredQuery_FieldFilter_Operator_ArrayContains;
- case Filter::Operator::In:
- return GCFSStructuredQuery_FieldFilter_Operator_In;
- case Filter::Operator::ArrayContainsAny:
- return GCFSStructuredQuery_FieldFilter_Operator_ArrayContainsAny;
- default:
- HARD_FAIL("Unhandled Filter::Operator: %s", filterOperator);
- }
- }
- - (Filter::Operator)decodedFieldFilterOperator:
- (GCFSStructuredQuery_FieldFilter_Operator)filterOperator {
- switch (filterOperator) {
- case GCFSStructuredQuery_FieldFilter_Operator_LessThan:
- return Filter::Operator::LessThan;
- case GCFSStructuredQuery_FieldFilter_Operator_LessThanOrEqual:
- return Filter::Operator::LessThanOrEqual;
- case GCFSStructuredQuery_FieldFilter_Operator_Equal:
- return Filter::Operator::Equal;
- case GCFSStructuredQuery_FieldFilter_Operator_GreaterThanOrEqual:
- return Filter::Operator::GreaterThanOrEqual;
- case GCFSStructuredQuery_FieldFilter_Operator_GreaterThan:
- return Filter::Operator::GreaterThan;
- case GCFSStructuredQuery_FieldFilter_Operator_ArrayContains:
- return Filter::Operator::ArrayContains;
- case GCFSStructuredQuery_FieldFilter_Operator_In:
- return Filter::Operator::In;
- case GCFSStructuredQuery_FieldFilter_Operator_ArrayContainsAny:
- return Filter::Operator::ArrayContainsAny;
- default:
- HARD_FAIL("Unhandled FieldFilter.operator: %s", filterOperator);
- }
- }
- #pragma mark Property Orders
- - (NSArray<GCFSStructuredQuery_Order *> *)encodedSortOrders:(const OrderByList &)orders {
- NSMutableArray<GCFSStructuredQuery_Order *> *protos = [NSMutableArray array];
- for (const OrderBy &order : orders) {
- [protos addObject:[self encodedSortOrder:order]];
- }
- return protos;
- }
- - (OrderByList)decodedSortOrders:(NSArray<GCFSStructuredQuery_Order *> *)protos {
- OrderByList result;
- result = result.reserve(protos.count);
- for (GCFSStructuredQuery_Order *orderProto in protos) {
- result = result.push_back([self decodedSortOrder:orderProto]);
- }
- return result;
- }
- - (GCFSStructuredQuery_Order *)encodedSortOrder:(const OrderBy &)sortOrder {
- GCFSStructuredQuery_Order *proto = [GCFSStructuredQuery_Order message];
- proto.field = [self encodedFieldPath:sortOrder.field()];
- if (sortOrder.ascending()) {
- proto.direction = GCFSStructuredQuery_Direction_Ascending;
- } else {
- proto.direction = GCFSStructuredQuery_Direction_Descending;
- }
- return proto;
- }
- - (OrderBy)decodedSortOrder:(GCFSStructuredQuery_Order *)proto {
- FieldPath fieldPath = FieldPath::FromServerFormat(util::MakeString(proto.field.fieldPath));
- Direction direction;
- switch (proto.direction) {
- case GCFSStructuredQuery_Direction_Ascending:
- direction = Direction::Ascending;
- break;
- case GCFSStructuredQuery_Direction_Descending:
- direction = Direction::Descending;
- break;
- default:
- HARD_FAIL("Unrecognized GCFSStructuredQuery_Direction %s", proto.direction);
- }
- return OrderBy(std::move(fieldPath), direction);
- }
- #pragma mark - Bounds/Cursors
- - (GCFSCursor *)encodedBound:(const Bound &)bound {
- GCFSCursor *proto = [GCFSCursor message];
- proto.before = bound.before();
- for (const FieldValue &fieldValue : bound.position()) {
- GCFSValue *value = [self encodedFieldValue:fieldValue];
- [proto.valuesArray addObject:value];
- }
- return proto;
- }
- - (std::shared_ptr<Bound>)decodedBound:(GCFSCursor *)proto {
- std::vector<FieldValue> indexComponents;
- for (GCFSValue *valueProto in proto.valuesArray) {
- FieldValue value = [self decodedFieldValue:valueProto];
- indexComponents.push_back(std::move(value));
- }
- return std::make_shared<Bound>(std::move(indexComponents), proto.before);
- }
- #pragma mark - WatchChange <= GCFSListenResponse proto
- - (std::unique_ptr<WatchChange>)decodedWatchChange:(GCFSListenResponse *)watchChange {
- switch (watchChange.responseTypeOneOfCase) {
- case GCFSListenResponse_ResponseType_OneOfCase_TargetChange:
- return [self decodedTargetChangeFromWatchChange:watchChange.targetChange];
- case GCFSListenResponse_ResponseType_OneOfCase_DocumentChange:
- return [self decodedDocumentChange:watchChange.documentChange];
- case GCFSListenResponse_ResponseType_OneOfCase_DocumentDelete:
- return [self decodedDocumentDelete:watchChange.documentDelete];
- case GCFSListenResponse_ResponseType_OneOfCase_DocumentRemove:
- return [self decodedDocumentRemove:watchChange.documentRemove];
- case GCFSListenResponse_ResponseType_OneOfCase_Filter:
- return [self decodedExistenceFilterWatchChange:watchChange.filter];
- default:
- HARD_FAIL("Unknown WatchChange.changeType %s", watchChange.responseTypeOneOfCase);
- }
- }
- - (SnapshotVersion)versionFromListenResponse:(GCFSListenResponse *)watchChange {
- // We have only reached a consistent snapshot for the entire stream if there is a read_time set
- // and it applies to all targets (i.e. the list of targets is empty). The backend is guaranteed to
- // send such responses.
- if (watchChange.responseTypeOneOfCase != GCFSListenResponse_ResponseType_OneOfCase_TargetChange) {
- return SnapshotVersion::None();
- }
- if (watchChange.targetChange.targetIdsArray.count != 0) {
- return SnapshotVersion::None();
- }
- return [self decodedVersion:watchChange.targetChange.readTime];
- }
- - (std::unique_ptr<WatchChange>)decodedTargetChangeFromWatchChange:(GCFSTargetChange *)change {
- WatchTargetChangeState state = [self decodedWatchTargetChangeState:change.targetChangeType];
- __block std::vector<TargetId> targetIDs;
- [change.targetIdsArray enumerateValuesWithBlock:^(int32_t value, NSUInteger idx, BOOL *stop) {
- targetIDs.push_back(value);
- }];
- ByteString resumeToken = MakeByteString(change.resumeToken);
- util::Status cause;
- if (change.hasCause) {
- cause =
- util::Status{static_cast<Error>(change.cause.code), util::MakeString(change.cause.message)};
- }
- return absl::make_unique<WatchTargetChange>(state, std::move(targetIDs), std::move(resumeToken),
- std::move(cause));
- }
- - (WatchTargetChangeState)decodedWatchTargetChangeState:(GCFSTargetChange_TargetChangeType)state {
- switch (state) {
- case GCFSTargetChange_TargetChangeType_NoChange:
- return WatchTargetChangeState::NoChange;
- case GCFSTargetChange_TargetChangeType_Add:
- return WatchTargetChangeState::Added;
- case GCFSTargetChange_TargetChangeType_Remove:
- return WatchTargetChangeState::Removed;
- case GCFSTargetChange_TargetChangeType_Current:
- return WatchTargetChangeState::Current;
- case GCFSTargetChange_TargetChangeType_Reset:
- return WatchTargetChangeState::Reset;
- default:
- HARD_FAIL("Unexpected TargetChange.state: %s", state);
- }
- }
- - (std::vector<TargetId>)decodedIntegerArray:(GPBInt32Array *)values {
- __block std::vector<TargetId> result;
- result.reserve(values.count);
- [values enumerateValuesWithBlock:^(int32_t value, NSUInteger idx, BOOL *stop) {
- result.push_back(value);
- }];
- return result;
- }
- - (std::unique_ptr<WatchChange>)decodedDocumentChange:(GCFSDocumentChange *)change {
- ObjectValue value = [self decodedFields:change.document.fields];
- DocumentKey key = [self decodedDocumentKey:change.document.name];
- SnapshotVersion version = [self decodedVersion:change.document.updateTime];
- HARD_ASSERT(version != SnapshotVersion::None(), "Got a document change with no snapshot version");
- // The document may soon be re-serialized back to protos in order to store it in local
- // persistence. Memoize the encoded form to avoid encoding it again.
- Document document(std::move(value), key, version, DocumentState::kSynced, Wrap(change.document));
- std::vector<TargetId> updatedTargetIDs = [self decodedIntegerArray:change.targetIdsArray];
- std::vector<TargetId> removedTargetIDs = [self decodedIntegerArray:change.removedTargetIdsArray];
- return absl::make_unique<DocumentWatchChange>(std::move(updatedTargetIDs),
- std::move(removedTargetIDs), std::move(key),
- std::move(document));
- }
- - (std::unique_ptr<WatchChange>)decodedDocumentDelete:(GCFSDocumentDelete *)change {
- DocumentKey key = [self decodedDocumentKey:change.document];
- // Note that version might be unset in which case we use SnapshotVersion::None()
- SnapshotVersion version = [self decodedVersion:change.readTime];
- NoDocument document(key, version, /* has_committed_mutations= */ false);
- std::vector<TargetId> removedTargetIDs = [self decodedIntegerArray:change.removedTargetIdsArray];
- return absl::make_unique<DocumentWatchChange>(
- std::vector<TargetId>{}, std::move(removedTargetIDs), std::move(key), std::move(document));
- }
- - (std::unique_ptr<WatchChange>)decodedDocumentRemove:(GCFSDocumentRemove *)change {
- DocumentKey key = [self decodedDocumentKey:change.document];
- std::vector<TargetId> removedTargetIDs = [self decodedIntegerArray:change.removedTargetIdsArray];
- return absl::make_unique<DocumentWatchChange>(
- std::vector<TargetId>{}, std::move(removedTargetIDs), std::move(key), absl::nullopt);
- }
- - (std::unique_ptr<WatchChange>)decodedExistenceFilterWatchChange:(GCFSExistenceFilter *)filter {
- ExistenceFilter existenceFilter{filter.count};
- TargetId targetID = filter.targetId;
- return absl::make_unique<ExistenceFilterWatchChange>(existenceFilter, targetID);
- }
- @end
- NS_ASSUME_NONNULL_END
|