activesupport 4.2.0 → 5.2.0

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of activesupport might be problematic. Click here for more details.

Files changed (254) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +366 -232
  3. data/MIT-LICENSE +2 -2
  4. data/README.rdoc +4 -5
  5. data/lib/active_support.rb +17 -7
  6. data/lib/active_support/all.rb +5 -3
  7. data/lib/active_support/array_inquirer.rb +48 -0
  8. data/lib/active_support/backtrace_cleaner.rb +7 -5
  9. data/lib/active_support/benchmarkable.rb +6 -4
  10. data/lib/active_support/builder.rb +3 -1
  11. data/lib/active_support/cache.rb +271 -177
  12. data/lib/active_support/cache/file_store.rb +41 -35
  13. data/lib/active_support/cache/mem_cache_store.rb +97 -88
  14. data/lib/active_support/cache/memory_store.rb +27 -30
  15. data/lib/active_support/cache/null_store.rb +7 -8
  16. data/lib/active_support/cache/redis_cache_store.rb +454 -0
  17. data/lib/active_support/cache/strategy/local_cache.rb +67 -34
  18. data/lib/active_support/cache/strategy/local_cache_middleware.rb +10 -9
  19. data/lib/active_support/callbacks.rb +654 -560
  20. data/lib/active_support/concern.rb +5 -3
  21. data/lib/active_support/concurrency/load_interlock_aware_monitor.rb +17 -0
  22. data/lib/active_support/concurrency/share_lock.rb +227 -0
  23. data/lib/active_support/configurable.rb +8 -5
  24. data/lib/active_support/core_ext.rb +3 -1
  25. data/lib/active_support/core_ext/array.rb +9 -6
  26. data/lib/active_support/core_ext/array/access.rb +29 -1
  27. data/lib/active_support/core_ext/array/conversions.rb +22 -18
  28. data/lib/active_support/core_ext/array/extract_options.rb +2 -0
  29. data/lib/active_support/core_ext/array/grouping.rb +11 -18
  30. data/lib/active_support/core_ext/array/inquiry.rb +19 -0
  31. data/lib/active_support/core_ext/array/prepend_and_append.rb +5 -3
  32. data/lib/active_support/core_ext/array/wrap.rb +7 -4
  33. data/lib/active_support/core_ext/benchmark.rb +3 -1
  34. data/lib/active_support/core_ext/big_decimal.rb +3 -1
  35. data/lib/active_support/core_ext/big_decimal/conversions.rb +10 -12
  36. data/lib/active_support/core_ext/class.rb +4 -3
  37. data/lib/active_support/core_ext/class/attribute.rb +41 -22
  38. data/lib/active_support/core_ext/class/attribute_accessors.rb +3 -1
  39. data/lib/active_support/core_ext/class/subclasses.rb +20 -8
  40. data/lib/active_support/core_ext/date.rb +6 -4
  41. data/lib/active_support/core_ext/date/acts_like.rb +3 -1
  42. data/lib/active_support/core_ext/date/blank.rb +14 -0
  43. data/lib/active_support/core_ext/date/calculations.rb +11 -9
  44. data/lib/active_support/core_ext/date/conversions.rb +31 -23
  45. data/lib/active_support/core_ext/date/zones.rb +4 -2
  46. data/lib/active_support/core_ext/date_and_time/calculations.rb +179 -56
  47. data/lib/active_support/core_ext/date_and_time/compatibility.rb +16 -0
  48. data/lib/active_support/core_ext/date_and_time/zones.rb +12 -12
  49. data/lib/active_support/core_ext/date_time.rb +7 -4
  50. data/lib/active_support/core_ext/date_time/acts_like.rb +4 -2
  51. data/lib/active_support/core_ext/date_time/blank.rb +14 -0
  52. data/lib/active_support/core_ext/date_time/calculations.rb +58 -20
  53. data/lib/active_support/core_ext/date_time/compatibility.rb +18 -0
  54. data/lib/active_support/core_ext/date_time/conversions.rb +16 -12
  55. data/lib/active_support/core_ext/digest/uuid.rb +7 -5
  56. data/lib/active_support/core_ext/enumerable.rb +107 -28
  57. data/lib/active_support/core_ext/file.rb +3 -1
  58. data/lib/active_support/core_ext/file/atomic.rb +38 -31
  59. data/lib/active_support/core_ext/hash.rb +11 -9
  60. data/lib/active_support/core_ext/hash/compact.rb +24 -15
  61. data/lib/active_support/core_ext/hash/conversions.rb +63 -43
  62. data/lib/active_support/core_ext/hash/deep_merge.rb +9 -13
  63. data/lib/active_support/core_ext/hash/except.rb +11 -8
  64. data/lib/active_support/core_ext/hash/indifferent_access.rb +4 -3
  65. data/lib/active_support/core_ext/hash/keys.rb +33 -27
  66. data/lib/active_support/core_ext/hash/reverse_merge.rb +5 -2
  67. data/lib/active_support/core_ext/hash/slice.rb +8 -8
  68. data/lib/active_support/core_ext/hash/transform_values.rb +16 -7
  69. data/lib/active_support/core_ext/integer.rb +5 -3
  70. data/lib/active_support/core_ext/integer/inflections.rb +3 -1
  71. data/lib/active_support/core_ext/integer/multiple.rb +2 -0
  72. data/lib/active_support/core_ext/integer/time.rb +11 -33
  73. data/lib/active_support/core_ext/kernel.rb +6 -5
  74. data/lib/active_support/core_ext/kernel/agnostics.rb +2 -0
  75. data/lib/active_support/core_ext/kernel/concern.rb +5 -1
  76. data/lib/active_support/core_ext/kernel/reporting.rb +4 -83
  77. data/lib/active_support/core_ext/kernel/singleton_class.rb +2 -0
  78. data/lib/active_support/core_ext/load_error.rb +3 -22
  79. data/lib/active_support/core_ext/marshal.rb +13 -10
  80. data/lib/active_support/core_ext/module.rb +14 -11
  81. data/lib/active_support/core_ext/module/aliasing.rb +6 -44
  82. data/lib/active_support/core_ext/module/anonymous.rb +12 -1
  83. data/lib/active_support/core_ext/module/attr_internal.rb +8 -9
  84. data/lib/active_support/core_ext/module/attribute_accessors.rb +43 -40
  85. data/lib/active_support/core_ext/module/attribute_accessors_per_thread.rb +150 -0
  86. data/lib/active_support/core_ext/module/concerning.rb +11 -12
  87. data/lib/active_support/core_ext/module/delegation.rb +121 -39
  88. data/lib/active_support/core_ext/module/deprecation.rb +4 -2
  89. data/lib/active_support/core_ext/module/introspection.rb +9 -9
  90. data/lib/active_support/core_ext/module/reachable.rb +5 -2
  91. data/lib/active_support/core_ext/module/redefine_method.rb +49 -0
  92. data/lib/active_support/core_ext/module/remove_method.rb +8 -3
  93. data/lib/active_support/core_ext/name_error.rb +22 -2
  94. data/lib/active_support/core_ext/numeric.rb +6 -3
  95. data/lib/active_support/core_ext/numeric/bytes.rb +22 -0
  96. data/lib/active_support/core_ext/numeric/conversions.rb +79 -74
  97. data/lib/active_support/core_ext/numeric/inquiry.rb +28 -0
  98. data/lib/active_support/core_ext/numeric/time.rb +35 -38
  99. data/lib/active_support/core_ext/object.rb +14 -13
  100. data/lib/active_support/core_ext/object/acts_like.rb +12 -1
  101. data/lib/active_support/core_ext/object/blank.rb +29 -4
  102. data/lib/active_support/core_ext/object/conversions.rb +6 -4
  103. data/lib/active_support/core_ext/object/deep_dup.rb +13 -4
  104. data/lib/active_support/core_ext/object/duplicable.rb +98 -45
  105. data/lib/active_support/core_ext/object/inclusion.rb +5 -3
  106. data/lib/active_support/core_ext/object/instance_variables.rb +3 -1
  107. data/lib/active_support/core_ext/object/json.rb +49 -19
  108. data/lib/active_support/core_ext/object/to_param.rb +3 -1
  109. data/lib/active_support/core_ext/object/to_query.rb +6 -4
  110. data/lib/active_support/core_ext/object/try.rb +70 -22
  111. data/lib/active_support/core_ext/object/with_options.rb +16 -3
  112. data/lib/active_support/core_ext/range.rb +7 -4
  113. data/lib/active_support/core_ext/range/conversions.rb +27 -7
  114. data/lib/active_support/core_ext/range/each.rb +19 -17
  115. data/lib/active_support/core_ext/range/include_range.rb +21 -19
  116. data/lib/active_support/core_ext/range/include_time_with_zone.rb +23 -0
  117. data/lib/active_support/core_ext/range/overlaps.rb +2 -0
  118. data/lib/active_support/core_ext/regexp.rb +6 -0
  119. data/lib/active_support/core_ext/securerandom.rb +25 -0
  120. data/lib/active_support/core_ext/string.rb +15 -13
  121. data/lib/active_support/core_ext/string/access.rb +9 -7
  122. data/lib/active_support/core_ext/string/behavior.rb +3 -1
  123. data/lib/active_support/core_ext/string/conversions.rb +8 -5
  124. data/lib/active_support/core_ext/string/exclude.rb +2 -0
  125. data/lib/active_support/core_ext/string/filters.rb +10 -8
  126. data/lib/active_support/core_ext/string/indent.rb +6 -4
  127. data/lib/active_support/core_ext/string/inflections.rb +61 -24
  128. data/lib/active_support/core_ext/string/inquiry.rb +3 -1
  129. data/lib/active_support/core_ext/string/multibyte.rb +15 -7
  130. data/lib/active_support/core_ext/string/output_safety.rb +35 -35
  131. data/lib/active_support/core_ext/string/starts_ends_with.rb +2 -0
  132. data/lib/active_support/core_ext/string/strip.rb +4 -5
  133. data/lib/active_support/core_ext/string/zones.rb +4 -2
  134. data/lib/active_support/core_ext/time.rb +7 -5
  135. data/lib/active_support/core_ext/time/acts_like.rb +3 -1
  136. data/lib/active_support/core_ext/time/calculations.rb +101 -51
  137. data/lib/active_support/core_ext/time/compatibility.rb +16 -0
  138. data/lib/active_support/core_ext/time/conversions.rb +20 -13
  139. data/lib/active_support/core_ext/time/zones.rb +41 -7
  140. data/lib/active_support/core_ext/uri.rb +5 -4
  141. data/lib/active_support/current_attributes.rb +195 -0
  142. data/lib/active_support/dependencies.rb +143 -160
  143. data/lib/active_support/dependencies/autoload.rb +2 -0
  144. data/lib/active_support/dependencies/interlock.rb +57 -0
  145. data/lib/active_support/deprecation.rb +12 -9
  146. data/lib/active_support/deprecation/behaviors.rb +41 -12
  147. data/lib/active_support/deprecation/constant_accessor.rb +52 -0
  148. data/lib/active_support/deprecation/instance_delegator.rb +17 -2
  149. data/lib/active_support/deprecation/method_wrappers.rb +54 -21
  150. data/lib/active_support/deprecation/proxy_wrappers.rb +56 -28
  151. data/lib/active_support/deprecation/reporting.rb +32 -12
  152. data/lib/active_support/descendants_tracker.rb +2 -0
  153. data/lib/active_support/digest.rb +20 -0
  154. data/lib/active_support/duration.rb +326 -30
  155. data/lib/active_support/duration/iso8601_parser.rb +125 -0
  156. data/lib/active_support/duration/iso8601_serializer.rb +55 -0
  157. data/lib/active_support/encrypted_configuration.rb +49 -0
  158. data/lib/active_support/encrypted_file.rb +99 -0
  159. data/lib/active_support/evented_file_update_checker.rb +205 -0
  160. data/lib/active_support/execution_wrapper.rb +128 -0
  161. data/lib/active_support/executor.rb +8 -0
  162. data/lib/active_support/file_update_checker.rb +63 -37
  163. data/lib/active_support/gem_version.rb +4 -2
  164. data/lib/active_support/gzip.rb +7 -5
  165. data/lib/active_support/hash_with_indifferent_access.rb +130 -30
  166. data/lib/active_support/i18n.rb +8 -6
  167. data/lib/active_support/i18n_railtie.rb +34 -14
  168. data/lib/active_support/inflections.rb +13 -11
  169. data/lib/active_support/inflector.rb +7 -5
  170. data/lib/active_support/inflector/inflections.rb +61 -12
  171. data/lib/active_support/inflector/methods.rb +161 -136
  172. data/lib/active_support/inflector/transliterate.rb +48 -27
  173. data/lib/active_support/json.rb +4 -2
  174. data/lib/active_support/json/decoding.rb +16 -13
  175. data/lib/active_support/json/encoding.rb +15 -57
  176. data/lib/active_support/key_generator.rb +25 -25
  177. data/lib/active_support/lazy_load_hooks.rb +50 -20
  178. data/lib/active_support/locale/en.yml +2 -0
  179. data/lib/active_support/log_subscriber.rb +13 -10
  180. data/lib/active_support/log_subscriber/test_helper.rb +14 -12
  181. data/lib/active_support/logger.rb +54 -3
  182. data/lib/active_support/logger_silence.rb +12 -7
  183. data/lib/active_support/logger_thread_safe_level.rb +33 -0
  184. data/lib/active_support/message_encryptor.rb +173 -51
  185. data/lib/active_support/message_verifier.rb +150 -17
  186. data/lib/active_support/messages/metadata.rb +71 -0
  187. data/lib/active_support/messages/rotation_configuration.rb +22 -0
  188. data/lib/active_support/messages/rotator.rb +56 -0
  189. data/lib/active_support/multibyte.rb +4 -2
  190. data/lib/active_support/multibyte/chars.rb +37 -24
  191. data/lib/active_support/multibyte/unicode.rb +100 -96
  192. data/lib/active_support/notifications.rb +11 -7
  193. data/lib/active_support/notifications/fanout.rb +10 -8
  194. data/lib/active_support/notifications/instrumenter.rb +27 -7
  195. data/lib/active_support/number_helper.rb +94 -68
  196. data/lib/active_support/number_helper/number_converter.rb +13 -11
  197. data/lib/active_support/number_helper/number_to_currency_converter.rb +9 -9
  198. data/lib/active_support/number_helper/number_to_delimited_converter.rb +9 -3
  199. data/lib/active_support/number_helper/number_to_human_converter.rb +11 -9
  200. data/lib/active_support/number_helper/number_to_human_size_converter.rb +9 -8
  201. data/lib/active_support/number_helper/number_to_percentage_converter.rb +3 -1
  202. data/lib/active_support/number_helper/number_to_phone_converter.rb +13 -4
  203. data/lib/active_support/number_helper/number_to_rounded_converter.rb +23 -56
  204. data/lib/active_support/number_helper/rounding_helper.rb +66 -0
  205. data/lib/active_support/option_merger.rb +3 -1
  206. data/lib/active_support/ordered_hash.rb +6 -4
  207. data/lib/active_support/ordered_options.rb +22 -4
  208. data/lib/active_support/per_thread_registry.rb +13 -6
  209. data/lib/active_support/proxy_object.rb +2 -0
  210. data/lib/active_support/rails.rb +16 -8
  211. data/lib/active_support/railtie.rb +43 -9
  212. data/lib/active_support/reloader.rb +131 -0
  213. data/lib/active_support/rescuable.rb +108 -53
  214. data/lib/active_support/security_utils.rb +17 -6
  215. data/lib/active_support/string_inquirer.rb +11 -3
  216. data/lib/active_support/subscriber.rb +15 -14
  217. data/lib/active_support/tagged_logging.rb +14 -11
  218. data/lib/active_support/test_case.rb +18 -46
  219. data/lib/active_support/testing/assertions.rb +137 -20
  220. data/lib/active_support/testing/autorun.rb +4 -2
  221. data/lib/active_support/testing/constant_lookup.rb +2 -1
  222. data/lib/active_support/testing/declarative.rb +3 -1
  223. data/lib/active_support/testing/deprecation.rb +14 -10
  224. data/lib/active_support/testing/file_fixtures.rb +36 -0
  225. data/lib/active_support/testing/isolation.rb +34 -25
  226. data/lib/active_support/testing/method_call_assertions.rb +43 -0
  227. data/lib/active_support/testing/setup_and_teardown.rb +12 -3
  228. data/lib/active_support/testing/stream.rb +44 -0
  229. data/lib/active_support/testing/tagged_logging.rb +3 -1
  230. data/lib/active_support/testing/time_helpers.rb +96 -27
  231. data/lib/active_support/time.rb +14 -12
  232. data/lib/active_support/time_with_zone.rb +195 -53
  233. data/lib/active_support/values/time_zone.rb +200 -61
  234. data/lib/active_support/values/unicode_tables.dat +0 -0
  235. data/lib/active_support/version.rb +3 -1
  236. data/lib/active_support/xml_mini.rb +69 -51
  237. data/lib/active_support/xml_mini/jdom.rb +116 -113
  238. data/lib/active_support/xml_mini/libxml.rb +17 -16
  239. data/lib/active_support/xml_mini/libxmlsax.rb +16 -18
  240. data/lib/active_support/xml_mini/nokogiri.rb +15 -15
  241. data/lib/active_support/xml_mini/nokogirisax.rb +15 -16
  242. data/lib/active_support/xml_mini/rexml.rb +17 -16
  243. metadata +55 -43
  244. data/lib/active_support/concurrency/latch.rb +0 -27
  245. data/lib/active_support/core_ext/big_decimal/yaml_conversions.rb +0 -14
  246. data/lib/active_support/core_ext/class/delegating_attributes.rb +0 -45
  247. data/lib/active_support/core_ext/date_time/zones.rb +0 -6
  248. data/lib/active_support/core_ext/kernel/debugger.rb +0 -10
  249. data/lib/active_support/core_ext/module/method_transplanting.rb +0 -11
  250. data/lib/active_support/core_ext/module/qualified_const.rb +0 -52
  251. data/lib/active_support/core_ext/object/itself.rb +0 -15
  252. data/lib/active_support/core_ext/struct.rb +0 -6
  253. data/lib/active_support/core_ext/thread.rb +0 -86
  254. data/lib/active_support/core_ext/time/marshal.rb +0 -30
