activesupport 7.0.8.7 → 7.1.0.beta1
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 +722 -314
- data/MIT-LICENSE +1 -1
- data/README.rdoc +4 -4
- data/lib/active_support/actionable_error.rb +3 -1
- data/lib/active_support/array_inquirer.rb +2 -0
- data/lib/active_support/backtrace_cleaner.rb +25 -5
- data/lib/active_support/benchmarkable.rb +1 -0
- data/lib/active_support/builder.rb +1 -1
- data/lib/active_support/cache/coder.rb +153 -0
- data/lib/active_support/cache/entry.rb +128 -0
- data/lib/active_support/cache/file_store.rb +36 -9
- data/lib/active_support/cache/mem_cache_store.rb +84 -68
- data/lib/active_support/cache/memory_store.rb +76 -24
- data/lib/active_support/cache/null_store.rb +6 -0
- data/lib/active_support/cache/redis_cache_store.rb +126 -131
- data/lib/active_support/cache/serializer_with_fallback.rb +175 -0
- data/lib/active_support/cache/strategy/local_cache.rb +20 -8
- data/lib/active_support/cache.rb +304 -246
- data/lib/active_support/callbacks.rb +38 -18
- data/lib/active_support/concern.rb +4 -2
- data/lib/active_support/concurrency/load_interlock_aware_monitor.rb +42 -3
- data/lib/active_support/concurrency/null_lock.rb +13 -0
- data/lib/active_support/configurable.rb +10 -0
- data/lib/active_support/core_ext/array/conversions.rb +2 -1
- data/lib/active_support/core_ext/array.rb +0 -1
- data/lib/active_support/core_ext/class/subclasses.rb +13 -10
- data/lib/active_support/core_ext/date/conversions.rb +1 -0
- data/lib/active_support/core_ext/date.rb +0 -1
- data/lib/active_support/core_ext/date_and_time/calculations.rb +10 -0
- data/lib/active_support/core_ext/date_time/conversions.rb +6 -2
- data/lib/active_support/core_ext/date_time.rb +0 -1
- data/lib/active_support/core_ext/digest/uuid.rb +1 -10
- data/lib/active_support/core_ext/enumerable.rb +3 -75
- data/lib/active_support/core_ext/erb/util.rb +196 -0
- data/lib/active_support/core_ext/hash/conversions.rb +1 -1
- data/lib/active_support/core_ext/module/attribute_accessors.rb +6 -0
- data/lib/active_support/core_ext/module/attribute_accessors_per_thread.rb +34 -16
- data/lib/active_support/core_ext/module/delegation.rb +40 -11
- data/lib/active_support/core_ext/module/deprecation.rb +15 -12
- data/lib/active_support/core_ext/module/introspection.rb +0 -1
- data/lib/active_support/core_ext/numeric/bytes.rb +9 -0
- data/lib/active_support/core_ext/numeric/conversions.rb +2 -0
- data/lib/active_support/core_ext/numeric.rb +0 -1
- data/lib/active_support/core_ext/object/deep_dup.rb +16 -0
- data/lib/active_support/core_ext/object/duplicable.rb +15 -24
- data/lib/active_support/core_ext/object/inclusion.rb +13 -5
- data/lib/active_support/core_ext/object/instance_variables.rb +22 -12
- data/lib/active_support/core_ext/object/json.rb +10 -2
- data/lib/active_support/core_ext/object/with.rb +44 -0
- data/lib/active_support/core_ext/object/with_options.rb +3 -3
- data/lib/active_support/core_ext/object.rb +1 -0
- data/lib/active_support/core_ext/pathname/blank.rb +16 -0
- data/lib/active_support/core_ext/pathname/existence.rb +2 -0
- data/lib/active_support/core_ext/pathname.rb +1 -0
- data/lib/active_support/core_ext/range/conversions.rb +28 -7
- data/lib/active_support/core_ext/range/{overlaps.rb → overlap.rb} +5 -3
- data/lib/active_support/core_ext/range.rb +1 -2
- data/lib/active_support/core_ext/securerandom.rb +24 -12
- data/lib/active_support/core_ext/string/filters.rb +20 -14
- data/lib/active_support/core_ext/string/inflections.rb +16 -5
- data/lib/active_support/core_ext/string/output_safety.rb +38 -174
- data/lib/active_support/core_ext/thread/backtrace/location.rb +12 -0
- data/lib/active_support/core_ext/time/calculations.rb +18 -2
- data/lib/active_support/core_ext/time/conversions.rb +2 -2
- data/lib/active_support/core_ext/time/zones.rb +4 -4
- data/lib/active_support/core_ext/time.rb +0 -1
- data/lib/active_support/current_attributes.rb +15 -6
- data/lib/active_support/dependencies/autoload.rb +17 -12
- data/lib/active_support/deprecation/behaviors.rb +53 -32
- data/lib/active_support/deprecation/constant_accessor.rb +5 -4
- data/lib/active_support/deprecation/deprecators.rb +104 -0
- data/lib/active_support/deprecation/disallowed.rb +3 -5
- data/lib/active_support/deprecation/instance_delegator.rb +31 -4
- data/lib/active_support/deprecation/method_wrappers.rb +6 -23
- data/lib/active_support/deprecation/proxy_wrappers.rb +37 -22
- data/lib/active_support/deprecation/reporting.rb +35 -21
- data/lib/active_support/deprecation.rb +32 -5
- data/lib/active_support/deprecator.rb +7 -0
- data/lib/active_support/descendants_tracker.rb +104 -132
- data/lib/active_support/duration/iso8601_serializer.rb +0 -2
- data/lib/active_support/duration.rb +2 -1
- data/lib/active_support/encrypted_configuration.rb +30 -9
- data/lib/active_support/encrypted_file.rb +8 -3
- data/lib/active_support/environment_inquirer.rb +22 -2
- data/lib/active_support/error_reporter/test_helper.rb +15 -0
- data/lib/active_support/error_reporter.rb +121 -35
- data/lib/active_support/execution_wrapper.rb +4 -4
- data/lib/active_support/file_update_checker.rb +4 -2
- data/lib/active_support/fork_tracker.rb +10 -2
- data/lib/active_support/gem_version.rb +4 -4
- data/lib/active_support/gzip.rb +2 -0
- data/lib/active_support/hash_with_indifferent_access.rb +35 -17
- data/lib/active_support/i18n.rb +1 -1
- data/lib/active_support/i18n_railtie.rb +20 -13
- data/lib/active_support/inflector/inflections.rb +2 -0
- data/lib/active_support/inflector/methods.rb +22 -10
- data/lib/active_support/inflector/transliterate.rb +3 -1
- data/lib/active_support/isolated_execution_state.rb +26 -22
- data/lib/active_support/json/decoding.rb +2 -1
- data/lib/active_support/json/encoding.rb +25 -43
- data/lib/active_support/key_generator.rb +9 -1
- data/lib/active_support/lazy_load_hooks.rb +6 -4
- data/lib/active_support/locale/en.yml +2 -0
- data/lib/active_support/log_subscriber.rb +78 -33
- data/lib/active_support/logger.rb +1 -1
- data/lib/active_support/logger_thread_safe_level.rb +9 -21
- data/lib/active_support/message_encryptor.rb +197 -53
- data/lib/active_support/message_encryptors.rb +140 -0
- data/lib/active_support/message_pack/cache_serializer.rb +23 -0
- data/lib/active_support/message_pack/extensions.rb +292 -0
- data/lib/active_support/message_pack/serializer.rb +63 -0
- data/lib/active_support/message_pack.rb +50 -0
- data/lib/active_support/message_verifier.rb +212 -93
- data/lib/active_support/message_verifiers.rb +134 -0
- data/lib/active_support/messages/codec.rb +65 -0
- data/lib/active_support/messages/metadata.rb +111 -45
- data/lib/active_support/messages/rotation_coordinator.rb +93 -0
- data/lib/active_support/messages/rotator.rb +34 -32
- data/lib/active_support/messages/serializer_with_fallback.rb +158 -0
- data/lib/active_support/multibyte/chars.rb +2 -0
- data/lib/active_support/multibyte/unicode.rb +9 -37
- data/lib/active_support/notifications/fanout.rb +239 -81
- data/lib/active_support/notifications/instrumenter.rb +71 -14
- data/lib/active_support/notifications.rb +1 -1
- data/lib/active_support/number_helper/number_converter.rb +2 -2
- data/lib/active_support/number_helper/number_to_human_size_converter.rb +1 -1
- data/lib/active_support/number_helper/number_to_phone_converter.rb +1 -0
- data/lib/active_support/ordered_hash.rb +3 -3
- data/lib/active_support/ordered_options.rb +14 -0
- data/lib/active_support/parameter_filter.rb +84 -69
- data/lib/active_support/proxy_object.rb +2 -0
- data/lib/active_support/railtie.rb +33 -21
- data/lib/active_support/reloader.rb +12 -4
- data/lib/active_support/rescuable.rb +2 -0
- data/lib/active_support/secure_compare_rotator.rb +16 -9
- data/lib/active_support/string_inquirer.rb +3 -1
- data/lib/active_support/subscriber.rb +9 -27
- data/lib/active_support/syntax_error_proxy.rb +49 -0
- data/lib/active_support/tagged_logging.rb +60 -24
- data/lib/active_support/test_case.rb +153 -6
- data/lib/active_support/testing/assertions.rb +25 -9
- data/lib/active_support/testing/autorun.rb +0 -2
- data/lib/active_support/testing/constant_stubbing.rb +32 -0
- data/lib/active_support/testing/deprecation.rb +25 -25
- data/lib/active_support/testing/error_reporter_assertions.rb +108 -0
- data/lib/active_support/testing/isolation.rb +1 -1
- data/lib/active_support/testing/method_call_assertions.rb +21 -8
- data/lib/active_support/testing/parallelize_executor.rb +8 -3
- data/lib/active_support/testing/stream.rb +1 -1
- data/lib/active_support/testing/strict_warnings.rb +38 -0
- data/lib/active_support/testing/time_helpers.rb +32 -14
- data/lib/active_support/time_with_zone.rb +4 -14
- data/lib/active_support/values/time_zone.rb +9 -7
- data/lib/active_support/version.rb +1 -1
- data/lib/active_support/xml_mini/jdom.rb +3 -10
- data/lib/active_support/xml_mini/nokogiri.rb +1 -1
- data/lib/active_support/xml_mini/nokogirisax.rb +1 -1
- data/lib/active_support/xml_mini/rexml.rb +1 -1
- data/lib/active_support/xml_mini.rb +2 -2
- data/lib/active_support.rb +13 -3
- metadata +106 -21
- data/lib/active_support/core_ext/array/deprecated_conversions.rb +0 -25
- data/lib/active_support/core_ext/date/deprecated_conversions.rb +0 -40
- data/lib/active_support/core_ext/date_time/deprecated_conversions.rb +0 -36
- data/lib/active_support/core_ext/numeric/deprecated_conversions.rb +0 -60
- data/lib/active_support/core_ext/range/deprecated_conversions.rb +0 -36
- data/lib/active_support/core_ext/range/include_time_with_zone.rb +0 -5
- data/lib/active_support/core_ext/time/deprecated_conversions.rb +0 -73
- data/lib/active_support/core_ext/uri.rb +0 -5
- data/lib/active_support/per_thread_registry.rb +0 -65
@@ -1,80 +1,146 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require "time"
|
4
|
+
require "active_support/json"
|
5
|
+
require_relative "serializer_with_fallback"
|
4
6
|
|
5
7
|
module ActiveSupport
|
6
8
|
module Messages # :nodoc:
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
9
|
+
module Metadata # :nodoc:
|
10
|
+
singleton_class.attr_accessor :use_message_serializer_for_metadata
|
11
|
+
|
12
|
+
ENVELOPE_SERIALIZERS = [
|
13
|
+
*SerializerWithFallback::SERIALIZERS.values,
|
14
|
+
ActiveSupport::JSON,
|
15
|
+
::JSON,
|
16
|
+
Marshal,
|
17
|
+
]
|
18
|
+
|
19
|
+
TIMESTAMP_SERIALIZERS = [
|
20
|
+
SerializerWithFallback::SERIALIZERS.fetch(:message_pack),
|
21
|
+
SerializerWithFallback::SERIALIZERS.fetch(:message_pack_allow_marshal),
|
22
|
+
]
|
12
23
|
|
13
|
-
|
14
|
-
|
24
|
+
ActiveSupport.on_load(:message_pack) do
|
25
|
+
ENVELOPE_SERIALIZERS << ActiveSupport::MessagePack
|
26
|
+
TIMESTAMP_SERIALIZERS << ActiveSupport::MessagePack
|
15
27
|
end
|
16
28
|
|
17
|
-
|
18
|
-
def
|
19
|
-
|
20
|
-
|
29
|
+
private
|
30
|
+
def serialize_with_metadata(data, **metadata)
|
31
|
+
has_metadata = metadata.any? { |k, v| v }
|
32
|
+
|
33
|
+
if has_metadata && !use_message_serializer_for_metadata?
|
34
|
+
data_string = serialize_to_json_safe_string(data)
|
35
|
+
envelope = wrap_in_metadata_legacy_envelope({ "message" => data_string }, **metadata)
|
36
|
+
serialize_to_json(envelope)
|
21
37
|
else
|
22
|
-
|
38
|
+
data = wrap_in_metadata_envelope({ "data" => data }, **metadata) if has_metadata
|
39
|
+
serialize(data)
|
23
40
|
end
|
24
41
|
end
|
25
42
|
|
26
|
-
def
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
43
|
+
def deserialize_with_metadata(message, **expected_metadata)
|
44
|
+
if dual_serialized_metadata_envelope_json?(message)
|
45
|
+
envelope = deserialize_from_json(message)
|
46
|
+
extracted = extract_from_metadata_envelope(envelope, **expected_metadata)
|
47
|
+
deserialize_from_json_safe_string(extracted["message"])
|
48
|
+
else
|
49
|
+
deserialized = deserialize(message)
|
50
|
+
if metadata_envelope?(deserialized)
|
51
|
+
extract_from_metadata_envelope(deserialized, **expected_metadata)["data"]
|
52
|
+
elsif expected_metadata.none? { |k, v| v }
|
53
|
+
deserialized
|
54
|
+
else
|
55
|
+
throw :invalid_message_content, "missing metadata"
|
36
56
|
end
|
37
57
|
end
|
58
|
+
end
|
38
59
|
|
39
|
-
|
40
|
-
|
60
|
+
def use_message_serializer_for_metadata?
|
61
|
+
Metadata.use_message_serializer_for_metadata && Metadata::ENVELOPE_SERIALIZERS.include?(serializer)
|
62
|
+
end
|
41
63
|
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
64
|
+
def wrap_in_metadata_envelope(hash, expires_at: nil, expires_in: nil, purpose: nil)
|
65
|
+
expiry = pick_expiry(expires_at, expires_in)
|
66
|
+
hash["exp"] = expiry if expiry
|
67
|
+
hash["pur"] = purpose.to_s if purpose
|
68
|
+
{ "_rails" => hash }
|
69
|
+
end
|
70
|
+
|
71
|
+
def wrap_in_metadata_legacy_envelope(hash, expires_at: nil, expires_in: nil, purpose: nil)
|
72
|
+
expiry = pick_expiry(expires_at, expires_in)
|
73
|
+
hash["exp"] = expiry
|
74
|
+
hash["pur"] = purpose
|
75
|
+
{ "_rails" => hash }
|
76
|
+
end
|
77
|
+
|
78
|
+
def extract_from_metadata_envelope(envelope, purpose: nil)
|
79
|
+
hash = envelope["_rails"]
|
48
80
|
|
49
|
-
|
50
|
-
|
81
|
+
if hash["exp"] && Time.now.utc >= parse_expiry(hash["exp"])
|
82
|
+
throw :invalid_message_content, "expired"
|
51
83
|
end
|
52
84
|
|
53
|
-
|
54
|
-
|
85
|
+
if hash["pur"] != purpose&.to_s
|
86
|
+
throw :invalid_message_content, "mismatched purpose"
|
55
87
|
end
|
56
|
-
end
|
57
88
|
|
58
|
-
|
59
|
-
|
60
|
-
end
|
89
|
+
hash
|
90
|
+
end
|
61
91
|
|
62
|
-
|
63
|
-
|
64
|
-
@purpose.to_s == purpose.to_s
|
92
|
+
def metadata_envelope?(object)
|
93
|
+
object.is_a?(Hash) && object.key?("_rails")
|
65
94
|
end
|
66
95
|
|
67
|
-
def
|
68
|
-
|
96
|
+
def dual_serialized_metadata_envelope_json?(string)
|
97
|
+
string.start_with?('{"_rails":{"message":')
|
98
|
+
end
|
99
|
+
|
100
|
+
def pick_expiry(expires_at, expires_in)
|
101
|
+
expiry = if expires_at
|
102
|
+
expires_at.utc
|
103
|
+
elsif expires_in
|
104
|
+
Time.now.utc.advance(seconds: expires_in)
|
105
|
+
end
|
106
|
+
|
107
|
+
unless Metadata::TIMESTAMP_SERIALIZERS.include?(serializer)
|
108
|
+
expiry = expiry&.iso8601(3)
|
109
|
+
end
|
110
|
+
|
111
|
+
expiry
|
69
112
|
end
|
70
113
|
|
71
|
-
def
|
72
|
-
if
|
114
|
+
def parse_expiry(expires_at)
|
115
|
+
if !expires_at.is_a?(String)
|
116
|
+
expires_at
|
117
|
+
elsif ActiveSupport.use_standard_json_time_format
|
73
118
|
Time.iso8601(expires_at)
|
74
119
|
else
|
75
120
|
Time.parse(expires_at)
|
76
121
|
end
|
77
122
|
end
|
123
|
+
|
124
|
+
def serialize_to_json(data)
|
125
|
+
ActiveSupport::JSON.encode(data)
|
126
|
+
end
|
127
|
+
|
128
|
+
def deserialize_from_json(serialized)
|
129
|
+
ActiveSupport::JSON.decode(serialized)
|
130
|
+
rescue ::JSON::ParserError => error
|
131
|
+
# Throw :invalid_message_format instead of :invalid_message_serialization
|
132
|
+
# because here a parse error is due to a bad message rather than an
|
133
|
+
# incompatible `self.serializer`.
|
134
|
+
throw :invalid_message_format, error
|
135
|
+
end
|
136
|
+
|
137
|
+
def serialize_to_json_safe_string(data)
|
138
|
+
encode(serialize(data), url_safe: false)
|
139
|
+
end
|
140
|
+
|
141
|
+
def deserialize_from_json_safe_string(string)
|
142
|
+
deserialize(decode(string, url_safe: false))
|
143
|
+
end
|
78
144
|
end
|
79
145
|
end
|
80
146
|
end
|
@@ -0,0 +1,93 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "active_support/core_ext/hash/slice"
|
4
|
+
|
5
|
+
module ActiveSupport
|
6
|
+
module Messages
|
7
|
+
class RotationCoordinator # :nodoc:
|
8
|
+
attr_accessor :transitional
|
9
|
+
|
10
|
+
def initialize(&secret_generator)
|
11
|
+
raise ArgumentError, "A secret generator block is required" unless secret_generator
|
12
|
+
@secret_generator = secret_generator
|
13
|
+
@rotate_options = []
|
14
|
+
@on_rotation = nil
|
15
|
+
@codecs = {}
|
16
|
+
end
|
17
|
+
|
18
|
+
def [](salt)
|
19
|
+
@codecs[salt] ||= build_with_rotations(salt)
|
20
|
+
end
|
21
|
+
|
22
|
+
def []=(salt, codec)
|
23
|
+
@codecs[salt] = codec
|
24
|
+
end
|
25
|
+
|
26
|
+
def rotate(**options, &block)
|
27
|
+
raise ArgumentError, "Options cannot be specified when using a block" if block && !options.empty?
|
28
|
+
changing_configuration!
|
29
|
+
|
30
|
+
@rotate_options << (block || options)
|
31
|
+
|
32
|
+
self
|
33
|
+
end
|
34
|
+
|
35
|
+
def rotate_defaults
|
36
|
+
rotate()
|
37
|
+
end
|
38
|
+
|
39
|
+
def clear_rotations
|
40
|
+
changing_configuration!
|
41
|
+
@rotate_options.clear
|
42
|
+
self
|
43
|
+
end
|
44
|
+
|
45
|
+
def on_rotation(&callback)
|
46
|
+
changing_configuration!
|
47
|
+
@on_rotation = callback
|
48
|
+
end
|
49
|
+
|
50
|
+
private
|
51
|
+
def changing_configuration!
|
52
|
+
if @codecs.any?
|
53
|
+
raise <<~MESSAGE
|
54
|
+
Cannot change #{self.class} configuration after it has already been applied.
|
55
|
+
|
56
|
+
The configuration has been applied with the following salts:
|
57
|
+
#{@codecs.keys.map { |salt| "- #{salt.inspect}" }.join("\n")}
|
58
|
+
MESSAGE
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
def normalize_options(options)
|
63
|
+
options = options.dup
|
64
|
+
|
65
|
+
options[:secret_generator] ||= @secret_generator
|
66
|
+
|
67
|
+
secret_generator_kwargs = options[:secret_generator].parameters.
|
68
|
+
filter_map { |type, name| name if type == :key || type == :keyreq }
|
69
|
+
options[:secret_generator_options] = options.extract!(*secret_generator_kwargs)
|
70
|
+
|
71
|
+
options[:on_rotation] = @on_rotation
|
72
|
+
|
73
|
+
options
|
74
|
+
end
|
75
|
+
|
76
|
+
def build_with_rotations(salt)
|
77
|
+
rotate_options = @rotate_options.map { |options| options.is_a?(Proc) ? options.(salt) : options }
|
78
|
+
transitional = self.transitional && rotate_options.first
|
79
|
+
rotate_options.compact!
|
80
|
+
rotate_options[0..1] = rotate_options[0..1].reverse if transitional
|
81
|
+
rotate_options = rotate_options.map { |options| normalize_options(options) }.uniq
|
82
|
+
|
83
|
+
raise "No options have been configured for #{salt}" if rotate_options.empty?
|
84
|
+
|
85
|
+
rotate_options.map { |options| build(salt.to_s, **options) }.reduce(&:fall_back_to)
|
86
|
+
end
|
87
|
+
|
88
|
+
def build(salt, secret_generator:, secret_generator_options:, **options)
|
89
|
+
raise NotImplementedError
|
90
|
+
end
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|
@@ -3,54 +3,56 @@
|
|
3
3
|
module ActiveSupport
|
4
4
|
module Messages
|
5
5
|
module Rotator # :nodoc:
|
6
|
-
def initialize(*
|
7
|
-
super(*
|
8
|
-
|
9
|
-
@options
|
6
|
+
def initialize(*args, on_rotation: nil, **options)
|
7
|
+
super(*args, **options)
|
8
|
+
@args = args
|
9
|
+
@options = options
|
10
10
|
@rotations = []
|
11
11
|
@on_rotation = on_rotation
|
12
12
|
end
|
13
13
|
|
14
|
-
def rotate(*
|
15
|
-
|
14
|
+
def rotate(*args, **options)
|
15
|
+
fall_back_to build_rotation(*args, **options)
|
16
16
|
end
|
17
17
|
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
super
|
23
|
-
rescue MessageEncryptor::InvalidMessage, MessageVerifier::InvalidSignature
|
24
|
-
run_rotations(on_rotation) { |encryptor| encryptor.decrypt_and_verify(*args, **options) } || raise
|
25
|
-
end
|
18
|
+
def fall_back_to(fallback)
|
19
|
+
@rotations << fallback
|
20
|
+
self
|
21
|
+
end
|
26
22
|
|
27
|
-
|
28
|
-
|
29
|
-
|
23
|
+
def read_message(message, on_rotation: @on_rotation, **options)
|
24
|
+
if @rotations.empty?
|
25
|
+
super(message, **options)
|
26
|
+
else
|
27
|
+
thrown, error = catch_rotation_error do
|
28
|
+
return super(message, **options)
|
30
29
|
end
|
31
|
-
end
|
32
30
|
|
33
|
-
|
34
|
-
|
31
|
+
@rotations.each do |rotation|
|
32
|
+
catch_rotation_error do
|
33
|
+
value = rotation.read_message(message, **options)
|
34
|
+
on_rotation&.call
|
35
|
+
return value
|
36
|
+
end
|
37
|
+
end
|
35
38
|
|
36
|
-
|
37
|
-
super || run_rotations(on_rotation) { |verifier| verifier.verified(*args, **options) }
|
39
|
+
throw thrown, error
|
38
40
|
end
|
39
|
-
|
40
|
-
private
|
41
|
-
def build_rotation(secret = @secret, options)
|
42
|
-
self.class.new(secret, **options)
|
43
|
-
end
|
44
41
|
end
|
45
42
|
|
46
43
|
private
|
47
|
-
def
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
44
|
+
def build_rotation(*args, **options)
|
45
|
+
self.class.new(*args, *@args.drop(args.length), **@options, **options)
|
46
|
+
end
|
47
|
+
|
48
|
+
def catch_rotation_error(&block)
|
49
|
+
error = catch :invalid_message_format do
|
50
|
+
error = catch :invalid_message_serialization do
|
51
|
+
return [nil, block.call]
|
52
52
|
end
|
53
|
+
return [:invalid_message_serialization, error]
|
53
54
|
end
|
55
|
+
[:invalid_message_format, error]
|
54
56
|
end
|
55
57
|
end
|
56
58
|
end
|
@@ -0,0 +1,158 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "active_support/core_ext/kernel/reporting"
|
4
|
+
require "active_support/notifications"
|
5
|
+
|
6
|
+
module ActiveSupport
|
7
|
+
module Messages # :nodoc:
|
8
|
+
module SerializerWithFallback # :nodoc:
|
9
|
+
def self.[](format)
|
10
|
+
if format.to_s.include?("message_pack") && !defined?(ActiveSupport::MessagePack)
|
11
|
+
require "active_support/message_pack"
|
12
|
+
end
|
13
|
+
|
14
|
+
SERIALIZERS.fetch(format)
|
15
|
+
end
|
16
|
+
|
17
|
+
def load(dumped)
|
18
|
+
format = detect_format(dumped)
|
19
|
+
|
20
|
+
if format == self.format
|
21
|
+
_load(dumped)
|
22
|
+
elsif format && fallback?(format)
|
23
|
+
payload = { serializer: SERIALIZERS.key(self), fallback: format, serialized: dumped }
|
24
|
+
ActiveSupport::Notifications.instrument("message_serializer_fallback.active_support", payload) do
|
25
|
+
payload[:deserialized] = SERIALIZERS[format]._load(dumped)
|
26
|
+
end
|
27
|
+
else
|
28
|
+
raise "Unsupported serialization format"
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
private
|
33
|
+
def detect_format(dumped)
|
34
|
+
case
|
35
|
+
when MessagePackWithFallback.dumped?(dumped)
|
36
|
+
:message_pack
|
37
|
+
when MarshalWithFallback.dumped?(dumped)
|
38
|
+
:marshal
|
39
|
+
when JsonWithFallback.dumped?(dumped)
|
40
|
+
:json
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
def fallback?(format)
|
45
|
+
format != :marshal
|
46
|
+
end
|
47
|
+
|
48
|
+
module AllowMarshal
|
49
|
+
private
|
50
|
+
def fallback?(format)
|
51
|
+
super || format == :marshal
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
55
|
+
module MarshalWithFallback
|
56
|
+
include SerializerWithFallback
|
57
|
+
extend self
|
58
|
+
|
59
|
+
def format
|
60
|
+
:marshal
|
61
|
+
end
|
62
|
+
|
63
|
+
def dump(object)
|
64
|
+
Marshal.dump(object)
|
65
|
+
end
|
66
|
+
|
67
|
+
def _load(dumped)
|
68
|
+
Marshal.load(dumped)
|
69
|
+
end
|
70
|
+
|
71
|
+
MARSHAL_SIGNATURE = "\x04\x08"
|
72
|
+
|
73
|
+
def dumped?(dumped)
|
74
|
+
dumped.start_with?(MARSHAL_SIGNATURE)
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
module JsonWithFallback
|
79
|
+
include SerializerWithFallback
|
80
|
+
extend self
|
81
|
+
|
82
|
+
def format
|
83
|
+
:json
|
84
|
+
end
|
85
|
+
|
86
|
+
def dump(object)
|
87
|
+
ActiveSupport::JSON.encode(object)
|
88
|
+
end
|
89
|
+
|
90
|
+
def _load(dumped)
|
91
|
+
ActiveSupport::JSON.decode(dumped)
|
92
|
+
end
|
93
|
+
|
94
|
+
JSON_START_WITH = /\A(?:[{\["]|-?\d|true|false|null)/
|
95
|
+
|
96
|
+
def dumped?(dumped)
|
97
|
+
JSON_START_WITH.match?(dumped)
|
98
|
+
end
|
99
|
+
|
100
|
+
private
|
101
|
+
def detect_format(dumped)
|
102
|
+
# Assume JSON format if format could not be determined.
|
103
|
+
super || :json
|
104
|
+
end
|
105
|
+
end
|
106
|
+
|
107
|
+
module JsonWithFallbackAllowMarshal
|
108
|
+
include JsonWithFallback
|
109
|
+
include AllowMarshal
|
110
|
+
extend self
|
111
|
+
end
|
112
|
+
|
113
|
+
module MessagePackWithFallback
|
114
|
+
include SerializerWithFallback
|
115
|
+
extend self
|
116
|
+
|
117
|
+
def format
|
118
|
+
:message_pack
|
119
|
+
end
|
120
|
+
|
121
|
+
def dump(object)
|
122
|
+
ActiveSupport::MessagePack.dump(object)
|
123
|
+
end
|
124
|
+
|
125
|
+
def _load(dumped)
|
126
|
+
ActiveSupport::MessagePack.load(dumped)
|
127
|
+
end
|
128
|
+
|
129
|
+
def dumped?(dumped)
|
130
|
+
available? && ActiveSupport::MessagePack.signature?(dumped)
|
131
|
+
end
|
132
|
+
|
133
|
+
private
|
134
|
+
def available?
|
135
|
+
return @available if defined?(@available)
|
136
|
+
silence_warnings { require "active_support/message_pack" }
|
137
|
+
@available = true
|
138
|
+
rescue LoadError
|
139
|
+
@available = false
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
module MessagePackWithFallbackAllowMarshal
|
144
|
+
include MessagePackWithFallback
|
145
|
+
include AllowMarshal
|
146
|
+
extend self
|
147
|
+
end
|
148
|
+
|
149
|
+
SERIALIZERS = {
|
150
|
+
marshal: MarshalWithFallback,
|
151
|
+
json: JsonWithFallback,
|
152
|
+
json_allow_marshal: JsonWithFallbackAllowMarshal,
|
153
|
+
message_pack: MessagePackWithFallback,
|
154
|
+
message_pack_allow_marshal: MessagePackWithFallbackAllowMarshal,
|
155
|
+
}
|
156
|
+
end
|
157
|
+
end
|
158
|
+
end
|
@@ -7,6 +7,8 @@ require "active_support/core_ext/module/delegation"
|
|
7
7
|
|
8
8
|
module ActiveSupport # :nodoc:
|
9
9
|
module Multibyte # :nodoc:
|
10
|
+
# = Active Support \Multibyte \Chars
|
11
|
+
#
|
10
12
|
# Chars enables you to work transparently with UTF-8 encoding in the Ruby
|
11
13
|
# String class without having extensive knowledge about the encoding. A
|
12
14
|
# Chars object accepts a string upon initialization and proxies String
|
@@ -22,43 +22,15 @@ module ActiveSupport
|
|
22
22
|
codepoints.pack("U*").unicode_normalize(:nfc).codepoints
|
23
23
|
end
|
24
24
|
|
25
|
-
#
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
return recode_windows1252_chars(string) if force
|
35
|
-
string.scrub { |bad| recode_windows1252_chars(bad) }
|
36
|
-
end
|
37
|
-
else
|
38
|
-
def tidy_bytes(string, force = false)
|
39
|
-
return string if string.empty?
|
40
|
-
return recode_windows1252_chars(string) if force
|
41
|
-
|
42
|
-
# We can't transcode to the same format, so we choose a nearly-identical encoding.
|
43
|
-
# We're going to 'transcode' bytes from UTF-8 when possible, then fall back to
|
44
|
-
# CP1252 when we get errors. The final string will be 'converted' back to UTF-8
|
45
|
-
# before returning.
|
46
|
-
reader = Encoding::Converter.new(Encoding::UTF_8, Encoding::UTF_16LE)
|
47
|
-
|
48
|
-
source = string.dup
|
49
|
-
out = "".force_encoding(Encoding::UTF_16LE)
|
50
|
-
|
51
|
-
loop do
|
52
|
-
reader.primitive_convert(source, out)
|
53
|
-
_, _, _, error_bytes, _ = reader.primitive_errinfo
|
54
|
-
break if error_bytes.nil?
|
55
|
-
out << error_bytes.encode(Encoding::UTF_16LE, Encoding::Windows_1252, invalid: :replace, undef: :replace)
|
56
|
-
end
|
57
|
-
|
58
|
-
reader.finish
|
59
|
-
|
60
|
-
out.encode!(Encoding::UTF_8)
|
61
|
-
end
|
25
|
+
# Replaces all ISO-8859-1 or CP1252 characters by their UTF-8 equivalent
|
26
|
+
# resulting in a valid UTF-8 string.
|
27
|
+
#
|
28
|
+
# Passing +true+ will forcibly tidy all bytes, assuming that the string's
|
29
|
+
# encoding is entirely CP1252 or ISO-8859-1.
|
30
|
+
def tidy_bytes(string, force = false)
|
31
|
+
return string if string.empty? || string.ascii_only?
|
32
|
+
return recode_windows1252_chars(string) if force
|
33
|
+
string.scrub { |bad| recode_windows1252_chars(bad) }
|
62
34
|
end
|
63
35
|
|
64
36
|
private
|