| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266 |
- /*
- * 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 "FirebaseDatabase/Sources/Core/View/FView.h"
- #import "FirebaseDatabase/Sources/Api/Private/FIRDatabaseQuery_Private.h"
- #import "FirebaseDatabase/Sources/Core/FQueryParams.h"
- #import "FirebaseDatabase/Sources/Core/FQuerySpec.h"
- #import "FirebaseDatabase/Sources/Core/FWriteTreeRef.h"
- #import "FirebaseDatabase/Sources/Core/Operation/FOperation.h"
- #import "FirebaseDatabase/Sources/Core/Operation/FOperationSource.h"
- #import "FirebaseDatabase/Sources/Core/Utilities/FPath.h"
- #import "FirebaseDatabase/Sources/Core/View/FCacheNode.h"
- #import "FirebaseDatabase/Sources/Core/View/FCancelEvent.h"
- #import "FirebaseDatabase/Sources/Core/View/FEventRegistration.h"
- #import "FirebaseDatabase/Sources/Core/View/FViewCache.h"
- #import "FirebaseDatabase/Sources/Core/View/Filter/FIndexedFilter.h"
- #import "FirebaseDatabase/Sources/FEventGenerator.h"
- #import "FirebaseDatabase/Sources/FViewProcessor.h"
- #import "FirebaseDatabase/Sources/FViewProcessorResult.h"
- #import "FirebaseDatabase/Sources/Public/FirebaseDatabase/FIRDatabaseQuery.h"
- #import "FirebaseDatabase/Sources/Snapshot/FEmptyNode.h"
- #import "FirebaseDatabase/Sources/Snapshot/FIndexedNode.h"
- #import "FirebaseDatabase/Sources/Snapshot/FNode.h"
- @interface FViewOperationResult ()
- @property(nonatomic, strong, readwrite) NSArray *changes;
- @property(nonatomic, strong, readwrite) NSArray *events;
- @end
- @implementation FViewOperationResult
- - (id)initWithChanges:(NSArray *)changes events:(NSArray *)events {
- self = [super init];
- if (self != nil) {
- self->_changes = changes;
- self->_events = events;
- }
- return self;
- }
- @end
- /**
- * A view represents a specific location and query that has 1 or more event
- * registrations.
- *
- * It does several things:
- * - Maintains the list of event registration for this location/query.
- * - Maintains a cache of the data visible for this location/query.
- * - Applies new operations (via applyOperation), updates the cache, and based
- * on the event registrations returns the set of events to be raised.
- */
- @interface FView ()
- @property(nonatomic, strong, readwrite) FQuerySpec *query;
- @property(nonatomic, strong) FViewProcessor *processor;
- @property(nonatomic, strong) FViewCache *viewCache;
- @property(nonatomic, strong) NSMutableArray *eventRegistrations;
- @property(nonatomic, strong) FEventGenerator *eventGenerator;
- @end
- @implementation FView
- - (id)initWithQuery:(FQuerySpec *)query
- initialViewCache:(FViewCache *)initialViewCache {
- self = [super init];
- if (self) {
- self.query = query;
- FIndexedFilter *indexFilter =
- [[FIndexedFilter alloc] initWithIndex:query.index];
- id<FNodeFilter> filter = query.params.nodeFilter;
- self.processor = [[FViewProcessor alloc] initWithFilter:filter];
- FCacheNode *initialServerCache = initialViewCache.cachedServerSnap;
- FCacheNode *initialEventCache = initialViewCache.cachedEventSnap;
- // Don't filter server node with other filter than index, wait for
- // tagged listen
- FIndexedNode *emptyIndexedNode =
- [FIndexedNode indexedNodeWithNode:[FEmptyNode emptyNode]
- index:query.index];
- FIndexedNode *serverSnap =
- [indexFilter updateFullNode:emptyIndexedNode
- withNewNode:initialServerCache.indexedNode
- accumulator:nil];
- FIndexedNode *eventSnap =
- [filter updateFullNode:emptyIndexedNode
- withNewNode:initialEventCache.indexedNode
- accumulator:nil];
- FCacheNode *newServerCache = [[FCacheNode alloc]
- initWithIndexedNode:serverSnap
- isFullyInitialized:initialServerCache.isFullyInitialized
- isFiltered:indexFilter.filtersNodes];
- FCacheNode *newEventCache = [[FCacheNode alloc]
- initWithIndexedNode:eventSnap
- isFullyInitialized:initialEventCache.isFullyInitialized
- isFiltered:filter.filtersNodes];
- self.viewCache = [[FViewCache alloc] initWithEventCache:newEventCache
- serverCache:newServerCache];
- self.eventRegistrations = [[NSMutableArray alloc] init];
- self.eventGenerator = [[FEventGenerator alloc] initWithQuery:query];
- }
- return self;
- }
- - (id<FNode>)serverCache {
- return self.viewCache.cachedServerSnap.node;
- }
- - (id<FNode>)eventCache {
- return self.viewCache.cachedEventSnap.node;
- }
- - (id<FNode>)completeServerCacheFor:(FPath *)path {
- id<FNode> cache = self.viewCache.completeServerSnap;
- if (cache) {
- // If this isn't a "loadsAllData" view, then cache isn't actually a
- // complete cache and we need to see if it contains the child we're
- // interested in.
- if ([self.query loadsAllData] ||
- (!path.isEmpty &&
- ![cache getImmediateChild:path.getFront].isEmpty)) {
- return [cache getChild:path];
- }
- }
- return nil;
- }
- - (BOOL)isEmpty {
- return self.eventRegistrations.count == 0;
- }
- - (void)addEventRegistration:(id<FEventRegistration>)eventRegistration {
- [self.eventRegistrations addObject:eventRegistration];
- }
- /**
- * @param eventRegistration If null, remove all callbacks.
- * @param cancelError If a cancelError is provided, appropriate cancel events
- * will be returned.
- * @return Cancel events, if cancelError was provided.
- */
- - (NSArray *)removeEventRegistration:(id<FEventRegistration>)eventRegistration
- cancelError:(NSError *)cancelError {
- NSMutableArray *cancelEvents = [[NSMutableArray alloc] init];
- if (cancelError != nil) {
- NSAssert(eventRegistration == nil,
- @"A cancel should cancel all event registrations.");
- FPath *path = self.query.path;
- for (id<FEventRegistration> registration in self.eventRegistrations) {
- FCancelEvent *maybeEvent =
- [registration createCancelEventFromError:cancelError path:path];
- if (maybeEvent) {
- [cancelEvents addObject:maybeEvent];
- }
- }
- }
- if (eventRegistration) {
- NSUInteger i = 0;
- while (i < self.eventRegistrations.count) {
- id<FEventRegistration> existing = self.eventRegistrations[i];
- if ([existing matches:eventRegistration]) {
- [self.eventRegistrations removeObjectAtIndex:i];
- } else {
- i++;
- }
- }
- } else {
- [self.eventRegistrations removeAllObjects];
- }
- return cancelEvents;
- }
- /**
- * Applies the given Operation, updates our cache, and returns the appropriate
- * events and changes
- */
- - (FViewOperationResult *)applyOperation:(id<FOperation>)operation
- writesCache:(FWriteTreeRef *)writesCache
- serverCache:(id<FNode>)optCompleteServerCache {
- if (operation.type == FOperationTypeMerge &&
- operation.source.queryParams != nil) {
- NSAssert(self.viewCache.completeServerSnap != nil,
- @"We should always have a full cache before handling merges");
- NSAssert(self.viewCache.completeEventSnap != nil,
- @"Missing event cache, even though we have a server cache");
- }
- FViewCache *oldViewCache = self.viewCache;
- FViewProcessorResult *result =
- [self.processor applyOperationOn:oldViewCache
- operation:operation
- writesCache:writesCache
- completeCache:optCompleteServerCache];
- NSAssert(result.viewCache.cachedServerSnap.isFullyInitialized ||
- !oldViewCache.cachedServerSnap.isFullyInitialized,
- @"Once a server snap is complete, it should never go back.");
- self.viewCache = result.viewCache;
- NSArray *events = [self
- generateEventsForChanges:result.changes
- eventCache:result.viewCache.cachedEventSnap.indexedNode
- registration:nil];
- return [[FViewOperationResult alloc] initWithChanges:result.changes
- events:events];
- }
- - (NSArray *)initialEvents:(id<FEventRegistration>)registration {
- FCacheNode *eventSnap = self.viewCache.cachedEventSnap;
- NSMutableArray *initialChanges = [[NSMutableArray alloc] init];
- [eventSnap.indexedNode.node enumerateChildrenUsingBlock:^(
- NSString *key, id<FNode> node, BOOL *stop) {
- FIndexedNode *indexed = [FIndexedNode indexedNodeWithNode:node];
- FChange *change = [[FChange alloc] initWithType:FIRDataEventTypeChildAdded
- indexedNode:indexed
- childKey:key];
- [initialChanges addObject:change];
- }];
- if (eventSnap.isFullyInitialized) {
- FChange *change = [[FChange alloc] initWithType:FIRDataEventTypeValue
- indexedNode:eventSnap.indexedNode];
- [initialChanges addObject:change];
- }
- return [self generateEventsForChanges:initialChanges
- eventCache:eventSnap.indexedNode
- registration:registration];
- }
- - (NSArray *)generateEventsForChanges:(NSArray *)changes
- eventCache:(FIndexedNode *)eventCache
- registration:(id<FEventRegistration>)registration {
- NSArray *registrations;
- if (registration == nil) {
- registrations = [[NSArray alloc] initWithArray:self.eventRegistrations];
- } else {
- registrations = [[NSArray alloc] initWithObjects:registration, nil];
- }
- return [self.eventGenerator generateEventsForChanges:changes
- eventCache:eventCache
- eventRegistrations:registrations];
- }
- - (NSString *)description {
- return [NSString stringWithFormat:@"FView (%@)", self.query];
- }
- @end
|