Sha256: b2c931a5744704d7c713e71f19159dab667580de54ff9bb8c672e4fbfa6ea66b

Contents?: true

Size: 1.06 KB

Versions: 1

Compression:

Stored size: 1.06 KB

Contents

require 'monitor'
require 'synchronizable/version'

# Synchronizable is intended to be injected into objects via Object#extend.
# After Object#extend(Synchronizable) is performed, the object's original
# methods will become synchronized via a per-instance lock.
module Synchronizable
  IGNORABLE_METHOD_OWNERS = [Object, Kernel, BasicObject]

  def self.extended(obj)
    # immediately create object-level lock
    obj.send(:__lock)

    # redefine all user-defined methods to utilize lock
    obj.methods.each do |m|
      next if IGNORABLE_METHOD_OWNERS.include?(obj.method(m).owner)
      obj.define_singleton_method(m) do |*args, &block|
        __lock.synchronize do
          super(*args, &block)
        end
      end
    end

    # define synchronize method that executes a block
    # protected with the internal instance lock
    obj.define_singleton_method(:synchronize) do |&block|
      __lock.synchronize(&block)
    end
  end

  private

  def __lock
    # Monitor is used instead of Mutex in order to support
    # re-entrant locking
    @__lock ||= Monitor.new
  end
end

Version data entries

1 entries across 1 versions & 1 rubygems

Version Path
synchronizable-0.0.2 lib/synchronizable.rb