| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- /*
- * 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 "FIRTransaction.h"
- #include <utility>
- #import "Firestore/Source/API/FIRDocumentReference+Internal.h"
- #import "Firestore/Source/API/FIRDocumentSnapshot+Internal.h"
- #import "Firestore/Source/API/FIRFirestore+Internal.h"
- #import "Firestore/Source/API/FIRTransaction+Internal.h"
- #import "Firestore/Source/API/FSTUserDataConverter.h"
- #import "Firestore/Source/Core/FSTTransaction.h"
- #import "Firestore/Source/Model/FSTDocument.h"
- #import "Firestore/Source/Util/FSTUsageValidation.h"
- #include "Firestore/core/src/firebase/firestore/util/hard_assert.h"
- using firebase::firestore::core::ParsedSetData;
- using firebase::firestore::core::ParsedUpdateData;
- NS_ASSUME_NONNULL_BEGIN
- #pragma mark - FIRTransaction
- @interface FIRTransaction ()
- - (instancetype)initWithTransaction:(FSTTransaction *)transaction
- firestore:(FIRFirestore *)firestore NS_DESIGNATED_INITIALIZER;
- @property(nonatomic, strong, readonly) FSTTransaction *internalTransaction;
- @property(nonatomic, strong, readonly) FIRFirestore *firestore;
- @end
- @implementation FIRTransaction (Internal)
- + (instancetype)transactionWithFSTTransaction:(FSTTransaction *)transaction
- firestore:(FIRFirestore *)firestore {
- return [[FIRTransaction alloc] initWithTransaction:transaction firestore:firestore];
- }
- @end
- @implementation FIRTransaction
- - (instancetype)initWithTransaction:(FSTTransaction *)transaction
- firestore:(FIRFirestore *)firestore {
- self = [super init];
- if (self) {
- _internalTransaction = transaction;
- _firestore = firestore;
- }
- return self;
- }
- - (FIRTransaction *)setData:(NSDictionary<NSString *, id> *)data
- forDocument:(FIRDocumentReference *)document {
- return [self setData:data forDocument:document merge:NO];
- }
- - (FIRTransaction *)setData:(NSDictionary<NSString *, id> *)data
- forDocument:(FIRDocumentReference *)document
- merge:(BOOL)merge {
- [self validateReference:document];
- ParsedSetData parsed = merge ? [self.firestore.dataConverter parsedMergeData:data fieldMask:nil]
- : [self.firestore.dataConverter parsedSetData:data];
- [self.internalTransaction setData:std::move(parsed) forDocument:document.key];
- return self;
- }
- - (FIRTransaction *)setData:(NSDictionary<NSString *, id> *)data
- forDocument:(FIRDocumentReference *)document
- mergeFields:(NSArray<id> *)mergeFields {
- [self validateReference:document];
- ParsedSetData parsed = [self.firestore.dataConverter parsedMergeData:data fieldMask:mergeFields];
- [self.internalTransaction setData:std::move(parsed) forDocument:document.key];
- return self;
- }
- - (FIRTransaction *)updateData:(NSDictionary<id, id> *)fields
- forDocument:(FIRDocumentReference *)document {
- [self validateReference:document];
- ParsedUpdateData parsed = [self.firestore.dataConverter parsedUpdateData:fields];
- [self.internalTransaction updateData:std::move(parsed) forDocument:document.key];
- return self;
- }
- - (FIRTransaction *)deleteDocument:(FIRDocumentReference *)document {
- [self validateReference:document];
- [self.internalTransaction deleteDocument:document.key];
- return self;
- }
- - (void)getDocument:(FIRDocumentReference *)document
- completion:(void (^)(FIRDocumentSnapshot *_Nullable document,
- NSError *_Nullable error))completion {
- [self validateReference:document];
- [self.internalTransaction
- lookupDocumentsForKeys:{document.key}
- completion:^(NSArray<FSTMaybeDocument *> *_Nullable documents,
- NSError *_Nullable error) {
- if (error) {
- completion(nil, error);
- return;
- }
- HARD_ASSERT(documents.count == 1,
- "Mismatch in docs returned from document lookup.");
- FSTMaybeDocument *internalDoc = documents.firstObject;
- if ([internalDoc isKindOfClass:[FSTDeletedDocument class]]) {
- FIRDocumentSnapshot *doc =
- [FIRDocumentSnapshot snapshotWithFirestore:self.firestore
- documentKey:document.key
- document:nil
- fromCache:NO
- hasPendingWrites:NO];
- completion(doc, nil);
- } else if ([internalDoc isKindOfClass:[FSTDocument class]]) {
- FIRDocumentSnapshot *doc =
- [FIRDocumentSnapshot snapshotWithFirestore:self.firestore
- documentKey:internalDoc.key
- document:(FSTDocument *)internalDoc
- fromCache:NO
- hasPendingWrites:NO];
- completion(doc, nil);
- } else {
- HARD_FAIL("BatchGetDocumentsRequest returned unexpected document type: %s",
- NSStringFromClass([internalDoc class]));
- }
- }];
- }
- - (FIRDocumentSnapshot *_Nullable)getDocument:(FIRDocumentReference *)document
- error:(NSError *__autoreleasing *)error {
- [self validateReference:document];
- dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
- __block FIRDocumentSnapshot *result;
- // We have to explicitly assign the innerError into a local to cause it to retain correctly.
- __block NSError *outerError = nil;
- [self getDocument:document
- completion:^(FIRDocumentSnapshot *_Nullable snapshot, NSError *_Nullable innerError) {
- result = snapshot;
- outerError = innerError;
- dispatch_semaphore_signal(semaphore);
- }];
- dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
- if (error) {
- *error = outerError;
- }
- return result;
- }
- - (void)validateReference:(FIRDocumentReference *)reference {
- if (reference.firestore != self.firestore) {
- FSTThrowInvalidArgument(@"Provided document reference is from a different Firestore instance.");
- }
- }
- @end
- NS_ASSUME_NONNULL_END
|