activesupport 7.2.2.2 → 8.1.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 (135) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +368 -152
  3. data/README.rdoc +1 -1
  4. data/lib/active_support/backtrace_cleaner.rb +73 -2
  5. data/lib/active_support/benchmark.rb +21 -0
  6. data/lib/active_support/benchmarkable.rb +3 -2
  7. data/lib/active_support/broadcast_logger.rb +61 -74
  8. data/lib/active_support/cache/file_store.rb +14 -4
  9. data/lib/active_support/cache/mem_cache_store.rb +27 -29
  10. data/lib/active_support/cache/memory_store.rb +11 -5
  11. data/lib/active_support/cache/null_store.rb +2 -2
  12. data/lib/active_support/cache/redis_cache_store.rb +43 -34
  13. data/lib/active_support/cache/strategy/local_cache.rb +72 -27
  14. data/lib/active_support/cache/strategy/local_cache_middleware.rb +7 -7
  15. data/lib/active_support/cache.rb +88 -20
  16. data/lib/active_support/callbacks.rb +28 -13
  17. data/lib/active_support/class_attribute.rb +33 -0
  18. data/lib/active_support/code_generator.rb +9 -0
  19. data/lib/active_support/concurrency/load_interlock_aware_monitor.rb +8 -62
  20. data/lib/active_support/concurrency/share_lock.rb +0 -1
  21. data/lib/active_support/concurrency/thread_monitor.rb +55 -0
  22. data/lib/active_support/configurable.rb +34 -0
  23. data/lib/active_support/configuration_file.rb +15 -6
  24. data/lib/active_support/continuous_integration.rb +145 -0
  25. data/lib/active_support/core_ext/array/conversions.rb +3 -3
  26. data/lib/active_support/core_ext/array.rb +7 -7
  27. data/lib/active_support/core_ext/benchmark.rb +4 -14
  28. data/lib/active_support/core_ext/big_decimal.rb +1 -1
  29. data/lib/active_support/core_ext/class/attribute.rb +26 -20
  30. data/lib/active_support/core_ext/class.rb +2 -2
  31. data/lib/active_support/core_ext/date/conversions.rb +2 -0
  32. data/lib/active_support/core_ext/date.rb +5 -5
  33. data/lib/active_support/core_ext/date_and_time/compatibility.rb +0 -35
  34. data/lib/active_support/core_ext/date_time/compatibility.rb +3 -5
  35. data/lib/active_support/core_ext/date_time/conversions.rb +4 -2
  36. data/lib/active_support/core_ext/date_time.rb +5 -5
  37. data/lib/active_support/core_ext/digest.rb +1 -1
  38. data/lib/active_support/core_ext/enumerable.rb +25 -8
  39. data/lib/active_support/core_ext/erb/util.rb +5 -5
  40. data/lib/active_support/core_ext/file.rb +1 -1
  41. data/lib/active_support/core_ext/hash/deep_merge.rb +1 -0
  42. data/lib/active_support/core_ext/hash/except.rb +0 -12
  43. data/lib/active_support/core_ext/hash.rb +8 -8
  44. data/lib/active_support/core_ext/integer.rb +3 -3
  45. data/lib/active_support/core_ext/kernel.rb +3 -3
  46. data/lib/active_support/core_ext/module/attr_internal.rb +3 -4
  47. data/lib/active_support/core_ext/module/introspection.rb +3 -0
  48. data/lib/active_support/core_ext/module.rb +11 -11
  49. data/lib/active_support/core_ext/numeric.rb +3 -3
  50. data/lib/active_support/core_ext/object/json.rb +24 -11
  51. data/lib/active_support/core_ext/object/to_query.rb +7 -1
  52. data/lib/active_support/core_ext/object/try.rb +2 -2
  53. data/lib/active_support/core_ext/object.rb +13 -13
  54. data/lib/active_support/core_ext/pathname.rb +2 -2
  55. data/lib/active_support/core_ext/range/overlap.rb +3 -3
  56. data/lib/active_support/core_ext/range/sole.rb +17 -0
  57. data/lib/active_support/core_ext/range.rb +4 -4
  58. data/lib/active_support/core_ext/securerandom.rb +24 -8
  59. data/lib/active_support/core_ext/string/filters.rb +3 -3
  60. data/lib/active_support/core_ext/string/multibyte.rb +12 -3
  61. data/lib/active_support/core_ext/string/output_safety.rb +19 -12
  62. data/lib/active_support/core_ext/string.rb +13 -13
  63. data/lib/active_support/core_ext/symbol.rb +1 -1
  64. data/lib/active_support/core_ext/thread/backtrace/location.rb +2 -7
  65. data/lib/active_support/core_ext/time/calculations.rb +7 -2
  66. data/lib/active_support/core_ext/time/compatibility.rb +2 -19
  67. data/lib/active_support/core_ext/time/conversions.rb +2 -0
  68. data/lib/active_support/core_ext/time.rb +5 -5
  69. data/lib/active_support/core_ext.rb +1 -1
  70. data/lib/active_support/current_attributes/test_helper.rb +2 -2
  71. data/lib/active_support/current_attributes.rb +27 -17
  72. data/lib/active_support/delegation.rb +25 -44
  73. data/lib/active_support/dependencies/interlock.rb +11 -5
  74. data/lib/active_support/dependencies.rb +6 -2
  75. data/lib/active_support/deprecation/reporting.rb +4 -21
  76. data/lib/active_support/deprecation.rb +1 -1
  77. data/lib/active_support/duration.rb +14 -10
  78. data/lib/active_support/editor.rb +70 -0
  79. data/lib/active_support/encrypted_configuration.rb +20 -2
  80. data/lib/active_support/error_reporter.rb +81 -4
  81. data/lib/active_support/event_reporter/test_helper.rb +32 -0
  82. data/lib/active_support/event_reporter.rb +592 -0
  83. data/lib/active_support/evented_file_update_checker.rb +5 -2
  84. data/lib/active_support/execution_context.rb +64 -7
  85. data/lib/active_support/execution_wrapper.rb +1 -1
  86. data/lib/active_support/file_update_checker.rb +8 -6
  87. data/lib/active_support/gem_version.rb +3 -3
  88. data/lib/active_support/gzip.rb +1 -0
  89. data/lib/active_support/hash_with_indifferent_access.rb +61 -38
  90. data/lib/active_support/i18n_railtie.rb +19 -11
  91. data/lib/active_support/inflector/inflections.rb +32 -15
  92. data/lib/active_support/inflector/methods.rb +2 -2
  93. data/lib/active_support/inflector/transliterate.rb +6 -8
  94. data/lib/active_support/isolated_execution_state.rb +17 -17
  95. data/lib/active_support/json/decoding.rb +6 -4
  96. data/lib/active_support/json/encoding.rb +157 -21
  97. data/lib/active_support/lazy_load_hooks.rb +1 -1
  98. data/lib/active_support/log_subscriber.rb +2 -6
  99. data/lib/active_support/logger_thread_safe_level.rb +6 -3
  100. data/lib/active_support/message_encryptors.rb +54 -2
  101. data/lib/active_support/message_pack/extensions.rb +6 -1
  102. data/lib/active_support/message_verifier.rb +9 -0
  103. data/lib/active_support/message_verifiers.rb +57 -3
  104. data/lib/active_support/messages/rotation_coordinator.rb +9 -0
  105. data/lib/active_support/messages/rotator.rb +10 -0
  106. data/lib/active_support/multibyte/chars.rb +12 -2
  107. data/lib/active_support/multibyte.rb +4 -0
  108. data/lib/active_support/notifications/fanout.rb +64 -43
  109. data/lib/active_support/notifications/instrumenter.rb +1 -1
  110. data/lib/active_support/number_helper.rb +22 -0
  111. data/lib/active_support/railtie.rb +32 -9
  112. data/lib/active_support/structured_event_subscriber.rb +99 -0
  113. data/lib/active_support/subscriber.rb +0 -5
  114. data/lib/active_support/syntax_error_proxy.rb +7 -0
  115. data/lib/active_support/tagged_logging.rb +5 -0
  116. data/lib/active_support/test_case.rb +67 -6
  117. data/lib/active_support/testing/assertions.rb +115 -27
  118. data/lib/active_support/testing/autorun.rb +5 -0
  119. data/lib/active_support/testing/error_reporter_assertions.rb +18 -1
  120. data/lib/active_support/testing/event_reporter_assertions.rb +227 -0
  121. data/lib/active_support/testing/isolation.rb +0 -2
  122. data/lib/active_support/testing/notification_assertions.rb +92 -0
  123. data/lib/active_support/testing/parallelization/server.rb +15 -2
  124. data/lib/active_support/testing/parallelization/worker.rb +8 -4
  125. data/lib/active_support/testing/parallelization.rb +25 -1
  126. data/lib/active_support/testing/tests_without_assertions.rb +1 -1
  127. data/lib/active_support/testing/time_helpers.rb +9 -4
  128. data/lib/active_support/time_with_zone.rb +36 -23
  129. data/lib/active_support/values/time_zone.rb +19 -10
  130. data/lib/active_support/xml_mini.rb +3 -2
  131. data/lib/active_support.rb +21 -9
  132. metadata +34 -12
  133. data/lib/active_support/core_ext/range/each.rb +0 -24
  134. data/lib/active_support/proxy_object.rb +0 -20
  135. data/lib/active_support/testing/strict_warnings.rb +0 -43