@@ -1,5 +1,7 @@
1
- require 'date'
2
- require 'active_support/core_ext/date_and_time/zones'
1
+ # frozen_string_literal: true
2
+
3
+ require "date"
4
+ require "active_support/core_ext/date_and_time/zones"
3
5
 
4
6
  class Date
5
7
  include DateAndTime::Zones
@@ -1,23 +1,38 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/object/try"
4
+
1
5
  module DateAndTime
2
6
  module Calculations
3
7
  DAYS_INTO_WEEK = {
4
- :monday => 0,
5
- :tuesday => 1,
6
- :wednesday => 2,
7
- :thursday => 3,
8
- :friday => 4,
9
- :saturday => 5,
10
- :sunday => 6
8
+ monday: 0,
9
+ tuesday: 1,
10
+ wednesday: 2,
11
+ thursday: 3,
12
+ friday: 4,
13
+ saturday: 5,
14
+ sunday: 6
11
15
  }
16
+ WEEKEND_DAYS = [ 6, 0 ]
12
17
 
13
18
  # Returns a new date/time representing yesterday.
14
19
  def yesterday
15
- advance(:days => -1)
20
+ advance(days: -1)
21
+ end
22
+
23
+ # Returns a new date/time the specified number of days ago.
24
+ def prev_day(days = 1)
25
+ advance(days: -days)
16
26
  end
