Uma maneira limpa de fazer isso é injetar o VirtualTime. Permite controlar o tempo. Primeiro instale o VirtualTime
Install-Package VirtualTime
Isso permite, por exemplo, acelerar o tempo 5 vezes mais em todas as chamadas para DateTime.Now ou UtcNow
var DateTime = DateTime.Now.ToVirtualTime(5);
Para tornar o tempo mais lento, por exemplo, 5 vezes mais lento,
var DateTime = DateTime.Now.ToVirtualTime(0.5);
Para fazer o tempo parar, faça
var DateTime = DateTime.Now.ToVirtualTime(0);
Retroceder no tempo ainda não foi testado
Aqui está um exemplo de teste:
[TestMethod]
public void it_should_make_time_move_faster()
{
int speedOfTimePerMs = 1000;
int timeToPassMs = 3000;
int expectedElapsedVirtualTime = speedOfTimePerMs * timeToPassMs;
DateTime whenTimeStarts = DateTime.Now;
ITime time = whenTimeStarts.ToVirtualTime(speedOfTimePerMs);
Thread.Sleep(timeToPassMs);
DateTime expectedTime = DateTime.Now.AddMilliseconds(expectedElapsedVirtualTime - timeToPassMs);
DateTime virtualTime = time.Now;
Assert.IsTrue(TestHelper.AreEqualWithinMarginOfError(expectedTime, virtualTime, MarginOfErrorMs));
}
Você pode conferir mais testes aqui:
https://github.com/VirtualTime/VirtualTime/blob/master/VirtualTimeLib.Tests/when_virtual_time_is_used.cs
O que a extensão DateTime.Now.ToVirtualTime oferece é uma instância do ITime que você passa para um método / classe que depende do ITime. some DateTime.Now.ToVirtualTime está configurado no contêiner DI de sua escolha
Aqui está outro exemplo de injeção em um controlador de classe
public class AlarmClock
{
private ITime DateTime;
public AlarmClock(ITime dateTime, int numberOfHours)
{
DateTime = dateTime;
SetTime = DateTime.UtcNow.AddHours(numberOfHours);
Task.Run(() =>
{
while (!IsAlarmOn)
{
IsAlarmOn = (SetTime - DateTime.UtcNow).TotalMilliseconds < 0;
}
});
}
public DateTime SetTime { get; set; }
public bool IsAlarmOn { get; set; }
}
[TestMethod]
public void it_can_be_injected_as_a_dependency()
{
//virtual time has to be 1000*3.75 faster to get to an hour
//in 1000 ms real time
var dateTime = DateTime.Now.ToVirtualTime(1000 * 3.75);
var numberOfHoursBeforeAlarmSounds = 1;
var alarmClock = new AlarmClock(dateTime, numberOfHoursBeforeAlarmSounds);
Assert.IsFalse(alarmClock.IsAlarmOn);
System.Threading.Thread.Sleep(1000);
Assert.IsTrue(alarmClock.IsAlarmOn);
}