@@ -8,24 +8,70 @@ module ActiveSupport
8
8
  delegate :use_standard_json_time_format, :use_standard_json_time_format=,
9
9
  :time_precision, :time_precision=,
10
10
  :escape_html_entities_in_json, :escape_html_entities_in_json=,
11
+ :escape_js_separators_in_json, :escape_js_separators_in_json=,
11
12
  :json_encoder, :json_encoder=,
12
13
  to: :'ActiveSupport::JSON::Encoding'
13
14
  end
14
15
 
15
16
  module JSON
16
- # Dumps objects in JSON (JavaScript Object Notation).
17
- # See http://www.json.org for more info.
18
- #
19
- # ActiveSupport::JSON.encode({ team: 'rails', players: '36' })
20
- # # => "{\"team\":\"rails\",\"players\":\"36\"}"
21
17
  class << self
18
+ # Dumps objects in JSON (JavaScript Object Notation).
19
+ # See http://www.json.org for more info.
20
+ #
21
+ # ActiveSupport::JSON.encode({ team: 'rails', players: '36' })
22
+ # # => "{\"team\":\"rails\",\"players\":\"36\"}"
23
+ #
24
+ # By default, it generates JSON that is safe to include in JavaScript, as
25
+ # it escapes U+2028 (Line Separator) and U+2029 (Paragraph Separator):
26
+ #
27
+ # ActiveSupport::JSON.encode({ key: "\u2028" })
28
+ # # => "{\"key\":\"\\u2028\"}"
29
+ #
30
+ # By default, it also generates JSON that is safe to include in HTML, as
31
+ # it escapes <tt><</tt>, <tt>></tt>, and <tt>&</tt>:
32
+ #
33
+ # ActiveSupport::JSON.encode({ key: "<>&" })
34
+ # # => "{\"key\":\"\\u003c\\u003e\\u0026\"}"
35
+ #
36
+ # This behavior can be changed with the +escape_html_entities+ option, or the
37
+ # global escape_html_entities_in_json configuration option.
38
+ #
39
+ # ActiveSupport::JSON.encode({ key: "<>&" }, escape_html_entities: false)
40
+ # # => "{\"key\":\"<>&\"}"
41
+ #
42
+ # For performance reasons, you can set the +escape+ option to false,
43
+ # which will skip all escaping:
44
+ #
45
+ # ActiveSupport::JSON.encode({ key: "\u2028<>&" }, escape: false)
46
+ # # => "{\"key\":\"\u2028<>&\"}"
22
47
  def encode(value, options = nil)
