activesupport 8.0.3 → 8.1.0
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 +310 -159
- data/lib/active_support/backtrace_cleaner.rb +71 -0
- data/lib/active_support/cache/mem_cache_store.rb +13 -13
- data/lib/active_support/cache/redis_cache_store.rb +36 -30
- data/lib/active_support/cache/strategy/local_cache.rb +16 -7
- data/lib/active_support/cache/strategy/local_cache_middleware.rb +7 -7
- data/lib/active_support/cache.rb +69 -6
- data/lib/active_support/callbacks.rb +20 -8
- data/lib/active_support/concurrency/load_interlock_aware_monitor.rb +8 -62
- data/lib/active_support/concurrency/thread_monitor.rb +55 -0
- data/lib/active_support/configurable.rb +34 -0
- data/lib/active_support/continuous_integration.rb +145 -0
- data/lib/active_support/core_ext/array.rb +7 -7
- data/lib/active_support/core_ext/benchmark.rb +4 -12
- data/lib/active_support/core_ext/big_decimal.rb +1 -1
- data/lib/active_support/core_ext/class/attribute.rb +8 -6
- data/lib/active_support/core_ext/class.rb +2 -2
- data/lib/active_support/core_ext/date.rb +5 -5
- data/lib/active_support/core_ext/date_and_time/compatibility.rb +0 -35
- data/lib/active_support/core_ext/date_time/compatibility.rb +3 -5
- data/lib/active_support/core_ext/date_time.rb +5 -5
- data/lib/active_support/core_ext/digest.rb +1 -1
- data/lib/active_support/core_ext/enumerable.rb +2 -2
- data/lib/active_support/core_ext/erb/util.rb +3 -3
- data/lib/active_support/core_ext/file.rb +1 -1
- data/lib/active_support/core_ext/hash.rb +8 -8
- data/lib/active_support/core_ext/integer.rb +3 -3
- data/lib/active_support/core_ext/kernel.rb +3 -3
- data/lib/active_support/core_ext/module.rb +11 -11
- data/lib/active_support/core_ext/numeric.rb +3 -3
- data/lib/active_support/core_ext/object/json.rb +8 -1
- data/lib/active_support/core_ext/object/to_query.rb +5 -0
- data/lib/active_support/core_ext/object.rb +13 -13
- data/lib/active_support/core_ext/pathname.rb +2 -2
- data/lib/active_support/core_ext/range.rb +4 -5
- data/lib/active_support/core_ext/string/multibyte.rb +10 -1
- data/lib/active_support/core_ext/string/output_safety.rb +19 -12
- data/lib/active_support/core_ext/string.rb +13 -13
- data/lib/active_support/core_ext/symbol.rb +1 -1
- data/lib/active_support/core_ext/time/calculations.rb +0 -7
- data/lib/active_support/core_ext/time/compatibility.rb +2 -27
- data/lib/active_support/core_ext/time.rb +5 -5
- data/lib/active_support/core_ext.rb +1 -1
- data/lib/active_support/current_attributes/test_helper.rb +2 -2
- data/lib/active_support/current_attributes.rb +13 -10
- data/lib/active_support/dependencies/interlock.rb +11 -5
- data/lib/active_support/dependencies.rb +6 -1
- data/lib/active_support/deprecation/reporting.rb +4 -2
- data/lib/active_support/deprecation.rb +1 -1
- data/lib/active_support/editor.rb +70 -0
- data/lib/active_support/error_reporter.rb +50 -6
- data/lib/active_support/event_reporter/test_helper.rb +32 -0
- data/lib/active_support/event_reporter.rb +592 -0
- data/lib/active_support/evented_file_update_checker.rb +5 -1
- data/lib/active_support/execution_context.rb +64 -7
- data/lib/active_support/file_update_checker.rb +7 -5
- data/lib/active_support/gem_version.rb +2 -2
- data/lib/active_support/gzip.rb +1 -0
- data/lib/active_support/hash_with_indifferent_access.rb +27 -7
- data/lib/active_support/i18n_railtie.rb +1 -2
- data/lib/active_support/inflector/inflections.rb +31 -15
- data/lib/active_support/inflector/transliterate.rb +6 -8
- data/lib/active_support/isolated_execution_state.rb +12 -15
- data/lib/active_support/json/decoding.rb +2 -2
- data/lib/active_support/json/encoding.rb +135 -17
- data/lib/active_support/log_subscriber.rb +2 -6
- data/lib/active_support/message_encryptors.rb +52 -0
- data/lib/active_support/message_pack/extensions.rb +5 -0
- data/lib/active_support/message_verifiers.rb +52 -0
- data/lib/active_support/messages/rotation_coordinator.rb +9 -0
- data/lib/active_support/messages/rotator.rb +5 -0
- data/lib/active_support/multibyte/chars.rb +8 -1
- data/lib/active_support/multibyte.rb +4 -0
- data/lib/active_support/notifications/fanout.rb +64 -42
- data/lib/active_support/notifications/instrumenter.rb +1 -1
- data/lib/active_support/railtie.rb +32 -15
- data/lib/active_support/structured_event_subscriber.rb +99 -0
- data/lib/active_support/subscriber.rb +0 -5
- data/lib/active_support/syntax_error_proxy.rb +3 -0
- data/lib/active_support/test_case.rb +61 -6
- data/lib/active_support/testing/assertions.rb +34 -6
- data/lib/active_support/testing/error_reporter_assertions.rb +18 -1
- data/lib/active_support/testing/event_reporter_assertions.rb +227 -0
- data/lib/active_support/testing/notification_assertions.rb +92 -0
- data/lib/active_support/testing/parallelization/server.rb +15 -2
- data/lib/active_support/testing/parallelization/worker.rb +4 -2
- data/lib/active_support/testing/parallelization.rb +25 -1
- data/lib/active_support/testing/tests_without_assertions.rb +1 -1
- data/lib/active_support/testing/time_helpers.rb +7 -3
- data/lib/active_support/time_with_zone.rb +22 -22
- data/lib/active_support/values/time_zone.rb +8 -1
- data/lib/active_support/xml_mini.rb +3 -2
- data/lib/active_support.rb +20 -15
- metadata +24 -17
- data/lib/active_support/core_ext/range/each.rb +0 -24
| @@ -56,6 +56,18 @@ module ActiveSupport | |
| 56 56 | 
             
                end
         | 
| 57 57 | 
             
                alias :filter :clean
         | 
| 58 58 |  | 
| 59 | 
            +
                # Given an array of Thread::Backtrace::Location objects, returns an array
         | 
| 60 | 
            +
                # with the clean ones:
         | 
| 61 | 
            +
                #
         | 
