# Some based on : https://gist.github.com/277289 # # This is a rudimentary script that allows us to # quickly determine if any gems are slowing down startup require 'benchmark' require 'fileutils' module RequireProfiler class << self attr_accessor :stats def profiling_enabled? @profiling_enabled end def profile start yield stop end def start(tmp_options={}) @start_time = Time.now [ ::Kernel, (class << ::Kernel; self; end) ].each do |klass| klass.class_eval do def require_with_profiling(path, *args) RequireProfiler.measure(path, caller, :require) { require_without_profiling(path, *args) } end alias require_without_profiling require alias require require_with_profiling def load_with_profiling(path, *args) RequireProfiler.measure(path, caller, :load) { load_without_profiling(path, *args) } end alias load_without_profiling load alias load load_with_profiling end end # This is necessary so we don't clobber Bundler.require on Rails 3 Kernel.class_eval { private :require, :load } @profiling_enabled = true end def stop @stop_time = Time.now [ ::Kernel, (class << ::Kernel; self; end) ].each do |klass| klass.class_eval do alias require require_without_profiling alias load load_without_profiling end end @profiling_enabled = false end def measure(path, full_backtrace, mechanism, &block) # Path may be a Pathname, convert to a String path = path.to_s @stack ||= [] self.stats ||= {} stat = self.stats.fetch(path){|key| self.stats[key] = {calls: 0, time: 0, parent_time: 0} } @stack << stat time = Time.now begin output = yield # do the require or load here ensure delta = Time.now - time stat[:time] += delta stat[:calls] += 1 @stack.pop @stack.each do |frame| frame[:parent_time] += delta end end output end def time_block start = Time.now yield Time.now - start end def gc_analyze ObjectSpace.garbage_collect gc_duration_start = time_block { ObjectSpace.garbage_collect } old_objs = ObjectSpace.count_objects yield ObjectSpace.garbage_collect gc_duration_finish = time_block { ObjectSpace.garbage_collect } new_objs = ObjectSpace.count_objects puts "New objects: #{(new_objs[:TOTAL] - new_objs[:FREE]) - (old_objs[:TOTAL] - old_objs[:FREE])}" puts "GC duration: #{gc_duration_finish}" puts "GC impact: #{gc_duration_finish - gc_duration_start}" end end end # RequireProfiler.gc_analyze do # # require 'mime-types' # require 'highline' # end # exit RequireProfiler.profile do Bundler.definition.dependencies.each do |dep| begin require dep.name rescue Exception # don't care end end end sorted = RequireProfiler.stats.to_a.sort{|a,b| b[1][:time] - b[1][:parent_time] <=> a[1][:time] - a[1][:parent_time]} sorted[0..120].each do |k, v| puts "#{k} : time #{v[:time] - v[:parent_time]} " end