23
- Encoding.json_encoder.new(options).encode(value)
48
+ if options.nil? || options.empty?
49
+ Encoding.encode_without_options(value)
50
+ elsif options == { escape: false }.freeze
51
+ Encoding.encode_without_escape(value)
52
+ else
53
+ Encoding.json_encoder.new(options).encode(value)
54
+ end
24
55
  end
25
56
  alias_method :dump, :encode
26
57
  end
27
58
 
28
59
  module Encoding # :nodoc:
60
+ U2028 = -"\u2028".b
61
+ U2029 = -"\u2029".b
62
+
63
+ ESCAPED_CHARS = {
64
+ U2028 => '\u2028'.b,
65
+ U2029 => '\u2029'.b,
66
+ ">".b => '\u003e'.b,
67
+ "<".b => '\u003c'.b,
68
+ "&".b => '\u0026'.b,
69
+ }
70
+
71
+ HTML_ENTITIES_REGEX = Regexp.union(*(ESCAPED_CHARS.keys - [U2028, U2029]))
72
+ FULL_ESCAPE_REGEX = Regexp.union(*ESCAPED_CHARS.keys)
73
+ JS_SEPARATORS_REGEX = Regexp.union(U2028, U2029)
74
+
29
75
  class JSONGemEncoder # :nodoc:
30
76
  attr_reader :options
31
77
 
@@ -36,21 +82,23 @@ module ActiveSupport
36
82
  # Encode the given object into a JSON string
37
83
  def encode(value)
38
84
  unless options.empty?
39
- value = value.as_json(options.dup)
85
+ value = value.as_json(options.dup.freeze)
40
86
  end
41
87
  json = stringify(jsonify(value))
42
88
 
43
- # Rails does more escaping than the JSON gem natively does (we
44
- # escape \u2028 and \u2029 and optionally >, <, & to work around
45
- # certain browser problems).
46
- if Encoding.escape_html_entities_in_json
47
- json.gsub!(">", '\u003e')
48
- json.gsub!("<", '\u003c')
49
- json.gsub!("&", '\u0026')
89
+ return json unless @options.fetch(:escape, true)
90
+
91
+ json.force_encoding(::Encoding::BINARY)
92
+ if @options.fetch(:escape_html_entities, Encoding.escape_html_entities_in_json)
93
+ if Encoding.escape_js_separators_in_json
94
+ json.gsub!(FULL_ESCAPE_REGEX, ESCAPED_CHARS)
95
+ else
96
+ json.gsub!(HTML_ENTITIES_REGEX, ESCAPED_CHARS)
97
+ end
98
+ elsif Encoding.escape_js_separators_in_json
99
+ json.gsub!(JS_SEPARATORS_REGEX, ESCAPED_CHARS)
50
100
  end
51
- json.gsub!("\u2028", '\u2028')
52
- json.gsub!("\u2029", '\u2029')
53
- json
101
+ json.force_encoding(::Encoding::UTF_8)
54
102
  end
55
103
 
56
104
  private
@@ -83,14 +131,75 @@ module ActiveSupport
83
131
  when Array
84
132
  value.map { |v| jsonify(v) }
85
133
  else
86
- jsonify value.as_json
134
+ if defined?(::JSON::Fragment) && ::JSON::Fragment === value
135
+ value
136
+ else
137
+ jsonify value.as_json
138
+ end
87
139
  end
88
140
  end
89
141
 
90
142
  # Encode a "jsonified" Ruby data structure using the JSON gem
91
143
  def stringify(jsonified)
92
- ::JSON.generate(jsonified, quirks_mode: true, max_nesting: false)
144
+ ::JSON.generate(jsonified)
145
+ end
146
+ end
147
+
148
+ # ruby/json 2.14.x yields non-String keys but doesn't let us know it's a key
149
+ if defined?(::JSON::Coder) && Gem::Version.new(::JSON::VERSION) >= Gem::Version.new("2.15.2")
150
+ class JSONGemCoderEncoder # :nodoc:
151
+ JSON_NATIVE_TYPES = [Hash, Array, Float, String, Symbol, Integer, NilClass, TrueClass, FalseClass, ::JSON::Fragment].freeze
152
+ CODER = ::JSON::Coder.new do |value, is_key|
153
+ json_value = value.as_json
154
+ # Keep compatibility by calling to_s on non-String keys
155
+ next json_value.to_s if is_key
156
+ # Handle objects returning self from as_json
157
+ if json_value.equal?(value)
158
+ next ::JSON::Fragment.new(::JSON.generate(json_value))
159
+ end
160
+ # Handle objects not returning JSON-native types from as_json
161
+ count = 5
162
+ until JSON_NATIVE_TYPES.include?(json_value.class)
163
+ raise SystemStackError if count == 0
164
+ json_value = json_value.as_json
165
+ count -= 1
166
+ end
167
+ json_value
168
+ end
169
+
170
+
171
+ def initialize(options = nil)
172
+ if options
173
+ options = options.dup
174
+ @escape = options.delete(:escape) { true }
175
+ @options = options.freeze
176
+ else
177
+ @escape = true
178
+ @options = {}.freeze
179
+ end
93
180
  end
