我正在使用timecop gem来存根日期。我也在使用factory_girl,以便更轻松地为我的规范创建对象。
这是我的工厂:
FactoryGirl.define do
factory :fiscal_year do
start_date {Date.today}
end_date {Date.today + 1.day}
trait(:sequenced_start_date) {sequence(:start_date, Date.today.year) {|n| Date.new(n, 10, 1) }}
trait(:sequenced_end_date) {sequence(:end_date, Date.today.year + 1) {|n| Date.new(n, 9, 30) }}
factory :sequenced_fiscal_year, parent: :fiscal_year,
traits: [:sequenced_start_date, :sequenced_end_date]
end
end
假设今天的实际日期是09/07/2016。
这是我的规格:
RSpec.describe FiscalYear, type: :model do
before(:example) do
Timecop.freeze(Time.local(2010))
end
after do
Timecop.return
end
describe "fake spec to show unexpected sequence behavior" do
it "tests something" do
puts "Today's frozen date thanks to Timecop: #{Date.today}"
fiscal_year = create(:sequenced_fiscal_year)
puts "Sequenced Date: #{fiscal_year.end_date}"
end
end
end
当运行规范时,输出是
Today's frozen date thanks to Timecop: 01/01/2010
Sequenced Date: 09/30/2017
在第一个输出行中:当我调用Date.today
时,Timecop按预期工作。它已成功冻结时间并返回冻结时间。
但是:第二行无法按预期工作。它显然使用系统Date
而不是Timecop的冻结Date
。排序的end_date
应该按当前冻结的日期顺序添加+1年。因此:我希望此处end_date
为09/30/2011
,而不是09/30/2017
。
如何制作,以便排序的end_date
属性与Timecop的Date
相比,而不是系统的Date
?
答案 0 :(得分:2)
问题在于我传递给sequence
方法的第二个参数。该参数在初始化时是静态/评估的。它不是动态的,所以timecop
没有机会在评估第二个参数之前将日期存根。
trait(:sequenced_end_date) {sequence(:end_date, Date.today.year + 1) {|n| Date.new(n, 9, 30) }}
传递给Date.today.year
方法的第二个参数的sequence
部分是静态的,并在初始化时进行评估。它不是动态的,并且在调用方法之前已经过评估。
我只需将这些方法更改为以下内容:
trait(:sequenced_year_start_date) {sequence(:start_date) {|n| Date.new(Date.today.year, 10, 1) }}
trait(:sequenced_year_end_date) {sequence(:end_date) {|n| Date.new(Date.today.year + n, 9, 30) }}
现在它按预期工作并返回以下内容:
Today's frozen date thanks to Timecop: 01/01/2010
Sequenced Date: 09/30/2011
块是动态的,因此在块中调用Date.today.year + n
使timecop有机会在执行此块之前首先删除Date
。