TimeScheduler 0.1.2-preview
See the version list below for details.
dotnet add package TimeScheduler --version 0.1.2-preview
NuGet\Install-Package TimeScheduler -Version 0.1.2-preview
<PackageReference Include="TimeScheduler" Version="0.1.2-preview" />
paket add TimeScheduler --version 0.1.2-preview
#r "nuget: TimeScheduler, 0.1.2-preview"
// Install TimeScheduler as a Cake Addin #addin nuget:?package=TimeScheduler&version=0.1.2-preview&prerelease // Install TimeScheduler as a Cake Tool #tool nuget:?package=TimeScheduler&version=0.1.2-preview&prerelease
Time Scheduler
A library that wraps common .NET scheduling and time related operations in an abstraction, that enables controlling time during testing.
Installation
Get the latest release from https://www.nuget.org/packages/TimeScheduler
Control time during tests
If a system under test (SUT) uses things like Task.Delay
, DateTimeOffset.UtcNow
, or PeriodicTimer
,
it becomes hard to create tests that runs fast and predictably.
The idea is to replace the use of e.g. Task.Delay
with an abstraction, the ITimeScheduler
, that in production
is represented by the DefaultScheduler
, that just uses the real Task.Delay
. During testing it is now possible to
pass in TestScheduler
, that allows the test to control the progress of time, making it possible to skip ahead,
e.g. 10 minutes, and also pause time, leading to fast and predictable tests.
As an example, lets test the "Stuff Service" below that performs a specific tasks every 10 second with an additional
1 second delay. We have two versions, one that uses the standard types in .NET, and one that uses the ITimeScheduler
.
// Version of stuff service that uses the built in DateTimeOffset, PeriodicTimer, and Task.Delay
public class StuffServiceSystem
{
private static readonly TimeSpan doStuffDelay = TimeSpan.FromSeconds(10);
private readonly List<DateTimeOffset> container;
public StuffServiceSystem(List<DateTimeOffset> container)
{
this.container = container;
}
public async Task DoStuff(CancellationToken cancelllationToken)
{
using var periodicTimer = new PeriodicTimer(doStuffDelay);
while (await periodicTimer.WaitForNextTickAsync(cancellationToken))
{
await Task.Delay(TimeSpan.FromSeconds(1));
container.Add(DateTimeOffset.UtcNow);
}
}
}
// Version of stuff service that uses the built in TimeScheduler
public class StuffServiceUsingTimeScheduler
{
private static readonly TimeSpan doStuffDelay = TimeSpan.FromSeconds(10);
private readonly ITimeScheduler scheduler;
private readonly List<DateTimeOffset> container;
public StuffServiceUsingTimeScheduler(ITimeScheduler scheduler, List<DateTimeOffset> container)
{
this.scheduler = scheduler;
this.container = container;
}
public async Task DoStuff(CancellationToken cancelllationToken)
{
using var periodicTimer = scheduler.PeriodicTimer(doStuffDelay);
while (await periodicTimer.WaitForNextTickAsync(cancellationToken))
{
await scheduler.Delay(TimeSpan.FromSeconds(1));
container.Add(scheduler.UtcNow);
}
}
}
The test, using xUnit and FluentAssertions, could look like this:
[Fact]
public void DoStuff_does_stuff_every_11_seconds()
{
// Arrange
var scheduler = new TestScheduler();
var container = new List<DateTimeOffset>();
var sut = new StuffServiceUsingTimeScheduler(scheduler, container);
// Act
_ = sut.DoStuff(CancellationToken.None);
scheduler.ForwardTime(TimeSpan.FromSeconds(11));
// Assert
container
.Should()
.ContainSingle()
.Which
.Should()
.Be(scheduler.UtcNow);
}
Writing a similar test for StuffServiceSystem
is both more simple and runs much slower.
[Fact]
public async Task DoStuff_does_stuff_every_11_seconds()
{
// Arrange
var container = new List<DateTimeOffset>();
var sut = new StuffServiceSystem(container);
// Act
_ = sut.DoStuff(CancellationToken.None);
await Task.Delay(TimeSpan.FromSeconds(11));
// Assert
container
.Should()
.ContainSingle()
.Which
.Should()
.BeCloseTo(DateTimeOffset.UtcNow, precision: TimeSpan.FromMilliseconds(50));
}
Set up in production
To use in production, pass in DefaultScheduler
to the types that depend on ITimeScheduler
.
This can be done directly, or via an IoC Container, e.g. .NETs built-in IServiceCollection
like so:
services.AddSingleton<ITimeScheduler, DefaultScheduler>();
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
-
net6.0
- No dependencies.
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated | |
---|---|---|---|
0.7.1 | 227 | 4/25/2023 | |
0.7.0 | 208 | 4/25/2023 | |
0.6.0 | 210 | 4/25/2023 | |
0.5.1 | 468 | 4/16/2023 | |
0.5.0 | 215 | 4/15/2023 | |
0.4.0-preview | 162 | 4/13/2023 | |
0.3.0 | 424 | 3/3/2023 | |
0.2.0 | 660 | 2/21/2023 | |
0.1.3-preview | 334 | 1/30/2023 | |
0.1.2-preview | 178 | 1/23/2023 | |
0.1.1-preview | 200 | 1/17/2023 | |
0.1.0-preview | 179 | 1/16/2023 |