17
27
 
18
28
  # Returns a new date/time representing tomorrow.
19
29
  def tomorrow
20
- advance(:days => 1)
30
+ advance(days: 1)
31
+ end
32
+
33
+ # Returns a new date/time the specified number of days in the future.
34
+ def next_day(days = 1)
35
+ advance(days: days)
21
36
  end
22
37
 
23
38
  # Returns true if the date/time is today.
@@ -35,90 +50,147 @@ module DateAndTime
35
50
  self > self.class.current
36
51
  end
37
52
 
53
+ # Returns true if the date/time falls on a Saturday or Sunday.
54
+ def on_weekend?
55
+ WEEKEND_DAYS.include?(wday)
56
+ end
57
+
58
+ # Returns true if the date/time does not fall on a Saturday or Sunday.
59
+ def on_weekday?
60
+ !WEEKEND_DAYS.include?(wday)
61
+ end
62
+
38
63
  # Returns a new date/time the specified number of days ago.
39
64
  def days_ago(days)
40
- advance(:days => -days)
65
+ advance(days: -days)
41
66
  end
42
67
 
43
68
  # Returns a new date/time the specified number of days in the future.
44
69
  def days_since(days)
45
- advance(:days => days)
70
+ advance(days: days)
46
71
  end
47
72
 
