# encoding: utf-8 require 'tmpdir' module Rack class RubyProf def initialize(app, options = {}) @app = app @options = options @tmpdir = options[:path] || Dir.tmpdir FileUtils.mkdir_p(@tmpdir) @printer_klasses = @options[:printers] || {::RubyProf::FlatPrinter => 'flat.txt', ::RubyProf::GraphPrinter => 'graph.txt', ::RubyProf::GraphHtmlPrinter => 'graph.html', ::RubyProf::CallStackPrinter => 'call_stack.html'} @skip_paths = options[:skip_paths] || [%r{^/assets}, %r{\.(css|js|png|jpeg|jpg|gif)$}] @only_paths = options[:only_paths] end def call(env) request = Rack::Request.new(env) if should_profile?(request.path) begin result = nil profile = ::RubyProf::Profile.profile(profiling_options) do result = @app.call(env) end if @options[:merge_fibers] profile.merge! end path = request.path.gsub('/', '-') path.slice!(0) print(profile, path) result end else @app.call(env) end end private def should_profile?(path) return false if paths_match?(path, @skip_paths) @only_paths ? paths_match?(path, @only_paths) : true end def paths_match?(path, paths) paths.any? { |skip_path| skip_path =~ path } end def profiling_options result = {} result[:measure_mode] = @options[:measure_mode] || ::RubyProf::WALL_TIME result[:track_allocations] = @options[:track_allocations] || false result[:exclude_common] = @options[:exclude_common] || false if @options[:ignore_existing_threads] result[:exclude_threads] = Thread.list.select {|thread| thread != Thread.current} end if @options[:request_thread_only] result[:include_threads] = [Thread.current] end result end def print_options result = {} result[:min_percent] = @options[:min_percent] || 1 result[:sort_method] = @options[:sort_method] || :total_time result end def print(profile, path) @printer_klasses.each do |printer_klass, base_name| printer = printer_klass.new(profile) if base_name.respond_to?(:call) base_name = base_name.call end if printer_klass == ::RubyProf::MultiPrinter printer.print(print_options.merge(:profile => "#{path}-#{base_name}")) elsif printer_klass == ::RubyProf::CallTreePrinter printer.print(print_options.merge(:profile => "#{path}-#{base_name}")) else file_name = ::File.join(@tmpdir, "#{path}-#{base_name}") ::File.open(file_name, 'wb') do |file| printer.print(file, print_options) end end end end end end