At the top-level a def adds a private method to Object.
I can think of three ways to get the top-level function:
(1) Use send to invoke the private method on Object itself (only works if the method is not a mutator since Object will be the receiver)
Object.send(:fn)
(2) Get a Method instance of the top-level method and bind it to the instance you want to invoke it on:
class Bar
def fn
Object.instance_method(:fn).bind(self).call
end
end
(3) Use super (assumes no super classes of Bar below Object redefine the function)
class Bar
def fn
super
end
end
UPDATE:
Since solution (2) is the preferable one (in my opinion) we can try to improve the syntax by defining a utility method on Object called super_method:
class Object
def super_method(base, meth, *args, &block)
if !self.kind_of?(base)
raise ArgumentError, "#{base} is not a superclass of #{self}"
end
base.instance_method(meth).bind(self).call(*args, &block)
end
end
Use like the following:
class Bar
def fn
super_method Object, :fn
end
end
Where the first argument to super_method must be a valid superclass of Bar, the second argument the method you want to invoke, and all remaining arguments (if any) are passed along as parameters to the selected method.