Class: ValidatesTimeliness::Formats
- Inherits:
-
Object
- Object
- ValidatesTimeliness::Formats
- Defined in:
- lib/validates_timeliness/formats.rb
Overview
A date and time format regular expression generator. Allows you to construct a date, time or datetime format using predefined tokens in a string. This makes it much easier to catalogue and customize the formats rather than dealing directly with regular expressions. The formats are then compiled into regular expressions for use validating date or time strings.
Formats can be added or removed to customize the set of valid date or time string values.
Constant Summary collapse
- @@time_formats =
Format tokens:
y = year m = month d = day h = hour n = minute s = second u = micro-seconds ampm = meridian (am or pm) with or without dots (e.g. am, a.m, or a.m.) _ = optional space tz = Timezone abbreviation (e.g. UTC, GMT, PST, EST) zo = Timezone offset (e.g. +10:00, -08:00, +1000) All other characters are considered literal. You can embed regexp in the format but no gurantees that it will remain intact. If you avoid the use of any token characters and regexp dots or backslashes as special characters in the regexp, it may well work as expected. For special characters use POSIX character clsses for safety.
Repeating tokens:
x = 1 or 2 digits for unit (e.g. 'h' means an hour can be '9' or '09') xx = 2 digits exactly for unit (e.g. 'hh' means an hour can only be '09')
Special Cases:
yy = 2 or 4 digit year yyyy = exactly 4 digit year mmm = month long name (e.g. 'Jul' or 'July') ddd = Day name of 3 to 9 letters (e.g. Wed or Wednesday) u = microseconds matches 1 to 6 digits Any other invalid combination of repeating tokens will be swallowed up by the next lowest length valid repeating token (e.g. yyy will be replaced with yy)
[ 'hh:nn:ss', 'hh-nn-ss', 'h:nn', 'h.nn', 'h nn', 'h-nn', 'h:nn_ampm', 'h.nn_ampm', 'h nn_ampm', 'h-nn_ampm', 'h_ampm' ]
- @@date_formats =
[ 'yyyy-mm-dd', 'yyyy/mm/dd', 'yyyy.mm.dd', 'm/d/yy', 'd/m/yy', 'm\d\yy', 'd\m\yy', 'd-m-yy', 'd.m.yy', 'd mmm yy' ]
- @@datetime_formats =
[ 'yyyy-mm-dd hh:nn:ss', 'yyyy-mm-dd h:nn', 'yyyy-mm-dd h:nn_ampm', 'yyyy-mm-dd hh:nn:ss.u', 'm/d/yy h:nn:ss', 'm/d/yy h:nn_ampm', 'm/d/yy h:nn', 'd/m/yy hh:nn:ss', 'd/m/yy h:nn_ampm', 'd/m/yy h:nn', 'ddd, dd mmm yyyy hh:nn:ss (zo|tz)', # RFC 822 'ddd mmm d hh:nn:ss zo yyyy', # Ruby time string 'yyyy-mm-ddThh:nn:ss(?:Z|zo)' # iso 8601 ]
- @@format_tokens =
All tokens available for format construction. The token array is made of token regexp, validation regexp and key for format proc mapping if any. If the token needs no format proc arg then the validation regexp should not have a capturing group, as all captured groups are passed to the format proc.
The token regexp should only use a capture group if ‘look-behind’ anchor is required. The first capture group will be considered a literal and put into the validation regexp string as-is. This is a hack.
[ { 'd' => [ /(\A|[^d])d{1}(?=[^d])/, '(\d{1,2})', :day ] }, #/ { 'ddd' => [ /d{3,}/, '(\w{3,9})' ] }, { 'dd' => [ /d{2,}/, '(\d{2})', :day ] }, { 'mmm' => [ /m{3,}/, '(\w{3,9})', :month ] }, { 'mm' => [ /m{2}/, '(\d{2})', :month ] }, { 'm' => [ /(\A|[^ap])m{1}/, '(\d{1,2})', :month ] }, { 'yyyy' => [ /y{4,}/, '(\d{4})', :year ] }, { 'yy' => [ /y{2,}/, '(\d{4}|\d{2})', :year ] }, { 'hh' => [ /h{2,}/, '(\d{2})', :hour ] }, { 'h' => [ /h{1}/, '(\d{1,2})', :hour ] }, { 'nn' => [ /n{2,}/, '(\d{2})', :min ] }, { 'n' => [ /n{1}/, '(\d{1,2})', :min ] }, { 'ss' => [ /s{2,}/, '(\d{2})', :sec ] }, { 's' => [ /s{1}/, '(\d{1,2})', :sec ] }, { 'u' => [ /u{1,}/, '(\d{1,6})', :usec ] }, { 'ampm' => [ /ampm/, '((?:[aApP])\.?[mM]\.?)', :meridian ] }, { 'zo' => [ /zo/, '([+-]\d{2}:?\d{2})', :offset ] }, { 'tz' => [ /tz/, '(?:[A-Z]{1,4})' ] }, { '_' => [ /_/, '\s?' ] } ]
- @@format_proc_args =
Arguments which will be passed to the format proc if matched in the time string. The key must be the key from the format tokens. The array consists of the arry position of the arg, the arg name, and the code to place in the time array slot. The position can be nil which means the arg won’t be placed in the array.
The code can be used to manipulate the arg value if required, otherwise should just be the arg name.
{ :year => [0, 'y', 'unambiguous_year(y)'], :month => [1, 'm', 'month_index(m)'], :day => [2, 'd', 'd'], :hour => [3, 'h', 'full_hour(h,md)'], :min => [4, 'n', 'n'], :sec => [5, 's', 's'], :usec => [6, 'u', 'microseconds(u)'], :offset => [7, 'z', 'offset_in_seconds(z)'], :meridian => [nil, 'md', nil] }
Class Method Summary collapse
- .abbr_month_names ⇒ Object
-
.add_formats(type, *add_formats) ⇒ Object
Adds new formats.
- .compile_format_expressions ⇒ Object
- .full_hour(hour, meridian) ⇒ Object
- .microseconds(usec) ⇒ Object
- .month_index(month) ⇒ Object
- .month_names ⇒ Object
- .offset_in_seconds(offset) ⇒ Object
-
.parse(string, type, options = {}) ⇒ Object
Loop through format expressions for type and call proc on matches.
-
.remove_formats(type, *remove_formats) ⇒ Object
Delete formats of specified type.
-
.remove_us_formats ⇒ Object
Removes formats where the 1 or 2 digit month comes first, to eliminate formats which are ambiguous with the European style of day then month.
- .unambiguous_year(year) ⇒ Object
Class Method Details
.abbr_month_names ⇒ Object
283 284 285 |
# File 'lib/validates_timeliness/formats.rb', line 283 def abbr_month_names defined?(I18n) ? I18n.t('date.abbr_month_names') : Date::ABBR_MONTHNAMES end |
.add_formats(type, *add_formats) ⇒ Object
Adds new formats. Must specify format type and can specify a :before option to nominate which format the new formats should be inserted in front on to take higher precedence. Error is raised if format already exists or if :before format is not found.
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 |
# File 'lib/validates_timeliness/formats.rb', line 229 def add_formats(type, *add_formats) formats = self.send("#{type}_formats") = {} = add_formats.pop if add_formats.last.is_a?(Hash) before = [:before] raise "Format for :before option #{format} was not found." if before && !formats.include?(before) add_formats.each do |format| raise "Format #{format} is already included in #{type} formats" if formats.include?(format) index = before ? formats.index(before) : -1 formats.insert(index, format) end compile_format_expressions end |
.compile_format_expressions ⇒ Object
176 177 178 179 180 |
# File 'lib/validates_timeliness/formats.rb', line 176 def compile_format_expressions @@time_expressions = compile_formats(@@time_formats) @@date_expressions = compile_formats(@@date_formats) @@datetime_expressions = compile_formats(@@datetime_formats) end |
.full_hour(hour, meridian) ⇒ Object
255 256 257 258 259 260 261 262 263 |
# File 'lib/validates_timeliness/formats.rb', line 255 def full_hour(hour, meridian) hour = hour.to_i return hour if meridian.nil? if meridian.delete('.').downcase == 'am' hour == 12 ? 0 : hour else hour == 12 ? hour : hour + 12 end end |
.microseconds(usec) ⇒ Object
287 288 289 |
# File 'lib/validates_timeliness/formats.rb', line 287 def microseconds(usec) (".#{usec}".to_f * 1_000_000).to_i end |
.month_index(month) ⇒ Object
274 275 276 277 |
# File 'lib/validates_timeliness/formats.rb', line 274 def month_index(month) return month.to_i if month.to_i.nonzero? abbr_month_names.index(month.capitalize) || month_names.index(month.capitalize) end |
.month_names ⇒ Object
279 280 281 |
# File 'lib/validates_timeliness/formats.rb', line 279 def month_names defined?(I18n) ? I18n.t('date.month_names') : Date::MONTHNAMES end |
.offset_in_seconds(offset) ⇒ Object
291 292 293 294 295 296 |
# File 'lib/validates_timeliness/formats.rb', line 291 def offset_in_seconds(offset) sign = offset =~ /^-/ ? -1 : 1 parts = offset.scan(/\d\d/).map {|p| p.to_f } parts[1] = parts[1].to_f / 60 (parts[0] + parts[1]) * sign * 3600 end |
.parse(string, type, options = {}) ⇒ Object
Loop through format expressions for type and call proc on matches. Allow pre or post match strings to exist if strict is false. Otherwise wrap regexp in start and end anchors. Returns time array if matches a format, nil otherwise.
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 |
# File 'lib/validates_timeliness/formats.rb', line 186 def parse(string, type, ={}) return string unless string.is_a?(String) .reverse_merge!(:strict => true) sets = if [:format] [:strict] = true [ send("#{type}_expressions").assoc([:format]) ] else expression_set(type, string) end matches = nil processor = sets.each do |format, regexp, proc| full = /\A#{regexp}\Z/ if [:strict] full ||= case type when :date then /\A#{regexp}/ when :time then /#{regexp}\Z/ when :datetime then /\A#{regexp}\Z/ end break(proc) if matches = full.match(string.strip) end last = [:include_offset] ? 8 : 7 if matches values = processor.call(*matches[1..last]) values[0..2] = dummy_date_for_time_type if type == :time return values end end |
.remove_formats(type, *remove_formats) ⇒ Object
Delete formats of specified type. Error raised if format not found.
216 217 218 219 220 221 222 223 |
# File 'lib/validates_timeliness/formats.rb', line 216 def remove_formats(type, *remove_formats) remove_formats.each do |format| unless self.send("#{type}_formats").delete(format) raise "Format #{format} not found in #{type} formats" end end compile_format_expressions end |
.remove_us_formats ⇒ Object
Removes formats where the 1 or 2 digit month comes first, to eliminate formats which are ambiguous with the European style of day then month. The mmm token is ignored as its not ambigous.
248 249 250 251 252 253 |
# File 'lib/validates_timeliness/formats.rb', line 248 def remove_us_formats us_format_regexp = /\Am{1,2}[^m]/ date_formats.reject! { |format| us_format_regexp =~ format } datetime_formats.reject! { |format| us_format_regexp =~ format } compile_format_expressions end |
.unambiguous_year(year) ⇒ Object
265 266 267 268 269 270 271 272 |
# File 'lib/validates_timeliness/formats.rb', line 265 def unambiguous_year(year) if year.length <= 2 century = Time.now.year.to_s[0..1].to_i century -= 1 if year.to_i >= ambiguous_year_threshold year = "#{century}#{year.rjust(2,'0')}" end year.to_i end |