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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +134 -1054
- data/lib/active_support/array_inquirer.rb +1 -1
- data/lib/active_support/backtrace_cleaner.rb +15 -3
- data/lib/active_support/broadcast_logger.rb +19 -18
- data/lib/active_support/cache/file_store.rb +15 -10
- data/lib/active_support/cache/mem_cache_store.rb +16 -74
- data/lib/active_support/cache/memory_store.rb +2 -1
- data/lib/active_support/cache/redis_cache_store.rb +16 -13
- data/lib/active_support/cache/serializer_with_fallback.rb +0 -23
- data/lib/active_support/cache.rb +62 -69
- data/lib/active_support/callbacks.rb +74 -113
- data/lib/active_support/code_generator.rb +15 -10
- data/lib/active_support/core_ext/array/conversions.rb +0 -2
- data/lib/active_support/core_ext/class/subclasses.rb +15 -35
- data/lib/active_support/core_ext/date/blank.rb +4 -0
- data/lib/active_support/core_ext/date/conversions.rb +0 -2
- data/lib/active_support/core_ext/date_and_time/compatibility.rb +28 -1
- data/lib/active_support/core_ext/date_time/blank.rb +4 -0
- data/lib/active_support/core_ext/date_time/conversions.rb +0 -4
- data/lib/active_support/core_ext/digest/uuid.rb +6 -0
- data/lib/active_support/core_ext/erb/util.rb +5 -0
- data/lib/active_support/core_ext/hash/keys.rb +4 -4
- data/lib/active_support/core_ext/module/attr_internal.rb +17 -6
- data/lib/active_support/core_ext/module/delegation.rb +20 -148
- data/lib/active_support/core_ext/module/deprecation.rb +1 -4
- data/lib/active_support/core_ext/numeric/conversions.rb +3 -3
- data/lib/active_support/core_ext/object/blank.rb +45 -1
- data/lib/active_support/core_ext/object/duplicable.rb +24 -15
- data/lib/active_support/core_ext/object/instance_variables.rb +11 -19
- data/lib/active_support/core_ext/object/json.rb +6 -4
- data/lib/active_support/core_ext/object/with.rb +5 -3
- data/lib/active_support/core_ext/pathname/blank.rb +4 -0
- data/lib/active_support/core_ext/range/overlap.rb +1 -1
- data/lib/active_support/core_ext/securerandom.rb +8 -24
- data/lib/active_support/core_ext/string/conversions.rb +1 -1
- data/lib/active_support/core_ext/string/filters.rb +1 -1
- data/lib/active_support/core_ext/string/multibyte.rb +1 -1
- data/lib/active_support/core_ext/string/output_safety.rb +0 -7
- data/lib/active_support/core_ext/time/calculations.rb +18 -28
- data/lib/active_support/core_ext/time/compatibility.rb +16 -0
- data/lib/active_support/core_ext/time/conversions.rb +0 -2
- data/lib/active_support/core_ext/time/zones.rb +1 -1
- data/lib/active_support/core_ext.rb +0 -1
- data/lib/active_support/current_attributes.rb +38 -40
- data/lib/active_support/delegation.rb +202 -0
- data/lib/active_support/dependencies/autoload.rb +0 -12
- data/lib/active_support/deprecation/constant_accessor.rb +47 -26
- data/lib/active_support/deprecation/proxy_wrappers.rb +9 -12
- data/lib/active_support/deprecation/reporting.rb +9 -4
- data/lib/active_support/deprecation.rb +8 -5
- data/lib/active_support/descendants_tracker.rb +9 -87
- data/lib/active_support/duration/iso8601_parser.rb +2 -2
- data/lib/active_support/duration/iso8601_serializer.rb +1 -2
- data/lib/active_support/duration.rb +11 -6
- data/lib/active_support/encrypted_file.rb +1 -1
- data/lib/active_support/error_reporter.rb +41 -3
- data/lib/active_support/evented_file_update_checker.rb +0 -1
- data/lib/active_support/execution_wrapper.rb +0 -1
- data/lib/active_support/file_update_checker.rb +1 -1
- data/lib/active_support/fork_tracker.rb +2 -38
- data/lib/active_support/gem_version.rb +2 -2
- data/lib/active_support/hash_with_indifferent_access.rb +6 -8
- data/lib/active_support/html_safe_translation.rb +7 -4
- data/lib/active_support/json/encoding.rb +1 -1
- data/lib/active_support/log_subscriber.rb +1 -12
- data/lib/active_support/logger.rb +15 -2
- data/lib/active_support/logger_thread_safe_level.rb +0 -8
- data/lib/active_support/message_pack/extensions.rb +15 -2
- data/lib/active_support/message_verifier.rb +12 -0
- data/lib/active_support/messages/codec.rb +1 -1
- data/lib/active_support/multibyte/chars.rb +2 -2
- data/lib/active_support/notifications/fanout.rb +4 -7
- data/lib/active_support/notifications/instrumenter.rb +32 -21
- data/lib/active_support/notifications.rb +28 -27
- data/lib/active_support/number_helper/number_converter.rb +2 -2
- data/lib/active_support/option_merger.rb +2 -2
- data/lib/active_support/ordered_options.rb +53 -15
- data/lib/active_support/proxy_object.rb +8 -5
- data/lib/active_support/railtie.rb +4 -11
- data/lib/active_support/string_inquirer.rb +1 -1
- data/lib/active_support/subscriber.rb +1 -0
- data/lib/active_support/syntax_error_proxy.rb +1 -11
- data/lib/active_support/tagged_logging.rb +4 -1
- data/lib/active_support/test_case.rb +3 -1
- data/lib/active_support/testing/assertions.rb +4 -4
- data/lib/active_support/testing/constant_stubbing.rb +30 -8
- data/lib/active_support/testing/deprecation.rb +5 -12
- data/lib/active_support/testing/isolation.rb +20 -8
- data/lib/active_support/testing/method_call_assertions.rb +2 -16
- data/lib/active_support/testing/parallelization/server.rb +3 -0
- data/lib/active_support/testing/setup_and_teardown.rb +2 -0
- data/lib/active_support/testing/strict_warnings.rb +8 -4
- data/lib/active_support/testing/tests_without_assertions.rb +19 -0
- data/lib/active_support/testing/time_helpers.rb +3 -3
- data/lib/active_support/time_with_zone.rb +8 -4
- data/lib/active_support/values/time_zone.rb +16 -7
- data/lib/active_support/xml_mini.rb +11 -2
- data/lib/active_support.rb +3 -2
- metadata +49 -18
- data/lib/active_support/deprecation/instance_delegator.rb +0 -65
- data/lib/active_support/ruby_features.rb +0 -7
@@ -2,6 +2,7 @@
|
|
2
2
|
|
3
3
|
require "active_support/core_ext/module/attribute_accessors"
|
4
4
|
require "active_support/core_ext/class/attribute"
|
5
|
+
require "active_support/core_ext/enumerable"
|
5
6
|
require "active_support/subscriber"
|
6
7
|
require "active_support/deprecation/proxy_wrappers"
|
7
8
|
|
@@ -61,10 +62,6 @@ module ActiveSupport
|
|
61
62
|
# that all logs are flushed, and it is called in Rails::Rack::Logger after a
|
62
63
|
# request finishes.
|
63
64
|
class LogSubscriber < Subscriber
|
64
|
-
# Embed in a String to clear all previous ANSI sequences.
|
65
|
-
CLEAR = ActiveSupport::Deprecation::DeprecatedObjectProxy.new("\e[0m", "CLEAR is deprecated! Use MODES[:clear] instead.", ActiveSupport.deprecator)
|
66
|
-
BOLD = ActiveSupport::Deprecation::DeprecatedObjectProxy.new("\e[1m", "BOLD is deprecated! Use MODES[:bold] instead.", ActiveSupport.deprecator)
|
67
|
-
|
68
65
|
# ANSI sequence modes
|
69
66
|
MODES = {
|
70
67
|
clear: 0,
|
@@ -181,14 +178,6 @@ module ActiveSupport
|
|
181
178
|
end
|
182
179
|
|
183
180
|
def mode_from(options)
|
184
|
-
if options.is_a?(TrueClass) || options.is_a?(FalseClass)
|
185
|
-
ActiveSupport.deprecator.warn(<<~MSG.squish)
|
186
|
-
Bolding log text with a positional boolean is deprecated and will be removed
|
187
|
-
in Rails 7.2. Use an option hash instead (eg. `color("my text", :red, bold: true)`).
|
188
|
-
MSG
|
189
|
-
options = { bold: options }
|
190
|
-
end
|
191
|
-
|
192
181
|
modes = MODES.values_at(*options.compact_blank.keys)
|
193
182
|
|
194
183
|
"\e[#{modes.join(";")}m" if modes.any?
|
@@ -13,6 +13,10 @@ module ActiveSupport
|
|
13
13
|
# logger = Logger.new(STDOUT)
|
14
14
|
# ActiveSupport::Logger.logger_outputs_to?(logger, STDOUT)
|
15
15
|
# # => true
|
16
|
+
#
|
17
|
+
# logger = Logger.new('/var/log/rails.log')
|
18
|
+
# ActiveSupport::Logger.logger_outputs_to?(logger, '/var/log/rails.log')
|
19
|
+
# # => true
|
16
20
|
def self.logger_outputs_to?(logger, *sources)
|
17
21
|
loggers = if logger.is_a?(BroadcastLogger)
|
18
22
|
logger.broadcasts
|
@@ -21,9 +25,9 @@ module ActiveSupport
|
|
21
25
|
end
|
22
26
|
|
23
27
|
logdevs = loggers.map { |logger| logger.instance_variable_get(:@logdev) }
|
24
|
-
logger_sources = logdevs.filter_map { |logdev| logdev.
|
28
|
+
logger_sources = logdevs.filter_map { |logdev| logdev.try(:filename) || logdev.try(:dev) }
|
25
29
|
|
26
|
-
(sources
|
30
|
+
normalize_sources(sources).intersect?(normalize_sources(logger_sources))
|
27
31
|
end
|
28
32
|
|
29
33
|
def initialize(*args, **kwargs)
|
@@ -38,5 +42,14 @@ module ActiveSupport
|
|
38
42
|
"#{String === msg ? msg : msg.inspect}\n"
|
39
43
|
end
|
40
44
|
end
|
45
|
+
|
46
|
+
private
|
47
|
+
def self.normalize_sources(sources)
|
48
|
+
sources.map do |source|
|
49
|
+
source = source.path if source.respond_to?(:path)
|
50
|
+
source = File.realpath(source) if source.is_a?(String) && File.exist?(source)
|
51
|
+
source
|
52
|
+
end
|
53
|
+
end
|
41
54
|
end
|
42
55
|
end
|
@@ -7,14 +7,6 @@ module ActiveSupport
|
|
7
7
|
module LoggerThreadSafeLevel # :nodoc:
|
8
8
|
extend ActiveSupport::Concern
|
9
9
|
|
10
|
-
Logger::Severity.constants.each do |severity|
|
11
|
-
class_eval(<<-EOT, __FILE__, __LINE__ + 1)
|
12
|
-
def #{severity.downcase}? # def debug?
|
13
|
-
Logger::#{severity} >= level # DEBUG >= level
|
14
|
-
end # end
|
15
|
-
EOT
|
16
|
-
end
|
17
|
-
|
18
10
|
def local_level
|
19
11
|
IsolatedExecutionState[local_level_key]
|
20
12
|
end
|
@@ -86,8 +86,9 @@ module ActiveSupport
|
|
86
86
|
unpacker: URI.method(:parse)
|
87
87
|
|
88
88
|
registry.register_type 14, IPAddr,
|
89
|
-
packer: :
|
90
|
-
unpacker: :
|
89
|
+
packer: method(:write_ipaddr),
|
90
|
+
unpacker: method(:read_ipaddr),
|
91
|
+
recursive: true
|
91
92
|
|
92
93
|
registry.register_type 15, Pathname,
|
93
94
|
packer: :to_s,
|
@@ -221,6 +222,18 @@ module ActiveSupport
|
|
221
222
|
Set.new(unpacker.read)
|
222
223
|
end
|
223
224
|
|
225
|
+
def write_ipaddr(ipaddr, packer)
|
226
|
+
if ipaddr.prefix < 32 || (ipaddr.ipv6? && ipaddr.prefix < 128)
|
227
|
+
packer.write("#{ipaddr}/#{ipaddr.prefix}")
|
228
|
+
else
|
229
|
+
packer.write(ipaddr.to_s)
|
230
|
+
end
|
231
|
+
end
|
232
|
+
|
233
|
+
def read_ipaddr(unpacker)
|
234
|
+
IPAddr.new(unpacker.read)
|
235
|
+
end
|
236
|
+
|
224
237
|
def write_hash_with_indifferent_access(hwia, packer)
|
225
238
|
packer.write(hwia.to_h)
|
226
239
|
end
|
@@ -30,6 +30,18 @@ module ActiveSupport
|
|
30
30
|
# self.current_user = User.find(id)
|
31
31
|
# end
|
32
32
|
#
|
33
|
+
# === Signing is not encryption
|
34
|
+
#
|
35
|
+
# The signed messages are not encrypted. The payload is merely encoded (Base64 by default) and can be decoded by
|
36
|
+
# anyone. The signature is just assuring that the message wasn't tampered with. For example:
|
37
|
+
#
|
38
|
+
# message = Rails.application.message_verifier('my_purpose').generate('never put secrets here')
|
39
|
+
# # => "BAhJIhtuZXZlciBwdXQgc2VjcmV0cyBoZXJlBjoGRVQ=--a0c1c0827919da5e949e989c971249355735e140"
|
40
|
+
# Base64.decode64(message.split("--").first) # no key needed
|
41
|
+
# # => 'never put secrets here'
|
42
|
+
#
|
43
|
+
# If you also need to encrypt the contents, you must use ActiveSupport::MessageEncryptor instead.
|
44
|
+
#
|
33
45
|
# === Confine messages to a specific purpose
|
34
46
|
#
|
35
47
|
# It's not recommended to use the same verifier for different purposes in your application.
|
@@ -28,7 +28,7 @@ module ActiveSupport
|
|
28
28
|
|
29
29
|
def decode(encoded, url_safe: @url_safe)
|
30
30
|
url_safe ? ::Base64.urlsafe_decode64(encoded) : ::Base64.strict_decode64(encoded)
|
31
|
-
rescue
|
31
|
+
rescue StandardError => error
|
32
32
|
throw :invalid_message_format, error
|
33
33
|
end
|
34
34
|
|
@@ -59,8 +59,8 @@ module ActiveSupport # :nodoc:
|
|
59
59
|
end
|
60
60
|
|
61
61
|
# Forward all undefined methods to the wrapped string.
|
62
|
-
def method_missing(method,
|
63
|
-
result = @wrapped_string.__send__(method,
|
62
|
+
def method_missing(method, ...)
|
63
|
+
result = @wrapped_string.__send__(method, ...)
|
64
64
|
if method.end_with?("!")
|
65
65
|
self if result
|
66
66
|
else
|
@@ -1,6 +1,5 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
require "mutex_m"
|
4
3
|
require "concurrent/map"
|
5
4
|
require "set"
|
6
5
|
require "active_support/core_ext/object/try"
|
@@ -49,15 +48,13 @@ module ActiveSupport
|
|
49
48
|
#
|
50
49
|
# This class is thread safe. All methods are reentrant.
|
51
50
|
class Fanout
|
52
|
-
include Mutex_m
|
53
|
-
|
54
51
|
def initialize
|
52
|
+
@mutex = Mutex.new
|
55
53
|
@string_subscribers = Concurrent::Map.new { |h, k| h.compute_if_absent(k) { [] } }
|
56
54
|
@other_subscribers = []
|
57
55
|
@all_listeners_for = Concurrent::Map.new
|
58
56
|
@groups_for = Concurrent::Map.new
|
59
57
|
@silenceable_groups_for = Concurrent::Map.new
|
60
|
-
super
|
61
58
|
end
|
62
59
|
|
63
60
|
def inspect # :nodoc:
|
@@ -67,7 +64,7 @@ module ActiveSupport
|
|
67
64
|
|
68
65
|
def subscribe(pattern = nil, callable = nil, monotonic: false, &block)
|
69
66
|
subscriber = Subscribers.new(pattern, callable || block, monotonic)
|
70
|
-
synchronize do
|
67
|
+
@mutex.synchronize do
|
71
68
|
case pattern
|
72
69
|
when String
|
73
70
|
@string_subscribers[pattern] << subscriber
|
@@ -83,7 +80,7 @@ module ActiveSupport
|
|
83
80
|
end
|
84
81
|
|
85
82
|
def unsubscribe(subscriber_or_name)
|
86
|
-
synchronize do
|
83
|
+
@mutex.synchronize do
|
87
84
|
case subscriber_or_name
|
88
85
|
when String
|
89
86
|
@string_subscribers[subscriber_or_name].clear
|
@@ -300,7 +297,7 @@ module ActiveSupport
|
|
300
297
|
|
301
298
|
def all_listeners_for(name)
|
302
299
|
# this is correctly done double-checked locking (Concurrent::Map's lookups have volatile semantics)
|
303
|
-
@all_listeners_for[name] || synchronize do
|
300
|
+
@all_listeners_for[name] || @mutex.synchronize do
|
304
301
|
# use synchronisation when accessing @subscribers
|
305
302
|
@all_listeners_for[name] ||=
|
306
303
|
@string_subscribers[name] + @other_subscribers.select { |s| s.subscribed_to?(name) }
|
@@ -104,7 +104,7 @@ module ActiveSupport
|
|
104
104
|
end
|
105
105
|
|
106
106
|
class Event
|
107
|
-
attr_reader :name, :
|
107
|
+
attr_reader :name, :transaction_id
|
108
108
|
attr_accessor :payload
|
109
109
|
|
110
110
|
def initialize(name, start, ending, transaction_id, payload)
|
@@ -117,9 +117,19 @@ module ActiveSupport
|
|
117
117
|
@cpu_time_finish = 0.0
|
118
118
|
@allocation_count_start = 0
|
119
119
|
@allocation_count_finish = 0
|
120
|
+
@gc_time_start = 0
|
121
|
+
@gc_time_finish = 0
|
120
122
|
end
|
121
123
|
|
122
|
-
def
|
124
|
+
def time
|
125
|
+
@time / 1000.0 if @time
|
126
|
+
end
|
127
|
+
|
128
|
+
def end
|
129
|
+
@end / 1000.0 if @end
|
130
|
+
end
|
131
|
+
|
132
|
+
def record # :nodoc:
|
123
133
|
start!
|
124
134
|
begin
|
125
135
|
yield payload if block_given?
|
@@ -136,12 +146,14 @@ module ActiveSupport
|
|
136
146
|
def start!
|
137
147
|
@time = now
|
138
148
|
@cpu_time_start = now_cpu
|
149
|
+
@gc_time_start = now_gc
|
139
150
|
@allocation_count_start = now_allocations
|
140
151
|
end
|
141
152
|
|
142
153
|
# Record information at the time this event finishes
|
143
154
|
def finish!
|
144
155
|
@cpu_time_finish = now_cpu
|
156
|
+
@gc_time_finish = now_gc
|
145
157
|
@end = now
|
146
158
|
@allocation_count_finish = now_allocations
|
147
159
|
end
|
@@ -165,28 +177,17 @@ module ActiveSupport
|
|
165
177
|
@allocation_count_finish - @allocation_count_start
|
166
178
|
end
|
167
179
|
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
EOM
|
173
|
-
[]
|
174
|
-
end
|
175
|
-
|
176
|
-
def parent_of?(event) # :nodoc:
|
177
|
-
ActiveSupport.deprecator.warn <<~EOM
|
178
|
-
ActiveSupport::Notifications::Event#parent_of? is deprecated and will
|
179
|
-
be removed in Rails 7.2.
|
180
|
-
EOM
|
181
|
-
start = (time - event.time) * 1000
|
182
|
-
start <= 0 && (start + duration >= event.duration)
|
180
|
+
# Returns the time spent in GC (in milliseconds) between the call to #start!
|
181
|
+
# and the call to #finish!
|
182
|
+
def gc_time
|
183
|
+
(@gc_time_finish - @gc_time_start) / 1_000_000.0
|
183
184
|
end
|
184
185
|
|
185
186
|
# Returns the difference in milliseconds between when the execution of the
|
186
187
|
# event started and when it ended.
|
187
188
|
#
|
188
|
-
# ActiveSupport::Notifications.subscribe('wait') do
|
189
|
-
# @event =
|
189
|
+
# ActiveSupport::Notifications.subscribe('wait') do |event|
|
190
|
+
# @event = event
|
190
191
|
# end
|
191
192
|
#
|
192
193
|
# ActiveSupport::Notifications.instrument('wait') do
|
@@ -195,7 +196,7 @@ module ActiveSupport
|
|
195
196
|
#
|
196
197
|
# @event.duration # => 1000.138
|
197
198
|
def duration
|
198
|
-
|
199
|
+
@end - @time
|
199
200
|
end
|
200
201
|
|
201
202
|
private
|
@@ -210,11 +211,21 @@ module ActiveSupport
|
|
210
211
|
Process.clock_gettime(Process::CLOCK_THREAD_CPUTIME_ID, :float_millisecond)
|
211
212
|
end
|
212
213
|
rescue
|
213
|
-
def now_cpu
|
214
|
+
def now_cpu
|
214
215
|
0.0
|
215
216
|
end
|
216
217
|
end
|
217
218
|
|
219
|
+
if GC.respond_to?(:total_time)
|
220
|
+
def now_gc
|
221
|
+
GC.total_time
|
222
|
+
end
|
223
|
+
else
|
224
|
+
def now_gc
|
225
|
+
0
|
226
|
+
end
|
227
|
+
end
|
228
|
+
|
218
229
|
if GC.stat.key?(:total_allocated_objects)
|
219
230
|
def now_allocations
|
220
231
|
GC.stat(:total_allocated_objects)
|
@@ -29,6 +29,16 @@ module ActiveSupport
|
|
29
29
|
# You can consume those events and the information they provide by registering
|
30
30
|
# a subscriber.
|
31
31
|
#
|
32
|
+
# ActiveSupport::Notifications.subscribe('render') do |event|
|
33
|
+
# event.name # => "render"
|
34
|
+
# event.duration # => 10 (in milliseconds)
|
35
|
+
# event.payload # => { extra: :information }
|
36
|
+
# event.allocations # => 1826 (objects)
|
37
|
+
# end
|
38
|
+
#
|
39
|
+
# +Event+ objects record CPU time and allocations. If you don't need this
|
40
|
+
# it's also possible to pass a block that accepts five arguments:
|
41
|
+
#
|
32
42
|
# ActiveSupport::Notifications.subscribe('render') do |name, start, finish, id, payload|
|
33
43
|
# name # => String, name of the event (such as 'render' from above)
|
34
44
|
# start # => Time, when the instrumented block started execution
|
@@ -42,20 +52,18 @@ module ActiveSupport
|
|
42
52
|
#
|
43
53
|
# ActiveSupport::Notifications.monotonic_subscribe('render') do |name, start, finish, id, payload|
|
44
54
|
# name # => String, name of the event (such as 'render' from above)
|
45
|
-
# start # =>
|
46
|
-
# finish # =>
|
55
|
+
# start # => Float, monotonic time when the instrumented block started execution
|
56
|
+
# finish # => Float, monotonic time when the instrumented block ended execution
|
47
57
|
# id # => String, unique ID for the instrumenter that fired the event
|
48
58
|
# payload # => Hash, the payload
|
49
59
|
# end
|
50
60
|
#
|
51
|
-
# The +start+ and +finish+ values above represent monotonic time.
|
52
|
-
#
|
53
61
|
# For instance, let's store all "render" events in an array:
|
54
62
|
#
|
55
63
|
# events = []
|
56
64
|
#
|
57
|
-
# ActiveSupport::Notifications.subscribe('render') do
|
58
|
-
# events <<
|
65
|
+
# ActiveSupport::Notifications.subscribe('render') do |event|
|
66
|
+
# events << event
|
59
67
|
# end
|
60
68
|
#
|
61
69
|
# That code returns right away, you are just subscribing to "render" events.
|
@@ -66,14 +74,10 @@ module ActiveSupport
|
|
66
74
|
# end
|
67
75
|
#
|
68
76
|
# event = events.first
|
69
|
-
# event.name
|
70
|
-
# event.duration
|
71
|
-
# event.payload
|
72
|
-
#
|
73
|
-
# The block in the <tt>subscribe</tt> call gets the name of the event, start
|
74
|
-
# timestamp, end timestamp, a string with a unique identifier for that event's instrumenter
|
75
|
-
# (something like "535801666f04d0298cd6"), and a hash with the payload, in
|
76
|
-
# that order.
|
77
|
+
# event.name # => "render"
|
78
|
+
# event.duration # => 10 (in milliseconds)
|
79
|
+
# event.payload # => { extra: :information }
|
80
|
+
# event.allocations # => 1826 (objects)
|
77
81
|
#
|
78
82
|
# If an exception happens during that particular instrumentation the payload will
|
79
83
|
# have a key <tt>:exception</tt> with an array of two elements as value: a string with
|
@@ -138,7 +142,7 @@ module ActiveSupport
|
|
138
142
|
# You can subscribe to some event temporarily while some block runs. For
|
139
143
|
# example, in
|
140
144
|
#
|
141
|
-
# callback = lambda {
|
145
|
+
# callback = lambda {|event| ... }
|
142
146
|
# ActiveSupport::Notifications.subscribed(callback, "sql.active_record") do
|
143
147
|
# ...
|
144
148
|
# end
|
@@ -161,7 +165,7 @@ module ActiveSupport
|
|
161
165
|
#
|
162
166
|
# The +subscribe+ method returns a subscriber object:
|
163
167
|
#
|
164
|
-
# subscriber = ActiveSupport::Notifications.subscribe("render") do
|
168
|
+
# subscriber = ActiveSupport::Notifications.subscribe("render") do |event|
|
165
169
|
# ...
|
166
170
|
# end
|
167
171
|
#
|
@@ -214,11 +218,15 @@ module ActiveSupport
|
|
214
218
|
# You can subscribe to events by passing a String to match exact event
|
215
219
|
# names, or by passing a Regexp to match all events that match a pattern.
|
216
220
|
#
|
217
|
-
#
|
218
|
-
#
|
221
|
+
# If the block passed to the method only takes one argument,
|
222
|
+
# it will yield an +Event+ object to the block:
|
223
|
+
#
|
224
|
+
# ActiveSupport::Notifications.subscribe(/render/) do |event|
|
225
|
+
# @event = event
|
219
226
|
# end
|
220
227
|
#
|
221
|
-
#
|
228
|
+
# Otherwise the +block+ will receive five arguments with information
|
229
|
+
# about the event:
|
222
230
|
#
|
223
231
|
# ActiveSupport::Notifications.subscribe('render') do |name, start, finish, id, payload|
|
224
232
|
# name # => String, name of the event (such as 'render' from above)
|
@@ -228,16 +236,9 @@ module ActiveSupport
|
|
228
236
|
# payload # => Hash, the payload
|
229
237
|
# end
|
230
238
|
#
|
231
|
-
# If the block passed to the method only takes one parameter,
|
232
|
-
# it will yield an event object to the block:
|
233
|
-
#
|
234
|
-
# ActiveSupport::Notifications.subscribe(/render/) do |event|
|
235
|
-
# @event = event
|
236
|
-
# end
|
237
|
-
#
|
238
239
|
# Raises an error if invalid event name type is passed:
|
239
240
|
#
|
240
|
-
# ActiveSupport::Notifications.subscribe(:render) {
|
241
|
+
# ActiveSupport::Notifications.subscribe(:render) {|event| ...}
|
241
242
|
# #=> ArgumentError (pattern must be specified as a String, Regexp or empty)
|
242
243
|
#
|
243
244
|
def subscribe(pattern = nil, callback = nil, &block)
|
@@ -164,11 +164,11 @@ module ActiveSupport
|
|
164
164
|
end
|
165
165
|
|
166
166
|
def translate_number_value_with_default(key, **i18n_options)
|
167
|
-
I18n.translate(key,
|
167
|
+
I18n.translate(key, default: default_value(key), scope: :number, **i18n_options)
|
168
168
|
end
|
169
169
|
|
170
170
|
def translate_in_locale(key, **i18n_options)
|
171
|
-
translate_number_value_with_default(key,
|
171
|
+
translate_number_value_with_default(key, locale: options[:locale], **i18n_options)
|
172
172
|
end
|
173
173
|
|
174
174
|
def default_value(key)
|
@@ -46,18 +46,14 @@ module ActiveSupport
|
|
46
46
|
super(key.to_sym, *identifiers)
|
47
47
|
end
|
48
48
|
|
49
|
-
def method_missing(
|
50
|
-
|
51
|
-
|
52
|
-
|
49
|
+
def method_missing(method, *args)
|
50
|
+
if method.end_with?("=")
|
51
|
+
self[method.name.chomp("=")] = args.first
|
52
|
+
elsif method.end_with?("!")
|
53
|
+
name_string = method.name.chomp("!")
|
54
|
+
self[name_string].presence || raise(KeyError.new(":#{name_string} is blank"))
|
53
55
|
else
|
54
|
-
|
55
|
-
|
56
|
-
if bangs
|
57
|
-
self[name_string].presence || raise(KeyError.new(":#{name_string} is blank"))
|
58
|
-
else
|
59
|
-
self[name_string]
|
60
|
-
end
|
56
|
+
self[method.name]
|
61
57
|
end
|
62
58
|
end
|
63
59
|
|
@@ -92,18 +88,60 @@ module ActiveSupport
|
|
92
88
|
# h.boy # => 'John'
|
93
89
|
class InheritableOptions < OrderedOptions
|
94
90
|
def initialize(parent = nil)
|
95
|
-
|
91
|
+
@parent = parent
|
92
|
+
if @parent.kind_of?(OrderedOptions)
|
96
93
|
# use the faster _get when dealing with OrderedOptions
|
97
|
-
super() { |h, k| parent._get(k) }
|
98
|
-
elsif parent
|
99
|
-
super() { |h, k| parent[k] }
|
94
|
+
super() { |h, k| @parent._get(k) }
|
95
|
+
elsif @parent
|
96
|
+
super() { |h, k| @parent[k] }
|
100
97
|
else
|
101
98
|
super()
|
99
|
+
@parent = {}
|
102
100
|
end
|
103
101
|
end
|
104
102
|
|
103
|
+
def to_h
|
104
|
+
@parent.merge(self)
|
105
|
+
end
|
106
|
+
|
107
|
+
def ==(other)
|
108
|
+
to_h == other.to_h
|
109
|
+
end
|
110
|
+
|
111
|
+
def inspect
|
112
|
+
"#<#{self.class.name} #{to_h.inspect}>"
|
113
|
+
end
|
114
|
+
|
115
|
+
def to_s
|
116
|
+
to_h.to_s
|
117
|
+
end
|
118
|
+
|
119
|
+
def pretty_print(pp)
|
120
|
+
pp.pp_hash(to_h)
|
121
|
+
end
|
122
|
+
|
123
|
+
alias_method :own_key?, :key?
|
124
|
+
private :own_key?
|
125
|
+
|
126
|
+
def key?(key)
|
127
|
+
super || @parent.key?(key)
|
128
|
+
end
|
129
|
+
|
130
|
+
def overridden?(key)
|
131
|
+
!!(@parent && @parent.key?(key) && own_key?(key.to_sym))
|
132
|
+
end
|
133
|
+
|
105
134
|
def inheritable_copy
|
106
135
|
self.class.new(self)
|
107
136
|
end
|
137
|
+
|
138
|
+
def to_a
|
139
|
+
entries
|
140
|
+
end
|
141
|
+
|
142
|
+
def each(&block)
|
143
|
+
to_h.each(&block)
|
144
|
+
self
|
145
|
+
end
|
108
146
|
end
|
109
147
|
end
|
@@ -1,11 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module ActiveSupport
|
4
|
-
|
5
|
-
#
|
6
|
-
# A class with no predefined methods that behaves similarly to Builder's
|
7
|
-
# BlankSlate. Used for proxy classes.
|
8
|
-
class ProxyObject < ::BasicObject
|
4
|
+
class ProxyObject < ::BasicObject # :nodoc:
|
9
5
|
undef_method :==
|
10
6
|
undef_method :equal?
|
11
7
|
|
@@ -13,5 +9,12 @@ module ActiveSupport
|
|
13
9
|
def raise(*args)
|
14
10
|
::Object.send(:raise, *args)
|
15
11
|
end
|
12
|
+
|
13
|
+
def self.inherited(_subclass)
|
14
|
+
::ActiveSupport.deprecator.warn(<<~MSG)
|
15
|
+
ActiveSupport::ProxyObject is deprecated and will be removed in Rails 8.0.
|
16
|
+
Use Ruby's built-in BasicObject instead.
|
17
|
+
MSG
|
18
|
+
end
|
16
19
|
end
|
17
20
|
end
|
@@ -89,10 +89,11 @@ module ActiveSupport
|
|
89
89
|
begin
|
90
90
|
TZInfo::DataSource.get
|
91
91
|
rescue TZInfo::DataSourceNotFound => e
|
92
|
-
raise e.exception
|
92
|
+
raise e.exception('tzinfo-data is not present. Please add gem "tzinfo-data" to your Gemfile and run bundle install')
|
93
93
|
end
|
94
94
|
require "active_support/core_ext/time/zones"
|
95
95
|
Time.zone_default = Time.find_zone!(app.config.time_zone)
|
96
|
+
config.eager_load_namespaces << TZInfo
|
96
97
|
end
|
97
98
|
|
98
99
|
# Sets the default week start
|
@@ -117,16 +118,8 @@ module ActiveSupport
|
|
117
118
|
|
118
119
|
initializer "active_support.set_configs" do |app|
|
119
120
|
app.config.active_support.each do |k, v|
|
120
|
-
|
121
|
-
|
122
|
-
elsif k == "remove_deprecated_time_with_zone_name"
|
123
|
-
ActiveSupport.deprecator.warn("config.active_support.remove_deprecated_time_with_zone_name is deprecated and will be removed in Rails 7.2.")
|
124
|
-
elsif k == "use_rfc4122_namespaced_uuids"
|
125
|
-
ActiveSupport.deprecator.warn("config.active_support.use_rfc4122_namespaced_uuids is deprecated and will be removed in Rails 7.2.")
|
126
|
-
else
|
127
|
-
k = "#{k}="
|
128
|
-
ActiveSupport.public_send(k, v) if ActiveSupport.respond_to? k
|
129
|
-
end
|
121
|
+
k = "#{k}="
|
122
|
+
ActiveSupport.public_send(k, v) if ActiveSupport.respond_to? k
|
130
123
|
end
|
131
124
|
end
|
132
125
|
|
@@ -67,6 +67,7 @@ module ActiveSupport
|
|
67
67
|
|
68
68
|
# Adds event subscribers for all new methods added to the class.
|
69
69
|
def method_added(event)
|
70
|
+
super
|
70
71
|
# Only public methods are added as subscribers, and only if a notifier
|
71
72
|
# has been set up. This means that subscribers will only be set up for
|
72
73
|
# classes that call #attach_to.
|
@@ -45,7 +45,7 @@ module ActiveSupport
|
|
45
45
|
|
46
46
|
private
|
47
47
|
def parse_message_for_trace
|
48
|
-
if
|
48
|
+
if __getobj__.to_s.start_with?("(eval")
|
49
49
|
# If the exception is coming from a call to eval, we need to keep
|
50
50
|
# the path of the file in which eval was called to ensure we can
|
51
51
|
# return the right source fragment to show the location of the
|
@@ -56,15 +56,5 @@ module ActiveSupport
|
|
56
56
|
__getobj__.to_s.split("\n")
|
57
57
|
end
|
58
58
|
end
|
59
|
-
|
60
|
-
if SyntaxError.method_defined?(:path) # Ruby 3.3+
|
61
|
-
def source_location_eval?
|
62
|
-
__getobj__.path.start_with?("(eval")
|
63
|
-
end
|
64
|
-
else # 3.2 and older versions of Ruby
|
65
|
-
def source_location_eval?
|
66
|
-
__getobj__.to_s.start_with?("(eval")
|
67
|
-
end
|
68
|
-
end
|
69
59
|
end
|
70
60
|
end
|
@@ -2,7 +2,6 @@
|
|
2
2
|
|
3
3
|
require "active_support/core_ext/module/delegation"
|
4
4
|
require "active_support/core_ext/object/blank"
|
5
|
-
require "logger"
|
6
5
|
require "active_support/logger"
|
7
6
|
|
8
7
|
module ActiveSupport
|
@@ -119,6 +118,10 @@ module ActiveSupport
|
|
119
118
|
|
120
119
|
if logger.formatter
|
121
120
|
logger.formatter = logger.formatter.clone
|
121
|
+
|
122
|
+
# Workaround for https://bugs.ruby-lang.org/issues/20250
|
123
|
+
# Can be removed when Ruby 3.4 is the least supported version.
|
124
|
+
logger.formatter.object_id if logger.formatter.is_a?(Proc)
|
122
125
|
else
|
123
126
|
# Ensure we set a default formatter so we aren't extending nil!
|
124
127
|
logger.formatter = ActiveSupport::Logger::SimpleFormatter.new
|