48
73
  # Returns a new date/time the specified number of weeks ago.
49
74
  def weeks_ago(weeks)
50
- advance(:weeks => -weeks)
75
+ advance(weeks: -weeks)
51
76
  end
52
77
 
53
78
  # Returns a new date/time the specified number of weeks in the future.
54
79
  def weeks_since(weeks)
55
- advance(:weeks => weeks)
80
+ advance(weeks: weeks)
56
81
  end
57
82
 
58
83
  # Returns a new date/time the specified number of months ago.
59
84
  def months_ago(months)
60
- advance(:months => -months)
85
+ advance(months: -months)
61
86
  end
62
87
 
63
88
  # Returns a new date/time the specified number of months in the future.
64
89
  def months_since(months)
65
- advance(:months => months)
90
+ advance(months: months)
66
91
  end
67
92
 
68
93
  # Returns a new date/time the specified number of years ago.
69
94
  def years_ago(years)
70
- advance(:years => -years)
95
+ advance(years: -years)
71
96
  end
72
97
 
73
98
  # Returns a new date/time the specified number of years in the future.
74
99
  def years_since(years)
75
- advance(:years => years)
100
+ advance(years: years)
76
101
  end
77
102
 
78
103
  # Returns a new date/time at the start of the month.
79
- # DateTime objects will have a time set to 0:00.
104
+ #
105
+ # today = Date.today # => Thu, 18 Jun 2015
106
+ # today.beginning_of_month # => Mon, 01 Jun 2015
107
+ #
108
+ # +DateTime+ objects will have a time set to 0:00.
109
+ #
110
+ # now = DateTime.current # => Thu, 18 Jun 2015 15:23:13 +0000
111
+ # now.beginning_of_month # => Mon, 01 Jun 2015 00:00:00 +0000
80
112
  def beginning_of_month
