2007-12-21 01:48:59 -06:00
|
|
|
require 'optparse'
|
|
|
|
require 'action_controller/integration'
|
|
|
|
|
|
|
|
module ActionController
|
|
|
|
class RequestProfiler
|
|
|
|
# Wrap up the integration session runner.
|
|
|
|
class Sandbox
|
|
|
|
include Integration::Runner
|
|
|
|
|
|
|
|
def self.benchmark(n, script)
|
|
|
|
new(script).benchmark(n)
|
|
|
|
end
|
|
|
|
|
|
|
|
def initialize(script_path)
|
|
|
|
@quiet = false
|
2008-05-17 23:22:34 -05:00
|
|
|
define_run_method(script_path)
|
2007-12-21 01:48:59 -06:00
|
|
|
reset!
|
|
|
|
end
|
|
|
|
|
2008-06-02 01:35:38 -05:00
|
|
|
def benchmark(n, profiling = false)
|
2007-12-21 01:48:59 -06:00
|
|
|
@quiet = true
|
|
|
|
print ' '
|
2008-05-17 23:22:34 -05:00
|
|
|
|
2007-12-21 01:48:59 -06:00
|
|
|
result = Benchmark.realtime do
|
|
|
|
n.times do |i|
|
2008-06-02 01:35:38 -05:00
|
|
|
run(profiling)
|
2008-05-17 23:22:34 -05:00
|
|
|
print_progress(i)
|
2007-12-21 01:48:59 -06:00
|
|
|
end
|
|
|
|
end
|
2008-05-17 23:22:34 -05:00
|
|
|
|
2007-12-21 01:48:59 -06:00
|
|
|
puts
|
|
|
|
result
|
|
|
|
ensure
|
|
|
|
@quiet = false
|
|
|
|
end
|
|
|
|
|
|
|
|
def say(message)
|
|
|
|
puts " #{message}" unless @quiet
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
2008-05-17 23:22:34 -05:00
|
|
|
def define_run_method(script_path)
|
|
|
|
script = File.read(script_path)
|
|
|
|
|
|
|
|
source = <<-end_source
|
2008-06-02 01:35:38 -05:00
|
|
|
def run(profiling = false)
|
|
|
|
if profiling
|
|
|
|
RubyProf.resume do
|
|
|
|
#{script}
|
|
|
|
end
|
|
|
|
else
|
|
|
|
#{script}
|
|
|
|
end
|
|
|
|
|
2008-05-17 23:22:34 -05:00
|
|
|
old_request_count = request_count
|
|
|
|
reset!
|
|
|
|
self.request_count = old_request_count
|
|
|
|
end
|
|
|
|
end_source
|
|
|
|
|
|
|
|
instance_eval source, script_path, 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def print_progress(i)
|
|
|
|
print "\n " if i % 60 == 0
|
|
|
|
print ' ' if i % 10 == 0
|
|
|
|
print '.'
|
|
|
|
$stdout.flush
|
2007-12-21 01:48:59 -06:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
attr_reader :options
|
|
|
|
|
|
|
|
def initialize(options = {})
|
|
|
|
@options = default_options.merge(options)
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
def self.run(args = nil, options = {})
|
|
|
|
profiler = new(options)
|
|
|
|
profiler.parse_options(args) if args
|
|
|
|
profiler.run
|
|
|
|
end
|
|
|
|
|
|
|
|
def run
|
|
|
|
sandbox = Sandbox.new(options[:script])
|
|
|
|
|
|
|
|
puts 'Warming up once'
|
|
|
|
|
|
|
|
elapsed = warmup(sandbox)
|
|
|
|
puts '%.2f sec, %d requests, %d req/sec' % [elapsed, sandbox.request_count, sandbox.request_count / elapsed]
|
|
|
|
puts "\n#{options[:benchmark] ? 'Benchmarking' : 'Profiling'} #{options[:n]}x"
|
|
|
|
|
|
|
|
options[:benchmark] ? benchmark(sandbox) : profile(sandbox)
|
|
|
|
end
|
|
|
|
|
|
|
|
def profile(sandbox)
|
|
|
|
load_ruby_prof
|
|
|
|
|
2008-06-02 01:35:38 -05:00
|
|
|
benchmark(sandbox, true)
|
|
|
|
results = RubyProf.stop
|
2007-12-21 01:48:59 -06:00
|
|
|
|
|
|
|
show_profile_results results
|
|
|
|
results
|
|
|
|
end
|
|
|
|
|
2008-06-02 01:35:38 -05:00
|
|
|
def benchmark(sandbox, profiling = false)
|
2007-12-21 01:48:59 -06:00
|
|
|
sandbox.request_count = 0
|
2008-06-02 01:35:38 -05:00
|
|
|
elapsed = sandbox.benchmark(options[:n], profiling).to_f
|
2007-12-21 01:48:59 -06:00
|
|
|
count = sandbox.request_count.to_i
|
|
|
|
puts '%.2f sec, %d requests, %d req/sec' % [elapsed, count, count / elapsed]
|
|
|
|
end
|
|
|
|
|
|
|
|
def warmup(sandbox)
|
2008-06-02 01:35:38 -05:00
|
|
|
Benchmark.realtime { sandbox.run(false) }
|
2007-12-21 01:48:59 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
def default_options
|
|
|
|
{ :n => 100, :open => 'open %s &' }
|
|
|
|
end
|
|
|
|
|
|
|
|
# Parse command-line options
|
|
|
|
def parse_options(args)
|
|
|
|
OptionParser.new do |opt|
|
|
|
|
opt.banner = "USAGE: #{$0} [options] [session script path]"
|
|
|
|
|
2008-05-17 23:22:34 -05:00
|
|
|
opt.on('-n', '--times [100]', 'How many requests to process. Defaults to 100.') { |v| options[:n] = v.to_i if v }
|
2007-12-21 01:48:59 -06:00
|
|
|
opt.on('-b', '--benchmark', 'Benchmark instead of profiling') { |v| options[:benchmark] = v }
|
2008-05-17 23:22:34 -05:00
|
|
|
opt.on('-m', '--measure [mode]', 'Which ruby-prof measure mode to use: process_time, wall_time, cpu_time, allocations, or memory. Defaults to process_time.') { |v| options[:measure] = v }
|
2007-12-21 01:48:59 -06:00
|
|
|
opt.on('--open [CMD]', 'Command to open profile results. Defaults to "open %s &"') { |v| options[:open] = v }
|
|
|
|
opt.on('-h', '--help', 'Show this help') { puts opt; exit }
|
|
|
|
|
|
|
|
opt.parse args
|
|
|
|
|
|
|
|
if args.empty?
|
|
|
|
puts opt
|
|
|
|
exit
|
|
|
|
end
|
|
|
|
options[:script] = args.pop
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
def load_ruby_prof
|
|
|
|
begin
|
2008-06-02 01:35:38 -05:00
|
|
|
gem 'ruby-prof', '>= 0.6.1'
|
2007-12-21 01:48:59 -06:00
|
|
|
require 'ruby-prof'
|
2008-05-17 23:22:34 -05:00
|
|
|
if mode = options[:measure]
|
|
|
|
RubyProf.measure_mode = RubyProf.const_get(mode.upcase)
|
|
|
|
end
|
2007-12-21 01:48:59 -06:00
|
|
|
rescue LoadError
|
|
|
|
abort '`gem install ruby-prof` to use the profiler'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def show_profile_results(results)
|
|
|
|
File.open "#{RAILS_ROOT}/tmp/profile-graph.html", 'w' do |file|
|
|
|
|
RubyProf::GraphHtmlPrinter.new(results).print(file)
|
|
|
|
`#{options[:open] % file.path}` if options[:open]
|
|
|
|
end
|
|
|
|
|
|
|
|
File.open "#{RAILS_ROOT}/tmp/profile-flat.txt", 'w' do |file|
|
|
|
|
RubyProf::FlatPrinter.new(results).print(file)
|
|
|
|
`#{options[:open] % file.path}` if options[:open]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|