tzinfo 1.2.6 → 2.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (145) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data.tar.gz.sig +1 -1
  4. data/.yardopts +3 -0
  5. data/CHANGES.md +482 -380
  6. data/LICENSE +12 -12
  7. data/README.md +368 -114
  8. data/lib/tzinfo.rb +64 -29
  9. data/lib/tzinfo/country.rb +141 -129
  10. data/lib/tzinfo/country_timezone.rb +70 -112
  11. data/lib/tzinfo/data_source.rb +389 -144
  12. data/lib/tzinfo/data_sources.rb +8 -0
  13. data/lib/tzinfo/data_sources/constant_offset_data_timezone_info.rb +56 -0
  14. data/lib/tzinfo/data_sources/country_info.rb +42 -0
  15. data/lib/tzinfo/data_sources/data_timezone_info.rb +91 -0
  16. data/lib/tzinfo/data_sources/linked_timezone_info.rb +33 -0
  17. data/lib/tzinfo/data_sources/ruby_data_source.rb +143 -0
  18. data/lib/tzinfo/data_sources/timezone_info.rb +47 -0
  19. data/lib/tzinfo/data_sources/transitions_data_timezone_info.rb +214 -0
  20. data/lib/tzinfo/data_sources/zoneinfo_data_source.rb +575 -0
  21. data/lib/tzinfo/data_sources/zoneinfo_reader.rb +286 -0
  22. data/lib/tzinfo/data_timezone.rb +33 -47
  23. data/lib/tzinfo/datetime_with_offset.rb +153 -0
  24. data/lib/tzinfo/format1.rb +10 -0
  25. data/lib/tzinfo/format1/country_definer.rb +17 -0
  26. data/lib/tzinfo/format1/country_index_definition.rb +64 -0
  27. data/lib/tzinfo/format1/timezone_definer.rb +64 -0
  28. data/lib/tzinfo/format1/timezone_definition.rb +39 -0
  29. data/lib/tzinfo/format1/timezone_index_definition.rb +77 -0
  30. data/lib/tzinfo/format2.rb +10 -0
  31. data/lib/tzinfo/format2/country_definer.rb +68 -0
  32. data/lib/tzinfo/format2/country_index_definer.rb +68 -0
  33. data/lib/tzinfo/format2/country_index_definition.rb +46 -0
  34. data/lib/tzinfo/format2/timezone_definer.rb +94 -0
  35. data/lib/tzinfo/format2/timezone_definition.rb +73 -0
  36. data/lib/tzinfo/format2/timezone_index_definer.rb +45 -0
  37. data/lib/tzinfo/format2/timezone_index_definition.rb +55 -0
  38. data/lib/tzinfo/info_timezone.rb +26 -21
  39. data/lib/tzinfo/linked_timezone.rb +33 -52
  40. data/lib/tzinfo/offset_timezone_period.rb +42 -0
  41. data/lib/tzinfo/string_deduper.rb +118 -0
  42. data/lib/tzinfo/time_with_offset.rb +128 -0
  43. data/lib/tzinfo/timestamp.rb +548 -0
  44. data/lib/tzinfo/timestamp_with_offset.rb +85 -0
  45. data/lib/tzinfo/timezone.rb +989 -502
  46. data/lib/tzinfo/timezone_offset.rb +84 -74
  47. data/lib/tzinfo/timezone_period.rb +151 -217
  48. data/lib/tzinfo/timezone_proxy.rb +70 -79
  49. data/lib/tzinfo/timezone_transition.rb +77 -109
  50. data/lib/tzinfo/transitions_timezone_period.rb +63 -0
  51. data/lib/tzinfo/untaint_ext.rb +18 -0
  52. data/lib/tzinfo/version.rb +7 -0
  53. data/lib/tzinfo/with_offset.rb +61 -0
  54. metadata +43 -99
  55. metadata.gz.sig +0 -0
  56. data/Rakefile +0 -107
  57. data/lib/tzinfo/country_index_definition.rb +0 -31
  58. data/lib/tzinfo/country_info.rb +0 -42
  59. data/lib/tzinfo/data_timezone_info.rb +0 -55
  60. data/lib/tzinfo/linked_timezone_info.rb +0 -26
  61. data/lib/tzinfo/offset_rationals.rb +0 -77
  62. data/lib/tzinfo/ruby_core_support.rb +0 -176
  63. data/lib/tzinfo/ruby_country_info.rb +0 -74
  64. data/lib/tzinfo/ruby_data_source.rb +0 -138
  65. data/lib/tzinfo/time_or_datetime.rb +0 -340
  66. data/lib/tzinfo/timezone_definition.rb +0 -36
  67. data/lib/tzinfo/timezone_index_definition.rb +0 -54
  68. data/lib/tzinfo/timezone_info.rb +0 -30
  69. data/lib/tzinfo/timezone_transition_definition.rb +0 -104
  70. data/lib/tzinfo/transition_data_timezone_info.rb +0 -274
  71. data/lib/tzinfo/zoneinfo_country_info.rb +0 -37
  72. data/lib/tzinfo/zoneinfo_data_source.rb +0 -496
  73. data/lib/tzinfo/zoneinfo_timezone_info.rb +0 -298
  74. data/test/tc_country.rb +0 -236
  75. data/test/tc_country_index_definition.rb +0 -69
  76. data/test/tc_country_info.rb +0 -16
  77. data/test/tc_country_timezone.rb +0 -173
  78. data/test/tc_data_source.rb +0 -218
  79. data/test/tc_data_timezone.rb +0 -99
  80. data/test/tc_data_timezone_info.rb +0 -18
  81. data/test/tc_info_timezone.rb +0 -34
  82. data/test/tc_linked_timezone.rb +0 -155
  83. data/test/tc_linked_timezone_info.rb +0 -23
  84. data/test/tc_offset_rationals.rb +0 -23
  85. data/test/tc_ruby_core_support.rb +0 -168
  86. data/test/tc_ruby_country_info.rb +0 -110
  87. data/test/tc_ruby_data_source.rb +0 -165
  88. data/test/tc_time_or_datetime.rb +0 -660
  89. data/test/tc_timezone.rb +0 -1359
  90. data/test/tc_timezone_definition.rb +0 -113
  91. data/test/tc_timezone_index_definition.rb +0 -73
  92. data/test/tc_timezone_info.rb +0 -11
  93. data/test/tc_timezone_london.rb +0 -143
  94. data/test/tc_timezone_melbourne.rb +0 -142
  95. data/test/tc_timezone_new_york.rb +0 -142
  96. data/test/tc_timezone_offset.rb +0 -126
  97. data/test/tc_timezone_period.rb +0 -555
  98. data/test/tc_timezone_proxy.rb +0 -136
  99. data/test/tc_timezone_transition.rb +0 -366
  100. data/test/tc_timezone_transition_definition.rb +0 -295
  101. data/test/tc_timezone_utc.rb +0 -27
  102. data/test/tc_transition_data_timezone_info.rb +0 -433
  103. data/test/tc_zoneinfo_country_info.rb +0 -78
  104. data/test/tc_zoneinfo_data_source.rb +0 -1204
  105. data/test/tc_zoneinfo_timezone_info.rb +0 -1234
  106. data/test/test_utils.rb +0 -188
  107. data/test/ts_all.rb +0 -7
  108. data/test/ts_all_ruby.rb +0 -5
  109. data/test/ts_all_zoneinfo.rb +0 -9
  110. data/test/tzinfo-data/tzinfo/data.rb +0 -8
  111. data/test/tzinfo-data/tzinfo/data/definitions/America/Argentina/Buenos_Aires.rb +0 -89
  112. data/test/tzinfo-data/tzinfo/data/definitions/America/New_York.rb +0 -315
  113. data/test/tzinfo-data/tzinfo/data/definitions/Australia/Melbourne.rb +0 -218
  114. data/test/tzinfo-data/tzinfo/data/definitions/EST.rb +0 -19
  115. data/test/tzinfo-data/tzinfo/data/definitions/Etc/GMT__m__1.rb +0 -21
  116. data/test/tzinfo-data/tzinfo/data/definitions/Etc/GMT__p__1.rb +0 -21
  117. data/test/tzinfo-data/tzinfo/data/definitions/Etc/UTC.rb +0 -21
  118. data/test/tzinfo-data/tzinfo/data/definitions/Europe/Amsterdam.rb +0 -261
  119. data/test/tzinfo-data/tzinfo/data/definitions/Europe/Andorra.rb +0 -186
  120. data/test/tzinfo-data/tzinfo/data/definitions/Europe/London.rb +0 -321
  121. data/test/tzinfo-data/tzinfo/data/definitions/Europe/Paris.rb +0 -265
  122. data/test/tzinfo-data/tzinfo/data/definitions/Europe/Prague.rb +0 -220
  123. data/test/tzinfo-data/tzinfo/data/definitions/UTC.rb +0 -16
  124. data/test/tzinfo-data/tzinfo/data/indexes/countries.rb +0 -927
  125. data/test/tzinfo-data/tzinfo/data/indexes/timezones.rb +0 -596
  126. data/test/tzinfo-data/tzinfo/data/version.rb +0 -14
  127. data/test/zoneinfo/America/Argentina/Buenos_Aires +0 -0
  128. data/test/zoneinfo/America/New_York +0 -0
  129. data/test/zoneinfo/Australia/Melbourne +0 -0
  130. data/test/zoneinfo/EST +0 -0
  131. data/test/zoneinfo/Etc/UTC +0 -0
  132. data/test/zoneinfo/Europe/Amsterdam +0 -0
  133. data/test/zoneinfo/Europe/Andorra +0 -0
  134. data/test/zoneinfo/Europe/London +0 -0
  135. data/test/zoneinfo/Europe/Paris +0 -0
  136. data/test/zoneinfo/Europe/Prague +0 -0
  137. data/test/zoneinfo/Factory +0 -0
  138. data/test/zoneinfo/iso3166.tab +0 -275
  139. data/test/zoneinfo/leapseconds +0 -61
  140. data/test/zoneinfo/posix/Europe/London +0 -0
  141. data/test/zoneinfo/posixrules +0 -0
  142. data/test/zoneinfo/right/Europe/London +0 -0
  143. data/test/zoneinfo/zone.tab +0 -439
  144. data/test/zoneinfo/zone1970.tab +0 -369
  145. data/tzinfo.gemspec +0 -21
