activesupport 7.1.3.2 → 7.2.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (102) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +134 -1054
  3. data/lib/active_support/array_inquirer.rb +1 -1
  4. data/lib/active_support/backtrace_cleaner.rb +15 -3
  5. data/lib/active_support/broadcast_logger.rb +19 -18
  6. data/lib/active_support/cache/file_store.rb +15 -10
  7. data/lib/active_support/cache/mem_cache_store.rb +16 -74
  8. data/lib/active_support/cache/memory_store.rb +2 -1
  9. data/lib/active_support/cache/redis_cache_store.rb +16 -13
  10. data/lib/active_support/cache/serializer_with_fallback.rb +0 -23
  11. data/lib/active_support/cache.rb +62 -69
  12. data/lib/active_support/callbacks.rb +74 -113
  13. data/lib/active_support/code_generator.rb +15 -10
  14. data/lib/active_support/core_ext/array/conversions.rb +0 -2
  15. data/lib/active_support/core_ext/class/subclasses.rb +15 -35
  16. data/lib/active_support/core_ext/date/blank.rb +4 -0
  17. data/lib/active_support/core_ext/date/conversions.rb +0 -2
  18. data/lib/active_support/core_ext/date_and_time/compatibility.rb +28 -1
  19. data/lib/active_support/core_ext/date_time/blank.rb +4 -0
  20. data/lib/active_support/core_ext/date_time/conversions.rb +0 -4
  21. data/lib/active_support/core_ext/digest/uuid.rb +6 -0
  22. data/lib/active_support/core_ext/erb/util.rb +5 -0
  23. data/lib/active_support/core_ext/hash/keys.rb +4 -4
  24. data/lib/active_support/core_ext/module/attr_internal.rb +17 -6
  25. data/lib/active_support/core_ext/module/delegation.rb +20 -148
  26. data/lib/active_support/core_ext/module/deprecation.rb +1 -4
  27. data/lib/active_support/core_ext/numeric/conversions.rb +3 -3
  28. data/lib/active_support/core_ext/object/blank.rb +45 -1
  29. data/lib/active_support/core_ext/object/duplicable.rb +24 -15
  30. data/lib/active_support/core_ext/object/instance_variables.rb +11 -19
  31. data/lib/active_support/core_ext/object/json.rb +6 -4
  32. data/lib/active_support/core_ext/object/with.rb +5 -3
  33. data/lib/active_support/core_ext/pathname/blank.rb +4 -0
  34. data/lib/active_support/core_ext/range/overlap.rb +1 -1
  35. data/lib/active_support/core_ext/securerandom.rb +8 -24
  36. data/lib/active_support/core_ext/string/conversions.rb +1 -1
  37. data/lib/active_support/core_ext/string/filters.rb +1 -1
  38. data/lib/active_support/core_ext/string/multibyte.rb +1 -1
  39. data/lib/active_support/core_ext/string/output_safety.rb +0 -7
  40. data/lib/active_support/core_ext/time/calculations.rb +18 -28
  41. data/lib/active_support/core_ext/time/compatibility.rb +16 -0
  42. data/lib/active_support/core_ext/time/conversions.rb +0 -2
  43. data/lib/active_support/core_ext/time/zones.rb +1 -1
  44. data/lib/active_support/core_ext.rb +0 -1
  45. data/lib/active_support/current_attributes.rb +38 -40
  46. data/lib/active_support/delegation.rb +202 -0
  47. data/lib/active_support/dependencies/autoload.rb +0 -12
  48. data/lib/active_support/deprecation/constant_accessor.rb +47 -26
  49. data/lib/active_support/deprecation/proxy_wrappers.rb +9 -12
  50. data/lib/active_support/deprecation/reporting.rb +9 -4
  51. data/lib/active_support/deprecation.rb +8 -5
  52. data/lib/active_support/descendants_tracker.rb +9 -87
  53. data/lib/active_support/duration/iso8601_parser.rb +2 -2
  54. data/lib/active_support/duration/iso8601_serializer.rb +1 -2
  55. data/lib/active_support/duration.rb +11 -6
  56. data/lib/active_support/encrypted_file.rb +1 -1
  57. data/lib/active_support/error_reporter.rb +41 -3
  58. data/lib/active_support/evented_file_update_checker.rb +0 -1
  59. data/lib/active_support/execution_wrapper.rb +0 -1
  60. data/lib/active_support/file_update_checker.rb +1 -1
  61. data/lib/active_support/fork_tracker.rb +2 -38
  62. data/lib/active_support/gem_version.rb +2 -2
  63. data/lib/active_support/hash_with_indifferent_access.rb +6 -8
  64. data/lib/active_support/html_safe_translation.rb +7 -4
  65. data/lib/active_support/json/encoding.rb +1 -1
  66. data/lib/active_support/log_subscriber.rb +1 -12
  67. data/lib/active_support/logger.rb +15 -2
  68. data/lib/active_support/logger_thread_safe_level.rb +0 -8
  69. data/lib/active_support/message_pack/extensions.rb +15 -2
  70. data/lib/active_support/message_verifier.rb +12 -0
  71. data/lib/active_support/messages/codec.rb +1 -1
  72. data/lib/active_support/multibyte/chars.rb +2 -2
  73. data/lib/active_support/notifications/fanout.rb +4 -7
  74. data/lib/active_support/notifications/instrumenter.rb +32 -21
  75. data/lib/active_support/notifications.rb +28 -27
  76. data/lib/active_support/number_helper/number_converter.rb +2 -2
  77. data/lib/active_support/option_merger.rb +2 -2
  78. data/lib/active_support/ordered_options.rb +53 -15
  79. data/lib/active_support/proxy_object.rb +8 -5
  80. data/lib/active_support/railtie.rb +4 -11
  81. data/lib/active_support/string_inquirer.rb +1 -1
  82. data/lib/active_support/subscriber.rb +1 -0
  83. data/lib/active_support/syntax_error_proxy.rb +1 -11
  84. data/lib/active_support/tagged_logging.rb +4 -1
  85. data/lib/active_support/test_case.rb +3 -1
  86. data/lib/active_support/testing/assertions.rb +4 -4
  87. data/lib/active_support/testing/constant_stubbing.rb +30 -8
  88. data/lib/active_support/testing/deprecation.rb +5 -12
  89. data/lib/active_support/testing/isolation.rb +20 -8
  90. data/lib/active_support/testing/method_call_assertions.rb +2 -16
  91. data/lib/active_support/testing/parallelization/server.rb +3 -0
  92. data/lib/active_support/testing/setup_and_teardown.rb +2 -0
  93. data/lib/active_support/testing/strict_warnings.rb +8 -4
  94. data/lib/active_support/testing/tests_without_assertions.rb +19 -0
  95. data/lib/active_support/testing/time_helpers.rb +3 -3
  96. data/lib/active_support/time_with_zone.rb +8 -4
  97. data/lib/active_support/values/time_zone.rb +16 -7
  98. data/lib/active_support/xml_mini.rb +11 -2
  99. data/lib/active_support.rb +3 -2
  100. metadata +49 -18
  101. data/lib/active_support/deprecation/instance_delegator.rb +0 -65
  102. data/lib/active_support/ruby_features.rb +0 -7
data/CHANGELOG.md CHANGED
@@ -1,1209 +1,289 @@
1
- ## Rails 7.1.3.2 (February 21, 2024) ##
1
+ ## Rails 7.2.2.2 (August 13, 2025) ##
2
2
 
3
3
  * No changes.
4
4
 
5
5
 
6
- ## Rails 7.1.3.1 (February 21, 2024) ##
6
+ ## Rails 7.2.2.1 (December 10, 2024) ##
7
7
 
8
8
  * No changes.
9
9
 
10
10
 
11
- ## Rails 7.1.3 (January 16, 2024) ##
11
+ ## Rails 7.2.2 (October 30, 2024) ##
12
12
 
13
- * Handle nil `backtrace_locations` in `ActiveSupport::SyntaxErrorProxy`.
13
+ * Include options when instrumenting `ActiveSupport::Cache::Store#delete` and `ActiveSupport::Cache::Store#delete_multi`.
14
14
 