81
- first_hour(change(:day => 1))
113
+ first_hour(change(day: 1))
82
114
  end
83
115
  alias :at_beginning_of_month :beginning_of_month
84
116
 
85
117
  # Returns a new date/time at the start of the quarter.
86
- # Example: 1st January, 1st July, 1st October.
87
- # DateTime objects will have a time set to 0:00.
118
+ #
119
+ # today = Date.today # => Fri, 10 Jul 2015
120
+ # today.beginning_of_quarter # => Wed, 01 Jul 2015
121
+ #
122
+ # +DateTime+ objects will have a time set to 0:00.
123
+ #
124
+ # now = DateTime.current # => Fri, 10 Jul 2015 18:41:29 +0000
125
+ # now.beginning_of_quarter # => Wed, 01 Jul 2015 00:00:00 +0000
88
126
  def beginning_of_quarter
89
127
  first_quarter_month = [10, 7, 4, 1].detect { |m| m <= month }
90
- beginning_of_month.change(:month => first_quarter_month)
128
+ beginning_of_month.change(month: first_quarter_month)
91
129
  end
92
130
  alias :at_beginning_of_quarter :beginning_of_quarter
93
131
 
94
132
  # Returns a new date/time at the end of the quarter.
95
- # Example: 31st March, 30th June, 30th September.
96
- # DateTime objects will have a time set to 23:59:59.
133
+ #
134
+ # today = Date.today # => Fri, 10 Jul 2015
135
+ # today.end_of_quarter # => Wed, 30 Sep 2015
136
+ #
137
+ # +DateTime+ objects will have a time set to 23:59:59.
138
+ #
139
+ # now = DateTime.current # => Fri, 10 Jul 2015 18:41:29 +0000
140
+ # now.end_of_quarter # => Wed, 30 Sep 2015 23:59:59 +0000
97
141
  def end_of_quarter