@@ -0,0 +1,286 @@
1
+ # encoding: UTF-8
2
+ # frozen_string_literal: true
3
+
4
+ module TZInfo
5
+ using UntaintExt if TZInfo.const_defined?(:UntaintExt)
6
+
7
+ module DataSources
8
+ # An {InvalidZoneinfoFile} exception is raised if an attempt is made to load
9
+ # an invalid zoneinfo file.
10
+ class InvalidZoneinfoFile < StandardError
11
+ end
12
+
13
+ # Reads compiled zoneinfo TZif (\0, 2 or 3) files.
14
+ class ZoneinfoReader #:nodoc:
15
+ # Initializes a new {ZoneinfoReader}.
16
+ #
17
+ # @param string_deduper [StringDeduper] a {StringDeduper} instance to use
18
+ # when deduping abbreviations.
19
+ def initialize(string_deduper)
20
+ @string_deduper = string_deduper
21
+ end
22
+
23
+ # Reads a zoneinfo structure from the given path. Returns either a
24
+ # {TimezoneOffset} that is constantly observed or an `Array`
25
+ # {TimezoneTransition}s.
26
+ #
27
+ # @param file_path [String] the path of a zoneinfo file.
28
+ # @return [Object] either a {TimezoneOffset} or an `Array` of
29
+ # {TimezoneTransition}s.
30
+ # @raise [SecurityError] if safe mode is enabled and `file_path` is
31
+ # tainted.
32
+ # @raise [InvalidZoneinfoFile] if `file_path`` does not refer to a valid
33
+ # zoneinfo file.
34
+ def read(file_path)
35
+ File.open(file_path, 'rb') { |file| parse(file) }
36
+ end
37
+
38
+ private
39
+
40
+ # Translates an unsigned 32-bit integer (as returned by unpack) to signed
41
+ # 32-bit.
42
+ #
43
+ # @param long [Integer] an unsigned 32-bit integer.
44
+ # @return [Integer] {long} translated to signed 32-bit.
45
+ def make_signed_int32(long)
46
+ long >= 0x80000000 ? long - 0x100000000 : long
47
+ end
48
+
49
+ # Translates a pair of unsigned 32-bit integers (as returned by unpack,
50
+ # most significant first) to a signed 64-bit integer.
51
+ #
52
+ # @param high [Integer] the most significant 32-bits.
53
+ # @param low [Integer] the least significant 32-bits.
54
+ # @return [Integer] {high} and {low} combined and translated to signed
55
+ # 64-bit.
56
+ def make_signed_int64(high, low)
57
+ unsigned = (high << 32) | low
58
+ unsigned >= 0x8000000000000000 ? unsigned - 0x10000000000000000 : unsigned
59
+ end
60
+
61
+ # Reads the given number of bytes from the given file and checks that the
62
+ # correct number of bytes could be read.
63
+ #
64
+ # @param file [IO] the file to read from.
65
+ # @param bytes [Integer] the number of bytes to read.
66
+ # @return [String] the bytes that were read.
67
+ # @raise [InvalidZoneinfoFile] if the number of bytes available didn't
68
+ # match the number requested.
69
+ def check_read(file, bytes)
70
+ result = file.read(bytes)
71
+
72
+ unless result && result.length == bytes
73
+ raise InvalidZoneinfoFile, "Expected #{bytes} bytes reading '#{file.path}', but got #{result ? result.length : 0} bytes"
74
+ end
75
+
76
+ result
77
+ end
78
+
79
+ # Zoneinfo files don't include the offset from standard time (std_offset)
80
+ # for DST periods. Derive the base offset (base_utc_offset) where DST is
81
+ # observed from either the previous or next non-DST period.
82
+ #
83
+ # @param transitions [Array<Hash>] an `Array` of transition hashes.
84
+ # @param offsets [Array<Hash>] an `Array` of offset hashes.
85
+ # @return [Integer] the index of the offset to be used prior to the first
86
+ # transition.
87
+ def derive_offsets(transitions, offsets)
88
+ # The first non-DST offset (if there is one) is the offset observed
89
+ # before the first transition. Fall back to the first DST offset if
90
+ # there are no non-DST offsets.
91
+ first_non_dst_offset_index = offsets.index {|o| !o[:is_dst] }
92
+ first_offset_index = first_non_dst_offset_index || 0
93
+ return first_offset_index if transitions.empty?
94
+
95
+ # Determine the base_utc_offset of the next non-dst offset at each transition.
96
+ base_utc_offset_from_next = nil
97
+
98
+ transitions.reverse_each do |transition|
99
+ offset = offsets[transition[:offset]]
100
+ if offset[:is_dst]
101
+ transition[:base_utc_offset_from_next] = base_utc_offset_from_next if base_utc_offset_from_next
102
+ else
103
+ base_utc_offset_from_next = offset[:observed_utc_offset]
104
+ end
105
+ end
106
+
107
+ base_utc_offset_from_previous = first_non_dst_offset_index ? offsets[first_non_dst_offset_index][:observed_utc_offset] : nil
108
+ defined_offsets = {}
109
+
110
+ transitions.each do |transition|
111
+ offset_index = transition[:offset]
112
+ offset = offsets[offset_index]
113
+ observed_utc_offset = offset[:observed_utc_offset]
114
+
115
+ if offset[:is_dst]
116
+ base_utc_offset_from_next = transition[:base_utc_offset_from_next]
117
+
118
+ difference_to_previous = (observed_utc_offset - (base_utc_offset_from_previous || observed_utc_offset)).abs
119
+ difference_to_next = (observed_utc_offset - (base_utc_offset_from_next || observed_utc_offset)).abs
120
+
121
+ base_utc_offset = if difference_to_previous == 3600
122
+ base_utc_offset_from_previous
123
+ elsif difference_to_next == 3600
124
+ base_utc_offset_from_next
125
+ elsif difference_to_previous > 0 && difference_to_next > 0
126
+ difference_to_previous < difference_to_next ? base_utc_offset_from_previous : base_utc_offset_from_next
127
+ elsif difference_to_previous > 0
128
+ base_utc_offset_from_previous
129
+ elsif difference_to_next > 0
130
+ base_utc_offset_from_next
131
+ else
132
+ # No difference, assume a 1 hour offset from standard time.
133
+ observed_utc_offset - 3600
134
+ end
135
+
136
+ if !offset[:base_utc_offset]
137
+ offset[:base_utc_offset] = base_utc_offset
138
+ defined_offsets[offset] = offset_index
139
+ elsif offset[:base_utc_offset] != base_utc_offset
140
+ # An earlier transition has already derived a different
141
+ # base_utc_offset. Define a new offset or reuse an existing identically
142
+ # defined offset.
143
+ new_offset = offset.dup
144
+ new_offset[:base_utc_offset] = base_utc_offset
145
+
146
+ offset_index = defined_offsets[new_offset]
147
+
148
+ unless offset_index
149
+ offsets << new_offset
150
+ offset_index = offsets.length - 1
151
+ defined_offsets[new_offset] = offset_index
152
+ end
153
+
154
+ transition[:offset] = offset_index
155
+ end
156
+ else
157
+ base_utc_offset_from_previous = observed_utc_offset
158
+ end
159
+ end
160
+
161
+ first_offset_index
162
+ end
163
+
164
+ # Parses a zoneinfo file and returns either a {TimezoneOffset} that is
165
+ # constantly observed or an `Array` of {TimezoneTransition}s.
166
+ #
167
+ # @param file [IO] the file to be read.
168
+ # @return [Object] either a {TimezoneOffset} or an `Array` of
169
+ # {TimezoneTransition}s.
170
+ # @raise [InvalidZoneinfoFile] if the file is not a valid zoneinfo file.
171
+ def parse(file)
172
+ magic, version, ttisgmtcnt, ttisstdcnt, leapcnt, timecnt, typecnt, charcnt =
173
+ check_read(file, 44).unpack('a4 a x15 NNNNNN')
174
+
175
+ if magic != 'TZif'
176
+ raise InvalidZoneinfoFile, "The file '#{file.path}' does not start with the expected header."
177
+ end
178
+
179
+ if version == '2' || version == '3'
180
+ # Skip the first 32-bit section and read the header of the second 64-bit section
181
+ file.seek(timecnt * 5 + typecnt * 6 + charcnt + leapcnt * 8 + ttisgmtcnt + ttisstdcnt, IO::SEEK_CUR)
182
+
183
+ prev_version = version
184
+
185
+ magic, version, ttisgmtcnt, ttisstdcnt, leapcnt, timecnt, typecnt, charcnt =
186
+ check_read(file, 44).unpack('a4 a x15 NNNNNN')
187
+
188
+ unless magic == 'TZif' && (version == prev_version)
189
+ raise InvalidZoneinfoFile, "The file '#{file.path}' contains an invalid 64-bit section header."
190
+ end
191
+
192
+ using_64bit = true
193
+ elsif version != '3' && version != '2' && version != "\0"
194
+ raise InvalidZoneinfoFile, "The file '#{file.path}' contains a version of the zoneinfo format that is not currently supported."
195
+ else
196
+ using_64bit = false
197
+ end
198
+
199
+ unless leapcnt == 0
200
+ raise InvalidZoneinfoFile, "The file '#{file.path}' contains leap second data. TZInfo requires zoneinfo files that omit leap seconds."
201
+ end
202
+
203
+ transitions = if using_64bit
204
+ timecnt.times.map do |i|
205
+ high, low = check_read(file, 8).unpack('NN'.freeze)
206
+ transition_time = make_signed_int64(high, low)
207
+ {at: transition_time}
208
+ end
209
+ else
210
+ timecnt.times.map do |i|
211
+ transition_time = make_signed_int32(check_read(file, 4).unpack('N'.freeze)[0])
212
+ {at: transition_time}
213
+ end
214
+ end
215
+
216
+ check_read(file, timecnt).unpack('C*'.freeze).each_with_index do |localtime_type, i|
217
+ raise InvalidZoneinfoFile, "Invalid offset referenced by transition in file '#{file.path}'." if localtime_type >= typecnt
218
+ transitions[i][:offset] = localtime_type
219
+ end
220
+
221
+ offsets = typecnt.times.map do |i|
222
+ gmtoff, isdst, abbrind = check_read(file, 6).unpack('NCC'.freeze)
223
+ gmtoff = make_signed_int32(gmtoff)
224
+ isdst = isdst == 1
225
+ {observed_utc_offset: gmtoff, is_dst: isdst, abbr_index: abbrind}
226
+ end
227
+
228
+ abbrev = check_read(file, charcnt)
229
+
230
+ # Derive the offsets from standard time (std_offset).
231
+ first_offset_index = derive_offsets(transitions, offsets)
232
+
233
+ offsets = offsets.map do |o|
234
+ observed_utc_offset = o[:observed_utc_offset]
235
+ base_utc_offset = o[:base_utc_offset]
236
+
237
+ if base_utc_offset
238
+ # DST offset with base_utc_offset derived by derive_offsets.
239
+ std_offset = observed_utc_offset - base_utc_offset
240
+ elsif o[:is_dst]
241
+ # DST offset unreferenced by a transition (offset in use before the
242
+ # first transition). No derived base UTC offset, so assume 1 hour
243
+ # DST.
244
+ base_utc_offset = observed_utc_offset - 3600
245
+ std_offset = 3600
246
+ else
247
+ # Non-DST offset.
248
+ base_utc_offset = observed_utc_offset
249
+ std_offset = 0
250
+ end
251
+
252
+ abbrev_start = o[:abbr_index]
253
+ raise InvalidZoneinfoFile, "Abbreviation index is out of range in file '#{file.path}'." unless abbrev_start < abbrev.length
254
+
255
+ abbrev_end = abbrev.index("\0", abbrev_start)
256
+ raise InvalidZoneinfoFile, "Missing abbreviation null terminator in file '#{file.path}'." unless abbrev_end
257
+
258
+ abbr = @string_deduper.dedupe(abbrev[abbrev_start...abbrev_end].force_encoding(Encoding::UTF_8).untaint)
259
+
260
+ TimezoneOffset.new(base_utc_offset, std_offset, abbr)
261
+ end
262
+
263
+ first_offset = offsets[first_offset_index]
264
+
265
+
266
+ if transitions.empty?
267
+ first_offset
268
+ else
269
+ previous_offset = first_offset
270
+ previous_at = nil
271
+
272
+ transitions.map do |t|
273
+ offset = offsets[t[:offset]]
274
+ at = t[:at]
275
+ raise InvalidZoneinfoFile, "Transition at #{at} is not later than the previous transition at #{previous_at} in file '#{file.path}'." if previous_at && previous_at >= at
276
+ tt = TimezoneTransition.new(offset, previous_offset, at)
277
+ previous_offset = offset
278
+ previous_at = at
279
+ tt
280
+ end
281
+ end
282
+ end
283
+ end
284
+ private_constant :ZoneinfoReader
285
+ end
286
+ end
@@ -1,56 +1,42 @@
1
+ # encoding: UTF-8
2
+ # frozen_string_literal: true
3
+
1
4
  module TZInfo
