module Skylight # @api private class Middleware class BodyProxy def initialize(body, &block) @body, @block, @closed = body, block, false end def respond_to?(*args) return false if args.first.to_s =~ /^to_ary$/ super or @body.respond_to?(*args) end def close return if @closed @closed = true begin @body.close if @body.respond_to? :close ensure @block.call end end def closed? @closed end # N.B. This method is a special case to address the bug described by # https://github.com/rack/rack/issues/434. # We are applying this special case for #each only. Future bugs of this # class will be handled by requesting users to patch their ruby # implementation, to save adding too many methods in this class. def each(*args, &block) @body.each(*args, &block) end def method_missing(*args, &block) super if args.first.to_s =~ /^to_ary$/ @body.__send__(*args, &block) end end def self.with_after_close(resp, &block) # Responses should be arrays but in some situations they aren't # e.g. https://github.com/ruby-grape/grape/issues/1041 # The safest approach seems to be to rely on implicit destructuring # since that is currently what Rack::Lint does. # See also https://github.com/rack/rack/issues/1239 status, headers, body = resp [status, headers, BodyProxy.new(body, &block)] end include Util::Logging # For Util::Logging attr_reader :config def initialize(app, opts={}) @app = app @config = opts[:config] end def call(env) if Skylight.tracing? error "Already instrumenting. Make sure the Middleware hasn't been added more than once." end if env["REQUEST_METHOD"] == "HEAD" t { "middleware skipping HEAD" } @app.call(env) else begin t { "middleware beginning trace" } trace = Skylight.trace "Rack", 'app.rack.request' resp = @app.call(env) if trace Middleware.with_after_close(resp) { trace.submit } else resp end rescue Exception t { "middleware exception: #{trace}"} trace.submit if trace raise end end end end end