98
142
  last_quarter_month = [3, 6, 9, 12].detect { |m| m >= month }
99
- beginning_of_month.change(:month => last_quarter_month).end_of_month
143
+ beginning_of_month.change(month: last_quarter_month).end_of_month
100
144
  end
101
145
  alias :at_end_of_quarter :end_of_quarter
102
146
 
103
- # Return a new date/time at the beginning of the year.
104
- # Example: 1st January.
105
- # DateTime objects will have a time set to 0:00.
147
+ # Returns a new date/time at the beginning of the year.
148
+ #
149
+ # today = Date.today # => Fri, 10 Jul 2015
150
+ # today.beginning_of_year # => Thu, 01 Jan 2015
151
+ #
152
+ # +DateTime+ objects will have a time set to 0:00.
153
+ #
154
+ # now = DateTime.current # => Fri, 10 Jul 2015 18:41:29 +0000
155
+ # now.beginning_of_year # => Thu, 01 Jan 2015 00:00:00 +0000
106
156
  def beginning_of_year
107
- change(:month => 1).beginning_of_month
157
+ change(month: 1).beginning_of_month
108
158
  end
109
159
  alias :at_beginning_of_year :beginning_of_year
110
160
 
111
161
  # Returns a new date/time representing the given day in the next week.
162
+ #
163
+ # today = Date.today # => Thu, 07 May 2015
164
+ # today.next_week # => Mon, 11 May 2015
165
+ #
112
166
  # The +given_day_in_next_week+ defaults to the beginning of the week
113
167
  # which is determined by +Date.beginning_of_week+ or +config.beginning_of_week+
114
- # when set. +DateTime+ objects have their time set to 0:00.
115
- def next_week(given_day_in_next_week = Date.beginning_of_week)
116
- first_hour(weeks_since(1).beginning_of_week.days_since(days_span(given_day_in_next_week)))
168
+ # when set.
169
+ #
170
+ # today = Date.today # => Thu, 07 May 2015
171
+ # today.next_week(:friday) # => Fri, 15 May 2015
172
+ #
173
+ # +DateTime+ objects have their time set to 0:00 unless +same_time+ is true.
174
+ #
175
+ # now = DateTime.current # => Thu, 07 May 2015 13:31:16 +0000
176
+ # now.next_week # => Mon, 11 May 2015 00:00:00 +0000
177
+ def next_week(given_day_in_next_week = Date.beginning_of_week, same_time: false)
178
+ result = first_hour(weeks_since(1).beginning_of_week.days_since(days_span(given_day_in_next_week)))
179
+ same_time ? copy_time_to(result) : result
180
+ end
181
+
182
+ # Returns a new date/time representing the next weekday.
183
+ def next_weekday
184
+ if next_day.on_weekend?
185
+ next_week(:monday, same_time: true)
186
+ else
187
+ next_day
188
+ end
117
189
  end
118
190
 
119
- # Short-hand for months_since(1).
120
- def next_month
121
- months_since(1)
191
+ # Returns a new date/time the specified number of months in the future.
192
+ def next_month(months = 1)
193
+ advance(months: months)
122
194
  end
123
195
 
124
196
  # Short-hand for months_since(3)
@@ -126,25 +198,40 @@ module DateAndTime
126
198
  months_since(3)
127
199
  end
128
200
 
129
- # Short-hand for years_since(1).
130
- def next_year
131
- years_since(1)
201
+ # Returns a new date/time the specified number of years in the future.
202
+ def next_year(years = 1)
203
+ advance(years: years)
132
204
  end
133
205
 
134
206
  # Returns a new date/time representing the given day in the previous week.
135
207
  # Week is assumed to start on +start_day+, default is
136
208
  # +Date.beginning_of_week+ or +config.beginning_of_week+ when set.