5
+ # Represents time zones that are defined by rules that set out when
6
+ # transitions occur.
7
+ class DataTimezone < InfoTimezone
8
+ # (see Timezone#period_for)
9
+ def period_for(time)
10
+ raise ArgumentError, 'time must be specified' unless time
11
+ timestamp = Timestamp.for(time)
12
+ raise ArgumentError, 'time must have a specified utc_offset' unless timestamp.utc_offset
13
+ info.period_for(timestamp)
14
+ end
2
15
 
3
- # A Timezone based on a DataTimezoneInfo.
4
- #
5
- # @private
6
- class DataTimezone < InfoTimezone #:nodoc:
7
-
8
- # Returns the TimezonePeriod for the given UTC time. utc can either be
9
- # a DateTime, Time or integer timestamp (Time.to_i). Any timezone
10
- # information in utc is ignored (it is treated as a UTC time).
11
- #
12
- # If no TimezonePeriod could be found, PeriodNotFound is raised.
13
- def period_for_utc(utc)
14
- info.period_for_utc(utc)
16
+ # (see Timezone#periods_for_local)
17
+ def periods_for_local(local_time)
18
+ raise ArgumentError, 'local_time must be specified' unless local_time
19
+ info.periods_for_local(Timestamp.for(local_time, :ignore))
15
20
  end
