Skip to content

Commit

Permalink
Listen sequence numbers (firebase#675)
Browse files Browse the repository at this point in the history
* Generate and save sequence numbers for listens

* Add documentation

* Fix include path

* Fix unavailable comment

* Review feedback
  • Loading branch information
Greg Soltis committed Jan 17, 2018
1 parent 898dc22 commit 77c95ef
Show file tree
Hide file tree
Showing 16 changed files with 217 additions and 19 deletions.
2 changes: 2 additions & 0 deletions Firestore/Example/Tests/Local/FSTLocalSerializerTests.m
Original file line number Diff line number Diff line change
Expand Up @@ -157,6 +157,7 @@ - (void)testEncodesQueryData {

FSTQueryData *queryData = [[FSTQueryData alloc] initWithQuery:query
targetID:targetID
listenSequenceNumber:10
purpose:FSTQueryPurposeListen
snapshotVersion:version
resumeToken:resumeToken];
Expand All @@ -166,6 +167,7 @@ - (void)testEncodesQueryData {

FSTPBTarget *expected = [FSTPBTarget message];
expected.targetId = targetID;
expected.lastListenSequenceNumber = 10;
expected.snapshotVersion.nanos = 1039000;
expected.resumeToken = [resumeToken copy];
expected.query.parent = queryTarget.parent;
Expand Down
80 changes: 70 additions & 10 deletions Firestore/Example/Tests/Local/FSTQueryCacheTests.m
Original file line number Diff line number Diff line change
Expand Up @@ -31,12 +31,18 @@

@implementation FSTQueryCacheTests {
FSTQuery *_queryRooms;
FSTListenSequenceNumber _previousSequenceNumber;
FSTTargetID _previousTargetID;
FSTTestSnapshotVersion _previousSnapshotVersion;
}

- (void)setUp {
[super setUp];

_queryRooms = FSTTestQuery(@"rooms");
_previousSequenceNumber = 1000;
_previousTargetID = 500;
_previousSnapshotVersion = 100;
}

/**
Expand All @@ -56,7 +62,7 @@ - (void)testReadQueryNotInCache {
- (void)testSetAndReadAQuery {
if ([self isTestBaseClass]) return;

FSTQueryData *queryData = [self queryDataWithQuery:_queryRooms targetID:1 version:1];
FSTQueryData *queryData = [self queryDataWithQuery:_queryRooms];
[self addQueryData:queryData];

FSTQueryData *result = [self.queryCache queryDataForQuery:_queryRooms];
Expand All @@ -74,14 +80,14 @@ - (void)testCanonicalIDCollision {
FSTQuery *q2 = [FSTTestQuery(@"a") queryByAddingFilter:FSTTestFilter(@"foo", @"==", @"1")];
XCTAssertEqualObjects(q1.canonicalID, q2.canonicalID);

FSTQueryData *data1 = [self queryDataWithQuery:q1 targetID:1 version:1];
FSTQueryData *data1 = [self queryDataWithQuery:q1];
[self addQueryData:data1];

// Using the other query should not return the query cache entry despite equal canonicalIDs.
XCTAssertNil([self.queryCache queryDataForQuery:q2]);
XCTAssertEqualObjects([self.queryCache queryDataForQuery:q1], data1);

FSTQueryData *data2 = [self queryDataWithQuery:q2 targetID:2 version:1];
FSTQueryData *data2 = [self queryDataWithQuery:q2];
[self addQueryData:data2];

XCTAssertEqualObjects([self.queryCache queryDataForQuery:q1], data1);
Expand All @@ -99,10 +105,12 @@ - (void)testCanonicalIDCollision {
- (void)testSetQueryToNewValue {
if ([self isTestBaseClass]) return;

FSTQueryData *queryData1 = [self queryDataWithQuery:_queryRooms targetID:1 version:1];
FSTQueryData *queryData1 =
[self queryDataWithQuery:_queryRooms targetID:1 listenSequenceNumber:10 version:1];
[self addQueryData:queryData1];

FSTQueryData *queryData2 = [self queryDataWithQuery:_queryRooms targetID:1 version:2];
FSTQueryData *queryData2 =
[self queryDataWithQuery:_queryRooms targetID:1 listenSequenceNumber:10 version:2];
[self addQueryData:queryData2];

FSTQueryData *result = [self.queryCache queryDataForQuery:_queryRooms];
Expand All @@ -115,7 +123,7 @@ - (void)testSetQueryToNewValue {
- (void)testRemoveQuery {
if ([self isTestBaseClass]) return;

FSTQueryData *queryData1 = [self queryDataWithQuery:_queryRooms targetID:1 version:1];
FSTQueryData *queryData1 = [self queryDataWithQuery:_queryRooms];
[self addQueryData:queryData1];

[self removeQueryData:queryData1];
Expand All @@ -127,7 +135,7 @@ - (void)testRemoveQuery {
- (void)testRemoveNonExistentQuery {
if ([self isTestBaseClass]) return;

FSTQueryData *queryData = [self queryDataWithQuery:_queryRooms targetID:1 version:1];
FSTQueryData *queryData = [self queryDataWithQuery:_queryRooms];

// no-op, but make sure it doesn't throw.
XCTAssertNoThrow([self removeQueryData:queryData]);
Expand All @@ -136,7 +144,7 @@ - (void)testRemoveNonExistentQuery {
- (void)testRemoveQueryRemovesMatchingKeysToo {
if ([self isTestBaseClass]) return;

FSTQueryData *rooms = [self queryDataWithQuery:_queryRooms targetID:1 version:1];
FSTQueryData *rooms = [self queryDataWithQuery:_queryRooms];
[self addQueryData:rooms];

FSTDocumentKey *key1 = FSTTestDocKey(@"rooms/foo");
Expand Down Expand Up @@ -204,14 +212,14 @@ - (void)testRemoveEmitsGarbageEvents {
[garbageCollector addGarbageSource:self.queryCache];
FSTAssertEqualSets([garbageCollector collectGarbage], @[]);

FSTQueryData *rooms = [self queryDataWithQuery:FSTTestQuery(@"rooms") targetID:1 version:1];
FSTQueryData *rooms = [self queryDataWithQuery:FSTTestQuery(@"rooms")];
FSTDocumentKey *room1 = FSTTestDocKey(@"rooms/bar");
FSTDocumentKey *room2 = FSTTestDocKey(@"rooms/foo");
[self addQueryData:rooms];
[self addMatchingKey:room1 forTargetID:rooms.targetID];
[self addMatchingKey:room2 forTargetID:rooms.targetID];

FSTQueryData *halls = [self queryDataWithQuery:FSTTestQuery(@"halls") targetID:2 version:1];
FSTQueryData *halls = [self queryDataWithQuery:FSTTestQuery(@"halls")];
FSTDocumentKey *hall1 = FSTTestDocKey(@"halls/bar");
FSTDocumentKey *hall2 = FSTTestDocKey(@"halls/foo");
[self addQueryData:halls];
Expand Down Expand Up @@ -249,13 +257,54 @@ - (void)testMatchingKeysForTargetID {
FSTAssertEqualSets([self.queryCache matchingKeysForTargetID:2], (@[ key1, key3 ]));
}

- (void)testHighestListenSequenceNumber {
if ([self isTestBaseClass]) return;

FSTQueryData *query1 = [[FSTQueryData alloc] initWithQuery:FSTTestQuery(@"rooms")
targetID:1
listenSequenceNumber:10
purpose:FSTQueryPurposeListen];
[self addQueryData:query1];
FSTQueryData *query2 = [[FSTQueryData alloc] initWithQuery:FSTTestQuery(@"halls")
targetID:2
listenSequenceNumber:20
purpose:FSTQueryPurposeListen];
[self addQueryData:query2];
XCTAssertEqual([self.queryCache highestListenSequenceNumber], 20);

// TargetIDs never come down.
[self removeQueryData:query2];
XCTAssertEqual([self.queryCache highestListenSequenceNumber], 20);

// A query with an empty result set still counts.
FSTQueryData *query3 = [[FSTQueryData alloc] initWithQuery:FSTTestQuery(@"garages")
targetID:42
listenSequenceNumber:100
purpose:FSTQueryPurposeListen];
[self addQueryData:query3];
XCTAssertEqual([self.queryCache highestListenSequenceNumber], 100);

[self removeQueryData:query1];
XCTAssertEqual([self.queryCache highestListenSequenceNumber], 100);

[self removeQueryData:query3];
XCTAssertEqual([self.queryCache highestListenSequenceNumber], 100);

// Verify that the highestTargetID even survives restarts.
[self.queryCache shutdown];
self.queryCache = [self.persistence queryCache];
[self.queryCache start];
XCTAssertEqual([self.queryCache highestListenSequenceNumber], 100);
}

- (void)testHighestTargetID {
if ([self isTestBaseClass]) return;

XCTAssertEqual([self.queryCache highestTargetID], 0);

FSTQueryData *query1 = [[FSTQueryData alloc] initWithQuery:FSTTestQuery(@"rooms")
targetID:1
listenSequenceNumber:10
purpose:FSTQueryPurposeListen];
FSTDocumentKey *key1 = FSTTestDocKey(@"rooms/bar");
FSTDocumentKey *key2 = FSTTestDocKey(@"rooms/foo");
Expand All @@ -265,6 +314,7 @@ - (void)testHighestTargetID {

FSTQueryData *query2 = [[FSTQueryData alloc] initWithQuery:FSTTestQuery(@"halls")
targetID:2
listenSequenceNumber:20
purpose:FSTQueryPurposeListen];
FSTDocumentKey *key3 = FSTTestDocKey(@"halls/foo");
[self addQueryData:query2];
Expand All @@ -278,6 +328,7 @@ - (void)testHighestTargetID {
// A query with an empty result set still counts.
FSTQueryData *query3 = [[FSTQueryData alloc] initWithQuery:FSTTestQuery(@"garages")
targetID:42
listenSequenceNumber:100
purpose:FSTQueryPurposeListen];
[self addQueryData:query3];
XCTAssertEqual([self.queryCache highestTargetID], 42);
Expand Down Expand Up @@ -319,12 +370,21 @@ - (void)testLastRemoteSnapshotVersion {
* Creates a new FSTQueryData object from the given parameters, synthesizing a resume token from
* the snapshot version.
*/
- (FSTQueryData *)queryDataWithQuery:(FSTQuery *)query {
return [self queryDataWithQuery:query
targetID:++_previousTargetID
listenSequenceNumber:++_previousSequenceNumber
version:++_previousSnapshotVersion];
}

- (FSTQueryData *)queryDataWithQuery:(FSTQuery *)query
targetID:(FSTTargetID)targetID
listenSequenceNumber:(FSTListenSequenceNumber)sequenceNumber
version:(FSTTestSnapshotVersion)version {
NSData *resumeToken = FSTTestResumeTokenFromSnapshotVersion(version);
return [[FSTQueryData alloc] initWithQuery:query
targetID:targetID
listenSequenceNumber:sequenceNumber
purpose:FSTQueryPurposeListen
snapshotVersion:FSTTestVersion(version)
resumeToken:resumeToken];
Expand Down
15 changes: 11 additions & 4 deletions Firestore/Example/Tests/Remote/FSTSerializerBetaTests.m
Original file line number Diff line number Diff line change
Expand Up @@ -396,20 +396,25 @@ - (void)testRoundTripSpecialFieldNames {

- (void)testEncodesListenRequestLabels {
FSTQuery *query = FSTTestQuery(@"collection/key");
FSTQueryData *queryData =
[[FSTQueryData alloc] initWithQuery:query targetID:2 purpose:FSTQueryPurposeListen];
FSTQueryData *queryData = [[FSTQueryData alloc] initWithQuery:query
targetID:2
listenSequenceNumber:3
purpose:FSTQueryPurposeListen];

NSDictionary<NSString *, NSString *> *result =
[self.serializer encodedListenRequestLabelsForQueryData:queryData];
XCTAssertNil(result);

queryData =
[[FSTQueryData alloc] initWithQuery:query targetID:2 purpose:FSTQueryPurposeLimboResolution];
queryData = [[FSTQueryData alloc] initWithQuery:query
targetID:2
listenSequenceNumber:3
purpose:FSTQueryPurposeLimboResolution];
result = [self.serializer encodedListenRequestLabelsForQueryData:queryData];
XCTAssertEqualObjects(result, @{@"goog-listen-tags" : @"limbo-document"});

queryData = [[FSTQueryData alloc] initWithQuery:query
targetID:2
listenSequenceNumber:3
purpose:FSTQueryPurposeExistenceFilterMismatch];
result = [self.serializer encodedListenRequestLabelsForQueryData:queryData];
XCTAssertEqualObjects(result, @{@"goog-listen-tags" : @"existence-filter-mismatch"});
Expand Down Expand Up @@ -627,6 +632,7 @@ - (void)testEncodesResumeTokens {
FSTQuery *q = FSTTestQuery(@"docs");
FSTQueryData *model = [[FSTQueryData alloc] initWithQuery:q
targetID:1
listenSequenceNumber:0
purpose:FSTQueryPurposeListen
snapshotVersion:[FSTSnapshotVersion noVersion]
resumeToken:FSTTestData(1, 2, 3, -1)];
Expand All @@ -647,6 +653,7 @@ - (void)testEncodesResumeTokens {
- (FSTQueryData *)queryDataForQuery:(FSTQuery *)query {
return [[FSTQueryData alloc] initWithQuery:query
targetID:1
listenSequenceNumber:0
purpose:FSTQueryPurposeListen
snapshotVersion:[FSTSnapshotVersion noVersion]
resumeToken:[NSData data]];
Expand Down
1 change: 1 addition & 0 deletions Firestore/Example/Tests/SpecTests/FSTSpecTests.m
Original file line number Diff line number Diff line change
Expand Up @@ -503,6 +503,7 @@ - (void)validateStateExpectations:(nullable NSDictionary *)expected {
expectedActiveTargets[@(targetID)] =
[[FSTQueryData alloc] initWithQuery:query
targetID:targetID
listenSequenceNumber:0
purpose:FSTQueryPurposeListen
snapshotVersion:[FSTSnapshotVersion noVersion]
resumeToken:resumeToken];
Expand Down
37 changes: 37 additions & 0 deletions Firestore/Source/Core/FSTListenSequence.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
/*
* Copyright 2018 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 <Foundation/Foundation.h>

#import "FSTTypes.h"

NS_ASSUME_NONNULL_BEGIN

/**
* FSTListenSequence is a monotonic sequence. It is initialized with a minimum value to
* exceed. All subsequent calls to next will return increasing values.
*/
@interface FSTListenSequence : NSObject

- (instancetype)initStartingAfter:(FSTListenSequenceNumber)after NS_DESIGNATED_INITIALIZER;

- (id)init NS_UNAVAILABLE;

- (FSTListenSequenceNumber)next;

@end

NS_ASSUME_NONNULL_END
34 changes: 34 additions & 0 deletions Firestore/Source/Core/FSTListenSequence.m
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
#import "FSTListenSequence.h"

NS_ASSUME_NONNULL_BEGIN

#pragma mark - FSTListenSequence

@interface FSTListenSequence () {
FSTListenSequenceNumber _previousSequenceNumber;
}

@end

@implementation FSTListenSequence

#pragma mark - Constructors

- (instancetype)initStartingAfter:(FSTListenSequenceNumber)after {
self = [super init];
if (self) {
_previousSequenceNumber = after;
}
return self;
}

#pragma mark - Public methods

- (FSTListenSequenceNumber)next {
_previousSequenceNumber++;
return _previousSequenceNumber;
}

@end

NS_ASSUME_NONNULL_END
5 changes: 5 additions & 0 deletions Firestore/Source/Core/FSTSyncEngine.m
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,10 @@

NS_ASSUME_NONNULL_BEGIN

// Limbo documents don't use persistence, and are eagerly GC'd. So, listens for them don't need
// real sequence numbers.
static const FSTListenSequenceNumber kIrrelevantSequenceNumber = -1;

#pragma mark - FSTQueryView

/**
Expand Down Expand Up @@ -490,6 +494,7 @@ - (void)trackLimboChange:(FSTLimboDocumentChange *)limboChange {
FSTQuery *query = [FSTQuery queryWithPath:key.path];
FSTQueryData *queryData = [[FSTQueryData alloc] initWithQuery:query
targetID:limboTargetID
listenSequenceNumber:kIrrelevantSequenceNumber
purpose:FSTQueryPurposeLimboResolution];
self.limboKeysByTarget[@(limboTargetID)] = key;
[self.remoteStore listenToTargetWithQueryData:queryData];
Expand Down
2 changes: 2 additions & 0 deletions Firestore/Source/Core/FSTTypes.h
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,8 @@ typedef int32_t FSTBatchID;

typedef int32_t FSTTargetID;

typedef int64_t FSTListenSequenceNumber;

typedef NSNumber FSTBoxedTargetID;

/**
Expand Down
Loading

0 comments on commit 77c95ef

Please sign in to comment.