15
- *Eugene Kenny*
15
+ *Adam Renberg Tamm*
16
16
 
17
- * Fix `ActiveSupport::JSON.encode` to prevent duplicate keys.
17
+ * Print test names when running `rails test -v` for parallel tests.
18
18
 
19
- If the same key exist in both String and Symbol form it could
20
- lead to the same key being emitted twice.
19
+ *John Hawthorn*, *Abeid Ahmed*
21
20
 
22
- *Manish Sharma*
23
21
 
24
- * Fix `ActiveSupport::Cache::Store#read_multi` when using a cache namespace
25
- and local cache strategy.
26
-
27
- *Mark Oleson*
28
-
29
- * Fix `Time.now/DateTime.now/Date.today` to return results in a system timezone after `#travel_to`.
30
-
31
- There is a bug in the current implementation of #travel_to:
32
- it remembers a timezone of its argument, and all stubbed methods start
33
- returning results in that remembered timezone. However, the expected
34
- behaviour is to return results in a system timezone.
35
-
36
- *Aleksei Chernenkov*
37
-
38
- * Fix `:unless_exist` option for `MemoryStore#write` (et al) when using a
39
- cache namespace.
40
-
41
- *S. Brent Faulkner*
42
-
43
- * Fix ActiveSupport::Deprecation to handle blaming generated code.
44
-
45
- *Jean Boussier*, *fatkodima*
46
-
47
-
48
- ## Rails 7.1.2 (November 10, 2023) ##
49
-
50
- * Fix `:expires_in` option for `RedisCacheStore#write_multi`.
51
-
52
- *fatkodima*
53
-
54
- * Fix deserialization of non-string "purpose" field in Message serializer
55
-
56
- *Jacopo Beschi*
57
-
58
- * Prevent global cache options being overwritten when setting dynamic options
59
- inside a `ActiveSupport::Cache::Store#fetch` block.
60
-
61
- *Yasha Krasnou*
62
-
63
- * Fix missing `require` resulting in `NoMethodError` when running
64
- `bin/rails secrets:show` or `bin/rails secrets:edit`.
65
-
66
- *Stephen Ierodiaconou*
67
-
68
- * Ensure `{down,up}case_first` returns non-frozen string.
69
-
70
- *Jonathan Hefner*
71
-
72
- * Fix `#to_fs(:human_size)` to correctly work with negative numbers.
73
-
74
- *Earlopain*
75
-
76
- * Fix `BroadcastLogger#dup` so that it duplicates the logger's `broadcasts`.
77
-
78
- *Andrew Novoselac*
79
-
80
- * Fix issue where `bootstrap.rb` overwrites the `level` of a `BroadcastLogger`'s `broadcasts`.
81
-
82
- *Andrew Novoselac*
83
-
84
- * Fix `ActiveSupport::Cache` to handle outdated Marshal payload from Rails 6.1 format.
85
-
86
- Active Support's Cache is supposed to treat a Marshal payload that can no longer be
87
- deserialized as a cache miss. It fail to do so for compressed payload in the Rails 6.1
88
- legacy format.
89
-
90
- *Jean Boussier*
91
-
92
- * Fix `OrderedOptions#dig` for array indexes.
93
-
94
- *fatkodima*
95
-
96
- * Fix time travel helpers to work when nested using with separate classes.
97
-
98
- *fatkodima*
99
-
100
- * Fix `delete_matched` for file cache store to work with keys longer than the
101
- max filename size.
102
-
103
- *fatkodima* and *Jonathan Hefner*
104
-
105
- * Fix compatibility with the `semantic_logger` gem.
106
-
107
- The `semantic_logger` gem doesn't behave exactly like stdlib logger in that
108
- `SemanticLogger#level` returns a Symbol while stdlib `Logger#level` returns an Integer.
109
-
110
- This caused the various `LogSubscriber` classes in Rails to break when assigned a
111
- `SemanticLogger` instance.
112
-
113
- *Jean Boussier*, *ojab*
114
-
115
- ## Rails 7.1.1 (October 11, 2023) ##
116
-
117
- * Add support for keyword arguments when delegating calls to custom loggers from `ActiveSupport::BroadcastLogger`.
118
-
119
- *Edouard Chin*
120
-
121
- * `NumberHelper`: handle objects responding `to_d`.
122
-
123
- *fatkodima*
124
-
125
- * Fix RedisCacheStore to properly set the TTL when incrementing or decrementing.
126
-
127
- This bug was only impacting Redis server older than 7.0.
128
-
129
- *Thomas Countz*
130
-
131
- * Fix MemoryStore to prevent race conditions when incrementing or decrementing.
132
-
133
- *Pierre Jambet*
134
-
135
-
136
- ## Rails 7.1.0 (October 05, 2023) ##
22
+ ## Rails 7.2.1.2 (October 23, 2024) ##
137
23
 
138
24
  * No changes.
139
25
 
140
26
 
141
- ## Rails 7.1.0.rc2 (October 01, 2023) ##
142
-
143
- * Fix `AS::MessagePack` with `ENV["RAILS_MAX_THREADS"]`.
144
-
145
- *Jonathan Hefner*
146
-
147
-
148
- ## Rails 7.1.0.rc1 (September 27, 2023) ##
27
+ ## Rails 7.2.1.1 (October 15, 2024) ##
149
28
 
150
- * Add a new public API for broadcasting logs
151
-
152
- This feature existed for a while but was until now a private API.
153
- Broadcasting log allows to send log message to difference sinks (STDOUT, a file ...) and
154
- is used by default in the development environment to write logs both on STDOUT and in the
155
- "development.log" file.
29
+ * No changes.
156
30
 
157
- Basic usage:
158
31
 
159
- ```ruby
160
- stdout_logger = Logger.new(STDOUT)
161
- file_logger = Logger.new("development.log")
162
- broadcast = ActiveSupport::BroadcastLogger.new(stdout_logger, file_logger)
32
+ ## Rails 7.2.1 (August 22, 2024) ##
163
33
 
164
- broadcast.info("Hello!") # The "Hello!" message is written on STDOUT and in the log file.
165
- ```
34
+ * No changes.
166
35
 
167
- Adding other sink(s) to the broadcast:
168
36
 
169
- ```ruby
170
- broadcast = ActiveSupport::BroadcastLogger.new
171
- broadcast.broadcast_to(Logger.new(STDERR))
172
- ```
37
+ ## Rails 7.2.0 (August 09, 2024) ##
173
38
 
174
- Remove a sink from the broadcast:
39
+ * Fix `delegate_missing_to allow_nil: true` when called with implict self
175
40
 
176
41
  ```ruby
177
- stdout_logger = Logger.new(STDOUT)
178
- broadcast = ActiveSupport::BroadcastLogger.new(stdout_logger)
179
-
180
- broadcast.stop_broadcasting_to(stdout_logger)
181
- ```
182
-
183
- *Edouard Chin*
184
-
185
- * Fix Range#overlap? not taking empty ranges into account on Ruby < 3.3
186
-
187
- *Nobuyoshi Nakada*, *Shouichi Kamiya*, *Hartley McGuire*
188
-
189
- * Use Ruby 3.3 Range#overlap? if available
190
-
191
- *Yasuo Honda*
192
-
193
-
194
- ## Rails 7.1.0.beta1 (September 13, 2023) ##
195
-
196
- * Add `bigdecimal` as Active Support dependency that is a bundled gem candidate for Ruby 3.4.
197
-
198
- `bigdecimal` 3.1.4 or higher version will be installed.
199
- Ruby 2.7 and 3.0 users who want `bigdecimal` version 2.0.0 or 3.0.0 behavior as a default gem,
200
- pin the `bigdecimal` version in your application Gemfile.
201
-
202
- *Koichi ITO*
203
-
204
- * Add `drb`, `mutex_m` and `base64` that are bundled gem candidates for Ruby 3.4
205
-
206
- *Yasuo Honda*
207
-
208
- * When using cache format version >= 7.1 or a custom serializer, expired and
209
- version-mismatched cache entries can now be detected without deserializing
210
- their values.
211
-
212
- *Jonathan Hefner*
213
-
214
- * Make all cache stores return a boolean for `#delete`
215
-
216
- Previously the `RedisCacheStore#delete` would return `1` if the entry
217
- exists and `0` otherwise. Now it returns true if the entry exists and false
218
- otherwise, just like the other stores.
42
+ class Person
43
+ delegate_missing_to :address, allow_nil: true
219
44
 
