-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
fix:
ITimer
returned by ManualTestProvider
such that timers creat…
…ed with a due time equal to zero will fire the timer callback immediately.
- Loading branch information
Showing
4 changed files
with
596 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
294 changes: 294 additions & 0 deletions
294
...eProviderExtensions.Tests/Microsoft.Extensions.Time.Testing.Test/FakeTimeProviderTests.cs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,294 @@ | ||
#if TargetMicrosoftTestTimeProvider | ||
using SutTimeProvider = Microsoft.Extensions.Time.Testing.FakeTimeProvider; | ||
#else | ||
using SutTimeProvider = TimeProviderExtensions.ManualTimeProvider; | ||
#endif | ||
|
||
// Licensed to the .NET Foundation under one or more agreements. | ||
// The .NET Foundation licenses this file to you under the MIT license. | ||
|
||
using System; | ||
using System.Threading; | ||
using System.Threading.Tasks; | ||
using Microsoft.Extensions.Time.Testing; | ||
using Xunit; | ||
|
||
namespace Microsoft.Extensions.Time.Testing.Test; | ||
|
||
public class FakeTimeProviderTests | ||
{ | ||
[Fact] | ||
public void DefaultCtor() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
|
||
var now = timeProvider.GetUtcNow(); | ||
var timestamp = timeProvider.GetTimestamp(); | ||
var frequency = timeProvider.TimestampFrequency; | ||
|
||
Assert.Equal(2000, now.Year); | ||
Assert.Equal(1, now.Month); | ||
Assert.Equal(1, now.Day); | ||
Assert.Equal(0, now.Hour); | ||
Assert.Equal(0, now.Minute); | ||
Assert.Equal(0, now.Second); | ||
Assert.Equal(0, now.Millisecond); | ||
Assert.Equal(TimeSpan.Zero, now.Offset); | ||
Assert.Equal(10_000_000, frequency); | ||
|
||
var timestamp2 = timeProvider.GetTimestamp(); | ||
var frequency2 = timeProvider.TimestampFrequency; | ||
now = timeProvider.GetUtcNow(); | ||
|
||
Assert.Equal(2000, now.Year); | ||
Assert.Equal(1, now.Month); | ||
Assert.Equal(1, now.Day); | ||
Assert.Equal(0, now.Hour); | ||
Assert.Equal(0, now.Minute); | ||
Assert.Equal(0, now.Second); | ||
Assert.Equal(0, now.Millisecond); | ||
Assert.Equal(10_000_000, frequency2); | ||
Assert.Equal(timestamp2, timestamp); | ||
} | ||
|
||
[Fact] | ||
public void RichCtor() | ||
{ | ||
var timeProvider = new SutTimeProvider(new DateTimeOffset(2001, 2, 3, 4, 5, 6, TimeSpan.Zero)); | ||
|
||
timeProvider.Advance(TimeSpan.FromMilliseconds(8)); | ||
var pnow = timeProvider.GetTimestamp(); | ||
var frequency = timeProvider.TimestampFrequency; | ||
var now = timeProvider.GetUtcNow(); | ||
|
||
Assert.Equal(2001, now.Year); | ||
Assert.Equal(2, now.Month); | ||
Assert.Equal(3, now.Day); | ||
Assert.Equal(4, now.Hour); | ||
Assert.Equal(5, now.Minute); | ||
Assert.Equal(6, now.Second); | ||
Assert.Equal(TimeSpan.Zero, now.Offset); | ||
Assert.Equal(8, now.Millisecond); | ||
Assert.Equal(10_000_000, frequency); | ||
|
||
timeProvider.Advance(TimeSpan.FromMilliseconds(8)); | ||
var pnow2 = timeProvider.GetTimestamp(); | ||
var frequency2 = timeProvider.TimestampFrequency; | ||
now = timeProvider.GetUtcNow(); | ||
|
||
Assert.Equal(2001, now.Year); | ||
Assert.Equal(2, now.Month); | ||
Assert.Equal(3, now.Day); | ||
Assert.Equal(4, now.Hour); | ||
Assert.Equal(5, now.Minute); | ||
Assert.Equal(6, now.Second); | ||
Assert.Equal(16, now.Millisecond); | ||
Assert.Equal(10_000_000, frequency2); | ||
Assert.True(pnow2 > pnow); | ||
} | ||
|
||
[Fact] | ||
public void LocalTimeZoneIsUtc() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
var localTimeZone = timeProvider.LocalTimeZone; | ||
|
||
Assert.Equal(TimeZoneInfo.Utc, localTimeZone); | ||
} | ||
|
||
[Fact] | ||
public void GetTimestampSyncWithUtcNow() | ||
{ | ||
var timeProvider = new SutTimeProvider(new DateTimeOffset(2001, 2, 3, 4, 5, 6, TimeSpan.Zero)); | ||
|
||
var initialTimeUtcNow = timeProvider.GetUtcNow(); | ||
var initialTimestamp = timeProvider.GetTimestamp(); | ||
|
||
timeProvider.SetUtcNow(timeProvider.GetUtcNow().AddMilliseconds(1234)); | ||
|
||
var finalTimeUtcNow = timeProvider.GetUtcNow(); | ||
var finalTimeTimestamp = timeProvider.GetTimestamp(); | ||
|
||
var utcDelta = finalTimeUtcNow - initialTimeUtcNow; | ||
var perfDelta = finalTimeTimestamp - initialTimestamp; | ||
var elapsedTime = timeProvider.GetElapsedTime(initialTimestamp, finalTimeTimestamp); | ||
|
||
Assert.Equal(1, utcDelta.Seconds); | ||
Assert.Equal(234, utcDelta.Milliseconds); | ||
Assert.Equal(1234D, utcDelta.TotalMilliseconds); | ||
Assert.Equal(1.234D, (double)perfDelta / timeProvider.TimestampFrequency, 3); | ||
Assert.Equal(1234, elapsedTime.TotalMilliseconds); | ||
} | ||
|
||
[Fact] | ||
public void AdvanceGoesForward() | ||
{ | ||
var timeProvider = new SutTimeProvider(new DateTimeOffset(2001, 2, 3, 4, 5, 6, TimeSpan.Zero)); | ||
|
||
var initialTimeUtcNow = timeProvider.GetUtcNow(); | ||
var initialTimestamp = timeProvider.GetTimestamp(); | ||
|
||
timeProvider.Advance(TimeSpan.FromMilliseconds(1234)); | ||
|
||
var finalTimeUtcNow = timeProvider.GetUtcNow(); | ||
var finalTimeTimestamp = timeProvider.GetTimestamp(); | ||
|
||
var utcDelta = finalTimeUtcNow - initialTimeUtcNow; | ||
var perfDelta = finalTimeTimestamp - initialTimestamp; | ||
var elapsedTime = timeProvider.GetElapsedTime(initialTimestamp, finalTimeTimestamp); | ||
|
||
Assert.Equal(1, utcDelta.Seconds); | ||
Assert.Equal(234, utcDelta.Milliseconds); | ||
Assert.Equal(1234D, utcDelta.TotalMilliseconds); | ||
Assert.Equal(1.234D, (double)perfDelta / timeProvider.TimestampFrequency, 3); | ||
Assert.Equal(1234, elapsedTime.TotalMilliseconds); | ||
} | ||
|
||
[Fact] | ||
public void ToStr() | ||
{ | ||
var dto = new DateTimeOffset(new DateTime(2022, 1, 2, 3, 4, 5, 6), TimeSpan.Zero); | ||
|
||
var timeProvider = new SutTimeProvider(dto); | ||
Assert.Equal("2022-01-02T03:04:05.006", timeProvider.ToString()); | ||
} | ||
|
||
private readonly TimeSpan _infiniteTimeout = TimeSpan.FromMilliseconds(-1); | ||
|
||
[Fact] | ||
public void Delay_InvalidArgs() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
_ = Assert.ThrowsAsync<ArgumentOutOfRangeException>(() => timeProvider.Delay(TimeSpan.FromTicks(-1), CancellationToken.None)); | ||
_ = Assert.ThrowsAsync<ArgumentOutOfRangeException>(() => timeProvider.Delay(_infiniteTimeout, CancellationToken.None)); | ||
} | ||
|
||
[Fact] | ||
public async Task Delay_Zero() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
var t = timeProvider.Delay(TimeSpan.Zero, CancellationToken.None); | ||
await t; | ||
|
||
Assert.True(t.IsCompleted && !t.IsFaulted); | ||
} | ||
|
||
[Fact] | ||
public async Task Delay_Timeout() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
|
||
var delay = timeProvider.Delay(TimeSpan.FromMilliseconds(1), CancellationToken.None); | ||
timeProvider.Advance(); | ||
await delay; | ||
|
||
Assert.True(delay.IsCompleted); | ||
Assert.False(delay.IsFaulted); | ||
Assert.False(delay.IsCanceled); | ||
} | ||
|
||
[Fact] | ||
public async Task Delay_Cancelled() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
|
||
using var cs = new CancellationTokenSource(); | ||
var delay = timeProvider.Delay(_infiniteTimeout, cs.Token); | ||
Assert.False(delay.IsCompleted); | ||
|
||
cs.Cancel(); | ||
|
||
#pragma warning disable VSTHRD003 // Avoid awaiting foreign Tasks | ||
await Assert.ThrowsAsync<TaskCanceledException>(async () => await delay); | ||
#pragma warning restore VSTHRD003 // Avoid awaiting foreign Tasks | ||
} | ||
|
||
[Fact] | ||
public async Task CreateSource() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
|
||
using var cts = timeProvider.CreateCancellationTokenSource(TimeSpan.FromMilliseconds(1)); | ||
timeProvider.Advance(); | ||
|
||
await Assert.ThrowsAsync<TaskCanceledException>(() => timeProvider.Delay(TimeSpan.FromTicks(1), cts.Token)); | ||
} | ||
|
||
[Fact] | ||
public async Task WaitAsync() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
var source = new TaskCompletionSource<bool>(); | ||
|
||
#if NET8_0_OR_GREATER | ||
await Assert.ThrowsAsync<TimeoutException>(() => source.Task.WaitAsync(TimeSpan.FromTicks(-1), timeProvider, CancellationToken.None)); | ||
#else | ||
await Assert.ThrowsAsync<ArgumentOutOfRangeException>(() => source.Task.WaitAsync(TimeSpan.FromTicks(-1), timeProvider, CancellationToken.None)); | ||
#endif | ||
await Assert.ThrowsAsync<ArgumentOutOfRangeException>(() => source.Task.WaitAsync(TimeSpan.FromMilliseconds(-2), timeProvider, CancellationToken.None)); | ||
|
||
var t = source.Task.WaitAsync(TimeSpan.FromSeconds(100000), timeProvider, CancellationToken.None); | ||
while (!t.IsCompleted) | ||
{ | ||
timeProvider.Advance(); | ||
await Task.Delay(1); | ||
_ = source.TrySetResult(true); | ||
} | ||
|
||
Assert.True(t.IsCompleted); | ||
Assert.False(t.IsFaulted); | ||
Assert.False(t.IsCanceled); | ||
} | ||
|
||
[Fact] | ||
public async Task WaitAsync_InfiniteTimeout() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
var source = new TaskCompletionSource<bool>(); | ||
|
||
var t = source.Task.WaitAsync(_infiniteTimeout, timeProvider, CancellationToken.None); | ||
while (!t.IsCompleted) | ||
{ | ||
timeProvider.Advance(); | ||
await Task.Delay(1); | ||
_ = source.TrySetResult(true); | ||
} | ||
|
||
Assert.True(t.IsCompleted); | ||
Assert.False(t.IsFaulted); | ||
Assert.False(t.IsCanceled); | ||
} | ||
|
||
[Fact] | ||
public async Task WaitAsync_Timeout() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
var source = new TaskCompletionSource<bool>(); | ||
|
||
var t = source.Task.WaitAsync(TimeSpan.FromMilliseconds(1), timeProvider, CancellationToken.None); | ||
while (!t.IsCompleted) | ||
{ | ||
timeProvider.Advance(); | ||
await Task.Delay(1); | ||
} | ||
|
||
Assert.True(t.IsCompleted); | ||
Assert.True(t.IsFaulted); | ||
Assert.False(t.IsCanceled); | ||
} | ||
|
||
[Fact] | ||
public async Task WaitAsync_Cancel() | ||
{ | ||
var timeProvider = new SutTimeProvider(); | ||
var source = new TaskCompletionSource<bool>(); | ||
using var cts = new CancellationTokenSource(); | ||
|
||
var t = source.Task.WaitAsync(_infiniteTimeout, timeProvider, cts.Token); | ||
cts.Cancel(); | ||
|
||
#pragma warning disable VSTHRD003 // Avoid awaiting foreign Tasks | ||
await Assert.ThrowsAsync<TaskCanceledException>(() => t).ConfigureAwait(false); | ||
#pragma warning restore VSTHRD003 // Avoid awaiting foreign Tasks | ||
} | ||
} |
Oops, something went wrong.