aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAaron Patterson <aaron.patterson@gmail.com>2012-03-21 13:06:56 -0700
committerAaron Patterson <aaron.patterson@gmail.com>2012-03-21 16:18:46 -0700
commitf08f8750a512f741acb004d0cebe210c5f949f28 (patch)
tree843cca4f460ef6652062e0f6d7b1227bdc764284
parent60736fec5397c331cc8791b884f3579ef29f4f09 (diff)
downloadrails-f08f8750a512f741acb004d0cebe210c5f949f28.tar.gz
rails-f08f8750a512f741acb004d0cebe210c5f949f28.tar.bz2
rails-f08f8750a512f741acb004d0cebe210c5f949f28.zip
start / finish events are sent by the instrumenter
-rw-r--r--activesupport/lib/active_support/notifications/fanout.rb42
-rw-r--r--activesupport/lib/active_support/notifications/instrumenter.rb6
2 files changed, 41 insertions, 7 deletions
diff --git a/activesupport/lib/active_support/notifications/fanout.rb b/activesupport/lib/active_support/notifications/fanout.rb
index bb1111e0c2..8ecafd2b2b 100644
--- a/activesupport/lib/active_support/notifications/fanout.rb
+++ b/activesupport/lib/active_support/notifications/fanout.rb
@@ -20,6 +20,14 @@ module ActiveSupport
@listeners_for.clear
end
+ def start(name, id, payload)
+ listeners_for(name).each { |s| s.start(name, id, payload) }
+ end
+
+ def finish(name, id, payload)
+ listeners_for(name).each { |s| s.finish(name, id, payload) }
+ end
+
def publish(name, *args)
listeners_for(name).each { |s| s.publish(name, *args) }
end
@@ -39,7 +47,7 @@ module ActiveSupport
module Subscribers # :nodoc:
def self.new(pattern, block)
if pattern
- Subscriber.new pattern, block
+ TimedSubscriber.new pattern, block
else
AllMessages.new pattern, block
end
@@ -51,8 +59,12 @@ module ActiveSupport
@delegate = delegate
end
- def publish(message, *args)
- @delegate.call(message, *args)
+ def start(name, id, payload)
+ raise NotImplementedError
+ end
+
+ def finish(name, id, payload)
+ raise NotImplementedError
end
def subscribed_to?(name)
@@ -65,7 +77,29 @@ module ActiveSupport
end
end
- class AllMessages < Subscriber # :nodoc:
+ class TimedSubscriber < Subscriber
+ def initialize(pattern, delegate)
+ @timestack = Hash.new { |h,id|
+ h[id] = Hash.new { |ids,name| ids[name] = [] }
+ }
+ super
+ end
+
+ def publish(name, *args)
+ @delegate.call name, *args
+ end
+
+ def start(name, id, payload)
+ @timestack[id][name].push Time.now
+ end
+
+ def finish(name, id, payload)
+ started = @timestack[id][name].pop
+ @delegate.call(name, started, Time.now, id, payload)
+ end
+ end
+
+ class AllMessages < TimedSubscriber # :nodoc:
def subscribed_to?(name)
true
end
diff --git a/activesupport/lib/active_support/notifications/instrumenter.rb b/activesupport/lib/active_support/notifications/instrumenter.rb
index 547df5c731..58e292c658 100644
--- a/activesupport/lib/active_support/notifications/instrumenter.rb
+++ b/activesupport/lib/active_support/notifications/instrumenter.rb
@@ -1,5 +1,6 @@
module ActiveSupport
module Notifications
+ # Instrumentors are stored in a thread local.
class Instrumenter
attr_reader :id
@@ -12,15 +13,14 @@ module ActiveSupport
# and publish it. Notice that events get sent even if an error occurs
# in the passed-in block
def instrument(name, payload={})
- started = Time.now
-
+ @notifier.start(name, @id, payload)
begin
yield
rescue Exception => e
payload[:exception] = [e.class.name, e.message]
raise e
ensure
- @notifier.publish(name, started, Time.now, @id, payload)
+ @notifier.finish(name, @id, payload)
end
end