2020-05-05 11:00:07 -04:00
* Add 3rd person aliases of `Symbol#start_with?` and `Symbol#end_with?` .
```ruby
:foo.starts_with?("f") # => true
:foo.ends_with?("o") # => true
```
*Ryuta Kamizono*
2020-04-29 19:35:05 -04:00
* Add override of unary plus for `ActiveSupport::Duration` .
`+ 1.second` is now identical to `+1.second` to prevent errors
where a seemingly innocent change of formatting leads to a change in the code behavior.
Before:
```ruby
+1.second.class
# => ActiveSupport::Duration
(+ 1.second).class
# => Integer
```
After:
```ruby
+1.second.class
# => ActiveSupport::Duration
(+ 1.second).class
# => ActiveSupport::Duration
```
Fixes #39079 .
*Roman Kushnir*
2020-03-22 10:11:58 -04:00
* Add subsec to `ActiveSupport::TimeWithZone#inspect` .
Before:
Time.at(1498099140).in_time_zone.inspect
# => "Thu, 22 Jun 2017 02:39:00 UTC +00:00"
Time.at(1498099140, 123456780, :nsec).in_time_zone.inspect
# => "Thu, 22 Jun 2017 02:39:00 UTC +00:00"
Time.at(1498099140 + Rational("1/3")).in_time_zone.inspect
# => "Thu, 22 Jun 2017 02:39:00 UTC +00:00"
After:
Time.at(1498099140).in_time_zone.inspect
2020-04-08 17:41:17 -04:00
# => "Thu, 22 Jun 2017 02:39:00.000000000 UTC +00:00"
2020-03-22 10:11:58 -04:00
Time.at(1498099140, 123456780, :nsec).in_time_zone.inspect
2020-04-08 17:41:17 -04:00
# => "Thu, 22 Jun 2017 02:39:00.123456780 UTC +00:00"
2020-03-22 10:11:58 -04:00
Time.at(1498099140 + Rational("1/3")).in_time_zone.inspect
2020-04-08 17:41:17 -04:00
# => "Thu, 22 Jun 2017 02:39:00.333333333 UTC +00:00"
2020-03-22 10:11:58 -04:00
*akinomaeni*
2020-03-31 03:05:49 -04:00
* Calling `ActiveSupport::TaggedLogging#tagged` without a block now returns a tagged logger.
```ruby
logger.tagged("BCX").info("Funky time!") # => [BCX] Funky time!
```
*Eugene Kenny*
2020-03-27 20:14:31 -04:00
* Align `Range#cover?` extension behavior with Ruby behavior for backwards ranges.
`(1..10).cover?(5..3)` now returns `false` , as it does in plain Ruby.
2020-04-15 08:23:24 -04:00
2020-03-27 20:14:31 -04:00
Also update `#include?` and `#===` behavior to match.
2020-04-15 08:23:24 -04:00
*Michael Groeneman*
2020-03-27 20:14:31 -04:00
2017-12-29 07:47:10 -05:00
* Update to TZInfo v2.0.0.
This changes the output of `ActiveSupport::TimeZone.utc_to_local` , but
can be controlled with the
2020-04-01 17:40:46 -04:00
`ActiveSupport.utc_to_local_returns_utc_offset_times` config.
2017-12-29 07:47:10 -05:00
New Rails 6.1 apps have it enabled by default, existing apps can upgrade
via the config in config/initializers/new_framework_defaults_6_1.rb
See the `utc_to_local_returns_utc_offset_times` documentation for details.
2020-04-15 08:23:24 -04:00
*Phil Ross* , *Jared Beck*
2017-12-29 07:47:10 -05:00
2020-03-19 23:30:14 -04:00
* Add Date and Time `#yesterday?` and `#tomorrow?` alongside `#today?` .
Aliased to `#prev_day?` and `#next_day?` to match the existing `#prev/next_day` methods.
*Jatin Dhankhar*
2020-03-17 16:56:41 -04:00
* Add `Enumerable#pick` to complement `ActiveRecord::Relation#pick` .
*Eugene Kenny*
2020-02-25 21:09:00 -05:00
* [Breaking change] `ActiveSupport::Callbacks#halted_callback_hook` now receive a 2nd argument:
`ActiveSupport::Callbacks#halted_callback_hook` now receive the name of the callback
being halted as second argument.
This change will allow you to differentiate which callbacks halted the chain
and act accordingly.
```ruby
class Book < ApplicationRecord
before_save { throw(:abort) }
before_create { throw(:abort) }
def halted_callback_hook(filter, callback_name)
Rails.logger.info("Book couldn't be #{callback_name}d")
end
Book.create # => "Book couldn't be created"
book.save # => "Book couldn't be saved"
end
```
*Edouard Chin*
2019-09-12 11:53:24 -04:00
* Support `prepend` with `ActiveSupport::Concern` .
Allows a module with `extend ActiveSupport::Concern` to be prepended.
module Imposter
extend ActiveSupport::Concern
# Same as `included` , except only run when prepended.
prepended do
end
end
class Person
prepend Imposter
end
2020-02-25 21:09:00 -05:00
Class methods are prepended to the base class, concerning is also
2020-02-14 12:03:05 -05:00
updated: `concerning :Imposter, prepend: true do` .
2019-09-12 11:53:24 -04:00
2020-02-25 00:14:54 -05:00
*Jason Karns* , *Elia Schito*
2019-09-12 11:53:24 -04:00
2020-01-08 06:45:10 -05:00
* Deprecate using `Range#include?` method to check the inclusion of a value
in a date time range. It is recommended to use `Range#cover?` method
instead of `Range#include?` to check the inclusion of a value
in a date time range.
*Vishal Telangre*
2020-01-03 07:44:14 -05:00
* Support added for a `round_mode` parameter, in all number helpers. (See: `BigDecimal::mode` .)
```ruby
number_to_currency(1234567890.50, precision: 0, round_mode: :half_down) # => "$1,234,567,890"
number_to_percentage(302.24398923423, precision: 5, round_mode: :down) # => "302.24398%"
number_to_rounded(389.32314, precision: 0, round_mode: :ceil) # => "390"
number_to_human_size(483989, precision: 2, round_mode: :up) # => "480 KB"
number_to_human(489939, precision: 2, round_mode: :floor) # => "480 Thousand"
485000.to_s(:human, precision: 2, round_mode: :half_even) # => "480 Thousand"
```
*Tom Lord*
2019-12-28 14:59:50 -05:00
* `Array#to_sentence` no longer returns a frozen string.
Before:
2019-12-29 23:54:35 -05:00
2019-12-28 14:59:50 -05:00
['one', 'two'].to_sentence.frozen?
# => true
After:
2019-12-29 23:54:35 -05:00
2019-12-28 14:59:50 -05:00
['one', 'two'].to_sentence.frozen?
# => false
*Nicolas Dular*
2018-12-17 05:17:01 -05:00
* When an instance of `ActiveSupport::Duration` is converted to an `iso8601` duration string, if `weeks` are mixed with `date` parts, the `week` part will be converted to days.
This keeps the parser and serializer on the same page.
```ruby
duration = ActiveSupport::Duration.build(1000000)
# 1 week, 4 days, 13 hours, 46 minutes, and 40.0 seconds
duration_iso = duration.iso8601
# P11DT13H46M40S
ActiveSupport::Duration.parse(duration_iso)
# 11 days, 13 hours, 46 minutes, and 40 seconds
duration = ActiveSupport::Duration.build(604800)
# 1 week
duration_iso = duration.iso8601
# P1W
ActiveSupport::Duration.parse(duration_iso)
# 1 week
```
*Abhishek Sarkar*
2019-10-17 11:15:15 -04:00
* Add block support to `ActiveSupport::Testing::TimeHelpers#travel_back` .
*Tim Masliuchenko*
2019-10-18 10:14:42 -04:00
* Update `ActiveSupport::Messages::Metadata#fresh?` to work for cookies with expiry set when
`ActiveSupport.parse_json_times = true` .
*Christian Gregg*
2019-10-13 13:18:29 -04:00
* Support symbolic links for `content_path` in `ActiveSupport::EncryptedFile` .
*Takumi Shotoku*
2019-09-20 18:02:11 -04:00
* Improve `Range#===` , `Range#include?` , and `Range#cover?` to work with beginless (startless)
and endless range targets.
2019-09-12 00:32:57 -04:00
2019-09-20 18:02:11 -04:00
*Allen Hsu* , *Andrew Hodgkinson*
2019-06-10 19:27:57 -04:00
2019-11-23 19:01:53 -05:00
* Don't use `Process#clock_gettime(CLOCK_THREAD_CPUTIME_ID)` on Solaris.
2019-09-03 08:11:40 -04:00
*Iain Beeston*
2019-08-22 11:11:42 -04:00
* Prevent `ActiveSupport::Duration.build(value)` from creating instances of
`ActiveSupport::Duration` unless `value` is of type `Numeric` .
Addresses the errant set of behaviours described in #37012 where
`ActiveSupport::Duration` comparisons would fail confusingly
or return unexpected results when comparing durations built from instances of `String` .
Before:
small_duration_from_string = ActiveSupport::Duration.build('9')
large_duration_from_string = ActiveSupport::Duration.build('100000000000000')
small_duration_from_int = ActiveSupport::Duration.build(9)
large_duration_from_string > small_duration_from_string
2019-11-23 19:20:00 -05:00
# => false
2019-08-22 11:11:42 -04:00
small_duration_from_string == small_duration_from_int
2019-11-23 19:20:00 -05:00
# => false
2019-08-22 11:11:42 -04:00
small_duration_from_int < large_duration_from_string
2019-11-23 19:20:00 -05:00
# => ArgumentError (comparison of ActiveSupport::Duration::Scalar with ActiveSupport::Duration failed)
2019-08-22 11:11:42 -04:00
large_duration_from_string > small_duration_from_int
2019-11-23 19:20:00 -05:00
# => ArgumentError (comparison of String with ActiveSupport::Duration failed)
2019-08-22 11:11:42 -04:00
After:
small_duration_from_string = ActiveSupport::Duration.build('9')
2019-11-23 19:20:00 -05:00
# => TypeError (can't build an ActiveSupport::Duration from a String)
2019-08-22 11:11:42 -04:00
*Alexei Emam*
2019-08-15 11:50:22 -04:00
* Add `ActiveSupport::Cache::Store#delete_multi` method to delete multiple keys from the cache store.
*Peter Zhu*
2019-08-07 12:31:33 -04:00
* Support multiple arguments in `HashWithIndifferentAccess` for `merge` and `update` methods, to
follow Ruby 2.6 addition.
*Wojciech Wnętrzak*
2019-11-23 19:20:00 -05:00
* Allow initializing `thread_mattr_*` attributes via `:default` option.
2019-05-21 14:45:04 -04:00
class Scraper
thread_mattr_reader :client, default: Api::Client.new
end
*Guilherme Mansur*
2019-06-05 03:25:18 -04:00
* Add `compact_blank` for those times when you want to remove #blank ? values from
2019-11-23 19:20:00 -05:00
an Enumerable (also `compact_blank!` on Hash, Array, ActionController::Parameters).
2019-06-05 03:25:18 -04:00
*Dana Sherson*
2019-11-23 19:20:00 -05:00
* Make ActiveSupport::Logger Fiber-safe.
2019-07-24 15:39:15 -04:00
Use `Fiber.current.__id__` in `ActiveSupport::Logger#local_level=` in order
to make log level local to Ruby Fibers in addition to Threads.
Example:
logger = ActiveSupport::Logger.new(STDOUT)
logger.level = 1
2019-11-23 19:20:00 -05:00
puts "Main is debug? #{logger.debug?}"
2019-07-24 15:39:15 -04:00
Fiber.new {
logger.local_level = 0
2019-11-23 19:20:00 -05:00
puts "Thread is debug? #{logger.debug?}"
2019-07-24 15:39:15 -04:00
}.resume
2019-11-23 19:20:00 -05:00
puts "Main is debug? #{logger.debug?}"
2019-07-24 15:39:15 -04:00
Before:
Main is debug? false
Thread is debug? true
Main is debug? true
After:
Main is debug? false
Thread is debug? true
Main is debug? false
2019-11-23 19:20:00 -05:00
Fixes #36752 .
2019-07-24 15:39:15 -04:00
*Alexander Varnin*
2019-07-11 12:43:50 -04:00
* Allow the `on_rotation` proc used when decrypting/verifying a message to be
passed at the constructor level.
2019-05-13 10:02:38 -04:00
Before:
2019-07-10 20:16:48 -04:00
crypt = ActiveSupport::MessageEncryptor.new('long_secret')
crypt.decrypt_and_verify(encrypted_message, on_rotation: proc { ... })
crypt.decrypt_and_verify(another_encrypted_message, on_rotation: proc { ... })
2019-05-13 10:02:38 -04:00
After:
2019-07-10 20:16:48 -04:00
crypt = ActiveSupport::MessageEncryptor.new('long_secret', on_rotation: proc { ... })
crypt.decrypt_and_verify(encrypted_message)
crypt.decrypt_and_verify(another_encrypted_message)
2019-05-13 10:02:38 -04:00
*Edouard Chin*
2018-07-25 17:04:34 -04:00
* `delegate_missing_to` would raise a `DelegationError` if the object
delegated to was `nil` . Now the `allow_nil` option has been added to enable
the user to specify they want `nil` returned in this case.
*Matthew Tanous*
2019-04-26 13:29:41 -04:00
* `truncate` would return the original string if it was too short to be truncated
and a frozen string if it were long enough to be truncated. Now truncate will
consistently return an unfrozen string regardless. This behavior is consistent
with `gsub` and `strip` .
2018-12-25 09:35:15 -05:00
2019-04-26 13:29:41 -04:00
Before:
2019-05-08 06:22:37 -04:00
'foobar'.truncate(5).frozen?
# => true
'foobar'.truncate(6).frozen?
# => false
2019-04-26 13:29:41 -04:00
After:
2019-05-08 06:22:37 -04:00
'foobar'.truncate(5).frozen?
# => false
'foobar'.truncate(6).frozen?
# => false
2019-04-26 13:29:41 -04:00
*Jordan Thomas*
2018-12-25 09:35:15 -05:00
2019-05-24 16:54:20 -04:00
2019-04-24 15:57:14 -04:00
Please check [6-0-stable ](https://github.com/rails/rails/blob/6-0-stable/activesupport/CHANGELOG.md ) for previous changes.