# encoding: utf-8 require 'erb' module RubyProf # Generates graph[link:files/examples/graph_html.html] profile reports as html. # To use the graph html printer: # # result = RubyProf.profile do # [code to profile] # end # # printer = RubyProf::GraphHtmlPrinter.new(result) # printer.print(STDOUT, :min_percent=>0) # # The Graph printer takes the following options in its print methods: # :filename - specify a file to use that contains the ERB # template to use, instead of the built-in self.template # # :template - specify an ERB template to use, instead of the # built-in self.template # # :editor_uri - Specifies editor uri scheme used for opening files # e.g. :atm or :mvim. For OS X default is :txmt. # Use RUBY_PROF_EDITOR_URI environment variable to overide. class GraphHtmlPrinter < AbstractPrinter include ERB::Util def setup_options(options) super(options) filename = options[:filename] template = filename ? File.read(filename).untaint : (options[:template] || self.template) @erb = ERB.new(template) @erb.filename = filename end def print(output = STDOUT, options = {}) @output = output setup_options(options) @editor = editor_uri @output << @erb.result(binding).split("\n").map(&:rstrip).join("\n") << "\n" end # Creates a link to a method. Note that we do not create # links to methods which are under the min_perecent # specified by the user, since they will not be # printed out. def create_link(thread, overall_time, method) total_percent = (method.total_time/overall_time) * 100 if total_percent < min_percent # Just return name h method.full_name else href = '#' + method_href(thread, method) "#{h method.full_name}" end end def method_href(thread, method) h(method.full_name.gsub(/[><#\.\?=:]/,"_") + "_" + thread.fiber_id.to_s) end def file_link(path, linenum) srcfile = File.expand_path(path) if srcfile =~ /\/ruby_runtime$/ "" else if @editor "#{linenum}" else "#{linenum}" end end end def template '

Profile Report: <%= RubyProf.measure_mode_string %>

<% for thread in @result.threads %> <% end %>
Thread ID Fiber ID Total Time
<%= thread.id %> <%= thread.fiber_id %> <%= thread.total_time %>
<% for thread in @result.threads methods = thread.methods total_time = thread.total_time %>

Thread <%= thread.id %>, Fiber: <%= thread.fiber_id %>

<% min_time = @options[:min_time] || (@options[:nonzero] ? 0.005 : nil) methods.sort_by(&sort_method).reverse_each do |method| total_percentage = (method.total_time/total_time) * 100 next if total_percentage < min_percent next if min_time && method.total_time < min_time self_percentage = (method.self_time/total_time) * 100 %> <% for caller in method.aggregate_parents.sort_by(&:total_time) next unless caller.parent next if min_time && caller.total_time < min_time %> <% end %> <% for callee in method.aggregate_children.sort_by(&:total_time).reverse next if min_time && callee.total_time < min_time %> <% end %> <% end %>
%Total %Self Total Self Wait Child Calls Name Line
    <%= sprintf("%.2f", caller.total_time) %> <%= sprintf("%.2f", caller.self_time) %> <%= sprintf("%.2f", caller.wait_time) %> <%= sprintf("%.2f", caller.children_time) %> <%= "#{caller.called}/#{method.called}" %> <%= create_link(thread, total_time, caller.parent.target) %> <%= file_link(caller.parent.target.source_file, caller.line) %>
<%= sprintf("%.2f%%", total_percentage) %> <%= sprintf("%.2f%%", self_percentage) %> <%= sprintf("%.2f", method.total_time) %> <%= sprintf("%.2f", method.self_time) %> <%= sprintf("%.2f", method.wait_time) %> <%= sprintf("%.2f", method.children_time) %> <%= sprintf("%i", method.called) %> <%= method.recursive? ? "*" : " "%><%= h method.full_name %> <%= file_link(method.source_file, method.line) %>
    <%= sprintf("%.2f", callee.total_time) %> <%= sprintf("%.2f", callee.self_time) %> <%= sprintf("%.2f", callee.wait_time) %> <%= sprintf("%.2f", callee.children_time) %> <%= "#{callee.called}/#{callee.target.called}" %> <%= create_link(thread, total_time, callee.target) %> <%= file_link(method.source_file, callee.line) %>
* indicates recursively called methods
<% end %> ' end end end