| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128 |
- /*
- * 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 "Firestore/Source/Remote/FSTExponentialBackoff.h"
- #include <random>
- #import "Firestore/Source/Util/FSTClasses.h"
- #import "Firestore/Source/Util/FSTDispatchQueue.h"
- #include "Firestore/core/src/firebase/firestore/util/log.h"
- #include "Firestore/core/src/firebase/firestore/util/secure_random.h"
- using firebase::firestore::util::SecureRandom;
- @interface FSTExponentialBackoff ()
- @property(nonatomic, strong) FSTDispatchQueue *dispatchQueue;
- @property(nonatomic, assign, readonly) FSTTimerID timerID;
- @property(nonatomic) double backoffFactor;
- @property(nonatomic) NSTimeInterval initialDelay;
- @property(nonatomic) NSTimeInterval maxDelay;
- @property(nonatomic) NSTimeInterval currentBase;
- @property(nonatomic) NSTimeInterval lastAttemptTime;
- @property(nonatomic, strong, nullable) FSTDelayedCallback *timerCallback;
- @end
- @implementation FSTExponentialBackoff {
- SecureRandom _secureRandom;
- }
- - (instancetype)initWithDispatchQueue:(FSTDispatchQueue *)dispatchQueue
- timerID:(FSTTimerID)timerID
- initialDelay:(NSTimeInterval)initialDelay
- backoffFactor:(double)backoffFactor
- maxDelay:(NSTimeInterval)maxDelay {
- if (self = [super init]) {
- _dispatchQueue = dispatchQueue;
- _timerID = timerID;
- _initialDelay = initialDelay;
- _backoffFactor = backoffFactor;
- _maxDelay = maxDelay;
- _lastAttemptTime = [[NSDate date] timeIntervalSince1970];
- [self reset];
- }
- return self;
- }
- - (void)reset {
- _currentBase = 0;
- }
- - (void)resetToMax {
- _currentBase = _maxDelay;
- }
- - (void)backoffAndRunBlock:(void (^)(void))block {
- [self cancel];
- // First schedule the block using the current base (which may be 0 and should be honored as such).
- NSTimeInterval desiredDelayWithJitter = _currentBase + [self jitterDelay];
- // Guard against lastAttemptTime being in the future due to a clock change.
- NSTimeInterval delaySoFar = MAX(0, [[NSDate date] timeIntervalSince1970] - self.lastAttemptTime);
- // Guard against the backoff delay already being past.
- NSTimeInterval remainingDelay = MAX(0, desiredDelayWithJitter - delaySoFar);
- if (_currentBase > 0) {
- LOG_DEBUG(
- "Backing off for %s seconds ("
- "base delay: %s seconds, "
- "delay with jitter: %s seconds, "
- "last attempt: %s seconds ago)",
- remainingDelay, _currentBase, desiredDelayWithJitter, delaySoFar);
- }
- FSTWeakify(self);
- self.timerCallback = [self.dispatchQueue
- dispatchAfterDelay:remainingDelay
- timerID:self.timerID
- block:^{
- FSTStrongify(self);
- if (self) {
- self.lastAttemptTime = [[NSDate date] timeIntervalSince1970];
- block();
- }
- }];
- // Apply backoff factor to determine next delay and ensure it is within bounds.
- _currentBase *= _backoffFactor;
- if (_currentBase < _initialDelay) {
- _currentBase = _initialDelay;
- }
- if (_currentBase > _maxDelay) {
- _currentBase = _maxDelay;
- }
- }
- - (void)cancel {
- if (self.timerCallback) {
- [self.timerCallback cancel];
- self.timerCallback = nil;
- }
- }
- /** Returns a random value in the range [-currentBase/2, currentBase/2] */
- - (NSTimeInterval)jitterDelay {
- std::uniform_real_distribution<double> dist;
- double random_double = dist(_secureRandom);
- return (random_double - 0.5) * _currentBase;
- }
- @end
|