16
-
17
- # Returns the set of TimezonePeriod instances that are valid for the given
18
- # local time as an array. If you just want a single period, use
19
- # period_for_local instead and specify how abiguities should be resolved.
20
- # Raises PeriodNotFound if no periods are found for the given time.
21
- def periods_for_local(local)
22
- info.periods_for_local(local)
21
+
22
+ # (see Timezone#transitions_up_to)
23
+ def transitions_up_to(to, from = nil)
24
+ raise ArgumentError, 'to must be specified' unless to
25
+ to_timestamp = Timestamp.for(to)
26
+ from_timestamp = from && Timestamp.for(from)
27
+
28
+ begin
29
+ info.transitions_up_to(to_timestamp, from_timestamp)
30
+ rescue ArgumentError => e
31
+ raise ArgumentError, e.message.gsub('_timestamp', '')
32
+ end
23
33
  end
24
-
25
- # Returns an Array of TimezoneTransition instances representing the times
26
- # where the UTC offset of the timezone changes.
27
- #
28
- # Transitions are returned up to a given date and time up to a given date
29
- # and time, specified in UTC (utc_to).
30
- #
31
- # A from date and time may also be supplied using the utc_from parameter
32
- # (also specified in UTC). If utc_from is not nil, only transitions from
33
- # that date and time onwards will be returned.
34
- #
35
- # Comparisons with utc_to are exclusive. Comparisons with utc_from are
36
- # inclusive. If a transition falls precisely on utc_to, it will be excluded.
37
- # If a transition falls on utc_from, it will be included.
38
- #
39
- # Transitions returned are ordered by when they occur, from earliest to
40
- # latest.
41
- #
42
- # utc_to and utc_from can be specified using either DateTime, Time or
43
- # integer timestamps (Time.to_i).
34
+
35
+ # Returns the canonical {Timezone} instance for this {DataTimezone}.
44
36
  #
