FIX: stop double counting net calls in logs

This commit is contained in:
Sam 2018-02-28 10:45:11 +11:00
parent 2a7b7add59
commit f295a18e94
3 changed files with 29 additions and 2 deletions

View File

@ -1,7 +1,16 @@
# see https://samsaffron.com/archive/2017/10/18/fastest-way-to-profile-a-method-in-ruby # see https://samsaffron.com/archive/2017/10/18/fastest-way-to-profile-a-method-in-ruby
class MethodProfiler class MethodProfiler
def self.patch(klass, methods, name) def self.patch(klass, methods, name, no_recurse: false)
patches = methods.map do |method_name| patches = methods.map do |method_name|
recurse_protection = ""
if no_recurse
recurse_protection = <<~RUBY
return #{method_name}__mp_unpatched(*args, &blk) if @mp_recurse_protect_#{method_name}
@mp_recurse_protect_#{method_name} = true
RUBY
end
<<~RUBY <<~RUBY
unless defined?(#{method_name}__mp_unpatched) unless defined?(#{method_name}__mp_unpatched)
alias_method :#{method_name}__mp_unpatched, :#{method_name} alias_method :#{method_name}__mp_unpatched, :#{method_name}
@ -9,6 +18,7 @@ class MethodProfiler
unless prof = Thread.current[:_method_profiler] unless prof = Thread.current[:_method_profiler]
return #{method_name}__mp_unpatched(*args, &blk) return #{method_name}__mp_unpatched(*args, &blk)
end end
#{recurse_protection}
begin begin
start = Process.clock_gettime(Process::CLOCK_MONOTONIC) start = Process.clock_gettime(Process::CLOCK_MONOTONIC)
#{method_name}__mp_unpatched(*args, &blk) #{method_name}__mp_unpatched(*args, &blk)
@ -16,6 +26,7 @@ class MethodProfiler
data = (prof[:#{name}] ||= {duration: 0.0, calls: 0}) data = (prof[:#{name}] ||= {duration: 0.0, calls: 0})
data[:duration] += Process.clock_gettime(Process::CLOCK_MONOTONIC) - start data[:duration] += Process.clock_gettime(Process::CLOCK_MONOTONIC) - start
data[:calls] += 1 data[:calls] += 1
#{"@mp_recurse_protect_#{method_name} = false" if no_recurse}
end end
end end
end end

View File

@ -27,7 +27,7 @@ class Middleware::RequestTracker
MethodProfiler.patch(Net::HTTP, [ MethodProfiler.patch(Net::HTTP, [
:request :request
], :net) ], :net, no_recurse: true)
MethodProfiler.patch(Excon::Connection, [ MethodProfiler.patch(Excon::Connection, [
:request :request

View File

@ -5,6 +5,22 @@ describe MethodProfiler do
class Sneetch class Sneetch
def beach def beach
end end
def recurse(count = 5)
if count > 0
recurse(count - 1)
end
end
end
it "can bypass recursion on demand" do
MethodProfiler.patch(Sneetch, [:recurse], :recurse, no_recurse: true)
MethodProfiler.start
Sneetch.new.recurse
result = MethodProfiler.stop
expect(result[:recurse][:calls]).to eq(1)
end end
it "can transfer data between threads" do it "can transfer data between threads" do