181
+
182
+ # Encode the given object into a JSON string
183
+ def encode(value)
184
+ value = value.as_json(@options) unless @options.empty?
185
+
186
+ json = CODER.dump(value)
187
+
188
+ return json unless @escape
189
+
190
+ json.force_encoding(::Encoding::BINARY)
191
+ if @options.fetch(:escape_html_entities, Encoding.escape_html_entities_in_json)
192
+ if Encoding.escape_js_separators_in_json
193
+ json.gsub!(FULL_ESCAPE_REGEX, ESCAPED_CHARS)
194
+ else
195
+ json.gsub!(HTML_ENTITIES_REGEX, ESCAPED_CHARS)
196
+ end
197
+ elsif Encoding.escape_js_separators_in_json
198
+ json.gsub!(JS_SEPARATORS_REGEX, ESCAPED_CHARS)
199
+ end
200
+ json.force_encoding(::Encoding::UTF_8)
201
+ end
202
+ end
94
203
  end
95
204
 
96
205
  class << self
@@ -102,18 +211,45 @@ module ActiveSupport
102
211
  # as a safety measure.
103
212
  attr_accessor :escape_html_entities_in_json
104
213
 
214
+ # If true, encode LINE SEPARATOR (U+2028) and PARAGRAPH SEPARATOR (U+2029)
215
+ # as escaped unicode sequences ('\u2028' and '\u2029').
216
+ # Historically these characters were not valid inside JavaScript strings
217
+ # but that changed in ECMAScript 2019. As such it's no longer a concern in
218
+ # modern browsers: https://caniuse.com/mdn-javascript_builtins_json_json_superset.
219
+ attr_accessor :escape_js_separators_in_json
220
+
105
221
  # Sets the precision of encoded time values.
106
222
  # Defaults to 3 (equivalent to millisecond precision)
107
223
  attr_accessor :time_precision
108
224
 
109
225
  # Sets the encoder used by \Rails to encode Ruby objects into JSON strings
110
226
  # in +Object#to_json+ and +ActiveSupport::JSON.encode+.
111
- attr_accessor :json_encoder
227
+ attr_reader :json_encoder
228
+
229
+ def json_encoder=(encoder)
230
+ @json_encoder = encoder
231
+ @encoder_without_options = encoder.new
232
+ @encoder_without_escape = encoder.new(escape: false)
233
+ end
234
+
235
+ def encode_without_options(value) # :nodoc:
236
+ @encoder_without_options.encode(value)
237
+ end
238
+
239
+ def encode_without_escape(value) # :nodoc:
240
+ @encoder_without_escape.encode(value)
241
+ end
112
242
  end
113
243
 
114
244
  self.use_standard_json_time_format = true
115
245
  self.escape_html_entities_in_json = true
116
- self.json_encoder = JSONGemEncoder
246
+ self.escape_js_separators_in_json = true
247
+ self.json_encoder =
248
+ if defined?(JSONGemCoderEncoder)
249
+ JSONGemCoderEncoder
250
+ else
251
+ JSONGemEncoder
252
+ end
117
253
  self.time_precision = 3
118
254
  end
119
255
  end
@@ -53,7 +53,7 @@ module ActiveSupport
53
53
  # loaded. If the component has already loaded, the block is executed
54
54
  # immediately.
55
55
  #
56
- # Options:
56
+ # ==== Options
57
57
  #
58
58
  # * <tt>:yield</tt> - Yields the object that run_load_hooks to +block+.
59
59
  # * <tt>:run_once</tt> - Given +block+ will run only once.
@@ -149,12 +149,6 @@ module ActiveSupport
149
149
  log_exception(event.name, e)
150
150
  end
151
151
 
152
- def publish_event(event)
153
- super if logger
154
- rescue => e
155
- log_exception(event.name, e)
156
- end
157
-
158
152
  attr_writer :event_levels # :nodoc:
159
153
 
160
154
  private
@@ -184,6 +178,8 @@ module ActiveSupport
184
178
  end
185
179
 
186
180
  def log_exception(name, e)
181
+ ActiveSupport.error_reporter.report(e, source: name)
182
+
187
183
  if logger
188
184
  logger.error "Could not log #{name.inspect} event. #{e.class}: #{e.message} #{e.backtrace}"
189
185
  end
@@ -7,6 +7,11 @@ module ActiveSupport
7
7
  module LoggerThreadSafeLevel # :nodoc:
8
8
  extend ActiveSupport::Concern
9
9
 
10
+ def initialize(...)
11
+ super
12
+ @local_level_key = :"logger_thread_safe_level_#{object_id}"
13
+ end
14
+
10
15
  def local_level
11
16
  IsolatedExecutionState[local_level_key]
12
17
  end
@@ -40,8 +45,6 @@ module ActiveSupport
40
45
  end
41
46
 
42
47
  private
