iot-firebase/IoT-iOS/Pods/FirebaseFirestore/Firestore/Source/API/FIRDocumentSnapshot.mm
2020-10-05 10:08:16 +01:00

328 lines
12 KiB
Plaintext

/*
* 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/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/api/document_reference.h"
#include "Firestore/core/src/api/document_snapshot.h"
#include "Firestore/core/src/api/firestore.h"
#include "Firestore/core/src/api/settings.h"
#include "Firestore/core/src/model/database_id.h"
#include "Firestore/core/src/model/document_key.h"
#include "Firestore/core/src/model/field_path.h"
#include "Firestore/core/src/model/field_value.h"
#include "Firestore/core/src/model/field_value_options.h"
#include "Firestore/core/src/nanopb/nanopb_util.h"
#include "Firestore/core/src/util/exception.h"
#include "Firestore/core/src/util/hard_assert.h"
#include "Firestore/core/src/util/log.h"
#include "Firestore/core/src/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