220
- The `FileStore` would return `nil` if the entry doesn't exists and returns
221
- `false` now as well.
222
-
223
- *Petrik de Heus*
224
-
225
- * Active Support cache stores now support replacing the default compressor via
226
- a `:compressor` option. The specified compressor must respond to `deflate`
227
- and `inflate`. For example:
228
-
229
- ```ruby
230
- module MyCompressor
231
- def self.deflate(string)
232
- # compression logic...
233
- end
234
-
235
- def self.inflate(compressed)
236
- # decompression logic...
237
- end
45
+ def address
46
+ nil
238
47
  end
239
48
 
240
- config.cache_store = :redis_cache_store, { compressor: MyCompressor }
241
- ```
242
-
243
- *Jonathan Hefner*
244
-
245
- * Active Support cache stores now support a `:serializer` option. Similar to
246
- the `:coder` option, serializers must respond to `dump` and `load`. However,
247
- serializers are only responsible for serializing a cached value, whereas
248
- coders are responsible for serializing the entire `ActiveSupport::Cache::Entry`
249
- instance. Additionally, the output from serializers can be automatically
250
- compressed, whereas coders are responsible for their own compression.
251
-
252
- Specifying a serializer instead of a coder also enables performance
253
- optimizations, including the bare string optimization introduced by cache
254
- format version 7.1.
255
-
256
- The `:serializer` and `:coder` options are mutually exclusive. Specifying
257
- both will raise an `ArgumentError`.
258
-
259
- *Jonathan Hefner*
260
-
261
- * Fix `ActiveSupport::Inflector.humanize(nil)` raising ``NoMethodError: undefined method `end_with?' for nil:NilClass``.
262
-
263
- *James Robinson*
264
-
265
- * Don't show secrets for `ActiveSupport::KeyGenerator#inspect`.
266
-
267
- Before:
268
-
269
- ```ruby
270
- ActiveSupport::KeyGenerator.new(secret).inspect
271
- "#<ActiveSupport::KeyGenerator:0x0000000104888038 ... @secret=\"\\xAF\\bFh]LV}q\\nl\\xB2U\\xB3 ... >"
272
- ```
273
-
274
- After:
275
-
276
- ```ruby
277
- ActiveSupport::KeyGenerator::Aes256Gcm(secret).inspect
278
- "#<ActiveSupport::KeyGenerator:0x0000000104888038>"
279
- ```
280
-
281
- *Petrik de Heus*
282
-
283
- * Improve error message when EventedFileUpdateChecker is used without a
284
- compatible version of the Listen gem
285
-
286
- *Hartley McGuire*
287
-
288
- * Add `:report` behavior for Deprecation
289
-
290
- Setting `config.active_support.deprecation = :report` uses the error
291
- reporter to report deprecation warnings to `ActiveSupport::ErrorReporter`.
292
-
293
- Deprecations are reported as handled errors, with a severity of `:warning`.
294
-
295
- Useful to report deprecations happening in production to your bug tracker.
296
-
297
- *Étienne Barrié*
298
-
299
- * Rename `Range#overlaps?` to `#overlap?` and add alias for backwards compatibility
300
-
301
- *Christian Schmidt*
302
-
303
- * Fix `EncryptedConfiguration` returning incorrect values for some `Hash`
304
- methods
305
-
306
- *Hartley McGuire*
307
-
308
- * Don't show secrets for `MessageEncryptor#inspect`.
309
-
310
- Before:
311
-
312
- ```ruby
313
- ActiveSupport::MessageEncryptor.new(secret, cipher: "aes-256-gcm").inspect
314
- "#<ActiveSupport::MessageEncryptor:0x0000000104888038 ... @secret=\"\\xAF\\bFh]LV}q\\nl\\xB2U\\xB3 ... >"
315
- ```
316
-
317
- After:
318
-
319
- ```ruby
320
- ActiveSupport::MessageEncryptor.new(secret, cipher: "aes-256-gcm").inspect
321
- "#<ActiveSupport::MessageEncryptor:0x0000000104888038>"
322
- ```
323
-
324
- *Petrik de Heus*
325
-
326
- * Don't show contents for `EncryptedConfiguration#inspect`.
327
-
328
- Before:
329
- ```ruby
330
- Rails.application.credentials.inspect
331
- "#<ActiveSupport::EncryptedConfiguration:0x000000010d2b38e8 ... @config={:secret=>\"something secret\"} ... @key_file_contents=\"915e4ea054e011022398dc242\" ...>"
332
- ```
333
-
334
- After:
335
- ```ruby
336
- Rails.application.credentials.inspect
337
- "#<ActiveSupport::EncryptedConfiguration:0x000000010d2b38e8>"
338
- ```
339
-
340
- *Petrik de Heus*
341
-
342
- * `ERB::Util.html_escape_once` always returns an `html_safe` string.
343
-
344
- This method previously maintained the `html_safe?` property of a string on the return
345
- value. Because this string has been escaped, however, not marking it as `html_safe` causes
346
- entities to be double-escaped.
347
-
348
- As an example, take this view snippet:
349
-
350
- ```html
351
- <p><%= html_escape_once("this & that &amp; the other") %></p>
352
- ```
353
-
354
- Before this change, that would be double-escaped and render as:
355
-
356
- ```html
357
- <p>this &amp;amp; that &amp;amp; the other</p>
358
- ```
359
-
360
- After this change, it renders correctly as:
361
-
362
- ```html
363
- <p>this &amp; that &amp; the other</p>
364
- ```
365
-
366
- Fixes #48256
367
-
368
- *Mike Dalessio*
369
-
370
- * Deprecate `SafeBuffer#clone_empty`.
371
-
372
- This method has not been used internally since Rails 4.2.0.
373
-
374
- *Mike Dalessio*
375
-
376
- * `MessageEncryptor`, `MessageVerifier`, and `config.active_support.message_serializer`
377
- now accept `:message_pack` and `:message_pack_allow_marshal` as serializers.
378
- These serializers require the [`msgpack` gem](https://rubygems.org/gems/msgpack)
379
- (>= 1.7.0).
380
-
381
- The Message Pack format can provide improved performance and smaller payload
382
- sizes. It also supports round-tripping some Ruby types that are not supported
383
- by JSON. For example:
384
-
385
- ```ruby
386
- verifier = ActiveSupport::MessageVerifier.new("secret")
387
- data = [{ a: 1 }, { b: 2 }.with_indifferent_access, 1.to_d, Time.at(0, 123)]
388
- message = verifier.generate(data)
389
-
390
- # BEFORE with config.active_support.message_serializer = :json
391
- verifier.verified(message)
392
- # => [{"a"=>1}, {"b"=>2}, "1.0", "1969-12-31T18:00:00.000-06:00"]
393
- verifier.verified(message).map(&:class)
394
- # => [Hash, Hash, String, String]
395
-
396
- # AFTER with config.active_support.message_serializer = :message_pack
397
- verifier.verified(message)
398
- # => [{:a=>1}, {"b"=>2}, 0.1e1, 1969-12-31 18:00:00.000123 -0600]
399
- verifier.verified(message).map(&:class)
400
- # => [Hash, ActiveSupport::HashWithIndifferentAccess, BigDecimal, Time]
401
- ```
402
-
403
- The `:message_pack` serializer can fall back to deserializing with
404
- `ActiveSupport::JSON` when necessary, and the `:message_pack_allow_marshal`
405
- serializer can fall back to deserializing with `Marshal` as well as
406
- `ActiveSupport::JSON`. Additionally, the `:marshal`, `:json`, and
407
- `:json_allow_marshal` serializers can now fall back to deserializing with
408
- `ActiveSupport::MessagePack` when necessary. These behaviors ensure old
409
- messages can still be read so that migration is easier.
410
-
411
- *Jonathan Hefner*
412
-
413
- * A new `7.1` cache format is available which includes an optimization for
414
- bare string values such as view fragments.
415
-
416
- The `7.1` cache format is used by default for new apps, and existing apps
417
- can enable the format by setting `config.load_defaults 7.1` or by setting
418
- `config.active_support.cache_format_version = 7.1` in `config/application.rb`
419
- or a `config/environments/*.rb` file.
420
-
421
- Cache entries written using the `6.1` or `7.0` cache formats can be read
422
- when using the `7.1` format. To perform a rolling deploy of a Rails 7.1
423
- upgrade, wherein servers that have not yet been upgraded must be able to
424
- read caches from upgraded servers, leave the cache format unchanged on the
425
- first deploy, then enable the `7.1` cache format on a subsequent deploy.
426
-
427
- *Jonathan Hefner*
428
-
429
- * Active Support cache stores can now use a preconfigured serializer based on
430
- `ActiveSupport::MessagePack` via the `:serializer` option:
431
-
432
- ```ruby
433
- config.cache_store = :redis_cache_store, { serializer: :message_pack }
434
- ```
435
-
436
- The `:message_pack` serializer can reduce cache entry sizes and improve
437
- performance, but requires the [`msgpack` gem](https://rubygems.org/gems/msgpack)
438
- (>= 1.7.0).
439
-
440
- The `:message_pack` serializer can read cache entries written by the default
441
- serializer, and the default serializer can now read entries written by the
442
- `:message_pack` serializer. These behaviors make it easy to migrate between
443
- serializer without invalidating the entire cache.
444
-
445
- *Jonathan Hefner*
446
-
447
- * `Object#deep_dup` no longer duplicate named classes and modules.
448
-
449
- Before:
450
-
451
- ```ruby
452
- hash = { class: Object, module: Kernel }
453
- hash.deep_dup # => {:class=>#<Class:0x00000001063ffc80>, :module=>#<Module:0x00000001063ffa00>}
454
- ```
455
-
456
- After:
49
+ def berliner?
50
+ city == "Berlin"
51
+ end
52
+ end
457
53
 
458
- ```ruby
459
- hash = { class: Object, module: Kernel }
460
- hash.deep_dup # => {:class=>Object, :module=>Kernel}
54
+ Person.new.city # => nil
55
+ Person.new.berliner? # undefined local variable or method `city' for an instance of Person (NameError)
461
56
  ```
462
57
 
463
58
  *Jean Boussier*
464
59
 
465
- * Consistently raise an `ArgumentError` if the `ActiveSupport::Cache` key is blank.
60
+ * Add `logger` as a dependency since it is a bundled gem candidate for Ruby 3.5
466
61
 
467
- *Joshua Young*
62
+ *Earlopain*
468
63
 
469
- * Deprecate usage of the singleton `ActiveSupport::Deprecation`.
64
+ * Define `Digest::UUID.nil_uuid`, which returns the so-called nil UUID.
470
65
 
471
- All usage of `ActiveSupport::Deprecation` as a singleton is deprecated, the most common one being
472
- `ActiveSupport::Deprecation.warn`. Gem authors should now create their own deprecator (`ActiveSupport::Deprecation`
473
- object), and use it to emit deprecation warnings.
66
+ *Xavier Noria*
474
67
 
475
- Calling any of the following without specifying a deprecator argument is also deprecated:
476
- * Module.deprecate
477
- * deprecate_constant
478
- * DeprecatedObjectProxy
479
- * DeprecatedInstanceVariableProxy
480
- * DeprecatedConstantProxy
481
- * deprecation-related test assertions
68
+ * Support `duration` type in `ActiveSupport::XmlMini`.
482
69
 
483
- Use of `ActiveSupport::Deprecation.silence` and configuration methods like `behavior=`, `disallowed_behavior=`,
484
- `disallowed_warnings=` should now be aimed at the [application's deprecators](https://api.rubyonrails.org/classes/Rails/Application.html#method-i-deprecators).
70
+ *heka1024*
485
71
 
486
- ```ruby
487
- Rails.application.deprecators.silence do
488
- # code that emits deprecation warnings
489
- end
490
- ```
72
+ * Remove deprecated `ActiveSupport::Notifications::Event#children` and `ActiveSupport::Notifications::Event#parent_of?`.
491
73
 
492
- If your gem has a Railtie or Engine, it's encouraged to add your deprecator to the application's deprecators, that
493
- way the deprecation related configuration options will apply to it as well, e.g.
494
- `config.active_support.report_deprecations` set to `false` in the production environment will also disable your
495
- deprecator.
496
-
497
- ```ruby
498
- initializer "my_gem.deprecator" do |app|
499
- app.deprecators[:my_gem] = MyGem.deprecator
500
- end
501
- ```
502
-
503
- *Étienne Barrié*
74
+ *Rafael Mendonça França*
504
75
 
505
- * Add `Object#with` to set and restore public attributes around a block
76
+ * Remove deprecated support to call the following methods without passing a deprecator:
506
77
 
507
- ```ruby
508
- client.timeout # => 5
509
- client.with(timeout: 1) do
510
- client.timeout # => 1
511
- end
512
- client.timeout # => 5
513
- ```
78
+ - `deprecate`
79
+ - `deprecate_constant`
80
+ - `ActiveSupport::Deprecation::DeprecatedObjectProxy.new`
81
+ - `ActiveSupport::Deprecation::DeprecatedInstanceVariableProxy.new`
82
+ - `ActiveSupport::Deprecation::DeprecatedConstantProxy.new`
83
+ - `assert_deprecated`
84
+ - `assert_not_deprecated`
85
+ - `collect_deprecations`
514
86
 
515
- *Jean Boussier*
87
+ *Rafael Mendonça França*
516
88
 
517
- * Remove deprecated support to generate incorrect RFC 4122 UUIDs when providing a namespace ID that is not one of the
518
- constants defined on `Digest::UUID`.
89
+ * Remove deprecated `ActiveSupport::Deprecation` delegation to instance.
519
90
 
520
91
  *Rafael Mendonça França*
521
92
 
522
- * Deprecate `config.active_support.use_rfc4122_namespaced_uuids`.
93
+ * Remove deprecated `SafeBuffer#clone_empty`.
523
94
 
524
95
  *Rafael Mendonça França*
525
96
 
526
- * Remove implicit conversion of objects into `String` by `ActiveSupport::SafeBuffer`.
97
+ * Remove deprecated `#to_default_s` from `Array`, `Date`, `DateTime` and `Time`.
527
98
 
528
99
  *Rafael Mendonça França*
529
100
 
530
- * Remove deprecated `active_support/core_ext/range/include_time_with_zone` file.
101
+ * Remove deprecated support to passing `Dalli::Client` instances to `MemCacheStore`.
531
102
 
532
103
  *Rafael Mendonça França*
533
104
 
534
- * Deprecate `config.active_support.remove_deprecated_time_with_zone_name`.
105
+ * Remove deprecated `config.active_support.use_rfc4122_namespaced_uuids`.
535
106
 
536
107
  *Rafael Mendonça França*
537
108
 
538
- * Remove deprecated override of `ActiveSupport::TimeWithZone.name`.
109
+ * Remove deprecated `config.active_support.remove_deprecated_time_with_zone_name`.
539
110
 
540
111
  *Rafael Mendonça França*
541
112
 
542
- * Deprecate `config.active_support.disable_to_s_conversion`.
113
+ * Remove deprecated `config.active_support.disable_to_s_conversion`.
543
114
 
544
115
  *Rafael Mendonça França*
545
116
 
546
- * Remove deprecated option to passing a format to `#to_s` in `Array`, `Range`, `Date`, `DateTime`, `Time`,
547
- `BigDecimal`, `Float` and, `Integer`.
117
+ * Remove deprecated support to bolding log text with positional boolean in `ActiveSupport::LogSubscriber#color`.
548
118
 
549
119
  *Rafael Mendonça França*
550
120
 
551
- * Remove deprecated `ActiveSupport::PerThreadRegistry`.
121
+ * Remove deprecated constants `ActiveSupport::LogSubscriber::CLEAR` and `ActiveSupport::LogSubscriber::BOLD`.
552
122
 
553
123
  *Rafael Mendonça França*
554
124
 
555
- * Remove deprecated override of `Enumerable#sum`.
125
+ * Remove deprecated support for `config.active_support.cache_format_version = 6.1`.
556
126
 
557
127
  *Rafael Mendonça França*
558
128
 
559
- * Deprecated initializing a `ActiveSupport::Cache::MemCacheStore` with an instance of `Dalli::Client`.
560
-
561
- Deprecate the undocumented option of providing an already-initialized instance of `Dalli::Client` to `ActiveSupport::Cache::MemCacheStore`. Such clients could be configured with unrecognized options, which could lead to unexpected behavior. Instead, provide addresses as documented.
129
+ * Remove deprecated `:pool_size` and `:pool_timeout` options for the cache storage.
562
130
 
563
- *aledustet*
131
+ *Rafael Mendonça França*
564
132
 
565
- * Stub `Time.new()` in `TimeHelpers#travel_to`
133
+ * Warn on tests without assertions.
566
134
 
567
- ```ruby
568
- travel_to Time.new(2004, 11, 24) do
569
- # Inside the `travel_to` block `Time.new` is stubbed
570
- assert_equal 2004, Time.new.year
571
- end
572
- ```
135
+ `ActiveSupport::TestCase` now warns when tests do not run any assertions.
136
+ This is helpful in detecting broken tests that do not perform intended assertions.
573
137
 
574
138
  *fatkodima*
575
139
 
576
- * Raise `ActiveSupport::MessageEncryptor::InvalidMessage` from
577
- `ActiveSupport::MessageEncryptor#decrypt_and_verify` regardless of cipher.
578
- Previously, when a `MessageEncryptor` was using a non-AEAD cipher such as
579
- AES-256-CBC, a corrupt or tampered message would raise
580
- `ActiveSupport::MessageVerifier::InvalidSignature`. Now, all ciphers raise
581
- the same error:
582
-
583
- ```ruby
584
- encryptor = ActiveSupport::MessageEncryptor.new("x" * 32, cipher: "aes-256-gcm")
585
- message = encryptor.encrypt_and_sign("message")
586
- encryptor.decrypt_and_verify(message.next)
587
- # => raises ActiveSupport::MessageEncryptor::InvalidMessage
588
-
589
- encryptor = ActiveSupport::MessageEncryptor.new("x" * 32, cipher: "aes-256-cbc")
590
- message = encryptor.encrypt_and_sign("message")
591
- encryptor.decrypt_and_verify(message.next)
592
- # BEFORE:
593
- # => raises ActiveSupport::MessageVerifier::InvalidSignature
594
- # AFTER:
595
- # => raises ActiveSupport::MessageEncryptor::InvalidMessage
596
- ```
597
-
598
- *Jonathan Hefner*
599
-
600
- * Support `nil` original values when using `ActiveSupport::MessageVerifier#verify`.
601
- Previously, `MessageVerifier#verify` did not work with `nil` original
602
- values, though both `MessageVerifier#verified` and
603
- `MessageEncryptor#decrypt_and_verify` do:
604
-
605
- ```ruby
606
- encryptor = ActiveSupport::MessageEncryptor.new(secret)
607
- message = encryptor.encrypt_and_sign(nil)
608
-
609
- encryptor.decrypt_and_verify(message)
610
- # => nil
611
-
612
- verifier = ActiveSupport::MessageVerifier.new(secret)
613
- message = verifier.generate(nil)
614
-
615
- verifier.verified(message)
616
- # => nil
617
-
618
- verifier.verify(message)
619
- # BEFORE:
620
- # => raises ActiveSupport::MessageVerifier::InvalidSignature
621
- # AFTER:
622
- # => nil
623
- ```
624
-
625
- *Jonathan Hefner*
626
-
627
- * Maintain `html_safe?` on html_safe strings when sliced with `slice`, `slice!`, or `chr` method.
628
-
629
- Previously, `html_safe?` was only maintained when the html_safe strings were sliced
630
- with `[]` method. Now, `slice`, `slice!`, and `chr` methods will maintain `html_safe?` like `[]` method.
631
-
632
- ```ruby
633
- string = "<div>test</div>".html_safe
634
- string.slice(0, 1).html_safe? # => true
635
- string.slice!(0, 1).html_safe? # => true
636
- # maintain html_safe? after the slice!
637
- string.html_safe? # => true
638
- string.chr.html_safe? # => true
639
- ```
640
-
641
- *Michael Go*
642
-
643
- * Add `Object#in?` support for open ranges.
644
-
645
- ```ruby
646
- assert Date.today.in?(..Date.tomorrow)
647
- assert_not Date.today.in?(Date.tomorrow..)
648
- ```
649
-
650
- *Ignacio Galindo*
651
-
652
- * `config.i18n.raise_on_missing_translations = true` now raises on any missing translation.
653
-
654
- Previously it would only raise when called in a view or controller. Now it will raise
655
- anytime `I18n.t` is provided an unrecognised key.
656
-
657
- If you do not want this behaviour, you can customise the i18n exception handler. See the
658
- upgrading guide or i18n guide for more information.
659
-
660
- *Alex Ghiculescu*
661
-
662
- * `ActiveSupport::CurrentAttributes` now raises if a restricted attribute name is used.
140
+ * Support `hexBinary` type in `ActiveSupport::XmlMini`.
663
141
 
664
- Attributes such as `set` and `reset` cannot be used as they clash with the
665
- `CurrentAttributes` public API.
142
+ *heka1024*
666
143
 
667
- *Alex Ghiculescu*
144
+ * Deprecate `ActiveSupport::ProxyObject` in favor of Ruby's built-in `BasicObject`.
668
145
 
669
- * `HashWithIndifferentAccess#transform_keys` now takes a Hash argument, just
670
- as Ruby's `Hash#transform_keys` does.
671
-
672
- *Akira Matsuda*
673
-
674
- * `delegate` now defines method with proper arity when delegating to a Class.
675
- With this change, it defines faster method (3.5x faster with no argument).
676
- However, in order to gain this benefit, the delegation target method has to
677
- be defined before declaring the delegation.
678
-
679
- ```ruby
680
- # This defines 3.5 times faster method than before
681
- class C
682
- def self.x() end
683
- delegate :x, to: :class
684
- end
685
-
686
- class C
687
- # This works but silently falls back to old behavior because
688
- # `delegate` cannot find the definition of `x`
689
- delegate :x, to: :class
690
- def self.x() end
691
- end
692
- ```
146
+ *Earlopain*
693
147
 
694
- *Akira Matsuda*
148
+ * `stub_const` now accepts a `exists: false` parameter to allow stubbing missing constants.
695
149
 
696
- * `assert_difference` message now includes what changed.
150
+ *Jean Boussier*
697
151
 
698
- This makes it easier to debug non-obvious failures.
152
+ * Make `ActiveSupport::BacktraceCleaner` copy filters and silencers on dup and clone.
699
153
 
700
- Before:
154
+ Previously the copy would still share the internal silencers and filters array,
155
+ causing state to leak.
701
156
 
702
- ```
703
- "User.count" didn't change by 32.
704
- Expected: 1611
705
- Actual: 1579
706
- ```
157
+ *Jean Boussier*
707
158
 
708
- After:
159
+ * Updating Astana with Western Kazakhstan TZInfo identifier.
709
160
 
710
- ```
711
- "User.count" didn't change by 32, but by 0.
712
- Expected: 1611
713
- Actual: 1579
714
- ```
715
-
716
- *Alex Ghiculescu*
161
+ *Damian Nelson*
717
162
 
718
- * Add ability to match exception messages to `assert_raises` assertion
163
+ * Add filename support for `ActiveSupport::Logger.logger_outputs_to?`.
719
164
 
720
- Instead of this
721
165
  ```ruby
722
- error = assert_raises(ArgumentError) do
723
- perform_service(param: 'exception')
724
- end
725
- assert_match(/incorrect param/i, error.message)
166
+ logger = Logger.new('/var/log/rails.log')
167
+ ActiveSupport::Logger.logger_outputs_to?(logger, '/var/log/rails.log')
726
168
  ```
727
169
 
728
- you can now write this
729
- ```ruby
730
- assert_raises(ArgumentError, match: /incorrect param/i) do
731
- perform_service(param: 'exception')
732
- end
733
- ```
734
-
735
- *fatkodima*
736
-
737
- * Add `Rails.env.local?` shorthand for `Rails.env.development? || Rails.env.test?`.
738
-
739
- *DHH*
170
+ *Christian Schmidt*
740
171
 
741
- * `ActiveSupport::Testing::TimeHelpers` now accepts named `with_usec` argument
742
- to `freeze_time`, `travel`, and `travel_to` methods. Passing true prevents
743
- truncating the destination time with `change(usec: 0)`.
172
+ * Include `IPAddr#prefix` when serializing an `IPAddr` using the
173
+ `ActiveSupport::MessagePack` serializer.
744
174
 
745
- *KevSlashNull*, and *serprex*
175
+ This change is backward and forward compatible — old payloads can
176
+ still be read, and new payloads will be readable by older versions of Rails.
746
177
 
747
- * `ActiveSupport::CurrentAttributes.resets` now accepts a method name
178
+ *Taiki Komaba*
748
179
 
749
- The block API is still the recommended approach, but now both APIs are supported:
180
+ * Add `default:` support for `ActiveSupport::CurrentAttributes.attribute`.
750
181
 
751
182
  ```ruby
752
183
  class Current < ActiveSupport::CurrentAttributes
753
- resets { Time.zone = nil }
754
- resets :clear_time_zone
184
+ attribute :counter, default: 0
755
185
  end
756
186
  ```
757
187
 
758
- *Alex Ghiculescu*
759
-
760
- * Ensure `ActiveSupport::Testing::Isolation::Forking` closes pipes
761
-
762
- Previously, `Forking.run_in_isolation` opened two ends of a pipe. The fork
763
- process closed the read end, wrote to it, and then terminated (which
764
- presumably closed the file descriptors on its end). The parent process
765
- closed the write end, read from it, and returned, never closing the read
766
- end.
767
-
768
- This resulted in an accumulation of open file descriptors, which could
769
- cause errors if the limit is reached.
770
-
771
- *Sam Bostock*
772
-
773
- * Fix `Time#change` and `Time#advance` for times around the end of Daylight
774
- Saving Time.
775
-
776
- Previously, when `Time#change` or `Time#advance` constructed a time inside
777
- the final stretch of Daylight Saving Time (DST), the non-DST offset would
778
- always be chosen for local times:
779
-
780
- ```ruby
781
- # DST ended just before 2021-11-07 2:00:00 AM in US/Eastern.
782
- ENV["TZ"] = "US/Eastern"
783
-
784
- time = Time.local(2021, 11, 07, 00, 59, 59) + 1
785
- # => 2021-11-07 01:00:00 -0400
786
- time.change(day: 07)
787
- # => 2021-11-07 01:00:00 -0500
788
- time.advance(seconds: 0)
789
- # => 2021-11-07 01:00:00 -0500
790
-
791
- time = Time.local(2021, 11, 06, 01, 00, 00)
792
- # => 2021-11-06 01:00:00 -0400
793
- time.change(day: 07)
794
- # => 2021-11-07 01:00:00 -0500
795
- time.advance(days: 1)
796
- # => 2021-11-07 01:00:00 -0500
797
- ```
798
-
799
- And the DST offset would always be chosen for times with a `TimeZone`
800
- object:
801
-
802
- ```ruby
803
- Time.zone = "US/Eastern"
804
-
805
- time = Time.new(2021, 11, 07, 02, 00, 00, Time.zone) - 3600
806
- # => 2021-11-07 01:00:00 -0500
807
- time.change(day: 07)
808
- # => 2021-11-07 01:00:00 -0400
809
- time.advance(seconds: 0)
810
- # => 2021-11-07 01:00:00 -0400
811
-
812
- time = Time.new(2021, 11, 8, 01, 00, 00, Time.zone)
813
- # => 2021-11-08 01:00:00 -0500
814
- time.change(day: 07)
815
- # => 2021-11-07 01:00:00 -0400
816
- time.advance(days: -1)
817
- # => 2021-11-07 01:00:00 -0400
818
- ```
819
-
820
- Now, `Time#change` and `Time#advance` will choose the offset that matches
821
- the original time's offset when possible:
822
-
823
- ```ruby
824
- ENV["TZ"] = "US/Eastern"
825
-
826
- time = Time.local(2021, 11, 07, 00, 59, 59) + 1
827
- # => 2021-11-07 01:00:00 -0400
828
- time.change(day: 07)
829
- # => 2021-11-07 01:00:00 -0400
830
- time.advance(seconds: 0)
831
- # => 2021-11-07 01:00:00 -0400
832
-
833
- time = Time.local(2021, 11, 06, 01, 00, 00)
834
- # => 2021-11-06 01:00:00 -0400
835
- time.change(day: 07)
836
- # => 2021-11-07 01:00:00 -0400
837
- time.advance(days: 1)
838
- # => 2021-11-07 01:00:00 -0400
839
-
840
- Time.zone = "US/Eastern"
841
-
842
- time = Time.new(2021, 11, 07, 02, 00, 00, Time.zone) - 3600
843
- # => 2021-11-07 01:00:00 -0500
844
- time.change(day: 07)
845
- # => 2021-11-07 01:00:00 -0500
846
- time.advance(seconds: 0)
847
- # => 2021-11-07 01:00:00 -0500
848
-
849
- time = Time.new(2021, 11, 8, 01, 00, 00, Time.zone)
850
- # => 2021-11-08 01:00:00 -0500
851
- time.change(day: 07)
852
- # => 2021-11-07 01:00:00 -0500
853
- time.advance(days: -1)
854
- # => 2021-11-07 01:00:00 -0500
855
- ```
856
-
857
- *Kevin Hall*, *Takayoshi Nishida*, and *Jonathan Hefner*
858
-
859
- * Fix MemoryStore to preserve entries TTL when incrementing or decrementing
860
-
861
- This is to be more consistent with how MemCachedStore and RedisCacheStore behaves.
862
-
863
- *Jean Boussier*
188
+ *Sean Doyle*
864
189
 
865
- * `Rails.error.handle` and `Rails.error.record` filter now by multiple error classes.
190
+ * Yield instance to `Object#with` block.
866
191
 
867
192
  ```ruby
868
- Rails.error.handle(IOError, ArgumentError) do
869
- 1 + '1' # raises TypeError
193
+ client.with(timeout: 5_000) do |c|
194
+ c.get("/commits")
870
195
  end
871
- 1 + 1 # TypeErrors are not IOErrors or ArgumentError, so this will *not* be handled
872
196
  ```
873
197
 
874
- *Martin Spickermann*
198
+ *Sean Doyle*
875
199
 
876
- * `Class#subclasses` and `Class#descendants` now automatically filter reloaded classes.
200
+ * Use logical core count instead of physical core count to determine the
201
+ default number of workers when parallelizing tests.
877
202
 
878
- Previously they could return old implementations of reloadable classes that have been
879
- dereferenced but not yet garbage collected.
880
-
881
- They now automatically filter such classes like `DescendantTracker#subclasses` and
882
- `DescendantTracker#descendants`.
883
-
884
- *Jean Boussier*
885
-
886
- * `Rails.error.report` now marks errors as reported to avoid reporting them twice.
203
+ *Jonathan Hefner*
887
204
 
888
- In some cases, users might want to report errors explicitly with some extra context
889
- before letting it bubble up.
205
+ * Fix `Time.now/DateTime.now/Date.today` to return results in a system timezone after `#travel_to`.
890
206
 
891
- This also allows to safely catch and report errors outside of the execution context.
207
+ There is a bug in the current implementation of #travel_to:
208
+ it remembers a timezone of its argument, and all stubbed methods start
209
+ returning results in that remembered timezone. However, the expected
210
+ behavior is to return results in a system timezone.
892
211
 
893
- *Jean Boussier*
212
+ *Aleksei Chernenkov*
894
213
 
895
- * Add `assert_error_reported` and `assert_no_error_reported`
214
+ * Add `ErrorReported#unexpected` to report precondition violations.
896
215
 
897
- Allows to easily asserts an error happened but was handled
216
+ For example:
898
217
 
899
218
  ```ruby
900
- report = assert_error_reported(IOError) do
219
+ def edit
220
+ if published?
221
+ Rails.error.unexpected("[BUG] Attempting to edit a published article, that shouldn't be possible")
222
+ return false
223
+ end
901
224
  # ...
902
225
  end
903
- assert_equal "Oops", report.error.message
904
- assert_equal "admin", report.context[:section]
905
- assert_equal :warning, report.severity
906
- assert_predicate report, :handled?
907
226
  ```
908
227
 
909
- *Jean Boussier*
910
-
911
- * `ActiveSupport::Deprecation` behavior callbacks can now receive the
912
- deprecator instance as an argument. This makes it easier for such callbacks
913
- to change their behavior based on the deprecator's state. For example,
914
- based on the deprecator's `debug` flag.
915
-
916
- 3-arity and splat-args callbacks such as the following will now be passed
917
- the deprecator instance as their third argument:
918
-
919
- * `->(message, callstack, deprecator) { ... }`
920
- * `->(*args) { ... }`
921
- * `->(message, *other_args) { ... }`
922
-
923
- 2-arity and 4-arity callbacks such as the following will continue to behave
924
- the same as before:
925
-
926
- * `->(message, callstack) { ... }`
927
- * `->(message, callstack, deprecation_horizon, gem_name) { ... }`
928
- * `->(message, callstack, *deprecation_details) { ... }`
929
-
930
- *Jonathan Hefner*
931
-
932
- * `ActiveSupport::Deprecation#disallowed_warnings` now affects the instance on
933
- which it is configured.
934
-
935
- This means that individual `ActiveSupport::Deprecation` instances can be
936
- configured with their own disallowed warnings, and the global
937
- `ActiveSupport::Deprecation.disallowed_warnings` now only affects the global
938
- `ActiveSupport::Deprecation.warn`.
939
-
940
- **Before**
941
-
942
- ```ruby
943
- ActiveSupport::Deprecation.disallowed_warnings = ["foo"]
944
- deprecator = ActiveSupport::Deprecation.new("2.0", "MyCoolGem")
945
- deprecator.disallowed_warnings = ["bar"]
946
-
947
- ActiveSupport::Deprecation.warn("foo") # => raise ActiveSupport::DeprecationException
948
- ActiveSupport::Deprecation.warn("bar") # => print "DEPRECATION WARNING: bar"
949
- deprecator.warn("foo") # => raise ActiveSupport::DeprecationException
950
- deprecator.warn("bar") # => print "DEPRECATION WARNING: bar"
951
- ```
952
-
953
- **After**
954
-
955
- ```ruby
956
- ActiveSupport::Deprecation.disallowed_warnings = ["foo"]
957
- deprecator = ActiveSupport::Deprecation.new("2.0", "MyCoolGem")
958
- deprecator.disallowed_warnings = ["bar"]
959
-
960
- ActiveSupport::Deprecation.warn("foo") # => raise ActiveSupport::DeprecationException
961
- ActiveSupport::Deprecation.warn("bar") # => print "DEPRECATION WARNING: bar"
962
- deprecator.warn("foo") # => print "DEPRECATION WARNING: foo"
963
- deprecator.warn("bar") # => raise ActiveSupport::DeprecationException
964
- ```
965
-
966
- Note that global `ActiveSupport::Deprecation` methods such as `ActiveSupport::Deprecation.warn`
967
- and `ActiveSupport::Deprecation.disallowed_warnings` have been deprecated.
968
-
969
- *Jonathan Hefner*
970
-
971
- * Add italic and underline support to `ActiveSupport::LogSubscriber#color`
972
-
973
- Previously, only bold text was supported via a positional argument.
974
- This allows for bold, italic, and underline options to be specified
975
- for colored logs.
976
-
977
- ```ruby
978
- info color("Hello world!", :red, bold: true, underline: true)
979
- ```
980
-
981
- *Gannon McGibbon*
982
-
983
- * Add `String#downcase_first` method.
984
-
985
- This method is the corollary of `String#upcase_first`.
986
-
987
- *Mark Schneider*
988
-
989
- * `thread_mattr_accessor` will call `.dup.freeze` on non-frozen default values.
990
-
991
- This provides a basic level of protection against different threads trying
992
- to mutate a shared default object.
993
-
994
- *Jonathan Hefner*
995
-
996
- * Add `raise_on_invalid_cache_expiration_time` config to `ActiveSupport::Cache::Store`
997
-
998
- Specifies if an `ArgumentError` should be raised if `Rails.cache` `fetch` or
999
- `write` are given an invalid `expires_at` or `expires_in` time.
1000
-
1001
- Options are `true`, and `false`. If `false`, the exception will be reported
1002
- as `handled` and logged instead. Defaults to `true` if `config.load_defaults >= 7.1`.
1003
-
1004
- *Trevor Turk*
1005
-
1006
- * `ActiveSupport::Cache::Store#fetch` now passes an options accessor to the block.
1007
-
1008
- It makes possible to override cache options:
1009
-
1010
- Rails.cache.fetch("3rd-party-token") do |name, options|
1011
- token = fetch_token_from_remote
1012
- # set cache's TTL to match token's TTL
1013
- options.expires_in = token.expires_in
1014
- token
1015
- end
1016
-
1017
- *Andrii Gladkyi*, *Jean Boussier*
1018
-
1019
- * `default` option of `thread_mattr_accessor` now applies through inheritance and
1020
- also across new threads.
1021
-
1022
- Previously, the `default` value provided was set only at the moment of defining
1023
- the attribute writer, which would cause the attribute to be uninitialized in
1024
- descendants and in other threads.
1025
-
1026
- Fixes #43312.
1027
-
1028
- *Thierry Deo*
1029
-
1030
- * Redis cache store is now compatible with redis-rb 5.0.
228
+ The above will raise an error in development and test, but only report the error in production.
1031
229
 
1032
230
  *Jean Boussier*
1033
231
 
1034
- * Add `skip_nil:` support to `ActiveSupport::Cache::Store#fetch_multi`.
1035
-
1036
- *Daniel Alfaro*
232
+ * Make the order of read_multi and write_multi notifications for `Cache::Store#fetch_multi` operations match the order they are executed in.
1037
233
 
1038
- * Add `quarter` method to date/time
234
+ *Adam Renberg Tamm*
1039
235
 
1040
- *Matt Swanson*
236
+ * Make return values of `Cache::Store#write` consistent.
1041
237
 
1042
- * Fix `NoMethodError` on custom `ActiveSupport::Deprecation` behavior.
238
+ The return value was not specified before. Now it returns `true` on a successful write,
239
+ `nil` if there was an error talking to the cache backend, and `false` if the write failed
240
+ for another reason (e.g. the key already exists and `unless_exist: true` was passed).
1043
241
 
1044
- `ActiveSupport::Deprecation.behavior=` was supposed to accept any object
1045
- that responds to `call`, but in fact its internal implementation assumed that
1046
- this object could respond to `arity`, so it was restricted to only `Proc` objects.
242
+ *Sander Verdonschot*
1047
243
 
1048
- This change removes this `arity` restriction of custom behaviors.
244
+ * Fix logged cache keys not always matching actual key used by cache action.
1049
245
 
1050
- *Ryo Nakamura*
1051
-
1052
- * Support `:url_safe` option for `MessageEncryptor`.
1053
-
1054
- The `MessageEncryptor` constructor now accepts a `:url_safe` option, similar
1055
- to the `MessageVerifier` constructor. When enabled, this option ensures
1056
- that messages use a URL-safe encoding.
246
+ *Hartley McGuire*
1057
247
 
1058
- *Jonathan Hefner*
248
+ * Improve error messages of `assert_changes` and `assert_no_changes`.
1059
249
 
1060
- * Add `url_safe` option to `ActiveSupport::MessageVerifier` initializer
250
+ `assert_changes` error messages now display objects with `.inspect` to make it easier
251
+ to differentiate nil from empty strings, strings from symbols, etc.
252
+ `assert_no_changes` error messages now surface the actual value.
1061
253
 
1062
- `ActiveSupport::MessageVerifier.new` now takes optional `url_safe` argument.
1063
- It can generate URL-safe strings by passing `url_safe: true`.
254
+ *pcreux*
1064
255
 
1065
- ```ruby
1066
- verifier = ActiveSupport::MessageVerifier.new(url_safe: true)
1067
- message = verifier.generate(data) # => URL-safe string
1068
- ```
256
+ * Fix `#to_fs(:human_size)` to correctly work with negative numbers.
1069
257
 
1070
- This option is `false` by default to be backwards compatible.
258
+ *Earlopain*
1071
259
 
1072
- *Shouichi Kamiya*
260
+ * Fix `BroadcastLogger#dup` so that it duplicates the logger's `broadcasts`.
1073
261
 
1074
- * Enable connection pooling by default for `MemCacheStore` and `RedisCacheStore`.
262
+ *Andrew Novoselac*
1075
263
 
1076
- If you want to disable connection pooling, set `:pool` option to `false` when configuring the cache store:
264
+ * Fix issue where `bootstrap.rb` overwrites the `level` of a `BroadcastLogger`'s `broadcasts`.
1077
265
 
1078
- ```ruby
1079
- config.cache_store = :mem_cache_store, "cache.example.com", pool: false
1080
- ```
266
+ *Andrew Novoselac*
1081
267
 
1082
- *fatkodima*
268
+ * Fix compatibility with the `semantic_logger` gem.
1083
269
 
1084
- * Add `force:` support to `ActiveSupport::Cache::Store#fetch_multi`.
270
+ The `semantic_logger` gem doesn't behave exactly like stdlib logger in that
271
+ `SemanticLogger#level` returns a Symbol while stdlib `Logger#level` returns an Integer.
1085
272
 
1086
- *fatkodima*
273
+ This caused the various `LogSubscriber` classes in Rails to break when assigned a
274
+ `SemanticLogger` instance.
1087
275
 
1088
- * Deprecated `:pool_size` and `:pool_timeout` options for configuring connection pooling in cache stores.
276
+ *Jean Boussier*, *ojab*
1089
277
 
1090
- Use `pool: true` to enable pooling with default settings:
278
+ * Fix MemoryStore to prevent race conditions when incrementing or decrementing.
1091
279
 
1092
- ```ruby
1093
- config.cache_store = :redis_cache_store, pool: true
1094
- ```
280
+ *Pierre Jambet*
1095
281
 
1096
- Or pass individual options via `:pool` option:
282
+ * Implement `HashWithIndifferentAccess#to_proc`.
1097
283
 
1098
- ```ruby
1099
- config.cache_store = :redis_cache_store, pool: { size: 10, timeout: 2 }
1100
- ```
284
+ Previously, calling `#to_proc` on `HashWithIndifferentAccess` object used inherited `#to_proc`
285
+ method from the `Hash` class, which was not able to access values using indifferent keys.
1101
286
 
1102
287
  *fatkodima*
1103
288
 
1104
- * Allow #increment and #decrement methods of `ActiveSupport::Cache::Store`
1105
- subclasses to set new values.
1106
-
1107
- Previously incrementing or decrementing an unset key would fail and return
1108
- nil. A default will now be assumed and the key will be created.
1109
-
1110
- *Andrej Blagojević*, *Eugene Kenny*
1111
-
1112
- * Add `skip_nil:` support to `RedisCacheStore`
1113
-
1114
- *Joey Paris*
1115
-
1116
- * `ActiveSupport::Cache::MemoryStore#write(name, val, unless_exist:true)` now
1117
- correctly writes expired keys.
1118
-
1119
- *Alan Savage*
1120
-
1121
- * `ActiveSupport::ErrorReporter` now accepts and forward a `source:` parameter.
1122
-
1123
- This allow libraries to signal the origin of the errors, and reporters
1124
- to easily ignore some sources.
1125
-
1126
- *Jean Boussier*
1127
-
1128
- * Fix and add protections for XSS in `ActionView::Helpers` and `ERB::Util`.
1129
-
1130
- Add the method `ERB::Util.xml_name_escape` to escape dangerous characters
1131
- in names of tags and names of attributes, following the specification of XML.
1132
-
1133
- *Álvaro Martín Fraguas*
1134
-
1135
- * Respect `ActiveSupport::Logger.new`'s `:formatter` keyword argument
1136
-
1137
- The stdlib `Logger::new` allows passing a `:formatter` keyword argument to
1138
- set the logger's formatter. Previously `ActiveSupport::Logger.new` ignored
1139
- that argument by always setting the formatter to an instance of
1140
- `ActiveSupport::Logger::SimpleFormatter`.
1141
-
1142
- *Steven Harman*
1143
-
1144
- * Deprecate preserving the pre-Ruby 2.4 behavior of `to_time`
1145
-
1146
- With Ruby 2.4+ the default for +to_time+ changed from converting to the
1147
- local system time to preserving the offset of the receiver. At the time Rails
1148
- supported older versions of Ruby so a compatibility layer was added to assist
1149
- in the migration process. From Rails 5.0 new applications have defaulted to
1150
- the Ruby 2.4+ behavior and since Rails 7.0 now only supports Ruby 2.7+
1151
- this compatibility layer can be safely removed.
1152
-
1153
- To minimize any noise generated the deprecation warning only appears when the
1154
- setting is configured to `false` as that is the only scenario where the
1155
- removal of the compatibility layer has any effect.
1156
-
1157
- *Andrew White*
1158
-
1159
- * `Pathname.blank?` only returns true for `Pathname.new("")`
1160
-
1161
- Previously it would end up calling `Pathname#empty?` which returned true
1162
- if the path existed and was an empty directory or file.
1163
-
1164
- That behavior was unlikely to be expected.
1165
-
1166
- *Jean Boussier*
1167
-
1168
- * Deprecate `Notification::Event`'s `#children` and `#parent_of?`
1169
-
1170
- *John Hawthorn*
1171
-
1172
- * Change the default serializer of `ActiveSupport::MessageVerifier` from
1173
- `Marshal` to `ActiveSupport::JSON` when using `config.load_defaults 7.1`.
1174
-
1175
- Messages serialized with `Marshal` can still be read, but new messages will
1176
- be serialized with `ActiveSupport::JSON`. For more information, see
1177
- https://guides.rubyonrails.org/v7.1/configuring.html#config-active-support-message-serializer.
1178
-
1179
- *Saba Kiaei*, *David Buckley*, and *Jonathan Hefner*
1180
-
1181
- * Change the default serializer of `ActiveSupport::MessageEncryptor` from
1182
- `Marshal` to `ActiveSupport::JSON` when using `config.load_defaults 7.1`.
1183
-
1184
- Messages serialized with `Marshal` can still be read, but new messages will
1185
- be serialized with `ActiveSupport::JSON`. For more information, see
1186
- https://guides.rubyonrails.org/v7.1/configuring.html#config-active-support-message-serializer.
1187
-
1188
- *Zack Deveau*, *Martin Gingras*, and *Jonathan Hefner*
1189
-
1190
- * Add `ActiveSupport::TestCase#stub_const` to stub a constant for the duration of a yield.
1191
-
1192
- *DHH*
1193
-
1194
- * Fix `ActiveSupport::EncryptedConfiguration` to be compatible with Psych 4
1195
-
1196
- *Stephen Sugden*
1197
-
1198
- * Improve `File.atomic_write` error handling
1199
-
1200
- *Daniel Pepper*
1201
-
1202
- * Fix `Class#descendants` and `DescendantsTracker#descendants` compatibility with Ruby 3.1.
1203
-
1204
- [The native `Class#descendants` was reverted prior to Ruby 3.1 release](https://bugs.ruby-lang.org/issues/14394#note-33),
1205
- but `Class#subclasses` was kept, breaking the feature detection.
1206
-
1207
- *Jean Boussier*
1208
-
1209
- Please check [7-0-stable](https://github.com/rails/rails/blob/7-0-stable/activesupport/CHANGELOG.md) for previous changes.
289
+ Please check [7-1-stable](https://github.com/rails/rails/blob/7-1-stable/activesupport/CHANGELOG.md) for previous changes.