43
- def local_level_key
44
- @local_level_key ||= :"logger_thread_safe_level_#{object_id}"
45
- end
48
+ attr_reader :local_level_key
46
49
  end
47
50
  end
@@ -26,10 +26,13 @@ module ActiveSupport
26
26
  # as the first rotation and <tt>transitional = true</tt>. Then, after all
27
27
  # servers have been updated, perform a second rolling deploy with
28
28
  # <tt>transitional = false</tt>.
29
+ #
30
+ #--
31
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#transitional
29
32
 
30
33
  ##
31
- # :method: initialize
32
- # :call-seq: initialize(&secret_generator)
34
+ # :singleton-method: new
35
+ # :call-seq: new(&secret_generator)
33
36
  #
34
37
  # Initializes a new instance. +secret_generator+ must accept a salt and a
35
38
  # +secret_length+ kwarg, and return a suitable secret (string) or secrets
@@ -43,6 +46,9 @@ module ActiveSupport
43
46
  # end
44
47
  #
45
48
  # encryptors.rotate(base: "...")
49
+ #
50
+ #--
51
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#initialize
46
52
 
47
53
  ##
48
54
  # :method: []
@@ -51,12 +57,18 @@ module ActiveSupport
51
57
  # Returns a MessageEncryptor configured with a secret derived from the
52
58
  # given +salt+, and options from #rotate. MessageEncryptor instances will
53
59
  # be memoized, so the same +salt+ will return the same instance.
60
+ #
61
+ #--
62
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#[]
54
63
 
55
64
  ##
56
65
  # :method: []=
57
66
  # :call-seq: []=(salt, encryptor)
58
67
  #
59
68
  # Overrides a MessageEncryptor instance associated with a given +salt+.
69
+ #
70
+ #--
71
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#[]=
60
72
 
61
73
  ##
62
74
  # :method: rotate
@@ -106,18 +118,55 @@ module ActiveSupport
106
118
  #
107
119
  # # Uses `serializer: Marshal, url_safe: false`.
108
120
  # encryptors[:baz]
121
+ #
122
+ #--
123
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#rotate
124
+
125
+ ##
126
+ # :method: prepend
127
+ # :call-seq:
128
+ # prepend(**options)
129
+ # prepend(&block)
130
+ #
131
+ # Just like #rotate, but prepends the given options or block to the list of
132
+ # option sets.
133
+ #
134
+ # This can be useful when you have an already-configured +MessageEncryptors+
135
+ # instance, but you want to override the way messages are encrypted.
136
+ #
137
+ # module ThirdParty
138
+ # ENCRYPTORS = ActiveSupport::MessageEncryptors.new { ... }.
139
+ # rotate(serializer: Marshal, url_safe: true).
140
+ # rotate(serializer: Marshal, url_safe: false)
141
+ # end
142
+ #
143
+ # ThirdParty.ENCRYPTORS.prepend(serializer: JSON, url_safe: true)
144
+ #
145
+ # # Uses `serializer: JSON, url_safe: true`.
146
+ # # Falls back to `serializer: Marshal, url_safe: true` or
147
+ # # `serializer: Marshal, url_safe: false`.
148
+ # ThirdParty.ENCRYPTORS[:foo]
149
+ #
150
+ #--
151
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#prepend
109
152
 
110
153
  ##
111
154
  # :method: rotate_defaults
112
155
  # :call-seq: rotate_defaults
113
156
  #
114
157
  # Invokes #rotate with the default options.
158
+ #
159
+ #--
160
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#rotate_defaults
115
161
 
116
162
  ##
117
163
  # :method: clear_rotations
118
164
  # :call-seq: clear_rotations
119
165
  #
120
166
  # Clears the list of option sets.
167
+ #
168
+ #--
169
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#clear_rotations
121
170
 
122
171
  ##
123
172
  # :method: on_rotation
@@ -129,6 +178,9 @@ module ActiveSupport
129
178
  # For example, this callback could log each time it is called, and thus
130
179
  # indicate whether old option sets are still in use or can be removed from
131
180
  # rotation.
181
+ #
182
+ #--
183
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#on_rotation
132
184
 
133
185
  ##
134
186
  private
@@ -4,9 +4,10 @@ require "bigdecimal"
4
4
  require "date"
5
5
  require "ipaddr"
6
6
  require "pathname"
7
- require "uri/generic"
7
+ require "uri"
8
8
  require "msgpack/bigint"