137
- # DateTime objects have their time set to 0:00.
138
- def prev_week(start_day = Date.beginning_of_week)
139
- first_hour(weeks_ago(1).beginning_of_week.days_since(days_span(start_day)))
209
+ # DateTime objects have their time set to 0:00 unless +same_time+ is true.
210
+ def prev_week(start_day = Date.beginning_of_week, same_time: false)
211
+ result = first_hour(weeks_ago(1).beginning_of_week.days_since(days_span(start_day)))
212
+ same_time ? copy_time_to(result) : result
140
213
  end
141
214
  alias_method :last_week, :prev_week
142
215
 
216
+ # Returns a new date/time representing the previous weekday.
217
+ def prev_weekday
218
+ if prev_day.on_weekend?
219
+ copy_time_to(beginning_of_week(:friday))
220
+ else
221
+ prev_day
222
+ end
223
+ end
224
+ alias_method :last_weekday, :prev_weekday
225
+
226
+ # Returns a new date/time the specified number of months ago.
227
+ def prev_month(months = 1)
228
+ advance(months: -months)
229
+ end
230
+
143
231
  # Short-hand for months_ago(1).
144
- def prev_month
232
+ def last_month
145
233
  months_ago(1)
146
234
  end
147
- alias_method :last_month, :prev_month
148
235
 
149
236
  # Short-hand for months_ago(3).
150
237
  def prev_quarter
@@ -152,11 +239,15 @@ module DateAndTime
152
239
  end
153
240
  alias_method :last_quarter, :prev_quarter
154
241
 
242
+ # Returns a new date/time the specified number of years ago.
243
+ def prev_year(years = 1)
244
+ advance(years: -years)
245
+ end
246
+
155
247
  # Short-hand for years_ago(1).
156
- def prev_year
248
+ def last_year
157
249
  years_ago(1)
158
250
  end
159
- alias_method :last_year, :prev_year
160
251
 
161
252
  # Returns the number of days to the start of the week on the given day.
162
253
  # Week is assumed to start on +start_day+, default is
@@ -209,12 +300,17 @@ module DateAndTime
209
300
  # Returns a new date/time representing the end of the year.
210
301
  # DateTime objects will have a time set to 23:59:59.
211
302
  def end_of_year
212
- change(:month => 12).end_of_month
303
+ change(month: 12).end_of_month
213
304
  end
214
305
  alias :at_end_of_year :end_of_year
215
306
 
307
+ # Returns a Range representing the whole day of the current date/time.
308
+ def all_day
309
+ beginning_of_day..end_of_day
310
+ end
311
+
216
312
  # Returns a Range representing the whole week of the current date/time.
217
- # Week starts on start_day, default is <tt>Date.week_start</tt> or <tt>config.week_start</tt> when set.
313
+ # Week starts on start_day, default is <tt>Date.beginning_of_week</tt> or <tt>config.beginning_of_week</tt> when set.
218
314
  def all_week(start_day = Date.beginning_of_week)
219
315
  beginning_of_week(start_day)..end_of_week(start_day)
220
316
  end
@@ -234,18 +330,45 @@ module DateAndTime
234
330
  beginning_of_year..end_of_year
235
331
  end
236
332
 
237
- private
238
-
239
- def first_hour(date_or_time)
240
- date_or_time.acts_like?(:time) ? date_or_time.beginning_of_day : date_or_time
333
+ # Returns a new date/time representing the next occurrence of the specified day of week.
334
+ #
335
+ # today = Date.today # => Thu, 14 Dec 2017
336
+ # today.next_occurring(:monday) # => Mon, 18 Dec 2017
337
+ # today.next_occurring(:thursday) # => Thu, 21 Dec 2017
338
+ def next_occurring(day_of_week)
339
+ current_day_number = wday != 0 ? wday - 1 : 6
340
+ from_now = DAYS_INTO_WEEK.fetch(day_of_week) - current_day_number
341
+ from_now += 7 unless from_now > 0
342
+ advance(days: from_now)
241
343
  end
242
344
 
243
- def last_hour(date_or_time)
244
- date_or_time.acts_like?(:time) ? date_or_time.end_of_day : date_or_time
345
+ # Returns a new date/time representing the previous occurrence of the specified day of week.
346
+ #
347
+ # today = Date.today # => Thu, 14 Dec 2017
348
+ # today.prev_occurring(:monday) # => Mon, 11 Dec 2017
349
+ # today.prev_occurring(:thursday) # => Thu, 07 Dec 2017
350
+ def prev_occurring(day_of_week)
351
+ current_day_number = wday != 0 ? wday - 1 : 6
352
+ ago = current_day_number - DAYS_INTO_WEEK.fetch(day_of_week)
353
+ ago += 7 unless ago > 0
354
+ advance(days: -ago)
245
355
  end
246
356
 