45
- # If utc_from is specified and utc_to is not greater than utc_from, then
46
- # transitions_up_to raises an ArgumentError exception.
47
- def transitions_up_to(utc_to, utc_from = nil)
48
- info.transitions_up_to(utc_to, utc_from)
49
- end
50
-
51
- # Returns the canonical zone for this Timezone.
37
+ # For a {DataTimezone}, this is always `self`.
52
38
  #
53
- # For a DataTimezone, this is always self.
39
+ # @return [Timezone] `self`.
54
40
  def canonical_zone
55
41
  self
56
42
  end
@@ -0,0 +1,153 @@
1
+ # encoding: UTF-8
2
+ # frozen_string_literal: true
3
+
4
+ require 'date'
5
+
6
+ module TZInfo
7
+ # A subclass of `DateTime` used to represent local times. {DateTimeWithOffset}
8
+ # holds a reference to the related {TimezoneOffset} and overrides various
9
+ # methods to return results appropriate for the {TimezoneOffset}. Certain
10
+ # operations will clear the associated {TimezoneOffset} (if the
11
+ # {TimezoneOffset} would not necessarily be valid for the result). Once the
12
+ # {TimezoneOffset} has been cleared, {DateTimeWithOffset} behaves identically
13
+ # to `DateTime`.
14
+ #
15
+ # Arithmetic performed on {DateTimeWithOffset} instances is _not_ time
16
+ # zone-aware. Regardless of whether transitions in the time zone are crossed,
17
+ # results of arithmetic operations will always maintain the same offset from
18
+ # UTC (`offset`). The associated {TimezoneOffset} will aways be cleared.
19
+ class DateTimeWithOffset < DateTime
20
+ include WithOffset
21
+
22
+ # @return [TimezoneOffset] the {TimezoneOffset} associated with this
23
+ # instance.
24
+ attr_reader :timezone_offset
25
+
26
+ # Sets the associated {TimezoneOffset}.
27
+ #
28
+ # @param timezone_offset [TimezoneOffset] a {TimezoneOffset} valid at the
29
+ # time and for the offset of this {DateTimeWithOffset}.
30
+ # @return [DateTimeWithOffset] `self`.
31
+ # @raise [ArgumentError] if `timezone_offset` is `nil`.
32
+ # @raise [ArgumentError] if `timezone_offset.observed_utc_offset` does not
33
+ # equal `self.offset * 86400`.
34
+ def set_timezone_offset(timezone_offset)
35
+ raise ArgumentError, 'timezone_offset must be specified' unless timezone_offset
36
+ raise ArgumentError, 'timezone_offset.observed_utc_offset does not match self.utc_offset' if offset * 86400 != timezone_offset.observed_utc_offset
37
+ @timezone_offset = timezone_offset
38
+ self
39
+ end
40
+
41
+ # An overridden version of `DateTime#to_time` that, if there is an
42
+ # associated {TimezoneOffset}, returns a {DateTimeWithOffset} with that
43
+ # offset.
44
+ #
45
+ # @return [Time] if there is an associated {TimezoneOffset}, a
46
+ # {TimeWithOffset} representation of this {DateTimeWithOffset}, otherwise
47
+ # a `Time` representation.
48
+ def to_time
49
+ if_timezone_offset(super) do |o,t|
50
+ # Ruby 2.4.0 changed the behaviour of to_time so that it preserves the
51
+ # offset instead of converting to the system local timezone.
52
+ #
53
+ # When self has an associated TimezonePeriod, this implementation will
54
+ # preserve the offset on all versions of Ruby.
55
+ TimeWithOffset.at(t.to_i, t.subsec * 1_000_000).set_timezone_offset(o)
56
+ end
57
+ end
58
+
59
+ # An overridden version of `DateTime#downto` that clears the associated
60
+ # {TimezoneOffset} of the returned or yielded instances.
61
+ def downto(min)
62
+ if block_given?
63
+ super {|dt| yield dt.clear_timezone_offset }
64
+ else
65
+ enum = super
66
+ enum.each {|dt| dt.clear_timezone_offset }
67
+ enum
68
+ end
69
+ end
70
+
71
+ # An overridden version of `DateTime#england` that preserves the associated
72
+ # {TimezoneOffset}.
73
+ #
74
+ # @return [DateTime]
75
+ def england
76
+ # super doesn't call #new_start on MRI, so each method has to be
77
+ # individually overridden.
78
+ if_timezone_offset(super) {|o,dt| dt.set_timezone_offset(o) }
79
+ end
80
+
81
+ # An overridden version of `DateTime#gregorian` that preserves the
82
+ # associated {TimezoneOffset}.
83
+ #
84
+ # @return [DateTime]
85
+ def gregorian
86
+ # super doesn't call #new_start on MRI, so each method has to be
87
+ # individually overridden.
88
+ if_timezone_offset(super) {|o,dt| dt.set_timezone_offset(o) }
89
+ end
90
+
91
+ # An overridden version of `DateTime#italy` that preserves the associated
92
+ # {TimezoneOffset}.
93
+ #
94
+ # @return [DateTime]
95
+ def italy
96
+ # super doesn't call #new_start on MRI, so each method has to be
97
+ # individually overridden.
98
+ if_timezone_offset(super) {|o,dt| dt.set_timezone_offset(o) }
99
+ end
100
+
101
+ # An overridden version of `DateTime#julian` that preserves the associated
102
+ # {TimezoneOffset}.
103
+ #
104
+ # @return [DateTime]
105
+ def julian
106
+ # super doesn't call #new_start on MRI, so each method has to be
107
+ # individually overridden.
108
+ if_timezone_offset(super) {|o,dt| dt.set_timezone_offset(o) }
109
+ end
110
+
111
+ # An overridden version of `DateTime#new_start` that preserves the
112
+ # associated {TimezoneOffset}.
113
+ #
114
+ # @return [DateTime]
115
+ def new_start(start = Date::ITALY)
116
+ if_timezone_offset(super) {|o,dt| dt.set_timezone_offset(o) }
117
+ end
118
+
119
+ # An overridden version of `DateTime#step` that clears the associated
120
+ # {TimezoneOffset} of the returned or yielded instances.
121
+ def step(limit, step = 1)
122
+ if block_given?
123
+ super {|dt| yield dt.clear_timezone_offset }
124
+ else
125
+ enum = super
126
+ enum.each {|dt| dt.clear_timezone_offset }
127
+ enum
128
+ end
129
+ end
130
+
131
+ # An overridden version of `DateTime#upto` that clears the associated
132
+ # {TimezoneOffset} of the returned or yielded instances.
133
+ def upto(max)
134
+ if block_given?
135
+ super {|dt| yield dt.clear_timezone_offset }
136
+ else
137
+ enum = super
138
+ enum.each {|dt| dt.clear_timezone_offset }
139
+ enum
140
+ end
141
+ end
142
+
143
+ protected
144
+
145
+ # Clears the associated {TimezoneOffset}.
146
+ #
147
+ # @return [DateTimeWithOffset] `self`.
148
+ def clear_timezone_offset
149
+ @timezone_offset = nil
150
+ self
151
+ end
152
+ end
153
+ end