9
9
  require "active_support/hash_with_indifferent_access"
10
+ require "active_support/core_ext/string/output_safety"
10
11
  require "active_support/time"
11
12
 
12
13
  module ActiveSupport
@@ -102,6 +103,10 @@ module ActiveSupport
102
103
  packer: method(:write_hash_with_indifferent_access),
103
104
  unpacker: method(:read_hash_with_indifferent_access),
104
105
  recursive: true
106
+
107
+ registry.register_type 18, ActiveSupport::SafeBuffer,
108
+ packer: :to_s,
109
+ unpacker: :new
105
110
  end
106
111
 
107
112
  def install_unregistered_type_error(registry)
@@ -154,6 +154,8 @@ module ActiveSupport
154
154
  # not URL-safe. In other words, they can contain "+" and "/". If you want to
155
155
  # generate URL-safe strings (in compliance with "Base 64 Encoding with URL
156
156
  # and Filename Safe Alphabet" in RFC 4648), you can pass +true+.
157
+ # Note that MessageVerifier will always accept both URL-safe and URL-unsafe
158
+ # encoded messages, to allow a smooth transition between the two settings.
157
159
  #
158
160
  # [+:force_legacy_metadata_serializer+]
159
161
  # Whether to use the legacy metadata serializer, which serializes the
@@ -318,6 +320,13 @@ module ActiveSupport
318
320
  end
319
321
 
320
322
  private
323
+ def decode(encoded, url_safe: @url_safe)
324
+ catch :invalid_message_format do
325
+ return super
326
+ end
327
+ super(encoded, url_safe: !url_safe)
328
+ end
329
+
321
330
  def sign_encoded(encoded)
322
331
  digest = generate_digest(encoded)
323
332
  encoded << SEPARATOR << digest
@@ -26,10 +26,13 @@ module ActiveSupport
26
26
  # as the first rotation and <tt>transitional = true</tt>. Then, after all
27
27
  # servers have been updated, perform a second rolling deploy with
28
28
  # <tt>transitional = false</tt>.
29
+ #
30
+ #--
31
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#transitional
29
32
 
30
33
  ##
31
- # :method: initialize
32
- # :call-seq: initialize(&secret_generator)
34
+ # :singleton-method: new
35
+ # :call-seq: new(&secret_generator)
33
36
  #
34
37
  # Initializes a new instance. +secret_generator+ must accept a salt, and
35
38
  # return a suitable secret (string). +secret_generator+ may also accept
@@ -42,6 +45,9 @@ module ActiveSupport
42
45
  # end
43
46
  #
44
47
  # verifiers.rotate(base: "...")
48
+ #
49
+ #--
50
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#initialize
45
51
 
46
52
  ##
47
53
  # :method: []
@@ -50,16 +56,24 @@ module ActiveSupport
50
56
  # Returns a MessageVerifier configured with a secret derived from the
51
57
  # given +salt+, and options from #rotate. MessageVerifier instances will
52
58
  # be memoized, so the same +salt+ will return the same instance.
59
+ #
60
+ #--
61
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#[]
53
62
 
54
63
  ##
55
64
  # :method: []=
56
65
  # :call-seq: []=(salt, verifier)
57
66
  #
58
67
  # Overrides a MessageVerifier instance associated with a given +salt+.
68
+ #
69
+ #--
70
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#[]=
59
71
 
60
72
  ##
61
73
  # :method: rotate
62
- # :call-seq: rotate(**options)
74
+ # :call-seq:
75
+ # rotate(**options)
76
+ # rotate(&block)
63
77
  #
64
78
  # Adds +options+ to the list of option sets. Messages will be signed using
65
79
  # the first set in the list. When verifying, however, each set will be
@@ -102,18 +116,55 @@ module ActiveSupport
102
116
  #
103
117
  # # Uses `serializer: Marshal, url_safe: false`.
104
118
  # verifiers[:baz]
