GOLF (and improve) #instance_interception

T

transfire

I have been working on this and was hoping maybe others would like to
take some wacks at it. It's a general purpose tool for prepending a
module in the inheritance chain. Here's an example of it's use. (code
follows):

module A
def f ; "F" ; end
def g ; "G" ; end

instance_interception do
def f( target, *args, &blk )
'{' + target.super + '}'
end
def g( target, *args, &blk )
'{' + target.super + '}'
end
end
end

class X
def f ; super ; end
include A
def g ; super ; end
end

x = X.new
x.f #=> "{F}"
x.g #=> "#{G}"


# instance_interception.rb

class Module

def instance_interception(&block)
@instance_interception ||= Module.new do
def self.append_features(mod)
append_features_without_instance_interception( mod )
end
end
@instance_interception.module_eval(&block) if block_given?
@instance_interception
end

private :instance_interception

alias_method :append_features_without_instance_interception,
:append_features

# Append features

def append_features( mod )

aspect = instance_interception
aspect.__send__( :append_features_without_instance_interception,
mod )

aspect.instance_methods.each do |meth|
if mod.method_defined?( meth )
aspect.advise( mod, meth )
end
end

append_features_without_instance_interception( mod )

#if mod.instance_of? Module
aspect.__send__( :append_features_without_instance_interception,
mod.__send__:)instance_interception) )
#end

end

# Apply the around advice.

def advise( mod, meth )
advice = instance_method( meth )
instance_target = mod.instance_method(meth)
mod.__send__( :define_method, meth ) { |*args| #, &blk|
target = instance_target.bind( self )
(class << target; self; end).class_eval { define_method( :super
){ call( *args ) } }
advice.bind( self ).call( target, *args ) #, &blk )
}
end

# If a method is added to the module/class that is advised.

def method_added( meth )
return if @method_added_short
if instance_interception.method_defined?( meth )
include instance_interception
@method_added_short = true
instance_interception.advise( self, meth )
@method_added_short = false
end
end

end
 
D

dblack

Hi --

I have been working on this and was hoping maybe others would like to
take some wacks at it. It's a general purpose tool for prepending a
module in the inheritance chain. Here's an example of it's use. (code
follows):

module A
def f ; "F" ; end
def g ; "G" ; end

Well, the golf part is easy, but I don't think it looks very good:

module A;def f;"F";end;def g;"G";end;end

Are you sure you want to golf this? :)


David

--
http://www.rubypowerandlight.com => Ruby/Rails training & consultancy
http://www.manning.com/black => RUBY FOR RAILS (reviewed on
Slashdot, 7/12/2006!)
http://dablog.rubypal.com => D[avid ]A[. ]B[lack's][ Web]log
(e-mail address removed) => me
 
T

transfire

Hi --



Well, the golf part is easy, but I don't think it looks very good:

module A;def f;"F";end;def g;"G";end;end

Are you sure you want to golf this? :)

Well, okay. I guess golf isn't exactly what I mean. Just *improve"
then. Thanks.

BTW, typo in example:

x.g #=> "#{G}"

should be

x.g #=> "{G}"

T.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
473,770
Messages
2,569,583
Members
45,073
Latest member
DarinCeden

Latest Threads

Top