| 62 | 
            +
                #     clean_locations = backtrace_cleaner.clean_locations(caller_locations)
         | 
| 63 | 
            +
                #
         | 
| 64 | 
            +
                # Filters and silencers receive strings as usual. However, the +path+
         | 
| 65 | 
            +
                # attributes of the locations in the returned array are the original,
         | 
| 66 | 
            +
                # unfiltered ones, since locations are immutable.
         | 
| 67 | 
            +
                def clean_locations(locations, kind = :silent)
         | 
| 68 | 
            +
                  locations.select { |location| clean_frame(location, kind) }
         | 
| 69 | 
            +
                end
         | 
| 70 | 
            +
             | 
| 59 71 | 
             
                # Returns the frame with all filters applied.
         | 
| 60 72 | 
             
                # returns +nil+ if the frame was silenced.
         | 
| 61 73 | 
             
                def clean_frame(frame, kind = :silent)
         | 
| @@ -74,6 +86,65 @@ module ActiveSupport | |
| 74 86 | 
             
                  end
         | 
| 75 87 | 
             
                end
         | 
| 76 88 |  | 
| 89 | 
            +
                # Thread.each_caller_location does not accept a start in Ruby < 3.4.
         | 
| 90 | 
            +
                if Thread.method(:each_caller_location).arity == 0
         | 
| 91 | 
            +
                  # Returns the first clean frame of the caller's backtrace, or +nil+.
         | 
| 92 | 
            +
                  #
         | 
| 93 | 
            +
                  # Frames are strings.
         | 
| 94 | 
            +
                  def first_clean_frame(kind = :silent)
         | 
| 95 | 
            +
                    caller_location_skipped = false
         | 
| 96 | 
            +
             | 
| 97 | 
            +
                    Thread.each_caller_location do |location|
         | 
| 98 | 
            +
                      unless caller_location_skipped
         | 
| 99 | 
            +
                        caller_location_skipped = true
         | 
| 100 | 
            +
                        next
         | 
| 101 | 
            +
                      end
         | 
| 102 | 
            +
             | 
| 103 | 
            +
                      frame = clean_frame(location, kind)
         | 
| 104 | 
            +
                      return frame if frame
         | 
| 105 | 
            +
                    end
         | 
| 106 | 
            +
                  end
         | 
| 107 | 
            +
             | 
| 108 | 
            +
                  # Returns the first clean location of the caller's call stack, or +nil+.
         | 
| 109 | 
            +
                  #
         | 
| 110 | 
            +
                  # Locations are Thread::Backtrace::Location objects. Since they are
         | 
| 111 | 
            +
                  # immutable, their +path+ attributes are the original ones, but filters
         | 
| 112 | 
            +
                  # are applied internally so silencers can still rely on them.
         | 
| 113 | 
            +
                  def first_clean_location(kind = :silent)
         | 
| 114 | 
            +
                    caller_location_skipped = false
         | 
| 115 | 
            +
             | 
| 116 | 
            +
                    Thread.each_caller_location do |location|
         | 
| 117 | 
            +
                      unless caller_location_skipped
         | 
| 118 | 
            +
                        caller_location_skipped = true
         | 
| 119 | 
            +
                        next
         | 
| 120 | 
            +
                      end
         | 
| 121 | 
            +
             | 
| 122 | 
            +
                      return location if clean_frame(location, kind)
         | 
| 123 | 
            +
                    end
         | 
| 124 | 
            +
                  end
         | 
| 125 | 
            +
                else
         | 
| 126 | 
            +
                  # Returns the first clean frame of the caller's backtrace, or +nil+.
         | 
| 127 | 
            +
                  #
         | 
| 128 | 
            +
                  # Frames are strings.
         | 
| 129 | 
            +
                  def first_clean_frame(kind = :silent)
         | 
| 130 | 
            +
                    Thread.each_caller_location(2) do |location|
         | 
| 131 | 
            +
                      frame = clean_frame(location, kind)
         | 
| 132 | 
            +
                      return frame if frame
         | 
| 133 | 
            +
                    end
         | 
| 134 | 
            +
                  end
         | 
| 135 | 
            +
             | 
| 136 | 
            +
                  # Returns the first clean location of the caller's call stack, or +nil+.
         | 
| 137 | 
            +
                  #
         | 
| 138 | 
            +
                  # Locations are Thread::Backtrace::Location objects. Since they are
         | 
| 139 | 
            +
                  # immutable, their +path+ attributes are the original ones, but filters
         | 
| 140 | 
            +
                  # are applied internally so silencers can still rely on them.
         | 
| 141 | 
            +
                  def first_clean_location(kind = :silent)
         | 
| 142 | 
            +
                    Thread.each_caller_location(2) do |location|
         | 
| 143 | 
            +
                      return location if clean_frame(location, kind)
         | 
| 144 | 
            +
                    end
         | 
| 145 | 
            +
                  end
         | 
| 146 | 
            +
                end
         | 
| 147 | 
            +
             | 
| 77 148 | 
             
                # Adds a filter from the block provided. Each line in the backtrace will be
         | 
| 78 149 | 
             
                # mapped against this filter.
         | 
| 79 150 | 
             
                #
         | 
| @@ -41,7 +41,6 @@ module ActiveSupport | |
| 41 41 |  | 
| 42 42 | 
             
                  prepend Strategy::LocalCache
         | 
| 43 43 |  | 
| 44 | 
            -
                  KEY_MAX_SIZE = 250
         | 
| 45 44 | 
             
                  ESCAPE_KEY_CHARS = /[\x00-\x20%\x7F-\xFF]/n
         | 
| 46 45 |  | 
| 47 46 | 
             
                  # Creates a new Dalli::Client instance with specified addresses and options.
         | 
| @@ -80,6 +79,7 @@ module ActiveSupport | |
| 80 79 | 
             
                    if options.key?(:cache_nils)
         | 
| 81 80 | 
             
                      options[:skip_nil] = !options.delete(:cache_nils)
         | 
| 82 81 | 
             
                    end
         | 
| 82 | 
            +
                    options[:max_key_size] ||= MAX_KEY_SIZE
         | 
| 83 83 | 
             
                    super(options)
         | 
| 84 84 |  | 
| 85 85 | 
             
                    unless [String, Dalli::Client, NilClass].include?(addresses.first.class)
         | 
| @@ -110,9 +110,6 @@ module ActiveSupport | |
| 110 110 | 
             
                  # * <tt>raw: true</tt> - Sends the value directly to the server as raw
         | 
| 111 111 | 
             
                  #   bytes. The value must be a string or number. You can use memcached
         | 
