1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
module TimeSpecs
class SubTime < Time; end
class MethodHolder
class << self
define_method(:now, &Time.method(:now))
define_method(:new, &Time.method(:new))
end
end
class Timezone
def initialize(options)
@offset = options[:offset]
end
def local_to_utc(t)
t - @offset
end
def utc_to_local(t)
t + @offset
end
end
class TimezoneMethodCallRecorder < Timezone
def initialize(options, &blk)
super(options)
@blk = blk
end
def local_to_utc(t)
@blk.call(t)
super
end
def utc_to_local(t)
@blk.call(t)
super
end
end
class TimeLikeArgumentRecorder
def self.result
arguments = []
zone = TimeSpecs::TimezoneMethodCallRecorder.new(offset: 0) do |obj|
arguments << obj
end
# ensure timezone's methods are called at least once
Time.new(2000, 1, 1, 12, 0, 0, zone)
return arguments[0]
end
end
Z = Struct.new(:offset, :abbr)
Zone = Struct.new(:std, :dst, :dst_range)
Zones = {
"Asia/Colombo" => Zone[Z[5*3600+30*60, "MMT"], nil, nil],
"PST" => Zone[Z[(-9*60*60), "PST"], nil, nil],
}
class TimezoneWithName < Timezone
attr_reader :name
def initialize(options)
@name = options[:name]
@std, @dst, @dst_range = *Zones[@name]
end
def dst?(t)
@dst_range&.cover?(t.mon)
end
def zone(t)
(dst?(t) ? @dst : @std)
end
def utc_offset(t)
zone(t)&.offset || 0
end
def abbr(t)
zone(t)&.abbr
end
def local_to_utc(t)
t - utc_offset(t)
end
def utc_to_local(t)
t + utc_offset(t)
end
end
class TimeWithFindTimezone < Time
def self.find_timezone(name)
TimezoneWithName.new(name: name.to_s)
end
end
TimezoneWithAbbr = TimezoneWithName
end
|