| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327 |
- /*
- * 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 "FIRDocumentSnapshot+Internal.h"
- #include <utility>
- #include <vector>
- #include "Firestore/core/src/firebase/firestore/util/warnings.h"
- #import "Firestore/Source/API/FIRDocumentReference+Internal.h"
- #import "Firestore/Source/API/FIRFieldPath+Internal.h"
- #import "Firestore/Source/API/FIRFirestore+Internal.h"
- #import "Firestore/Source/API/FIRGeoPoint+Internal.h"
- #import "Firestore/Source/API/FIRSnapshotMetadata+Internal.h"
- #import "Firestore/Source/API/FIRTimestamp+Internal.h"
- #import "Firestore/Source/API/converters.h"
- #include "Firestore/core/src/firebase/firestore/api/document_reference.h"
- #include "Firestore/core/src/firebase/firestore/api/document_snapshot.h"
- #include "Firestore/core/src/firebase/firestore/api/firestore.h"
- #include "Firestore/core/src/firebase/firestore/api/settings.h"
- #include "Firestore/core/src/firebase/firestore/model/database_id.h"
- #include "Firestore/core/src/firebase/firestore/model/document_key.h"
- #include "Firestore/core/src/firebase/firestore/model/field_path.h"
- #include "Firestore/core/src/firebase/firestore/model/field_value.h"
- #include "Firestore/core/src/firebase/firestore/model/field_value_options.h"
- #include "Firestore/core/src/firebase/firestore/nanopb/nanopb_util.h"
- #include "Firestore/core/src/firebase/firestore/util/exception.h"
- #include "Firestore/core/src/firebase/firestore/util/hard_assert.h"
- #include "Firestore/core/src/firebase/firestore/util/log.h"
- #include "Firestore/core/src/firebase/firestore/util/string_apple.h"
- namespace util = firebase::firestore::util;
- using firebase::Timestamp;
- using firebase::firestore::GeoPoint;
- using firebase::firestore::api::DocumentSnapshot;
- using firebase::firestore::api::Firestore;
- using firebase::firestore::api::MakeFIRGeoPoint;
- using firebase::firestore::api::MakeFIRTimestamp;
- using firebase::firestore::api::SnapshotMetadata;
- using firebase::firestore::model::DatabaseId;
- using firebase::firestore::model::Document;
- using firebase::firestore::model::DocumentKey;
- using firebase::firestore::model::FieldPath;
- using firebase::firestore::model::FieldValue;
- using firebase::firestore::model::FieldValueOptions;
- using firebase::firestore::model::ObjectValue;
- using firebase::firestore::model::ServerTimestampBehavior;
- using firebase::firestore::nanopb::MakeNSData;
- using firebase::firestore::util::MakeString;
- using firebase::firestore::util::ThrowInvalidArgument;
- NS_ASSUME_NONNULL_BEGIN
- namespace {
- /**
- * Converts a public FIRServerTimestampBehavior into its internal equivalent.
- */
- ServerTimestampBehavior InternalServerTimestampBehavior(FIRServerTimestampBehavior behavior) {
- switch (behavior) {
- case FIRServerTimestampBehaviorNone:
- return ServerTimestampBehavior::kNone;
- case FIRServerTimestampBehaviorEstimate:
- return ServerTimestampBehavior::kEstimate;
- case FIRServerTimestampBehaviorPrevious:
- return ServerTimestampBehavior::kPrevious;
- default:
- HARD_FAIL("Unexpected server timestamp option: %s", behavior);
- }
- }
- } // namespace
- @implementation FIRDocumentSnapshot {
- DocumentSnapshot _snapshot;
- FIRSnapshotMetadata *_cachedMetadata;
- }
- - (instancetype)initWithSnapshot:(DocumentSnapshot &&)snapshot {
- if (self = [super init]) {
- _snapshot = std::move(snapshot);
- }
- return self;
- }
- - (instancetype)initWithFirestore:(std::shared_ptr<Firestore>)firestore
- documentKey:(DocumentKey)documentKey
- document:(const absl::optional<Document> &)document
- metadata:(SnapshotMetadata)metadata {
- DocumentSnapshot wrapped;
- if (document.has_value()) {
- wrapped =
- DocumentSnapshot::FromDocument(std::move(firestore), document.value(), std::move(metadata));
- } else {
- wrapped = DocumentSnapshot::FromNoDocument(std::move(firestore), std::move(documentKey),
- std::move(metadata));
- }
- return [self initWithSnapshot:std::move(wrapped)];
- }
- - (instancetype)initWithFirestore:(std::shared_ptr<Firestore>)firestore
- documentKey:(DocumentKey)documentKey
- document:(const absl::optional<Document> &)document
- fromCache:(bool)fromCache
- hasPendingWrites:(bool)hasPendingWrites {
- return [self initWithFirestore:firestore
- documentKey:std::move(documentKey)
- document:document
- metadata:SnapshotMetadata(hasPendingWrites, fromCache)];
- }
- // NSObject Methods
- - (BOOL)isEqual:(nullable id)other {
- if (other == self) return YES;
- // self class could be FIRDocumentSnapshot or subtype. So we compare with base type explicitly.
- if (![other isKindOfClass:[FIRDocumentSnapshot class]]) return NO;
- return _snapshot == static_cast<FIRDocumentSnapshot *>(other)->_snapshot;
- }
- - (NSUInteger)hash {
- return _snapshot.Hash();
- }
- @dynamic exists;
- - (BOOL)exists {
- return _snapshot.exists();
- }
- - (const absl::optional<Document> &)internalDocument {
- return _snapshot.internal_document();
- }
- - (FIRDocumentReference *)reference {
- return [[FIRDocumentReference alloc] initWithReference:_snapshot.CreateReference()];
- }
- - (NSString *)documentID {
- return util::MakeNSString(_snapshot.document_id());
- }
- @dynamic metadata;
- - (FIRSnapshotMetadata *)metadata {
- if (!_cachedMetadata) {
- _cachedMetadata = [[FIRSnapshotMetadata alloc] initWithMetadata:_snapshot.metadata()];
- }
- return _cachedMetadata;
- }
- - (nullable NSDictionary<NSString *, id> *)data {
- return [self dataWithServerTimestampBehavior:FIRServerTimestampBehaviorNone];
- }
- - (nullable NSDictionary<NSString *, id> *)dataWithServerTimestampBehavior:
- (FIRServerTimestampBehavior)serverTimestampBehavior {
- FieldValueOptions options = [self optionsForServerTimestampBehavior:serverTimestampBehavior];
- absl::optional<ObjectValue> data = _snapshot.GetData();
- if (!data) return nil;
- return [self convertedObject:data->GetInternalValue() options:options];
- }
- - (nullable id)valueForField:(id)field {
- return [self valueForField:field serverTimestampBehavior:FIRServerTimestampBehaviorNone];
- }
- - (nullable id)valueForField:(id)field
- serverTimestampBehavior:(FIRServerTimestampBehavior)serverTimestampBehavior {
- FieldPath fieldPath;
- if ([field isKindOfClass:[NSString class]]) {
- fieldPath = FieldPath::FromDotSeparatedString(MakeString(field));
- } else if ([field isKindOfClass:[FIRFieldPath class]]) {
- fieldPath = ((FIRFieldPath *)field).internalValue;
- } else {
- ThrowInvalidArgument("Subscript key must be an NSString or FIRFieldPath.");
- }
- absl::optional<FieldValue> fieldValue = _snapshot.GetValue(fieldPath);
- FieldValueOptions options = [self optionsForServerTimestampBehavior:serverTimestampBehavior];
- return !fieldValue ? nil : [self convertedValue:*fieldValue options:options];
- }
- - (nullable id)objectForKeyedSubscript:(id)key {
- return [self valueForField:key];
- }
- - (FieldValueOptions)optionsForServerTimestampBehavior:
- (FIRServerTimestampBehavior)serverTimestampBehavior {
- SUPPRESS_DEPRECATED_DECLARATIONS_BEGIN()
- return FieldValueOptions(InternalServerTimestampBehavior(serverTimestampBehavior),
- _snapshot.firestore()->settings().timestamps_in_snapshots_enabled());
- SUPPRESS_END()
- }
- - (id)convertedValue:(FieldValue)value options:(const FieldValueOptions &)options {
- switch (value.type()) {
- case FieldValue::Type::Null:
- return [NSNull null];
- case FieldValue::Type::Boolean:
- return value.boolean_value() ? @YES : @NO;
- case FieldValue::Type::Integer:
- return @(value.integer_value());
- case FieldValue::Type::Double:
- return @(value.double_value());
- case FieldValue::Type::Timestamp:
- return [self convertedTimestamp:value options:options];
- case FieldValue::Type::ServerTimestamp:
- return [self convertedServerTimestamp:value options:options];
- case FieldValue::Type::String:
- return util::MakeNSString(value.string_value());
- case FieldValue::Type::Blob:
- return MakeNSData(value.blob_value());
- case FieldValue::Type::Reference:
- return [self convertedReference:value];
- case FieldValue::Type::GeoPoint:
- return MakeFIRGeoPoint(value.geo_point_value());
- case FieldValue::Type::Array:
- return [self convertedArray:value.array_value() options:options];
- case FieldValue::Type::Object:
- return [self convertedObject:value.object_value() options:options];
- }
- UNREACHABLE();
- }
- - (id)convertedTimestamp:(const FieldValue &)value options:(const FieldValueOptions &)options {
- FIRTimestamp *wrapped = MakeFIRTimestamp(value.timestamp_value());
- if (options.timestamps_in_snapshots_enabled()) {
- return wrapped;
- } else {
- return [wrapped dateValue];
- }
- }
- - (id)convertedServerTimestamp:(const FieldValue &)value
- options:(const FieldValueOptions &)options {
- const auto &sts = value.server_timestamp_value();
- switch (options.server_timestamp_behavior()) {
- case ServerTimestampBehavior::kNone:
- return [NSNull null];
- case ServerTimestampBehavior::kEstimate: {
- FieldValue local_write_time = FieldValue::FromTimestamp(sts.local_write_time());
- return [self convertedTimestamp:local_write_time options:options];
- }
- case ServerTimestampBehavior::kPrevious:
- return sts.previous_value() ? [self convertedValue:*sts.previous_value() options:options]
- : [NSNull null];
- }
- UNREACHABLE();
- }
- - (id)convertedReference:(const FieldValue &)value {
- const auto &ref = value.reference_value();
- const DatabaseId &refDatabase = ref.database_id();
- const DatabaseId &database = _snapshot.firestore()->database_id();
- if (refDatabase != database) {
- LOG_WARN("Document %s contains a document reference within a different database (%s/%s) which "
- "is not supported. It will be treated as a reference within the current database "
- "(%s/%s) instead.",
- _snapshot.CreateReference().Path(), refDatabase.project_id(),
- refDatabase.database_id(), database.project_id(), database.database_id());
- }
- const DocumentKey &key = ref.key();
- return [[FIRDocumentReference alloc] initWithKey:key firestore:_snapshot.firestore()];
- }
- - (NSArray<id> *)convertedArray:(const FieldValue::Array &)arrayContents
- options:(const FieldValueOptions &)options {
- NSMutableArray *result = [NSMutableArray arrayWithCapacity:arrayContents.size()];
- for (const FieldValue &value : arrayContents) {
- [result addObject:[self convertedValue:value options:options]];
- }
- return result;
- }
- - (NSDictionary<NSString *, id> *)convertedObject:(const FieldValue::Map &)objectValue
- options:(const FieldValueOptions &)options {
- NSMutableDictionary *result = [NSMutableDictionary dictionary];
- for (const auto &kv : objectValue) {
- const std::string &key = kv.first;
- const FieldValue &value = kv.second;
- result[util::MakeNSString(key)] = [self convertedValue:value options:options];
- }
- return result;
- }
- @end
- @implementation FIRQueryDocumentSnapshot
- - (NSDictionary<NSString *, id> *)data {
- NSDictionary<NSString *, id> *data = [super data];
- HARD_ASSERT(data, "Document in a QueryDocumentSnapshot should exist");
- return data;
- }
- - (NSDictionary<NSString *, id> *)dataWithServerTimestampBehavior:
- (FIRServerTimestampBehavior)serverTimestampBehavior {
- NSDictionary<NSString *, id> *data =
- [super dataWithServerTimestampBehavior:serverTimestampBehavior];
- HARD_ASSERT(data, "Document in a QueryDocumentSnapshot should exist");
- return data;
- }
- @end
- NS_ASSUME_NONNULL_END
|