247
- def days_span(day)
248
- (DAYS_INTO_WEEK[day] - DAYS_INTO_WEEK[Date.beginning_of_week]) % 7
249
- end
357
+ private
358
+ def first_hour(date_or_time)
359
+ date_or_time.acts_like?(:time) ? date_or_time.beginning_of_day : date_or_time
360
+ end
361
+
362
+ def last_hour(date_or_time)
363
+ date_or_time.acts_like?(:time) ? date_or_time.end_of_day : date_or_time
364
+ end
365
+
366
+ def days_span(day)
367
+ (DAYS_INTO_WEEK[day] - DAYS_INTO_WEEK[Date.beginning_of_week]) % 7
368
+ end
369
+
370
+ def copy_time_to(other)
371
+ other.change(hour: hour, min: min, sec: sec, nsec: try(:nsec))
372
+ end
250
373
  end
251
374
  end
@@ -0,0 +1,16 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/module/attribute_accessors"
4
+
5
+ module DateAndTime
6
+ module Compatibility
7
+ # If true, +to_time+ preserves the timezone offset of receiver.
8
+ #
9
+ # NOTE: With Ruby 2.4+ the default for +to_time+ changed from
10
+ # converting to the local system time, to preserving the offset
11
+ # of the receiver. For backwards compatibility we're overriding
12
+ # this behavior, but new apps will have an initializer that sets
13
+ # this to true, because the new behavior is preferred.
14
+ mattr_accessor :preserve_timezone, instance_writer: false, default: false
15
+ end
16
+ end
@@ -1,11 +1,13 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module DateAndTime
2
4
  module Zones
3
5
  # Returns the simultaneous time in <tt>Time.zone</tt> if a zone is given or
4
6
  # if Time.zone_default is set. Otherwise, it returns the current time.
5
7
  #
6
8
  # Time.zone = 'Hawaii' # => 'Hawaii'
7
- # DateTime.utc(2000).in_time_zone # => Fri, 31 Dec 1999 14:00:00 HST -10:00
8
- # Date.new(2000).in_time_zone # => Sat, 01 Jan 2000 00:00:00 HST -10:00
9
+ # Time.utc(2000).in_time_zone # => Fri, 31 Dec 1999 14:00:00 HST -10:00
10
+ # Date.new(2000).in_time_zone # => Sat, 01 Jan 2000 00:00:00 HST -10:00
9
11
  #
10
12
  # This method is similar to Time#localtime, except that it uses <tt>Time.zone</tt> as the local zone
11
13
  # instead of the operating system's time zone.
@@ -14,8 +16,7 @@ module DateAndTime
14
16
  # and the conversion will be based on that zone instead of <tt>Time.zone</tt>.
15
17
  #
16
18
  # Time.utc(2000).in_time_zone('Alaska') # => Fri, 31 Dec 1999 15:00:00 AKST -09:00
17
- # DateTime.utc(2000).in_time_zone('Alaska') # => Fri, 31 Dec 1999 15:00:00 AKST -09:00
18
- # Date.new(2000).in_time_zone('Alaska') # => Sat, 01 Jan 2000 00:00:00 AKST -09:00
19
+ # Date.new(2000).in_time_zone('Alaska') # => Sat, 01 Jan 2000 00:00:00 AKST -09:00
19
20
  def in_time_zone(zone = ::Time.zone)
20
21
  time_zone = ::Time.find_zone! zone
21
22
  time = acts_like?(:time) ? self : nil
@@ -23,19 +24,18 @@ module DateAndTime
23
24
  if time_zone
24
25
  time_with_zone(time, time_zone)
25
26
  else
26
- time || self.to_time
27
+ time || to_time
27
28
  end
28
29
  end
29
30
 
30
31
  private
31
32
 
32
- def time_with_zone(time, zone)
33
- if time
34
- ActiveSupport::TimeWithZone.new(time.utc? ? time : time.getutc, zone)
35
- else
36
- ActiveSupport::TimeWithZone.new(nil, zone, to_time(:utc))
33
+ def time_with_zone(time, zone)
34
+ if time
35
+ ActiveSupport::TimeWithZone.new(time.utc? ? time : time.getutc, zone)
36
+ else
37
+ ActiveSupport::TimeWithZone.new(nil, zone, to_time(:utc))
38
+ end
37
39
  end
38
- end
39
40
  end
40
41
  end
41
-
@@ -1,4 +1,7 @@
1
- require 'active_support/core_ext/date_time/acts_like'
2
- require 'active_support/core_ext/date_time/calculations'
3
- require 'active_support/core_ext/date_time/conversions'
4
- require 'active_support/core_ext/date_time/zones'
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/date_time/acts_like"
4
+ require "active_support/core_ext/date_time/blank"
5
+ require "active_support/core_ext/date_time/calculations"
6
+ require "active_support/core_ext/date_time/compatibility"
7
+ require "active_support/core_ext/date_time/conversions"