lib/rspec/mocks/instance_method_stasher.rb
module RSpec module Mocks # @private class InstanceMethodStasher def initialize(klass, method) @klass = klass @method = method @method_is_stashed = false end # @private def method_is_stashed? @method_is_stashed end # @private def stash return if !method_defined_directly_on_klass? || @method_is_stashed @klass.__send__(:alias_method, stashed_method_name, @method) @method_is_stashed = true end private # @private def method_defined_directly_on_klass? method_defined_on_klass? && method_owned_by_klass? end # @private def method_defined_on_klass?(klass = @klass) klass.method_defined?(@method) || klass.private_method_defined?(@method) end if ::UnboundMethod.method_defined?(:owner) # @private def method_owned_by_klass? owner = @klass.instance_method(@method).owner # On Ruby 2.0.0+ the owner of a method on a class which has been # `prepend`ed may actually be an instance, e.g. # `#<MyClass:0x007fbb94e3cd10>`, rather than the expected `MyClass`. owner = owner.class unless owner.is_a? Class # On 1.8 (and some 1.9s -- e.g. rubinius) aliased methods # can report the wrong owner. Example: # class MyClass # class << self # alias alternate_new new # end # end # # MyClass.owner(:alternate_new) returns `Class` on 1.8, # but we need to consider the owner to be `MyClass` because # it is not actually available on `Class` but is on `MyClass`. # Hence, we verify that the owner actually has the method defined. # If the given owner does not have the method defined, we assume # that the method is actually owned by @klass. owner == @klass || !(method_defined_on_klass?(owner)) end else # @private def method_owned_by_klass? # On 1.8.6, which does not support Method#owner, we have no choice but # to assume it's defined on the klass even if it may be defined on # a superclass. true end end public # @private def stashed_method_name "obfuscated_by_rspec_mocks__#{@method}" end # @private def restore return unless @method_is_stashed if @klass.__send__(:method_defined?, @method) @klass.__send__(:undef_method, @method) end @klass.__send__(:alias_method, @method, stashed_method_name) @klass.__send__(:remove_method, stashed_method_name) @method_is_stashed = false end end end end