Add support for day name symbols

This commit is contained in:
Alex Dunae 2007-12-19 21:18:56 +00:00
parent 8f3f04d2b2
commit 20d513c455
2 changed files with 34 additions and 8 deletions

View file

@ -49,6 +49,7 @@ module Holidays
WEEKS = {:first => 1, :second => 2, :third => 3, :fourth => 4, :fifth => 5, :last => -1}
MONTH_LENGTHS = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
DAY_SYMBOLS = Date::DAYNAMES.collect { |n| n.downcase.intern }
# Get all holidays on a given date.
#
@ -94,25 +95,32 @@ module Holidays
dates.each do |year, months|
months.each do |month|
next unless hbm = @@holidays_by_month[month]
hbm.each do |h|
next unless in_region?(regions, h[:regions])
# Skip informal holidays unless they have been requested
next if h[:type] == :informal and not informal
if h[:function]
# Holiday definition requires a calculation
result = call_proc(h[:function], year)
# Procs may return either Date or an integer representing mday
if result.kind_of?(Date)
month = result.month
mday = result.mday
else
day = result
mday = result
end
else
# Calculate the mday
mday = h[:mday] || Date.calculate_mday(year, month, h[:week], h[:wday])
end
# Silently skip bad mdays
begin
date = Date.new(year, month, mday)
date = Date.civil(year, month, mday)
rescue; next; end
# If the :observed option is set, calculate the date when the holiday
@ -121,7 +129,6 @@ module Holidays
date = call_proc(h[:observed], date)
end
if date.between?(start_date, end_date)
holidays << {:date => date, :name => h[:name], :regions => h[:regions]}
end
@ -352,7 +359,7 @@ class Date
# [<tt>year</tt>] Integer.
# [<tt>month</tt>] Integer from 1-12.
# [<tt>week</tt>] One of <tt>:first</tt>, <tt>:second</tt>, <tt>:third</tt>,
# <tt>:fourth</tt> or <tt>:fifth</tt>.
# <tt>:fourth</tt>, <tt>:fifth</tt> or <tt>:last</tt>.
# [<tt>wday</tt>] Day of the week as an integer from 0 (Sunday) to 6
# (Saturday) or as a symbol (e.g. <tt>:monday</tt>).
#
@ -364,7 +371,7 @@ class Date
# => 7
#
# Third Thursday of December, 2008:
# Date.calculate_mday(2008, 12, :third, 4)
# Date.calculate_mday(2008, 12, :third, :thursday)
# => 18
#
# Last Monday of January, 2008:
@ -375,7 +382,12 @@ class Date
def self.calculate_mday(year, month, week, wday)
raise ArgumentError, "Week parameter must be one of Holidays::WEEKS (provided #{week})." unless WEEKS.include?(week) or WEEKS.has_value?(week)
unless wday.kind_of?(Numeric) and wday.between?(0,6) or DAY_SYMBOLS.index(wday)
raise ArgumentError, "Wday parameter must be an integer between 0 and 6 or one of Date::DAY_SYMBOLS."
end
week = WEEKS[week] if week.kind_of?(Symbol)
wday = DAY_SYMBOLS.index(wday) if wday.kind_of?(Symbol)
# :first, :second, :third, :fourth or :fifth
if week > 0

View file

@ -21,7 +21,7 @@ class DateTests < Test::Unit::TestCase
# Labour day
assert_equal 3, Date.calculate_mday(2007, 9, :first, 1)
assert_equal 1, Date.calculate_mday(2008, 9, :first, 1)
assert_equal 1, Date.calculate_mday(2008, 9, :first, :monday)
assert_equal 7, Date.calculate_mday(2009, 9, :first, 1)
assert_equal 5, Date.calculate_mday(2011, 9, :first, 1)
assert_equal 5, Date.calculate_mday(2050, 9, :first, 1)
@ -29,14 +29,14 @@ class DateTests < Test::Unit::TestCase
# Canadian thanksgiving
assert_equal 8, Date.calculate_mday(2007, 10, :second, 1)
assert_equal 13, Date.calculate_mday(2008, 10, :second, 1)
assert_equal 13, Date.calculate_mday(2008, 10, :second, :monday)
assert_equal 12, Date.calculate_mday(2009, 10, :second, 1)
assert_equal 11, Date.calculate_mday(2010, 10, :second, 1)
# Misc
assert_equal 21, Date.calculate_mday(2008, 1, :third, 1)
assert_equal 1, Date.calculate_mday(2007, 1, :first, 1)
assert_equal 2, Date.calculate_mday(2007, 3, :first, 5)
assert_equal 2, Date.calculate_mday(2007, 3, :first, :friday)
end
def test_mday_allows_integers_or_symbols
@ -67,6 +67,20 @@ class DateTests < Test::Unit::TestCase
end
end
def test_mday_requires_valid_day
assert_raises ArgumentError do
Date.calculate_mday(2008, 1, 1, :october)
end
assert_raises ArgumentError do
Date.calculate_mday(2008, 1, 1, nil)
end
assert_raises ArgumentError do
Date.calculate_mday(2008, 1, 1, 7)
end
end
def test_holiday?
assert Date.civil(2008,1,1).holiday?('ca')
end