activesupport/lib/active_support/concurrency/load_interlock_aware_monitor.rb

Summary

Maintainability
A
0 mins
Test Coverage
# frozen_string_literal: true

require "monitor"

module ActiveSupport
  module Concurrency
    module LoadInterlockAwareMonitorMixin # :nodoc:
      EXCEPTION_NEVER = { Exception => :never }.freeze
      EXCEPTION_IMMEDIATE = { Exception => :immediate }.freeze
      private_constant :EXCEPTION_NEVER, :EXCEPTION_IMMEDIATE

      # Enters an exclusive section, but allows dependency loading while blocked
      def mon_enter
        mon_try_enter ||
          ActiveSupport::Dependencies.interlock.permit_concurrent_loads { super }
      end

      def synchronize(&block)
        Thread.handle_interrupt(EXCEPTION_NEVER) do
          mon_enter

          begin
            Thread.handle_interrupt(EXCEPTION_IMMEDIATE, &block)
          ensure
            mon_exit
          end
        end
      end
    end
    # A monitor that will permit dependency loading while blocked waiting for
    # the lock.
    class LoadInterlockAwareMonitor < Monitor
      include LoadInterlockAwareMonitorMixin
    end

    class ThreadLoadInterlockAwareMonitor # :nodoc:
      prepend LoadInterlockAwareMonitorMixin

      def initialize
        @owner = nil
        @count = 0
        @mutex = Mutex.new
      end

      private
        def mon_try_enter
          if @owner != Thread.current
            return false unless @mutex.try_lock
            @owner = Thread.current
          end
          @count += 1
        end

        def mon_enter
          @mutex.lock if @owner != Thread.current
          @owner = Thread.current
          @count += 1
        end

        def mon_exit
          unless @owner == Thread.current
            raise ThreadError, "current thread not owner"
          end

          @count -= 1
          return unless @count == 0
          @owner = nil
          @mutex.unlock
        end
    end
  end
end