119
+ #
120
+ #--
121
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#rotate
122
+
123
+ ##
124
+ # :method: prepend
125
+ # :call-seq:
126
+ # prepend(**options)
127
+ # prepend(&block)
128
+ #
129
+ # Just like #rotate, but prepends the given options or block to the list of
130
+ # option sets.
131
+ #
132
+ # This can be useful when you have an already-configured +MessageVerifiers+
133
+ # instance, but you want to override the way messages are signed.
134
+ #
135
+ # module ThirdParty
136
+ # VERIFIERS = ActiveSupport::MessageVerifiers.new { ... }.
137
+ # rotate(serializer: Marshal, url_safe: true).
138
+ # rotate(serializer: Marshal, url_safe: false)
139
+ # end
140
+ #
141
+ # ThirdParty.VERIFIERS.prepend(serializer: JSON, url_safe: true)
142
+ #
143
+ # # Uses `serializer: JSON, url_safe: true`.
144
+ # # Falls back to `serializer: Marshal, url_safe: true` or
145
+ # # `serializer: Marshal, url_safe: false`.
146
+ # ThirdParty.VERIFIERS[:foo]
147
+ #
148
+ #--
149
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#prepend
105
150
 
106
151
  ##
107
152
  # :method: rotate_defaults
108
153
  # :call-seq: rotate_defaults
109
154
  #
110
155
  # Invokes #rotate with the default options.
156
+ #
157
+ #--
158
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#rotate_defaults
111
159
 
112
160
  ##
113
161
  # :method: clear_rotations
114
162
  # :call-seq: clear_rotations
115
163
  #
116
164
  # Clears the list of option sets.
165
+ #
166
+ #--
167
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#clear_rotations
117
168
 
118
169
  ##
119
170
  # :method: on_rotation
@@ -125,6 +176,9 @@ module ActiveSupport
125
176
  # For example, this callback could log each time it is called, and thus
126
177
  # indicate whether old option sets are still in use or can be removed from
127
178
  # rotation.
179
+ #
180
+ #--
181
+ # Implemented by ActiveSupport::Messages::RotationCoordinator#on_rotation
128
182
 
129
183
  ##
130
184
  private
@@ -32,6 +32,15 @@ module ActiveSupport
32
32
  self
33
33
  end
34
34
 
35
+ def prepend(**options, &block)
36
+ raise ArgumentError, "Options cannot be specified when using a block" if block && !options.empty?
37
+ changing_configuration!
38
+
39
+ @rotate_options.unshift(block || options)
40
+
41
+ self
42
+ end
43
+
35
44
  def rotate_defaults
36
45
  rotate()
37
46
  end
@@ -15,6 +15,11 @@ module ActiveSupport
15
15
  fall_back_to build_rotation(*args, **options)
16
16
  end
17
17
 
18
+ def on_rotation(&on_rotation)
19
+ @on_rotation = on_rotation
20
+ self
21
+ end
22
+
18
23
  def fall_back_to(fallback)
19
24
  @rotations << fallback
20
25
  self
@@ -40,6 +45,11 @@ module ActiveSupport
40
45
  end
41
46
  end
42
47
 
48
+ def initialize_dup(*)
49
+ super
50
+ @rotations = @rotations.dup
51
+ end
52
+
43
53
  private
44
54
  def build_rotation(*args, **options)
45
55
  self.class.new(*args, *@args.drop(args.length), **@options, **options)
@@ -53,9 +53,19 @@ module ActiveSupport # :nodoc:
53
53
  delegate :<=>, :=~, :match?, :acts_like_string?, to: :wrapped_string
54
54
 
55
55
  # Creates a new Chars instance by wrapping _string_.
56
- def initialize(string)
56
+ def initialize(string, deprecation: true)
57
+ if deprecation
58
+ ActiveSupport.deprecator.warn(
59
+ "ActiveSupport::Multibyte::Chars is deprecated and will be removed in Rails 8.2. " \
60
+ "Use normal string methods instead."
61
+ )
62
+ end
63
+
57
64
  @wrapped_string = string
58
- @wrapped_string.force_encoding(Encoding::UTF_8) unless @wrapped_string.frozen?
65
+ if string.encoding != Encoding::UTF_8
66
+ @wrapped_string = @wrapped_string.dup
67
+ @wrapped_string.force_encoding(Encoding::UTF_8)
68
+ end
59
69
  end
60
70
 
61
71
  # Forward all undefined methods to the wrapped string.
@@ -12,6 +12,10 @@ module ActiveSupport # :nodoc:
12
12
  #
13
13
  # ActiveSupport::Multibyte.proxy_class = CharsForUTF32
14
14
  def self.proxy_class=(klass)
15
+ ActiveSupport.deprecator.warn(
16
+ "ActiveSupport::Multibyte.proxy_class= is deprecated and will be removed in Rails 8.2. " \
17
+ "Use normal string methods instead."
18
+ )
15
19
  @proxy_class = klass
16
20
  end
17
21