public async Task RecordErrorAndWait_RetrySettingsObeyed() { RetrySettings retrySettings = RetrySettings.FromExponentialBackoff( maxAttempts: int.MaxValue, // Ignored in SqlResultStream initialBackoff: TimeSpan.FromSeconds(1), maxBackoff: TimeSpan.FromSeconds(5), backoffMultiplier: 2.0, ignored => false, // Ignored in SqlResultStream RetrySettings.NoJitter); var mock = new Mock <IScheduler>(MockBehavior.Strict); mock.Setup(s => s.Delay(TimeSpan.FromSeconds(1), default)).Returns(Task.FromResult(0)); mock.Setup(s => s.Delay(TimeSpan.FromSeconds(2), default)).Returns(Task.FromResult(0)); mock.Setup(s => s.Delay(TimeSpan.FromSeconds(4), default)).Returns(Task.FromResult(0)); // Retry maxes out at 5 seconds mock.Setup(s => s.Delay(TimeSpan.FromSeconds(5), default)).Returns(Task.FromResult(0)); // After reset mock.Setup(s => s.Delay(TimeSpan.FromSeconds(1), default)).Returns(Task.FromResult(0)); var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang")); var state = new RetryState(new FakeClock(), mock.Object, retrySettings, s_callSettings); await state.WaitAsync(exception, default); await state.WaitAsync(exception, default); await state.WaitAsync(exception, default); await state.WaitAsync(exception, default); state.Reset(); await state.WaitAsync(exception, default); }
public async Task ResetDeadline() { var clock = new FakeClock(0); var scheduler = new AdvanceFakeClockScheduler(clock); var callSettings = CallSettings.FromExpiration(Expiration.FromDeadline(new DateTime(TimeSpan.FromSeconds(7).Ticks, DateTimeKind.Utc))); var state = new RetryState(clock, scheduler, s_retrySettings, callSettings); var retryInfo = new Rpc.RetryInfo { RetryDelay = Duration.FromTimeSpan(TimeSpan.FromSeconds(3)) }; Metadata trailers = new Metadata { { RetryState.RetryInfoKey, retryInfo.ToByteArray() } }; var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang"), trailers); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); // Reset does not change the absolute deadline of the call. // The next retry attempt will therefore fail. state.Reset(); Assert.True(state.CanRetry(exception)); await Assert.ThrowsAsync <RpcException>(() => state.WaitAsync(exception, default)); Assert.Equal(TimeSpan.FromSeconds(6).Ticks, clock.GetCurrentDateTimeUtc().Ticks); }
public async Task RecordErrorAndWait_BackoffSettingsObeyed() { var settings = new BackoffSettings(TimeSpan.FromSeconds(1), TimeSpan.FromSeconds(5), 2.0); var mock = new Mock <IScheduler>(MockBehavior.Strict); mock.Setup(s => s.Delay(TimeSpan.FromSeconds(1), default)).Returns(Task.FromResult(0)); mock.Setup(s => s.Delay(TimeSpan.FromSeconds(2), default)).Returns(Task.FromResult(0)); mock.Setup(s => s.Delay(TimeSpan.FromSeconds(4), default)).Returns(Task.FromResult(0)); // Retry maxes out at 5 seconds mock.Setup(s => s.Delay(TimeSpan.FromSeconds(5), default)).Returns(Task.FromResult(0)); // After reset mock.Setup(s => s.Delay(TimeSpan.FromSeconds(1), default)).Returns(Task.FromResult(0)); var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang")); var state = new RetryState(mock.Object, settings, RetrySettings.NoJitter, maxConsecutiveErrors: 5); await state.RecordErrorAndWaitAsync(exception, default); await state.RecordErrorAndWaitAsync(exception, default); await state.RecordErrorAndWaitAsync(exception, default); await state.RecordErrorAndWaitAsync(exception, default); state.Reset(); await state.RecordErrorAndWaitAsync(exception, default); }
public async Task CanRetry_MaxConsecutiveRetries_WithReset() { var state = new RetryState( new NoOpScheduler(), s_retrySettings, maxConsecutiveErrors: 2); var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang")); Assert.True(state.CanRetry(exception)); await state.RecordErrorAndWaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await state.RecordErrorAndWaitAsync(exception, default); state.Reset(); Assert.True(state.CanRetry(exception)); await state.RecordErrorAndWaitAsync(exception, default); Assert.True(state.CanRetry(exception)); }
public async Task CanRetry_WithReset() { var state = new RetryState( new FakeClock(), new NoOpScheduler(), s_retrySettings, s_callSettings); var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang")); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); state.Reset(); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); Assert.True(state.CanRetry(exception)); }
public async Task CanRetry_MaxConsecutiveRetries_WithReset() { var state = new RetryState( new NoOpScheduler(), new BackoffSettings(TimeSpan.FromSeconds(1), TimeSpan.FromSeconds(15), 2.0), RetrySettings.NoJitter, maxConsecutiveErrors: 2); var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang")); Assert.True(state.CanRetry(exception)); await state.RecordErrorAndWaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await state.RecordErrorAndWaitAsync(exception, default); state.Reset(); Assert.True(state.CanRetry(exception)); await state.RecordErrorAndWaitAsync(exception, default); Assert.True(state.CanRetry(exception)); }
public async Task ResetTimeout() { var clock = new FakeClock(0); var scheduler = new AdvanceFakeClockScheduler(clock); var callSettings = CallSettings.FromExpiration(Expiration.FromTimeout(TimeSpan.FromSeconds(7))); var state = new RetryState(clock, scheduler, s_retrySettings, callSettings); var retryInfo = new Rpc.RetryInfo { RetryDelay = Duration.FromTimeSpan(TimeSpan.FromSeconds(3)) }; Metadata trailers = new Metadata { { RetryState.RetryInfoKey, retryInfo.ToByteArray() } }; var exception = new RpcException(new Status(StatusCode.Unavailable, "Bang"), trailers); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); // Reset should set the deadline of the call to CurrentTime + Timeout. // That means that we can do two new retries without a timeout exception. state.Reset(); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await state.WaitAsync(exception, default); Assert.True(state.CanRetry(exception)); await Assert.ThrowsAsync <RpcException>(() => state.WaitAsync(exception, default)); // Verify that the clock has been advanced 12 seconds. Assert.Equal(TimeSpan.FromSeconds(12).Ticks, clock.GetCurrentDateTimeUtc().Ticks); }
// See https://github.com/googleapis/google-cloud-java/blob/master/google-cloud-clients/google-cloud-spanner/src/main/java/com/google/cloud/spanner/SpannerImpl.java#L2674 private async Task <PartialResultSet> ComputeNextAsync(CancellationToken cancellationToken) { // The retry state is local to the method as we're not trying to handle callers retrying. RetryState retryState = new RetryState(_client.Settings.Clock ?? SystemClock.Instance, _client.Settings.Scheduler ?? SystemScheduler.Instance, _retrySettings, _callSettings); while (true) { // If we've successfully read to the end of the stream and emptied the buffer, we've read all the responses. if (_finished && _buffer.Count == 0) { return(null); } // Buffer contains items up to a resume token or has reached capacity: flush. if (_buffer.Count > 0 && (_finished || !_safeToRetry || !_buffer.Last.Value.ResumeToken.IsEmpty)) { var firstResult = _buffer.First.Value; _buffer.RemoveFirst(); return(firstResult); } try { if (_grpcCall == null) { // Note: no cancellation token here; if we've been given a short cancellation token, // it ought to apply to just the MoveNext call, not the original request. _grpcCall = _request.ExecuteStreaming(_client, _callSettings); } bool hasNext = await _grpcCall.ResponseStream .MoveNext(cancellationToken) .WithSessionExpiryChecking(_session) .ConfigureAwait(false); retryState.Reset(); if (hasNext) { var next = _grpcCall.ResponseStream.Current; var hasResumeToken = !next.ResumeToken.IsEmpty; if (hasResumeToken) { _request.ResumeToken = next.ResumeToken; _safeToRetry = true; } // If the buffer is empty and this result has a resume token or we cannot resume safely // anyway, we can yield it immediately rather than placing it in the buffer to be // returned on the next iteration. if ((hasResumeToken || !_safeToRetry) && _buffer.Count == 0) { return(next); } _buffer.AddLast(next); if (_buffer.Count > _maxBufferSize && !hasResumeToken) { // We need to flush without a restart token. Errors encountered until we see // such a token will fail the read. _safeToRetry = false; } } else { _finished = true; // Let the next iteration of the loop return 0 or buffered data. } } catch (RpcException e) when(e.StatusCode == StatusCode.Cancelled && cancellationToken.IsCancellationRequested) { // gRPC throws RpcException, but it's more idiomatic to see an OperationCanceledException cancellationToken.ThrowIfCancellationRequested(); } catch (RpcException e) when(_safeToRetry && retryState.CanRetry(e)) { _client.Settings.Logger.Warn($"Exception when reading from result stream. Retrying.", e); await retryState.WaitAsync(e, cancellationToken).ConfigureAwait(false); // Clear anything we've received since the previous response that contained a resume token _buffer.Clear(); _grpcCall.Dispose(); _grpcCall = null; } } }