aboutsummaryrefslogtreecommitdiffhomepage
path: root/Firestore/Example/Tests/Util/FSTDispatchQueueTests.mm
blob: 5ef860ca4b96a66e55c2f46ed3063727ace8b28e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
/*
 * 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 "Firestore/Source/Util/FSTDispatchQueue.h"

#import <XCTest/XCTest.h>

#import "Firestore/Example/Tests/Util/XCTestCase+Await.h"

// In these generic tests the specific TimerIDs don't matter.
static const FSTTimerID timerID1 = FSTTimerIDListenStreamConnectionBackoff;
static const FSTTimerID timerID2 = FSTTimerIDListenStreamIdle;
static const FSTTimerID timerID3 = FSTTimerIDWriteStreamConnectionBackoff;

@interface FSTDispatchQueueTests : XCTestCase
@end

@implementation FSTDispatchQueueTests {
  FSTDispatchQueue *_queue;
  NSMutableArray *_completedSteps;
  NSArray *_expectedSteps;
  XCTestExpectation *_expectation;
}

- (void)setUp {
  [super setUp];
  dispatch_queue_t dispatch_queue =
      dispatch_queue_create("FSTDispatchQueueTests", DISPATCH_QUEUE_SERIAL);
  _queue = [[FSTDispatchQueue alloc] initWithQueue:dispatch_queue];
  _completedSteps = [NSMutableArray array];
  _expectedSteps = nil;
}

/**
 * Helper to return a block that adds @(n) to _completedSteps when run and fulfils _expectation if
 * the _completedSteps match the _expectedSteps.
 */
- (void (^)())blockForStep:(int)n {
  return ^void() {
    [self->_completedSteps addObject:@(n)];
    if (self->_expectedSteps && self->_completedSteps.count >= self->_expectedSteps.count) {
      XCTAssertEqualObjects(self->_completedSteps, self->_expectedSteps);
      [self->_expectation fulfill];
    }
  };
}

- (void)testCanScheduleCallbacksInTheFuture {
  _expectation = [self expectationWithDescription:@"Expected steps"];
  _expectedSteps = @[ @1, @2, @3, @4 ];
  [_queue dispatchAsync:[self blockForStep:1]];
  [_queue dispatchAfterDelay:0.005 timerID:timerID1 block:[self blockForStep:4]];
  [_queue dispatchAfterDelay:0.001 timerID:timerID2 block:[self blockForStep:3]];
  [_queue dispatchAsync:[self blockForStep:2]];

  [self awaitExpectations];
}

- (void)testCanCancelDelayedCallbacks {
  _expectation = [self expectationWithDescription:@"Expected steps"];
  _expectedSteps = @[ @1, @3 ];
  // Queue everything from the queue to ensure nothing completes before we cancel.
  [_queue dispatchAsync:^{
    [_queue dispatchAsyncAllowingSameQueue:[self blockForStep:1]];
    FSTDelayedCallback *step2Timer =
        [_queue dispatchAfterDelay:.001 timerID:timerID1 block:[self blockForStep:2]];
    [_queue dispatchAfterDelay:.005 timerID:timerID2 block:[self blockForStep:3]];

    XCTAssertTrue([_queue containsDelayedCallbackWithTimerID:timerID1]);
    [step2Timer cancel];
    XCTAssertFalse([_queue containsDelayedCallbackWithTimerID:timerID1]);
  }];

  [self awaitExpectations];
}

- (void)testCanManuallyDrainAllDelayedCallbacksForTesting {
  [_queue dispatchAsync:[self blockForStep:1]];
  [_queue dispatchAfterDelay:20 timerID:timerID1 block:[self blockForStep:4]];
  [_queue dispatchAfterDelay:10 timerID:timerID2 block:[self blockForStep:3]];
  [_queue dispatchAsync:[self blockForStep:2]];

  [_queue runDelayedCallbacksUntil:FSTTimerIDAll];
  XCTAssertEqualObjects(_completedSteps, (@[ @1, @2, @3, @4 ]));
}

- (void)testCanManuallyDrainSpecificDelayedCallbacksForTesting {
  [_queue dispatchAsync:[self blockForStep:1]];
  [_queue dispatchAfterDelay:20 timerID:timerID1 block:[self blockForStep:5]];
  [_queue dispatchAfterDelay:10 timerID:timerID2 block:[self blockForStep:3]];
  [_queue dispatchAfterDelay:15 timerID:timerID3 block:[self blockForStep:4]];
  [_queue dispatchAsync:[self blockForStep:2]];

  [_queue runDelayedCallbacksUntil:timerID3];
  XCTAssertEqualObjects(_completedSteps, (@[ @1, @2, @3, @4 ]));
}

@end