| 112 112 | 
             
                  #   direct operations like +increment+ and +decrement+ only on raw values.
         | 
| 113 | 
            -
                  #
         | 
| 114 | 
            -
                  # * <tt>unless_exist: true</tt> - Prevents overwriting an existing cache
         | 
| 115 | 
            -
                  #   entry.
         | 
| 116 113 |  | 
| 117 114 | 
             
                  # Increment a cached integer value using the memcached incr atomic operator.
         | 
| 118 115 | 
             
                  # Returns the updated value.
         | 
| @@ -129,6 +126,11 @@ module ActiveSupport | |
| 129 126 | 
             
                  #
         | 
| 130 127 | 
             
                  # Incrementing a non-numeric value, or a value written without
         | 
| 131 128 | 
             
                  # <tt>raw: true</tt>, will fail and return +nil+.
         | 
| 129 | 
            +
                  #
         | 
| 130 | 
            +
                  # To read the value later, call #read_counter:
         | 
| 131 | 
            +
                  #
         | 
| 132 | 
            +
                  #   cache.increment("baz") # => 7
         | 
| 133 | 
            +
                  #   cache.read_counter("baz") # 7
         | 
| 132 134 | 
             
                  def increment(name, amount = 1, options = nil)
         | 
| 133 135 | 
             
                    options = merged_options(options)
         | 
| 134 136 | 
             
                    key = normalize_key(name, options)
         | 
| @@ -155,6 +157,11 @@ module ActiveSupport | |
| 155 157 | 
             
                  #
         | 
| 156 158 | 
             
                  # Decrementing a non-numeric value, or a value written without
         | 
| 157 159 | 
             
                  # <tt>raw: true</tt>, will fail and return +nil+.
         | 
| 160 | 
            +
                  #
         | 
| 161 | 
            +
                  # To read the value later, call #read_counter:
         | 
| 162 | 
            +
                  #
         | 
| 163 | 
            +
                  #   cache.decrement("baz") # => 3
         | 
| 164 | 
            +
                  #   cache.read_counter("baz") # 3
         | 
| 158 165 | 
             
                  def decrement(name, amount = 1, options = nil)
         | 
| 159 166 | 
             
                    options = merged_options(options)
         | 
| 160 167 | 
             
                    key = normalize_key(name, options)
         | 
| @@ -249,19 +256,12 @@ module ActiveSupport | |
| 249 256 | 
             
                    # before applying the regular expression to ensure we are escaping all
         | 
| 250 257 | 
             
                    # characters properly.
         | 
| 251 258 | 
             
                    def normalize_key(key, options)
         | 
| 252 | 
            -
                      key =  | 
| 259 | 
            +
                      key = expand_and_namespace_key(key, options)
         | 
| 253 260 | 
             
                      if key
         | 
| 254 261 | 
             
                        key = key.dup.force_encoding(Encoding::ASCII_8BIT)
         | 
| 255 262 | 
             
                        key = key.gsub(ESCAPE_KEY_CHARS) { |match| "%#{match.getbyte(0).to_s(16).upcase}" }
         | 
| 256 | 
            -
             | 
| 257 | 
            -
                        if key.size > KEY_MAX_SIZE
         | 
| 258 | 
            -
                          key_separator = ":hash:"
         | 
| 259 | 
            -
                          key_hash = ActiveSupport::Digest.hexdigest(key)
         | 
| 260 | 
            -
                          key_trim_size = KEY_MAX_SIZE - key_separator.size - key_hash.size
         | 
| 261 | 
            -
                          key = "#{key[0, key_trim_size]}#{key_separator}#{key_hash}"
         | 
| 262 | 
            -
                        end
         | 
| 263 263 | 
             
                      end
         | 
| 264 | 
            -
                      key
         | 
| 264 | 
            +
                      truncate_key(key)
         | 
| 265 265 | 
             
                    end
         | 
| 266 266 |  | 
| 267 267 | 
             
                    def deserialize_entry(payload, raw: false, **)
         | 
| @@ -35,9 +35,6 @@ module ActiveSupport | |
| 35 35 | 
             
                #   +Redis::Distributed+ 4.0.1+ for distributed mget support.
         | 
| 36 36 | 
             
                # * +delete_matched+ support for Redis KEYS globs.
         | 
| 37 37 | 
             
                class RedisCacheStore < Store
         | 
| 38 | 
            -
                  # Keys are truncated with the Active Support digest if they exceed 1kB
         | 
| 39 | 
            -
                  MAX_KEY_BYTESIZE = 1024
         | 
| 40 | 
            -
             | 
