* lib/ostruct.rb: a patch from Florian Gross <florgro@gmail.com>
merged to allow recursive inspect (and to_s) for OpenStruct. [ruby-core:05532] * lib/observer.rb: a patch from nornagon <nornagon@gmail.com> merged to allow arbitrary names for update methods. [ruby-core:05416] * eval.c (rb_f_fcall): new method to avoid inefficiency of obj.instance_eval{send(...)} tricks. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@9081 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
This commit is contained in:
parent
f5ac36f1a3
commit
48653d5ef0
@ -33,10 +33,9 @@ module Find
|
|||||||
# See the +Find+ module documentation for an example.
|
# See the +Find+ module documentation for an example.
|
||||||
#
|
#
|
||||||
def find(*paths) # :yield: path
|
def find(*paths) # :yield: path
|
||||||
paths.collect!{|d| d.dup}
|
paths.collect!{|d| open(d){}; d.dup}
|
||||||
while file = paths.shift
|
while file = paths.shift
|
||||||
catch(:prune) do
|
catch(:prune) do
|
||||||
next unless File.exist? file
|
|
||||||
yield file.dup.taint
|
yield file.dup.taint
|
||||||
begin
|
begin
|
||||||
if File.lstat(file).directory? then
|
if File.lstat(file).directory? then
|
||||||
|
@ -87,11 +87,11 @@ module MonitorMixin
|
|||||||
class Timeout < Exception; end
|
class Timeout < Exception; end
|
||||||
|
|
||||||
def wait(timeout = nil)
|
def wait(timeout = nil)
|
||||||
@monitor.instance_eval {mon_check_owner()}
|
@monitor.fcall(:mon_check_owner)
|
||||||
timer = create_timer(timeout)
|
timer = create_timer(timeout)
|
||||||
|
|
||||||
Thread.critical = true
|
Thread.critical = true
|
||||||
count = @monitor.instance_eval {mon_exit_for_cond()}
|
count = @monitor.fcall(:mon_exit_for_cond)
|
||||||
@waiters.push(Thread.current)
|
@waiters.push(Thread.current)
|
||||||
|
|
||||||
begin
|
begin
|
||||||
@ -107,7 +107,7 @@ module MonitorMixin
|
|||||||
if @waiters.include?(Thread.current) # interrupted?
|
if @waiters.include?(Thread.current) # interrupted?
|
||||||
@waiters.delete(Thread.current)
|
@waiters.delete(Thread.current)
|
||||||
end
|
end
|
||||||
@monitor.instance_eval {mon_enter_for_cond(count)}
|
@monitor.fcall(:mon_enter_for_cond, count)
|
||||||
Thread.critical = false
|
Thread.critical = false
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -125,7 +125,7 @@ module MonitorMixin
|
|||||||
end
|
end
|
||||||
|
|
||||||
def signal
|
def signal
|
||||||
@monitor.instance_eval {mon_check_owner()}
|
@monitor.fcall(:mon_check_owner)
|
||||||
Thread.critical = true
|
Thread.critical = true
|
||||||
t = @waiters.shift
|
t = @waiters.shift
|
||||||
t.wakeup if t
|
t.wakeup if t
|
||||||
@ -134,7 +134,7 @@ module MonitorMixin
|
|||||||
end
|
end
|
||||||
|
|
||||||
def broadcast
|
def broadcast
|
||||||
@monitor.instance_eval {mon_check_owner()}
|
@monitor.fcall(:mon_check_owner)
|
||||||
Thread.critical = true
|
Thread.critical = true
|
||||||
for t in @waiters
|
for t in @waiters
|
||||||
t.wakeup
|
t.wakeup
|
||||||
@ -172,7 +172,7 @@ module MonitorMixin
|
|||||||
|
|
||||||
def self.extend_object(obj)
|
def self.extend_object(obj)
|
||||||
super(obj)
|
super(obj)
|
||||||
obj.instance_eval {mon_initialize()}
|
obj.fcall(:mon_initialize)
|
||||||
end
|
end
|
||||||
|
|
||||||
#
|
#
|
||||||
|
@ -118,14 +118,15 @@ module Observable
|
|||||||
|
|
||||||
#
|
#
|
||||||
# Add +observer+ as an observer on this object. +observer+ will now receive
|
# Add +observer+ as an observer on this object. +observer+ will now receive
|
||||||
# notifications.
|
# notifications. The second optional argument specifies a method to notify
|
||||||
|
# updates, of which default value is +update+.
|
||||||
#
|
#
|
||||||
def add_observer(observer)
|
def add_observer(observer, func=:update)
|
||||||
@observer_peers = [] unless defined? @observer_peers
|
@observer_peers = {} unless defined? @observer_peers
|
||||||
unless observer.respond_to? :update
|
unless observer.respond_to? func
|
||||||
raise NoMethodError, "observer needs to respond to `update'"
|
raise NoMethodError, "observer does not respond to `#{func.to_s}'"
|
||||||
end
|
end
|
||||||
@observer_peers.push observer
|
@observer_peers[observer] = func
|
||||||
end
|
end
|
||||||
|
|
||||||
#
|
#
|
||||||
@ -181,9 +182,9 @@ module Observable
|
|||||||
def notify_observers(*arg)
|
def notify_observers(*arg)
|
||||||
if defined? @observer_state and @observer_state
|
if defined? @observer_state and @observer_state
|
||||||
if defined? @observer_peers
|
if defined? @observer_peers
|
||||||
for i in @observer_peers.dup
|
@observer_peers.each { |k, v|
|
||||||
i.update(*arg)
|
k.send v, *arg
|
||||||
end
|
}
|
||||||
end
|
end
|
||||||
@observer_state = false
|
@observer_state = false
|
||||||
end
|
end
|
||||||
|
@ -47,7 +47,7 @@ class OpenStruct
|
|||||||
@table = {}
|
@table = {}
|
||||||
if hash
|
if hash
|
||||||
for k,v in hash
|
for k,v in hash
|
||||||
@table[k.to_sym] = v
|
@table[k.to_sym] = v
|
||||||
new_ostruct_member(k)
|
new_ostruct_member(k)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -68,11 +68,11 @@ class OpenStruct
|
|||||||
end
|
end
|
||||||
|
|
||||||
def new_ostruct_member(name)
|
def new_ostruct_member(name)
|
||||||
|
name = name.to_sym
|
||||||
unless self.respond_to?(name)
|
unless self.respond_to?(name)
|
||||||
self.instance_eval %{
|
meta = class << self; self; end
|
||||||
def #{name}; @table[:#{name}]; end
|
meta.send(:define_method, name) { @table[name] }
|
||||||
def #{name}=(x); @table[:#{name}] = x; end
|
meta.send(:define_method, :"#{name}=") { |x| @table[name] = x }
|
||||||
}
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -81,14 +81,14 @@ class OpenStruct
|
|||||||
len = args.length
|
len = args.length
|
||||||
if mname =~ /=$/
|
if mname =~ /=$/
|
||||||
if len != 1
|
if len != 1
|
||||||
raise ArgumentError, "wrong number of arguments (#{len} for 1)", caller(1)
|
raise ArgumentError, "wrong number of arguments (#{len} for 1)", caller(1)
|
||||||
end
|
end
|
||||||
if self.frozen?
|
if self.frozen?
|
||||||
raise TypeError, "can't modify frozen #{self.class}", caller(1)
|
raise TypeError, "can't modify frozen #{self.class}", caller(1)
|
||||||
end
|
end
|
||||||
mname.chop!
|
mname.chop!
|
||||||
@table[mname.intern] = args[0]
|
|
||||||
self.new_ostruct_member(mname)
|
self.new_ostruct_member(mname)
|
||||||
|
@table[mname.intern] = args[0]
|
||||||
elsif len == 0
|
elsif len == 0
|
||||||
@table[mid]
|
@table[mid]
|
||||||
else
|
else
|
||||||
@ -103,16 +103,35 @@ class OpenStruct
|
|||||||
@table.delete name.to_sym
|
@table.delete name.to_sym
|
||||||
end
|
end
|
||||||
|
|
||||||
|
InspectKey = :__inspect_key__ # :nodoc:
|
||||||
|
|
||||||
#
|
#
|
||||||
# Returns a string containing a detailed summary of the keys and values.
|
# Returns a string containing a detailed summary of the keys and values.
|
||||||
#
|
#
|
||||||
def inspect
|
def inspect
|
||||||
str = "<#{self.class}"
|
str = "#<#{self.class}"
|
||||||
for k,v in @table
|
|
||||||
str << " #{k}=#{v.inspect}"
|
Thread.current[InspectKey] ||= []
|
||||||
|
if Thread.current[InspectKey].include?(self) then
|
||||||
|
str << " ..."
|
||||||
|
else
|
||||||
|
first = true
|
||||||
|
for k,v in @table
|
||||||
|
str << "," unless first
|
||||||
|
first = false
|
||||||
|
|
||||||
|
Thread.current[InspectKey] << v
|
||||||
|
begin
|
||||||
|
str << " #{k}=#{v.inspect}"
|
||||||
|
ensure
|
||||||
|
Thread.current[InspectKey].pop
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
str << ">"
|
str << ">"
|
||||||
end
|
end
|
||||||
|
alias :to_s :inspect
|
||||||
|
|
||||||
attr_reader :table # :nodoc:
|
attr_reader :table # :nodoc:
|
||||||
protected :table
|
protected :table
|
||||||
|
Loading…
x
Reference in New Issue
Block a user