I have a method that uses DateTime.now to perform a search on some data, I want to test the method with various dates but I don't know how to stub DateTime.now nor can I get it working with Timecop ( if it even works like that ).
With time cop I tried
it 'has the correct amount if falls in the previous month' do
t = "25 May".to_datetime
Timecop.travel(t)
puts DateTime.now
expect(@employee.monthly_sales).to eq 150
end
when I run the spec I can see that puts DateTime.now gives 2015-05-25T01:00:00+01:00
but having the same puts DateTime.now within the method I'm testing outputs 2015-07-24T08:57:53+01:00
(todays date).
How can I accomplish this?
------------------update---------------------------------------------------
I was setting up the records (@employee, etc.) in a before(:all)
block which seems to have caused the problem. It only works when the setup is done after the Timecop do
block. Why is this the case?
TL;DR: The problem was that
DateTime.now
was called inEmployee
beforeTimecop.freeze
was called in the specs.Timecop mocks the constructor of
Time
,Date
andDateTime
. Any instance created betweenfreeze
andreturn
(or inside afreeze
block) will be mocked.Any instance created before
freeze
or afterreturn
won't be affected because Timecop doesn't mess with existing objects.From the README (my emphasis):
So it is essential to call
Timecop.freeze
before you create theTime
object you want to mock. If youfreeze
in an RSpecbefore
block, this will be run beforesubject
is evaluated. However, if you have abefore
block where you set up your subject (@employee
in your case), and have anotherbefore
block in a nesteddescribe
, then your subject is already set up, having calledDateTime.new
before you froze time.What happens if you add the following to your
Employee
Then you run the following spec:
Instead of using a
freeze
block, you can alsofreeze
andreturn
in rspecbefore
andafter
hooks:Off-topic, but maybe have a look at http://betterspecs.org/