| 41 38 | 
             
                  DEFAULT_REDIS_OPTIONS = {
         | 
| 42 39 | 
             
                    connect_timeout:    1,
         | 
| 43 40 | 
             
                    read_timeout:       1,
         | 
| @@ -106,20 +103,29 @@ module ActiveSupport | |
| 106 103 | 
             
                      end
         | 
| 107 104 | 
             
                  end
         | 
| 108 105 |  | 
| 109 | 
            -
                  attr_reader :max_key_bytesize
         | 
| 110 106 | 
             
                  attr_reader :redis
         | 
| 111 107 |  | 
| 112 108 | 
             
                  # Creates a new Redis cache store.
         | 
| 113 109 | 
             
                  #
         | 
| 114 | 
            -
                  # There are  | 
| 115 | 
            -
                  # | 
| 116 | 
            -
                  #  | 
| 117 | 
            -
                  #  | 
| 118 | 
            -
                  # instance.
         | 
| 110 | 
            +
                  # There are a few ways to provide the Redis client used by the cache:
         | 
| 111 | 
            +
                  #
         | 
| 112 | 
            +
                  # 1. The +:redis+ param can be:
         | 
| 113 | 
            +
                  #    - A Redis instance.
         | 
| 114 | 
            +
                  #    - A +ConnectionPool+ instance wrapping a Redis instance.
         | 
| 115 | 
            +
                  #    - A block that returns a Redis instance.
         | 
| 116 | 
            +
                  #
         | 
| 117 | 
            +
                  # 2. The +:url+ param can be:
         | 
| 118 | 
            +
                  #    - A string used to create a Redis instance.
         | 
| 119 | 
            +
                  #    - An array of strings used to create a +Redis::Distributed+ instance.
         | 
| 120 | 
            +
                  #
         | 
| 121 | 
            +
                  # If the final Redis instance is not already a +ConnectionPool+, it will
         | 
| 122 | 
            +
                  # be wrapped in one using +ActiveSupport::Cache::Store::DEFAULT_POOL_OPTIONS+.
         | 
| 123 | 
            +
                  # These options can be overridden with the +:pool+ param, or the pool can be
         | 
| 124 | 
            +
                  # disabled with +:pool: false+.
         | 
| 119 125 | 
             
                  #
         | 
| 120 126 | 
             
                  #   Option  Class       Result
         | 
| 121 | 
            -
                  #   :redis  Proc    ->  options[:redis].call
         | 
| 122 127 | 
             
                  #   :redis  Object  ->  options[:redis]
         | 
| 128 | 
            +
                  #   :redis  Proc    ->  options[:redis].call
         | 
| 123 129 | 
             
                  #   :url    String  ->  Redis.new(url: …)
         | 
| 124 130 | 
             
                  #   :url    Array   ->  Redis::Distributed.new([{ url: … }, { url: … }, …])
         | 
| 125 131 | 
             
                  #
         | 
| @@ -148,14 +154,17 @@ module ActiveSupport | |
| 148 154 | 
             
                  #   cache.exist?('bar') # => false
         | 
| 149 155 | 
             
                  def initialize(error_handler: DEFAULT_ERROR_HANDLER, **redis_options)
         | 
| 150 156 | 
             
                    universal_options = redis_options.extract!(*UNIVERSAL_OPTIONS)
         | 
| 157 | 
            +
                    redis = redis_options[:redis]
         | 
| 158 | 
            +
             | 
| 159 | 
            +
                    already_pool = redis.instance_of?(::ConnectionPool) ||
         | 
| 160 | 
            +
                                   (redis.respond_to?(:wrapped_pool) && redis.wrapped_pool.instance_of?(::ConnectionPool))
         | 
| 151 161 |  | 
| 152 | 
            -
                    if pool_options = self.class.send(:retrieve_pool_options, redis_options)
         | 
| 162 | 
            +
                    if !already_pool && pool_options = self.class.send(:retrieve_pool_options, redis_options)
         | 
| 153 163 | 
             
                      @redis = ::ConnectionPool.new(pool_options) { self.class.build_redis(**redis_options) }
         | 
| 154 164 | 
             
                    else
         | 
| 155 165 | 
             
                      @redis = self.class.build_redis(**redis_options)
         | 
| 156 166 | 
             
                    end
         | 
| 157 167 |  | 
| 158 | 
            -
                    @max_key_bytesize = MAX_KEY_BYTESIZE
         | 
| 159 168 | 
             
                    @error_handler = error_handler
         | 
| 160 169 |  | 
| 161 170 | 
             
                    super(universal_options)
         | 
| @@ -213,7 +222,7 @@ module ActiveSupport | |
| 213 222 | 
             
                        nodes.each do |node|
         | 
| 214 223 | 
             
                          begin
         | 
| 215 224 | 
             
                            cursor, keys = node.scan(cursor, match: pattern, count: SCAN_BATCH_SIZE)
         | 
| 216 | 
            -
                            node. | 
| 225 | 
            +
                            node.unlink(*keys) unless keys.empty?
         | 
| 217 226 | 
             
                          end until cursor == "0"
         | 
| 218 227 | 
             
                        end
         | 
| 219 228 | 
             
                      end
         | 
| @@ -236,6 +245,11 @@ module ActiveSupport | |
| 236 245 | 
             
                  # Incrementing a non-numeric value, or a value written without
         | 
| 237 246 | 
             
                  # <tt>raw: true</tt>, will fail and return +nil+.
         | 
| 238 247 | 
             
                  #
         | 
| 248 | 
            +
                  # To read the value later, call #read_counter:
         | 
| 249 | 
            +
                  #
         | 
| 250 | 
            +
                  #   cache.increment("baz") # => 7
         | 
| 251 | 
            +
                  #   cache.read_counter("baz") # 7
         | 
| 252 | 
            +
                  #
         | 
| 239 253 | 
             
                  # Failsafe: Raises errors.
         | 
| 240 254 | 
             
                  def increment(name, amount = 1, options = nil)
         | 
| 241 255 | 
             
                    options = merged_options(options)
         | 
| @@ -263,6 +277,11 @@ module ActiveSupport | |
| 263 277 | 
             
                  # Decrementing a non-numeric value, or a value written without
         | 
| 264 278 | 
             
                  # <tt>raw: true</tt>, will fail and return +nil+.
         | 
| 265 279 | 
             
                  #
         | 
| 280 | 
            +
                  # To read the value later, call #read_counter:
         | 
| 281 | 
            +
                  #
         | 
| 282 | 
            +
                  #   cache.decrement("baz") # => 3
         | 
| 283 | 
            +
                  #   cache.read_counter("baz") # 3
         | 
| 284 | 
            +
                  #
         | 
| 266 285 | 
             
                  # Failsafe: Raises errors.
         | 
| 267 286 | 
             
                  def decrement(name, amount = 1, options = nil)
         | 
| 268 287 | 
             
                    options = merged_options(options)
         | 
| @@ -384,14 +403,16 @@ module ActiveSupport | |
| 384 403 | 
             
                    # Delete an entry from the cache.
         | 
| 385 404 | 
             
                    def delete_entry(key, **options)
         | 
| 386 405 | 
             
                      failsafe :delete_entry, returning: false do
         | 
| 387 | 
            -
                        redis.then { |c| c. | 
| 406 | 
            +
                        redis.then { |c| c.unlink(key) == 1 }
         | 
| 388 407 | 
             
                      end
         | 
| 389 408 | 
             
                    end
         | 
| 390 409 |  | 
| 391 410 | 
             
                    # Deletes multiple entries in the cache. Returns the number of entries deleted.
         | 
| 392 411 | 
             
                    def delete_multi_entries(entries, **_options)
         | 
| 412 | 
            +
                      return 0 if entries.empty?
         | 
| 413 | 
            +
             | 
| 393 414 | 
             
                      failsafe :delete_multi_entries, returning: 0 do
         | 
| 394 | 
            -
                        redis.then { |c| c. | 
| 415 | 
            +
                        redis.then { |c| c.unlink(*entries) }
         | 
| 395 416 | 
             
                      end
         | 
| 396 417 | 
             
                    end
         | 
| 397 418 |  | 
| @@ -410,21 +431,6 @@ module ActiveSupport | |
| 410 431 | 
             
                      end
         | 
| 411 432 | 
             
                    end
         | 
| 412 433 |  | 
| 413 | 
            -
                    # Truncate keys that exceed 1kB.
         | 
| 414 | 
            -
                    def normalize_key(key, options)
         | 
| 415 | 
            -
                      truncate_key super&.b
         | 
| 416 | 
            -
                    end
         | 
| 417 | 
            -
             | 
| 418 | 
            -
                    def truncate_key(key)
         | 
| 419 | 
            -
                      if key && key.bytesize > max_key_bytesize
         | 
| 420 | 
            -
                        suffix = ":hash:#{ActiveSupport::Digest.hexdigest(key)}"
         | 
| 421 | 
            -
                        truncate_at = max_key_bytesize - suffix.bytesize
         | 
| 422 | 
            -
                        "#{key.byteslice(0, truncate_at)}#{suffix}"
         | 
| 423 | 
            -
                      else
         | 
| 424 | 
            -
                        key
         | 
| 425 | 
            -
                      end
         | 
| 426 | 
            -
                    end
         | 
| 427 | 
            -
             | 
| 428 434 | 
             
                    def deserialize_entry(payload, raw: false, **)
         | 
| 429 435 | 
             
                      if raw && !payload.nil?
         | 
| 430 436 | 
             
                        Entry.new(payload)
         | 
| @@ -68,12 +68,25 @@ module ActiveSupport | |
| 68 68 | 
             
                      use_temporary_local_cache(LocalStore.new, &block)
         | 
| 69 69 | 
             
                    end
         | 
| 70 70 |  | 
| 71 | 
            +
                    # Set a new local cache.
         | 
| 72 | 
            +
                    def new_local_cache
         | 
| 73 | 
            +
                      LocalCacheRegistry.set_cache_for(local_cache_key, LocalStore.new)
         | 
| 74 | 
            +
                    end
         | 
| 75 | 
            +
             | 
| 76 | 
            +
                    # Unset the current local cache.
         | 
| 77 | 
            +
                    def unset_local_cache
         | 
| 78 | 
            +
                      LocalCacheRegistry.set_cache_for(local_cache_key, nil)
         | 
| 79 | 
            +
                    end
         | 
| 80 | 
            +
             | 
| 81 | 
            +
                    # The current local cache.
         | 
| 82 | 
            +
                    def local_cache
         | 
| 83 | 
            +
                      LocalCacheRegistry.cache_for(local_cache_key)
         | 
| 84 | 
            +
                    end
         | 
| 85 | 
            +
             | 
| 71 86 | 
             
                    # Middleware class can be inserted as a Rack handler to be local cache for the
         | 
| 72 87 | 
             
                    # duration of request.
         | 
| 73 88 | 
             
                    def middleware
         | 
| 74 | 
            -
                      @middleware ||= Middleware.new(
         | 
| 75 | 
            -
                        "ActiveSupport::Cache::Strategy::LocalCache",
         | 
| 76 | 
            -
                        local_cache_key)
         | 
| 89 | 
            +
                      @middleware ||= Middleware.new("ActiveSupport::Cache::Strategy::LocalCache", self)
         | 
| 77 90 | 
             
                    end
         | 
| 78 91 |  | 
| 79 92 | 
             
                    def clear(options = nil) # :nodoc:
         | 
| @@ -214,10 +227,6 @@ module ActiveSupport | |
| 214 227 | 
             
                        @local_cache_key ||= "#{self.class.name.underscore}_local_cache_#{object_id}".gsub(/[\/-]/, "_").to_sym
         | 
| 215 228 | 
             
                      end
         | 
| 216 229 |  | 
| 217 | 
            -
                      def local_cache
         | 
| 218 | 
            -
                        LocalCacheRegistry.cache_for(local_cache_key)
         | 
| 219 | 
            -
                      end
         | 
| 220 | 
            -
             | 
| 221 230 | 
             
                      def bypass_local_cache(&block)
         | 
| 222 231 | 
             
                        use_temporary_local_cache(nil, &block)
         | 
| 223 232 | 
             
                      end
         | 
| @@ -11,11 +11,12 @@ module ActiveSupport | |
| 11 11 | 
             
                    # This class wraps up local storage for middlewares. Only the middleware method should
         | 
| 12 12 | 
             
                    # construct them.
         | 
| 13 13 | 
             
                    class Middleware # :nodoc:
         | 
| 14 | 
            -
                      attr_reader :name | 
| 14 | 
            +
                      attr_reader :name
         | 
| 15 | 
            +
                      attr_accessor :cache
         | 
| 15 16 |  | 
| 16 | 
            -
                      def initialize(name,  | 
| 17 | 
            +
                      def initialize(name, cache)
         | 
| 17 18 | 
             
                        @name = name
         | 
| 18 | 
            -
                        @ | 
| 19 | 
            +
                        @cache = cache
         | 
| 19 20 | 
             
                        @app = nil
         | 
| 20 21 | 
             
                      end
         | 
| 21 22 |  | 
| @@ -25,18 +26,17 @@ module ActiveSupport | |
| 25 26 | 
             
                      end
         | 
| 26 27 |  | 
| 27 28 | 
             
                      def call(env)
         | 
| 28 | 
            -
                         | 
| 29 | 
            +
                        cache.new_local_cache
         | 
| 29 30 | 
             
                        response = @app.call(env)
         | 
| 30 31 | 
             
                        response[2] = ::Rack::BodyProxy.new(response[2]) do
         | 
| 31 | 
            -
                           | 
| 32 | 
            +
                          cache.unset_local_cache
         | 
| 32 33 | 
             
                        end
         | 
| 33 34 | 
             
                        cleanup_on_body_close = true
         | 
| 34 35 | 
             
                        response
         | 
| 35 36 | 
             
                      rescue Rack::Utils::InvalidParameterError
         | 
| 36 37 | 
             
                        [400, {}, []]
         | 
| 37 38 | 
             
                      ensure
         | 
| 38 | 
            -
                         | 
| 39 | 
            -
                          cleanup_on_body_close
         | 
| 39 | 
            +
                        cache.unset_local_cache unless cleanup_on_body_close
         | 
| 40 40 | 
             
                      end
         | 
| 41 41 | 
             
                    end
         | 
| 42 42 | 
             
                  end
         | 
    
        data/lib/active_support/cache.rb
    CHANGED
    
    | @@ -36,6 +36,7 @@ module ActiveSupport | |
| 36 36 | 
             
                  :serializer,
         | 
| 37 37 | 
             
                  :skip_nil,
         | 
| 38 38 | 
             
                  :raw,
         | 
| 39 | 
            +
                  :max_key_size,
         | 
| 39 40 | 
             
                ]
         | 
| 40 41 |  | 
| 41 42 | 
             
                # Mapping of canonical option names to aliases that a store will recognize.
         | 
| @@ -187,6 +188,12 @@ module ActiveSupport | |
| 187 188 | 
             
                #   @last_mod_time = Time.now  # Invalidate the entire cache by changing namespace
         | 
| 188 189 | 
             
                #
         | 
| 189 190 | 
             
                class Store
         | 
| 191 | 
            +
                  # Default +ConnectionPool+ options
         | 
| 192 | 
            +
                  DEFAULT_POOL_OPTIONS = { size: 5, timeout: 5 }.freeze
         | 
| 193 | 
            +
             | 
| 194 | 
            +
                  # Keys are truncated with the Active Support digest if they exceed the limit.
         | 
| 195 | 
            +
                  MAX_KEY_SIZE = 250
         | 
| 196 | 
            +
             | 
| 190 197 | 
             
                  cattr_accessor :logger, instance_writer: true
         | 
| 191 198 | 
             
                  cattr_accessor :raise_on_invalid_cache_expiration_time, default: false
         | 
| 192 199 |  | 
| @@ -195,9 +202,6 @@ module ActiveSupport | |
| 195 202 |  | 
| 196 203 | 
             
                  class << self
         | 
| 197 204 | 
             
                    private
         | 
| 198 | 
            -
                      DEFAULT_POOL_OPTIONS = { size: 5, timeout: 5 }.freeze
         | 
| 199 | 
            -
                      private_constant :DEFAULT_POOL_OPTIONS
         | 
| 200 | 
            -
             | 
| 201 205 | 
             
                      def retrieve_pool_options(options)
         | 
| 202 206 | 
             
                        if options.key?(:pool)
         | 
| 203 207 | 
             
                          pool_options = options.delete(:pool)
         | 
| @@ -299,6 +303,9 @@ module ActiveSupport | |
| 299 303 | 
             
                    @options[:compress] = true unless @options.key?(:compress)
         | 
| 300 304 | 
             
                    @options[:compress_threshold] ||= DEFAULT_COMPRESS_LIMIT
         | 
| 301 305 |  | 
| 306 | 
            +
                    @max_key_size = @options.delete(:max_key_size)
         | 
| 307 | 
            +
                    @max_key_size = MAX_KEY_SIZE if @max_key_size.nil? # allow 'false' as a value
         | 
| 308 | 
            +
             | 
| 302 309 | 
             
                    @coder = @options.delete(:coder) do
         | 
| 303 310 | 
             
                      legacy_serializer = Cache.format_version < 7.1 && !@options[:serializer]
         | 
| 304 311 | 
             
                      serializer = @options.delete(:serializer) || default_serializer
         | 
| @@ -557,7 +564,7 @@ module ActiveSupport | |
| 557 564 |  | 
| 558 565 | 
             
                    instrument_multi :write_multi, normalized_hash, options do |payload|
         | 
| 559 566 | 
             
                      entries = hash.each_with_object({}) do |(name, value), memo|
         | 
| 560 | 
            -
                        memo[normalize_key(name, options)] = Entry.new(value, **options | 
| 567 | 
            +
                        memo[normalize_key(name, options)] = Entry.new(value, **options, version: normalize_version(name, options))
         | 
| 561 568 | 
             
                      end
         | 
| 562 569 |  | 
| 563 570 | 
             
                      write_multi_entries entries, **options
         | 
| @@ -659,13 +666,15 @@ module ActiveSupport | |
| 659 666 | 
             
                  #   version, the read will be treated as a cache miss. This feature is
         | 
| 660 667 | 
             
                  #   used to support recyclable cache keys.
         | 
| 661 668 | 
             
                  #
         | 
| 669 | 
            +
                  # * +:unless_exist+ - Prevents overwriting an existing cache entry.
         | 
| 670 | 
            +
                  #
         | 
| 662 671 | 
             
                  # Other options will be handled by the specific cache store implementation.
         | 
| 663 672 | 
             
                  def write(name, value, options = nil)
         | 
| 664 673 | 
             
                    options = merged_options(options)
         | 
| 665 674 | 
             
                    key = normalize_key(name, options)
         | 
| 666 675 |  | 
| 667 676 | 
             
                    instrument(:write, key, options) do
         | 
| 668 | 
            -
                      entry = Entry.new(value, **options | 
| 677 | 
            +
                      entry = Entry.new(value, **options, version: normalize_version(name, options))
         | 
| 669 678 | 
             
                      write_entry(key, entry, **options)
         | 
| 670 679 | 
             
                    end
         | 
| 671 680 | 
             
                  end
         | 
| @@ -742,6 +751,32 @@ module ActiveSupport | |
| 742 751 | 
             
                    raise NotImplementedError.new("#{self.class.name} does not support decrement")
         | 
| 743 752 | 
             
                  end
         | 
| 744 753 |  | 
| 754 | 
            +
                  # Reads a counter that was set by #increment / #decrement.
         | 
| 755 | 
            +
                  #
         | 
| 756 | 
            +
                  #   cache.write_counter("foo", 1)
         | 
| 757 | 
            +
                  #   cache.read_counter("foo") # => 1
         | 
| 758 | 
            +
                  #   cache.increment("foo")
         | 
| 759 | 
            +
                  #   cache.read_counter("foo") # => 2
         | 
| 760 | 
            +
                  #
         | 
| 761 | 
            +
                  # Options are passed to the underlying cache implementation.
         | 
| 762 | 
            +
                  def read_counter(name, **options)
         | 
| 763 | 
            +
                    options = merged_options(options).merge(raw: true)
         | 
| 764 | 
            +
                    read(name, **options)&.to_i
         | 
| 765 | 
            +
                  end
         | 
| 766 | 
            +
             | 
| 767 | 
            +
                  # Writes a counter that can then be modified by #increment / #decrement.
         | 
| 768 | 
            +
                  #
         | 
| 769 | 
            +
                  #   cache.write_counter("foo", 1)
         | 
| 770 | 
            +
                  #   cache.read_counter("foo") # => 1
         | 
| 771 | 
            +
                  #   cache.increment("foo")
         | 
| 772 | 
            +
                  #   cache.read_counter("foo") # => 2
         | 
| 773 | 
            +
                  #
         | 
| 774 | 
            +
                  # Options are passed to the underlying cache implementation.
         | 
| 775 | 
            +
                  def write_counter(name, value, **options)
         | 
| 776 | 
            +
                    options = merged_options(options).merge(raw: true)
         | 
| 777 | 
            +
                    write(name, value.to_i, **options)
         | 
| 778 | 
            +
                  end
         | 
| 779 | 
            +
             | 
| 745 780 | 
             
                  # Cleans up the cache by removing expired entries.
         | 
| 746 781 | 
             
                  #
         | 
| 747 782 | 
             
                  # Options are passed to the underlying cache implementation.
         | 
| @@ -761,6 +796,17 @@ module ActiveSupport | |
| 761 796 | 
             
                    raise NotImplementedError.new("#{self.class.name} does not support clear")
         | 
| 762 797 | 
             
                  end
         | 
| 763 798 |  | 
| 799 | 
            +
                  # Get the current namespace
         | 
| 800 | 
            +
                  def namespace
         | 
| 801 | 
            +
                    @options[:namespace]
         | 
| 802 | 
            +
                  end
         | 
| 803 | 
            +
             | 
| 804 | 
            +
                  # Set the current namespace. Note, this will be ignored if custom
         | 
| 805 | 
            +
                  # options are passed to cache wills with a namespace key.
         | 
| 806 | 
            +
                  def namespace=(namespace)
         | 
| 807 | 
            +
                    @options[:namespace] = namespace
         | 
| 808 | 
            +
                  end
         | 
| 809 | 
            +
             | 
| 764 810 | 
             
                  private
         | 
| 765 811 | 
             
                    def default_serializer
         | 
| 766 812 | 
             
                      case Cache.format_version
         | 
| @@ -927,16 +973,33 @@ module ActiveSupport | |
| 927 973 | 
             
                      options
         | 
| 928 974 | 
             
                    end
         | 
| 929 975 |  | 
| 930 | 
            -
                    # Expands and  | 
| 976 | 
            +
                    # Expands, namespaces and truncates the cache key.
         | 
| 931 977 | 
             
                    # Raises an exception when the key is +nil+ or an empty string.
         | 
| 932 978 | 
             
                    # May be overridden by cache stores to do additional normalization.
         | 
| 933 979 | 
             
                    def normalize_key(key, options = nil)
         | 
| 980 | 
            +
                      key = expand_and_namespace_key(key, options)
         | 
| 981 | 
            +
                      truncate_key(key)
         | 
| 982 | 
            +
                    end
         | 
| 983 | 
            +
             | 
| 984 | 
            +
                    def expand_and_namespace_key(key, options = nil)
         | 
| 934 985 | 
             
                      str_key = expanded_key(key)
         | 
| 935 986 | 
             
                      raise(ArgumentError, "key cannot be blank") if !str_key || str_key.empty?
         | 
| 936 987 |  | 
| 937 988 | 
             
                      namespace_key str_key, options
         | 
| 938 989 | 
             
                    end
         | 
| 939 990 |  | 
| 991 | 
            +
                    def truncate_key(key)
         | 
| 992 | 
            +
                      if key && @max_key_size && key.bytesize > @max_key_size
         | 
| 993 | 
            +
                        suffix = ":hash:#{ActiveSupport::Digest.hexdigest(key)}"
         | 
| 994 | 
            +
                        truncate_at = @max_key_size - suffix.bytesize
         | 
| 995 | 
            +
                        key = key.byteslice(0, truncate_at)
         | 
| 996 | 
            +
                        key.scrub!("")
         | 
| 997 | 
            +
                        "#{key}#{suffix}"
         | 
| 998 | 
            +
                      else
         | 
| 999 | 
            +
                        key
         | 
| 1000 | 
            +
                      end
         | 
| 1001 | 
            +
                    end
         | 
| 1002 | 
            +
             | 
| 940 1003 | 
             
                    # Prefix the key with a namespace string:
         | 
| 941 1004 | 
             
                    #
         | 
| 942 1005 | 
             
                    #   namespace_key 'foo', namespace: 'cache'
         | 
| @@ -4,6 +4,7 @@ require "active_support/concern" | |
| 4 4 | 
             
            require "active_support/descendants_tracker"
         | 
| 5 5 | 
             
            require "active_support/core_ext/array/extract_options"
         | 
| 6 6 | 
             
            require "active_support/core_ext/class/attribute"
         | 
| 7 | 
            +
            require "active_support/core_ext/module/redefine_method"
         | 
| 7 8 | 
             
            require "active_support/core_ext/string/filters"
         | 
| 8 9 | 
             
            require "active_support/core_ext/object/blank"
         | 
| 9 10 |  | 
| @@ -573,7 +574,7 @@ module ActiveSupport | |
| 573 574 | 
             
                      @name = name
         | 
| 574 575 | 
             
                      @config = {
         | 
| 575 576 | 
             
                        scope: [:kind],
         | 
| 576 | 
            -
                        terminator:  | 
| 577 | 
            +
                        terminator: DEFAULT_TERMINATOR
         | 
| 577 578 | 
             
                      }.merge!(config)
         | 
| 578 579 | 
             
                      @chain = []
         | 
| 579 580 | 
             
                      @all_callbacks = nil
         | 
| @@ -661,8 +662,8 @@ module ActiveSupport | |
| 661 662 | 
             
                        @chain.delete_if { |c| callback.duplicates?(c) }
         | 
| 662 663 | 
             
                      end
         | 
| 663 664 |  | 
| 664 | 
            -
                       | 
| 665 | 
            -
                         | 
| 665 | 
            +
                      class DefaultTerminator # :nodoc:
         | 
| 666 | 
            +
                        def call(target, result_lambda)
         | 
| 666 667 | 
             
                          terminate = true
         | 
| 667 668 | 
             
                          catch(:abort) do
         | 
| 668 669 | 
             
                            result_lambda.call
         | 
| @@ -671,6 +672,7 @@ module ActiveSupport | |
| 671 672 | 
             
                          terminate
         | 
| 672 673 | 
             
                        end
         | 
| 673 674 | 
             
                      end
         | 
| 675 | 
            +
                      DEFAULT_TERMINATOR = DefaultTerminator.new.freeze
         | 
| 674 676 | 
             
                  end
         | 
| 675 677 |  | 
| 676 678 | 
             
                  module ClassMethods
         | 
| @@ -904,12 +906,13 @@ module ActiveSupport | |
| 904 906 | 
             
                      names.each do |name|
         | 
| 905 907 | 
             
                        name = name.to_sym
         | 
| 906 908 |  | 
| 907 | 
            -
                        ([self] + self.descendants).each do |target|
         | 
| 908 | 
            -
                          target.set_callbacks name, CallbackChain.new(name, options)
         | 
| 909 | 
            -
                        end
         | 
| 910 | 
            -
             | 
| 911 909 | 
             
                        module_eval <<-RUBY, __FILE__, __LINE__ + 1
         | 
| 912 | 
            -
                          def _run_#{name}_callbacks | 
| 910 | 
            +
                          def _run_#{name}_callbacks
         | 
| 911 | 
            +
                            yield if block_given?
         | 
| 912 | 
            +
                          end
         | 
| 913 | 
            +
                          silence_redefinition_of_method(:_run_#{name}_callbacks)
         | 
| 914 | 
            +
             | 
| 915 | 
            +
                          def _run_#{name}_callbacks!(&block)
         | 
| 913 916 | 
             
                            run_callbacks #{name.inspect}, &block
         | 
| 914 917 | 
             
                          end
         | 
| 915 918 |  | 
| @@ -925,6 +928,10 @@ module ActiveSupport | |
| 925 928 | 
             
                            __callbacks[#{name.inspect}]
         | 
| 926 929 | 
             
                          end
         | 
| 927 930 | 
             
                        RUBY
         | 
| 931 | 
            +
             | 
| 932 | 
            +
                        ([self] + self.descendants).each do |target|
         | 
| 933 | 
            +
                          target.set_callbacks name, CallbackChain.new(name, options)
         | 
| 934 | 
            +
                        end
         | 
| 928 935 | 
             
                      end
         | 
| 929 936 | 
             
                    end
         | 
| 930 937 |  | 
| @@ -940,6 +947,11 @@ module ActiveSupport | |
| 940 947 | 
             
                        unless singleton_class.private_method_defined?(:__class_attr__callbacks, false)
         | 
| 941 948 | 
             
                          self.__callbacks = __callbacks.dup
         | 
| 942 949 | 
             
                        end
         | 
| 950 | 
            +
                        name = name.to_sym
         | 
| 951 | 
            +
                        callbacks_was = self.__callbacks[name.to_sym]
         | 
| 952 | 
            +
                        if (callbacks_was.nil? || callbacks_was.empty?) && !callbacks.empty?
         | 
| 953 | 
            +
                          alias_method("_run_#{name}_callbacks", "_run_#{name}_callbacks!")
         | 
| 954 | 
            +
                        end
         | 
| 943 955 | 
             
                        self.__callbacks[name.to_sym] = callbacks
         | 
| 944 956 | 
             
                        self.__callbacks
         | 
| 945 957 | 
             
                      end
         | 
| @@ -4,69 +4,15 @@ require "monitor" | |
| 4 4 |  | 
| 5 5 | 
             
            module ActiveSupport
         | 
| 6 6 | 
             
              module Concurrency
         | 
| 7 | 
            -
                module LoadInterlockAwareMonitorMixin # :nodoc:
         | 
| 8 | 
            -
                  EXCEPTION_NEVER = { Exception => :never }.freeze
         | 
| 9 | 
            -
                  EXCEPTION_IMMEDIATE = { Exception => :immediate }.freeze
         | 
| 10 | 
            -
                  private_constant :EXCEPTION_NEVER, :EXCEPTION_IMMEDIATE
         | 
| 11 | 
            -
             | 
| 12 | 
            -
                  # Enters an exclusive section, but allows dependency loading while blocked
         | 
| 13 | 
            -
                  def mon_enter
         | 
| 14 | 
            -
                    mon_try_enter ||
         | 
| 15 | 
            -
                      ActiveSupport::Dependencies.interlock.permit_concurrent_loads { super }
         | 
| 16 | 
            -
                  end
         | 
| 17 | 
            -
             | 
| 18 | 
            -
                  def synchronize(&block)
         | 
| 19 | 
            -
                    Thread.handle_interrupt(EXCEPTION_NEVER) do
         | 
| 20 | 
            -
                      mon_enter
         | 
| 21 | 
            -
             | 
| 22 | 
            -
                      begin
         | 
| 23 | 
            -
                        Thread.handle_interrupt(EXCEPTION_IMMEDIATE, &block)
         | 
| 24 | 
            -
                      ensure
         | 
| 25 | 
            -
                        mon_exit
         | 
| 26 | 
            -
                      end
         | 
| 27 | 
            -
                    end
         | 
| 28 | 
            -
                  end
         | 
| 29 | 
            -
                end
         | 
| 30 7 | 
             
                # A monitor that will permit dependency loading while blocked waiting for
         | 
| 31 8 | 
             
                # the lock.
         | 
| 32 | 
            -
                 | 
| 33 | 
            -
                   | 
| 34 | 
            -
             | 
| 35 | 
            -
             | 
| 36 | 
            -
             | 
| 37 | 
            -
             | 
| 38 | 
            -
             | 
| 39 | 
            -
             | 
| 40 | 
            -
                    @owner = nil
         | 
| 41 | 
            -
                    @count = 0
         | 
| 42 | 
            -
                    @mutex = Mutex.new
         | 
| 43 | 
            -
                  end
         | 
| 44 | 
            -
             | 
| 45 | 
            -
                  private
         | 
| 46 | 
            -
                    def mon_try_enter
         | 
| 47 | 
            -
                      if @owner != Thread.current
         | 
| 48 | 
            -
                        return false unless @mutex.try_lock
         | 
| 49 | 
            -
                        @owner = Thread.current
         | 
| 50 | 
            -
                      end
         | 
| 51 | 
            -
                      @count += 1
         | 
| 52 | 
            -
                    end
         | 
| 53 | 
            -
             | 
| 54 | 
            -
                    def mon_enter
         | 
| 55 | 
            -
                      @mutex.lock if @owner != Thread.current
         | 
| 56 | 
            -
                      @owner = Thread.current
         | 
| 57 | 
            -
                      @count += 1
         | 
| 58 | 
            -
                    end
         | 
| 59 | 
            -
             | 
| 60 | 
            -
                    def mon_exit
         | 
| 61 | 
            -
                      unless @owner == Thread.current
         | 
| 62 | 
            -
                        raise ThreadError, "current thread not owner"
         | 
| 63 | 
            -
                      end
         | 
| 64 | 
            -
             | 
| 65 | 
            -
                      @count -= 1
         | 
| 66 | 
            -
                      return unless @count == 0
         | 
| 67 | 
            -
                      @owner = nil
         | 
| 68 | 
            -
                      @mutex.unlock
         | 
| 69 | 
            -
                    end
         | 
| 70 | 
            -
                end
         | 
| 9 | 
            +
                LoadInterlockAwareMonitor = ActiveSupport::Deprecation::DeprecatedConstantProxy.new(
         | 
| 10 | 
            +
                  "ActiveSupport::Concurrency::LoadInterlockAwareMonitor",
         | 
| 11 | 
            +
                  "::Monitor",
         | 
| 12 | 
            +
                  ActiveSupport.deprecator,
         | 
| 13 | 
            +
                  message: "ActiveSupport::Concurrency::LoadInterlockAwareMonitor is deprecated and will be " \
         | 
| 14 | 
            +
                           "removed in Rails 9.0. Use Monitor directly instead, as the loading interlock is " \
         | 
| 15 | 
            +
                           "no longer used."
         | 
| 16 | 
            +
                )
         | 
| 71 17 | 
             
              end
         | 
| 72 18 | 
             
            end
         |