stackprof 0.2.12 → 0.2.26
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/.github/workflows/ci.yml +43 -0
- data/.gitignore +1 -1
- data/CHANGELOG.md +17 -2
- data/README.md +66 -51
- data/Rakefile +21 -25
- data/bin/stackprof +115 -71
- data/ext/stackprof/extconf.rb +6 -0
- data/ext/stackprof/stackprof.c +392 -84
- data/lib/stackprof/autorun.rb +19 -0
- data/lib/stackprof/middleware.rb +8 -2
- data/lib/stackprof/report.rb +280 -16
- data/lib/stackprof/truffleruby.rb +37 -0
- data/lib/stackprof.rb +22 -1
- data/stackprof.gemspec +11 -3
- data/test/fixtures/profile.dump +1 -0
- data/test/fixtures/profile.json +1 -0
- data/test/test_middleware.rb +36 -17
- data/test/test_report.rb +25 -1
- data/test/test_stackprof.rb +153 -15
- data/test/test_truffleruby.rb +18 -0
- data/vendor/FlameGraph/flamegraph.pl +751 -85
- metadata +16 -23
- data/.travis.yml +0 -8
- data/Gemfile.lock +0 -27
| @@ -0,0 +1,19 @@ | |
| 1 | 
            +
            require "stackprof"
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            options = {}
         | 
| 4 | 
            +
            options[:mode] = ENV["STACKPROF_MODE"].to_sym if ENV.key?("STACKPROF_MODE")
         | 
| 5 | 
            +
            options[:interval] = Integer(ENV["STACKPROF_INTERVAL"]) if ENV.key?("STACKPROF_INTERVAL")
         | 
| 6 | 
            +
            options[:raw] = true if ENV["STACKPROF_RAW"]
         | 
| 7 | 
            +
            options[:ignore_gc] = true if ENV["STACKPROF_IGNORE_GC"]
         | 
| 8 | 
            +
             | 
| 9 | 
            +
            at_exit do
         | 
| 10 | 
            +
              StackProf.stop
         | 
| 11 | 
            +
              output_path = ENV.fetch("STACKPROF_OUT") do
         | 
| 12 | 
            +
                require "tempfile"
         | 
| 13 | 
            +
                Tempfile.create(["stackprof", ".dump"]).path
         | 
| 14 | 
            +
              end
         | 
| 15 | 
            +
              StackProf.results(output_path)
         | 
| 16 | 
            +
              $stderr.puts("StackProf results dumped at: #{output_path}")
         | 
| 17 | 
            +
            end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
            StackProf.start(**options)
         | 
    
        data/lib/stackprof/middleware.rb
    CHANGED
    
    | @@ -13,12 +13,18 @@ module StackProf | |
| 13 13 | 
             
                  Middleware.enabled  = options[:enabled]
         | 
| 14 14 | 
             
                  options[:path]      = 'tmp/' if options[:path].to_s.empty?
         | 
| 15 15 | 
             
                  Middleware.path     = options[:path]
         | 
| 16 | 
            +
                  Middleware.metadata = options[:metadata] || {}
         | 
| 16 17 | 
             
                  at_exit{ Middleware.save } if options[:save_at_exit]
         | 
| 17 18 | 
             
                end
         | 
| 18 19 |  | 
| 19 20 | 
             
                def call(env)
         | 
| 20 21 | 
             
                  enabled = Middleware.enabled?(env)
         | 
| 21 | 
            -
                  StackProf.start( | 
| 22 | 
            +
                  StackProf.start(
         | 
| 23 | 
            +
                    mode:     Middleware.mode,
         | 
| 24 | 
            +
                    interval: Middleware.interval,
         | 
| 25 | 
            +
                    raw:      Middleware.raw,
         | 
| 26 | 
            +
                    metadata: Middleware.metadata,
         | 
| 27 | 
            +
                  ) if enabled
         | 
| 22 28 | 
             
                  @app.call(env)
         | 
| 23 29 | 
             
                ensure
         | 
| 24 30 | 
             
                  if enabled
         | 
| @@ -31,7 +37,7 @@ module StackProf | |
| 31 37 | 
             
                end
         | 
| 32 38 |  | 
| 33 39 | 
             
                class << self
         | 
| 34 | 
            -
                  attr_accessor :enabled, :mode, :interval, :raw, :path
         | 
| 40 | 
            +
                  attr_accessor :enabled, :mode, :interval, :raw, :path, :metadata
         | 
| 35 41 |  | 
| 36 42 | 
             
                  def enabled?(env)
         | 
| 37 43 | 
             
                    if enabled.respond_to?(:call)
         | 
    
        data/lib/stackprof/report.rb
    CHANGED
    
    | @@ -1,8 +1,44 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 1 3 | 
             
            require 'pp'
         | 
| 2 | 
            -
            require 'digest/ | 
| 4 | 
            +
            require 'digest/sha2'
         | 
| 5 | 
            +
            require 'json'
         | 
| 3 6 |  | 
| 4 7 | 
             
            module StackProf
         | 
| 5 8 | 
             
              class Report
         | 
| 9 | 
            +
                MARSHAL_SIGNATURE = "\x04\x08"
         | 
| 10 | 
            +
             | 
| 11 | 
            +
                class << self
         | 
| 12 | 
            +
                  def from_file(file)
         | 
| 13 | 
            +
                    if (content = IO.binread(file)).start_with?(MARSHAL_SIGNATURE)
         | 
| 14 | 
            +
                      new(Marshal.load(content))
         | 
| 15 | 
            +
                    else
         | 
| 16 | 
            +
                      from_json(JSON.parse(content))
         | 
| 17 | 
            +
                    end
         | 
| 18 | 
            +
                  end
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                  def from_json(json)
         | 
| 21 | 
            +
                    new(parse_json(json))
         | 
| 22 | 
            +
                  end
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                  def parse_json(json)
         | 
| 25 | 
            +
                    json.keys.each do |key|
         | 
| 26 | 
            +
                      value = json.delete(key)
         | 
| 27 | 
            +
                      from_json(value) if value.is_a?(Hash)
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                      new_key = case key
         | 
| 30 | 
            +
                      when /\A[0-9]*\z/
         | 
| 31 | 
            +
                        key.to_i
         | 
| 32 | 
            +
                      else
         | 
| 33 | 
            +
                        key.to_sym
         | 
| 34 | 
            +
                      end
         | 
| 35 | 
            +
             | 
| 36 | 
            +
                      json[new_key] = value
         | 
| 37 | 
            +
                    end
         | 
| 38 | 
            +
                    json
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
                end
         | 
| 41 | 
            +
             | 
| 6 42 | 
             
                def initialize(data)
         | 
| 7 43 | 
             
                  @data = data
         | 
| 8 44 | 
             
                end
         | 
| @@ -16,7 +52,7 @@ module StackProf | |
| 16 52 | 
             
                def normalized_frames
         | 
| 17 53 | 
             
                  id2hash = {}
         | 
| 18 54 | 
             
                  @data[:frames].each do |frame, info|
         | 
| 19 | 
            -
                    id2hash[frame.to_s] = info[:hash] = Digest:: | 
| 55 | 
            +
                    id2hash[frame.to_s] = info[:hash] = Digest::SHA256.hexdigest("#{info[:name]}#{info[:file]}#{info[:line]}")
         | 
| 20 56 | 
             
                  end
         | 
| 21 57 | 
             
                  @data[:frames].inject(Hash.new) do |hash, (frame, info)|
         | 
| 22 58 | 
             
                    info = hash[id2hash[frame.to_s]] = info.dup
         | 
| @@ -38,7 +74,7 @@ module StackProf | |
| 38 74 | 
             
                end
         | 
| 39 75 |  | 
| 40 76 | 
             
                def max_samples
         | 
| 41 | 
            -
                  @data[:max_samples] ||= frames.max_by{ | | 
| 77 | 
            +
                  @data[:max_samples] ||= @data[:frames].values.max_by{ |frame| frame[:samples] }[:samples]
         | 
| 42 78 | 
             
                end
         | 
| 43 79 |  | 
| 44 80 | 
             
                def files
         | 
| @@ -68,6 +104,11 @@ module StackProf | |
| 68 104 | 
             
                  f.puts Marshal.dump(@data.reject{|k,v| k == :files })
         | 
| 69 105 | 
             
                end
         | 
| 70 106 |  | 
| 107 | 
            +
                def print_json(f=STDOUT)
         | 
| 108 | 
            +
                  require "json"
         | 
| 109 | 
            +
                  f.puts JSON.generate(@data, max_nesting: false)
         | 
| 110 | 
            +
                end
         | 
| 111 | 
            +
             | 
| 71 112 | 
             
                def print_stackcollapse
         | 
| 72 113 | 
             
                  raise "profile does not include raw samples (add `raw: true` to collecting StackProf.run)" unless raw = data[:raw]
         | 
| 73 114 |  | 
| @@ -80,18 +121,20 @@ module StackProf | |
| 80 121 | 
             
                  end
         | 
| 81 122 | 
             
                end
         | 
| 82 123 |  | 
| 83 | 
            -
                def  | 
| 124 | 
            +
                def print_timeline_flamegraph(f=STDOUT, skip_common=true)
         | 
| 125 | 
            +
                  print_flamegraph(f, skip_common, false)
         | 
| 126 | 
            +
                end
         | 
| 127 | 
            +
             | 
| 128 | 
            +
                def print_alphabetical_flamegraph(f=STDOUT, skip_common=true)
         | 
| 129 | 
            +
                  print_flamegraph(f, skip_common, true)
         | 
| 130 | 
            +
                end
         | 
| 131 | 
            +
             | 
| 132 | 
            +
                def print_flamegraph(f, skip_common, alphabetical=false)
         | 
| 84 133 | 
             
                  raise "profile does not include raw samples (add `raw: true` to collecting StackProf.run)" unless raw = data[:raw]
         | 
| 85 134 |  | 
| 86 | 
            -
                  stacks =  | 
| 87 | 
            -
             | 
| 88 | 
            -
                   | 
| 89 | 
            -
                  while len = raw.shift
         | 
| 90 | 
            -
                    max_y = len if len > max_y
         | 
| 91 | 
            -
                    stack = raw.slice!(0, len+1)
         | 
| 92 | 
            -
                    stacks << stack
         | 
| 93 | 
            -
                    max_x += stack.last
         | 
| 94 | 
            -
                  end
         | 
| 135 | 
            +
                  stacks, max_x, max_y = flamegraph_stacks(raw)
         | 
| 136 | 
            +
             | 
| 137 | 
            +
                  stacks.sort! if alphabetical
         | 
| 95 138 |  | 
| 96 139 | 
             
                  f.puts 'flamegraph(['
         | 
| 97 140 | 
             
                  max_y.times do |y|
         | 
| @@ -141,13 +184,233 @@ module StackProf | |
| 141 184 | 
             
                  f.puts '])'
         | 
| 142 185 | 
             
                end
         | 
| 143 186 |  | 
| 187 | 
            +
                def flamegraph_stacks(raw)
         | 
| 188 | 
            +
                  stacks = []
         | 
| 189 | 
            +
                  max_x = 0
         | 
| 190 | 
            +
                  max_y = 0
         | 
| 191 | 
            +
                  idx = 0
         | 
| 192 | 
            +
             | 
| 193 | 
            +
                  while len = raw[idx]
         | 
| 194 | 
            +
                    idx += 1
         | 
| 195 | 
            +
                    max_y = len if len > max_y
         | 
| 196 | 
            +
                    stack = raw.slice(idx, len+1)
         | 
| 197 | 
            +
                    idx += len+1
         | 
| 198 | 
            +
                    stacks << stack
         | 
| 199 | 
            +
                    max_x += stack.last
         | 
| 200 | 
            +
                  end
         | 
| 201 | 
            +
             | 
| 202 | 
            +
                  return stacks, max_x, max_y
         | 
| 203 | 
            +
                end
         | 
| 204 | 
            +
             | 
| 144 205 | 
             
                def flamegraph_row(f, x, y, weight, addr)
         | 
| 145 | 
            -
                  frame = frames[addr]
         | 
| 206 | 
            +
                  frame = @data[:frames][addr]
         | 
| 146 207 | 
             
                  f.print ',' if @rows_started
         | 
| 147 208 | 
             
                  @rows_started = true
         | 
| 148 209 | 
             
                  f.puts %{{"x":#{x},"y":#{y},"width":#{weight},"frame_id":#{addr},"frame":#{frame[:name].dump},"file":#{frame[:file].dump}}}
         | 
| 149 210 | 
             
                end
         | 
| 150 211 |  | 
| 212 | 
            +
                def convert_to_d3_flame_graph_format(name, stacks, depth)
         | 
| 213 | 
            +
                  weight = 0
         | 
| 214 | 
            +
                  children = []
         | 
| 215 | 
            +
                  stacks.chunk do |stack|
         | 
| 216 | 
            +
                    if depth == stack.length - 1
         | 
| 217 | 
            +
                      :leaf
         | 
| 218 | 
            +
                    else
         | 
| 219 | 
            +
                      stack[depth]
         | 
| 220 | 
            +
                    end
         | 
| 221 | 
            +
                  end.each do |val, child_stacks|
         | 
| 222 | 
            +
                    if val == :leaf
         | 
| 223 | 
            +
                      child_stacks.each do |stack|
         | 
| 224 | 
            +
                        weight += stack.last
         | 
| 225 | 
            +
                      end
         | 
| 226 | 
            +
                    else
         | 
| 227 | 
            +
                      frame = @data[:frames][val]
         | 
| 228 | 
            +
                      child_name = "#{ frame[:name] } : #{ frame[:file] } : #{ frame[:line] }"
         | 
| 229 | 
            +
                      child_data = convert_to_d3_flame_graph_format(child_name, child_stacks, depth + 1)
         | 
| 230 | 
            +
                      weight += child_data["value"]
         | 
| 231 | 
            +
                      children << child_data
         | 
| 232 | 
            +
                    end
         | 
| 233 | 
            +
                  end
         | 
| 234 | 
            +
             | 
| 235 | 
            +
                  {
         | 
| 236 | 
            +
                    "name" => name,
         | 
| 237 | 
            +
                    "value" => weight,
         | 
| 238 | 
            +
                    "children" => children,
         | 
| 239 | 
            +
                  }
         | 
| 240 | 
            +
                end
         | 
| 241 | 
            +
             | 
| 242 | 
            +
                def print_d3_flamegraph(f=STDOUT, skip_common=true)
         | 
| 243 | 
            +
                  raise "profile does not include raw samples (add `raw: true` to collecting StackProf.run)" unless raw = data[:raw]
         | 
| 244 | 
            +
             | 
| 245 | 
            +
                  stacks, * = flamegraph_stacks(raw)
         | 
| 246 | 
            +
             | 
| 247 | 
            +
                  # d3-flame-grpah supports only alphabetical flamegraph
         | 
| 248 | 
            +
                  stacks.sort!
         | 
| 249 | 
            +
             | 
| 250 | 
            +
                  require "json"
         | 
| 251 | 
            +
                  json = JSON.generate(convert_to_d3_flame_graph_format("<root>", stacks, 0), max_nesting: false)
         | 
| 252 | 
            +
             | 
| 253 | 
            +
                  # This html code is almost copied from d3-flame-graph sample code.
         | 
| 254 | 
            +
                  # (Apache License 2.0)
         | 
| 255 | 
            +
                  # https://github.com/spiermar/d3-flame-graph/blob/gh-pages/index.html
         | 
| 256 | 
            +
             | 
| 257 | 
            +
                  f.print <<-END
         | 
| 258 | 
            +
            <!DOCTYPE html>
         | 
| 259 | 
            +
            <html lang="en">
         | 
| 260 | 
            +
              <head>
         | 
| 261 | 
            +
                <meta charset="utf-8">
         | 
| 262 | 
            +
                <meta http-equiv="X-UA-Compatible" content="IE=edge">
         | 
| 263 | 
            +
                <meta name="viewport" content="width=device-width, initial-scale=1">
         | 
| 264 | 
            +
             | 
| 265 | 
            +
                <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css">
         | 
| 266 | 
            +
                <link rel="stylesheet" type="text/css" href="https://cdn.jsdelivr.net/gh/spiermar/d3-flame-graph@2.0.3/dist/d3-flamegraph.css">
         | 
| 267 | 
            +
             | 
| 268 | 
            +
                <style>
         | 
| 269 | 
            +
             | 
| 270 | 
            +
                /* Space out content a bit */
         | 
| 271 | 
            +
                body {
         | 
| 272 | 
            +
                  padding-top: 20px;
         | 
| 273 | 
            +
                  padding-bottom: 20px;
         | 
| 274 | 
            +
                }
         | 
| 275 | 
            +
             | 
| 276 | 
            +
                /* Custom page header */
         | 
| 277 | 
            +
                .header {
         | 
| 278 | 
            +
                  padding-bottom: 20px;
         | 
| 279 | 
            +
                  padding-right: 15px;
         | 
| 280 | 
            +
                  padding-left: 15px;
         | 
| 281 | 
            +
                  border-bottom: 1px solid #e5e5e5;
         | 
| 282 | 
            +
                }
         | 
| 283 | 
            +
             | 
| 284 | 
            +
                /* Make the masthead heading the same height as the navigation */
         | 
| 285 | 
            +
                .header h3 {
         | 
| 286 | 
            +
                  margin-top: 0;
         | 
| 287 | 
            +
                  margin-bottom: 0;
         | 
| 288 | 
            +
                  line-height: 40px;
         | 
| 289 | 
            +
                }
         | 
| 290 | 
            +
             | 
| 291 | 
            +
                /* Customize container */
         | 
| 292 | 
            +
                .container {
         | 
| 293 | 
            +
                  max-width: 990px;
         | 
| 294 | 
            +
                }
         | 
| 295 | 
            +
             | 
| 296 | 
            +
                address {
         | 
| 297 | 
            +
                  text-align: right;
         | 
| 298 | 
            +
                }
         | 
| 299 | 
            +
                </style>
         | 
| 300 | 
            +
             | 
| 301 | 
            +
                <title>stackprof (mode: #{ data[:mode] })</title>
         | 
| 302 | 
            +
             | 
| 303 | 
            +
                <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries -->
         | 
| 304 | 
            +
                <!--[if lt IE 9]>
         | 
| 305 | 
            +
                  <script src="https://oss.maxcdn.com/html5shiv/3.7.2/html5shiv.min.js"></script>
         | 
| 306 | 
            +
                  <script src="https://oss.maxcdn.com/respond/1.4.2/respond.min.js"></script>
         | 
| 307 | 
            +
                <![endif]-->
         | 
| 308 | 
            +
              </head>
         | 
| 309 | 
            +
              <body>
         | 
| 310 | 
            +
                <div class="container">
         | 
| 311 | 
            +
                  <div class="header clearfix">
         | 
| 312 | 
            +
                    <nav>
         | 
| 313 | 
            +
                      <div class="pull-right">
         | 
| 314 | 
            +
                        <form class="form-inline" id="form">
         | 
| 315 | 
            +
                          <a class="btn" href="javascript: resetZoom();">Reset zoom</a>
         | 
| 316 | 
            +
                          <a class="btn" href="javascript: clear();">Clear</a>
         | 
| 317 | 
            +
                          <div class="form-group">
         | 
| 318 | 
            +
                            <input type="text" class="form-control" id="term">
         | 
| 319 | 
            +
                          </div>
         | 
| 320 | 
            +
                          <a class="btn btn-primary" href="javascript: search();">Search</a>
         | 
| 321 | 
            +
                        </form>
         | 
| 322 | 
            +
                      </div>
         | 
| 323 | 
            +
                    </nav>
         | 
| 324 | 
            +
                    <h3 class="text-muted">stackprof (mode: #{ data[:mode] })</h3>
         | 
| 325 | 
            +
                  </div>
         | 
| 326 | 
            +
                  <div id="chart">
         | 
| 327 | 
            +
                  </div>
         | 
| 328 | 
            +
                  <address>
         | 
| 329 | 
            +
                    powered by <a href="https://github.com/spiermar/d3-flame-graph">d3-flame-graph</a>
         | 
| 330 | 
            +
                  </address>
         | 
| 331 | 
            +
                  <hr>
         | 
| 332 | 
            +
                  <div id="details">
         | 
| 333 | 
            +
                  </div>
         | 
| 334 | 
            +
                </div>
         | 
| 335 | 
            +
             | 
| 336 | 
            +
                <!-- D3.js -->
         | 
| 337 | 
            +
                <script src="https://d3js.org/d3.v4.min.js" charset="utf-8"></script>
         | 
| 338 | 
            +
             | 
| 339 | 
            +
                <!-- d3-tip -->
         | 
| 340 | 
            +
                <script type="text/javascript" src=https://cdnjs.cloudflare.com/ajax/libs/d3-tip/0.9.1/d3-tip.min.js></script>
         | 
| 341 | 
            +
             | 
| 342 | 
            +
                <!-- d3-flamegraph -->
         | 
| 343 | 
            +
                <script type="text/javascript" src="https://cdn.jsdelivr.net/gh/spiermar/d3-flame-graph@2.0.3/dist/d3-flamegraph.min.js"></script>
         | 
| 344 | 
            +
             | 
| 345 | 
            +
                <script type="text/javascript">
         | 
| 346 | 
            +
                var flameGraph = d3.flamegraph()
         | 
| 347 | 
            +
                  .width(960)
         | 
| 348 | 
            +
                  .cellHeight(18)
         | 
| 349 | 
            +
                  .transitionDuration(750)
         | 
| 350 | 
            +
                  .minFrameSize(5)
         | 
| 351 | 
            +
                  .transitionEase(d3.easeCubic)
         | 
| 352 | 
            +
                  .sort(true)
         | 
| 353 | 
            +
                  //Example to sort in reverse order
         | 
| 354 | 
            +
                  //.sort(function(a,b){ return d3.descending(a.name, b.name);})
         | 
| 355 | 
            +
                  .title("")
         | 
| 356 | 
            +
                  .onClick(onClick)
         | 
| 357 | 
            +
                  .differential(false)
         | 
| 358 | 
            +
                  .selfValue(false);
         | 
| 359 | 
            +
             | 
| 360 | 
            +
             | 
| 361 | 
            +
                // Example on how to use custom tooltips using d3-tip.
         | 
| 362 | 
            +
                // var tip = d3.tip()
         | 
| 363 | 
            +
                //   .direction("s")
         | 
| 364 | 
            +
                //   .offset([8, 0])
         | 
| 365 | 
            +
                //   .attr('class', 'd3-flame-graph-tip')
         | 
| 366 | 
            +
                //   .html(function(d) { return "name: " + d.data.name + ", value: " + d.data.value; });
         | 
| 367 | 
            +
             | 
| 368 | 
            +
                // flameGraph.tooltip(tip);
         | 
| 369 | 
            +
             | 
| 370 | 
            +
                var details = document.getElementById("details");
         | 
| 371 | 
            +
                flameGraph.setDetailsElement(details);
         | 
| 372 | 
            +
             | 
| 373 | 
            +
                // Example on how to use custom labels
         | 
| 374 | 
            +
                // var label = function(d) {
         | 
| 375 | 
            +
                //  return "name: " + d.name + ", value: " + d.value;
         | 
| 376 | 
            +
                // }
         | 
| 377 | 
            +
                // flameGraph.label(label);
         | 
| 378 | 
            +
             | 
| 379 | 
            +
                // Example of how to set fixed chart height
         | 
| 380 | 
            +
                // flameGraph.height(540);
         | 
| 381 | 
            +
             | 
| 382 | 
            +
                d3.select("#chart")
         | 
| 383 | 
            +
                    .datum(#{ json })
         | 
| 384 | 
            +
                    .call(flameGraph);
         | 
| 385 | 
            +
             | 
| 386 | 
            +
                document.getElementById("form").addEventListener("submit", function(event){
         | 
| 387 | 
            +
                  event.preventDefault();
         | 
| 388 | 
            +
                  search();
         | 
| 389 | 
            +
                });
         | 
| 390 | 
            +
             | 
| 391 | 
            +
                function search() {
         | 
| 392 | 
            +
                  var term = document.getElementById("term").value;
         | 
| 393 | 
            +
                  flameGraph.search(term);
         | 
| 394 | 
            +
                }
         | 
| 395 | 
            +
             | 
| 396 | 
            +
                function clear() {
         | 
| 397 | 
            +
                  document.getElementById('term').value = '';
         | 
| 398 | 
            +
                  flameGraph.clear();
         | 
| 399 | 
            +
                }
         | 
| 400 | 
            +
             | 
| 401 | 
            +
                function resetZoom() {
         | 
| 402 | 
            +
                  flameGraph.resetZoom();
         | 
| 403 | 
            +
                }
         | 
| 404 | 
            +
             | 
| 405 | 
            +
                function onClick(d) {
         | 
| 406 | 
            +
                  console.info("Clicked on " + d.data.name);
         | 
| 407 | 
            +
                }
         | 
| 408 | 
            +
                </script>
         | 
| 409 | 
            +
              </body>
         | 
| 410 | 
            +
            </html>
         | 
| 411 | 
            +
                  END
         | 
| 412 | 
            +
                end
         | 
| 413 | 
            +
             | 
| 151 414 | 
             
                def print_graphviz(options = {}, f = STDOUT)
         | 
| 152 415 | 
             
                  if filter = options[:filter]
         | 
| 153 416 | 
             
                    mark_stack = []
         | 
| @@ -185,7 +448,7 @@ module StackProf | |
| 185 448 | 
             
                    call, total = info.values_at(:samples, :total_samples)
         | 
| 186 449 | 
             
                    break if total < node_minimum || (limit && index >= limit)
         | 
| 187 450 |  | 
| 188 | 
            -
                    sample = ''
         | 
| 451 | 
            +
                    sample = ''.dup
         | 
| 189 452 | 
             
                    sample << "#{call} (%2.1f%%)\\rof " % (call*100.0/overall_samples) if call < total
         | 
| 190 453 | 
             
                    sample << "#{total} (%2.1f%%)\\r" % (total*100.0/overall_samples)
         | 
| 191 454 | 
             
                    fontsize = (1.0 * call / max_samples) * 28 + 10
         | 
| @@ -429,7 +692,8 @@ module StackProf | |
| 429 692 | 
             
                      end
         | 
| 430 693 | 
             
                    end
         | 
| 431 694 | 
             
                  end
         | 
| 695 | 
            +
                rescue SystemCallError
         | 
| 696 | 
            +
                  f.puts "        SOURCE UNAVAILABLE"
         | 
| 432 697 | 
             
                end
         | 
| 433 | 
            -
             | 
| 434 698 | 
             
              end
         | 
| 435 699 | 
             
            end
         | 
| @@ -0,0 +1,37 @@ | |
| 1 | 
            +
            module StackProf
         | 
| 2 | 
            +
              # Define the same methods as stackprof.c
         | 
| 3 | 
            +
              class << self
         | 
| 4 | 
            +
                def running?
         | 
| 5 | 
            +
                  false
         | 
| 6 | 
            +
                end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                def run(*args)
         | 
| 9 | 
            +
                  unimplemented
         | 
| 10 | 
            +
                end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                def start(*args)
         | 
| 13 | 
            +
                  unimplemented
         | 
| 14 | 
            +
                end
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                def stop
         | 
| 17 | 
            +
                  unimplemented
         | 
| 18 | 
            +
                end
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                def results(*args)
         | 
| 21 | 
            +
                  unimplemented
         | 
| 22 | 
            +
                end
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                def sample
         | 
| 25 | 
            +
                  unimplemented
         | 
| 26 | 
            +
                end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                def use_postponed_job!
         | 
| 29 | 
            +
                  # noop
         | 
| 30 | 
            +
                end
         | 
| 31 | 
            +
             | 
| 32 | 
            +
                private def unimplemented
         | 
| 33 | 
            +
                  raise "Use --cpusampler=flamegraph or --cpusampler instead of StackProf on TruffleRuby.\n" \
         | 
| 34 | 
            +
                        "See https://www.graalvm.org/tools/profiling/ and `ruby --help:cpusampler` for more details."
         | 
| 35 | 
            +
                end
         | 
| 36 | 
            +
              end
         | 
| 37 | 
            +
            end
         | 
    
        data/lib/stackprof.rb
    CHANGED
    
    | @@ -1,4 +1,25 @@ | |
| 1 | 
            -
             | 
| 1 | 
            +
            if RUBY_ENGINE == 'truffleruby'
         | 
| 2 | 
            +
              require "stackprof/truffleruby"
         | 
| 3 | 
            +
            else
         | 
| 4 | 
            +
              require "stackprof/stackprof"
         | 
| 5 | 
            +
            end
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            if defined?(RubyVM::YJIT) && RubyVM::YJIT.enabled?
         | 
| 8 | 
            +
              if RUBY_VERSION < "3.3"
         | 
| 9 | 
            +
                # On 3.3 we don't need postponed jobs:
         | 
| 10 | 
            +
                # https://github.com/ruby/ruby/commit/a1dc1a3de9683daf5a543d6f618e17aabfcb8708
         | 
| 11 | 
            +
                StackProf.use_postponed_job!
         | 
| 12 | 
            +
              end
         | 
| 13 | 
            +
            elsif RUBY_VERSION == "3.2.0"
         | 
| 14 | 
            +
              # 3.2.0 crash is the signal is received at the wrong time.
         | 
| 15 | 
            +
              # Fixed in https://github.com/ruby/ruby/pull/7116
         | 
| 16 | 
            +
              # The fix is backported in 3.2.1: https://bugs.ruby-lang.org/issues/19336
         | 
| 17 | 
            +
              StackProf.use_postponed_job!
         | 
| 18 | 
            +
            end
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            module StackProf
         | 
| 21 | 
            +
              VERSION = '0.2.26'
         | 
| 22 | 
            +
            end
         | 
| 2 23 |  | 
| 3 24 | 
             
            StackProf.autoload :Report, "stackprof/report.rb"
         | 
| 4 25 | 
             
            StackProf.autoload :Middleware, "stackprof/middleware.rb"
         | 
    
        data/stackprof.gemspec
    CHANGED
    
    | @@ -1,11 +1,18 @@ | |
| 1 1 | 
             
            Gem::Specification.new do |s|
         | 
| 2 2 | 
             
              s.name = 'stackprof'
         | 
| 3 | 
            -
              s.version = '0.2. | 
| 3 | 
            +
              s.version = '0.2.26'
         | 
| 4 4 | 
             
              s.homepage = 'http://github.com/tmm1/stackprof'
         | 
| 5 5 |  | 
| 6 6 | 
             
              s.authors = 'Aman Gupta'
         | 
| 7 7 | 
             
              s.email   = 'aman@tmm1.net'
         | 
| 8 8 |  | 
| 9 | 
            +
              s.metadata = {
         | 
| 10 | 
            +
                'bug_tracker_uri'   => 'https://github.com/tmm1/stackprof/issues',
         | 
| 11 | 
            +
                'changelog_uri'     => "https://github.com/tmm1/stackprof/blob/v#{s.version}/CHANGELOG.md",
         | 
| 12 | 
            +
                'documentation_uri' => "https://www.rubydoc.info/gems/stackprof/#{s.version}",
         | 
| 13 | 
            +
                'source_code_uri'   => "https://github.com/tmm1/stackprof/tree/v#{s.version}"
         | 
| 14 | 
            +
              }
         | 
| 15 | 
            +
             | 
| 9 16 | 
             
              s.files = `git ls-files`.split("\n")
         | 
| 10 17 | 
             
              s.extensions = 'ext/stackprof/extconf.rb'
         | 
| 11 18 |  | 
| @@ -14,12 +21,13 @@ Gem::Specification.new do |s| | |
| 14 21 | 
             
              s.executables << 'stackprof-flamegraph.pl'
         | 
| 15 22 | 
             
              s.executables << 'stackprof-gprof2dot.py'
         | 
| 16 23 |  | 
| 17 | 
            -
              s.summary = 'sampling callstack-profiler for ruby 2. | 
| 24 | 
            +
              s.summary = 'sampling callstack-profiler for ruby 2.2+'
         | 
| 18 25 | 
             
              s.description = 'stackprof is a fast sampling profiler for ruby code, with cpu, wallclock and object allocation samplers.'
         | 
| 19 26 |  | 
| 27 | 
            +
              s.required_ruby_version = '>= 2.2'
         | 
| 28 | 
            +
             | 
| 20 29 | 
             
              s.license = 'MIT'
         | 
| 21 30 |  | 
| 22 31 | 
             
              s.add_development_dependency 'rake-compiler', '~> 0.9'
         | 
| 23 | 
            -
              s.add_development_dependency 'mocha', '~> 0.14'
         | 
| 24 32 | 
             
              s.add_development_dependency 'minitest', '~> 5.0'
         | 
| 25 33 | 
             
            end
         | 
| @@ -0,0 +1 @@ | |
| 1 | 
            +
            {:	modeI"cpu:ET
         | 
| @@ -0,0 +1 @@ | |
| 1 | 
            +
            { "mode": "cpu" }
         | 
    
        data/test/test_middleware.rb
    CHANGED
    
    | @@ -2,9 +2,9 @@ $:.unshift File.expand_path('../../lib', __FILE__) | |
| 2 2 | 
             
            require 'stackprof'
         | 
| 3 3 | 
             
            require 'stackprof/middleware'
         | 
| 4 4 | 
             
            require 'minitest/autorun'
         | 
| 5 | 
            -
            require ' | 
| 5 | 
            +
            require 'tmpdir'
         | 
| 6 6 |  | 
| 7 | 
            -
            class StackProf::MiddlewareTest <  | 
| 7 | 
            +
            class StackProf::MiddlewareTest < Minitest::Test
         | 
| 8 8 |  | 
| 9 9 | 
             
              def test_path_default
         | 
| 10 10 | 
             
                StackProf::Middleware.new(Object.new)
         | 
| @@ -19,23 +19,36 @@ class StackProf::MiddlewareTest < MiniTest::Test | |
| 19 19 | 
             
              end
         | 
| 20 20 |  | 
| 21 21 | 
             
              def test_save_default
         | 
| 22 | 
            -
                StackProf::Middleware.new(Object.new | 
| 23 | 
            -
             | 
| 24 | 
            -
             | 
| 25 | 
            -
                 | 
| 26 | 
            -
             | 
| 27 | 
            -
             | 
| 28 | 
            -
             | 
| 22 | 
            +
                middleware = StackProf::Middleware.new(->(env) { 100.times { Object.new } },
         | 
| 23 | 
            +
                                                      save_every: 1,
         | 
| 24 | 
            +
                                                      enabled: true)
         | 
| 25 | 
            +
                Dir.mktmpdir do |dir|
         | 
| 26 | 
            +
                  Dir.chdir(dir) { middleware.call({}) }
         | 
| 27 | 
            +
                  dir = File.join(dir, "tmp")
         | 
| 28 | 
            +
                  assert File.directory? dir
         | 
| 29 | 
            +
                  profile = Dir.entries(dir).reject { |x| File.directory?(x) }.first
         | 
| 30 | 
            +
                  assert profile
         | 
| 31 | 
            +
                  assert_equal "stackprof", profile.split("-")[0]
         | 
| 32 | 
            +
                  assert_equal "cpu", profile.split("-")[1]
         | 
| 33 | 
            +
                  assert_equal Process.pid.to_s, profile.split("-")[2]
         | 
| 34 | 
            +
                end
         | 
| 29 35 | 
             
              end
         | 
| 30 36 |  | 
| 31 37 | 
             
              def test_save_custom
         | 
| 32 | 
            -
                StackProf::Middleware.new( | 
| 33 | 
            -
             | 
| 34 | 
            -
             | 
| 35 | 
            -
             | 
| 36 | 
            -
                 | 
| 37 | 
            -
             | 
| 38 | 
            -
             | 
| 38 | 
            +
                middleware = StackProf::Middleware.new(->(env) { 100.times { Object.new } },
         | 
| 39 | 
            +
                                                      path: "foo/",
         | 
| 40 | 
            +
                                                      save_every: 1,
         | 
| 41 | 
            +
                                                      enabled: true)
         | 
| 42 | 
            +
                Dir.mktmpdir do |dir|
         | 
| 43 | 
            +
                  Dir.chdir(dir) { middleware.call({}) }
         | 
| 44 | 
            +
                  dir = File.join(dir, "foo")
         | 
| 45 | 
            +
                  assert File.directory? dir
         | 
| 46 | 
            +
                  profile = Dir.entries(dir).reject { |x| File.directory?(x) }.first
         | 
| 47 | 
            +
                  assert profile
         | 
| 48 | 
            +
                  assert_equal "stackprof", profile.split("-")[0]
         | 
| 49 | 
            +
                  assert_equal "cpu", profile.split("-")[1]
         | 
| 50 | 
            +
                  assert_equal Process.pid.to_s, profile.split("-")[2]
         | 
| 51 | 
            +
                end
         | 
| 39 52 | 
             
              end
         | 
| 40 53 |  | 
| 41 54 | 
             
              def test_enabled_should_use_a_proc_if_passed
         | 
| @@ -64,4 +77,10 @@ class StackProf::MiddlewareTest < MiniTest::Test | |
| 64 77 | 
             
                StackProf::Middleware.new(Object.new, raw: true)
         | 
| 65 78 | 
             
                assert StackProf::Middleware.raw
         | 
| 66 79 | 
             
              end
         | 
| 67 | 
            -
             | 
| 80 | 
            +
             | 
| 81 | 
            +
              def test_metadata
         | 
| 82 | 
            +
                metadata = { key: 'value' }
         | 
| 83 | 
            +
                StackProf::Middleware.new(Object.new, metadata: metadata)
         | 
| 84 | 
            +
                assert_equal metadata, StackProf::Middleware.metadata
         | 
| 85 | 
            +
              end
         | 
| 86 | 
            +
            end unless RUBY_ENGINE == 'truffleruby'
         | 
    
        data/test/test_report.rb
    CHANGED
    
    | @@ -2,7 +2,7 @@ $:.unshift File.expand_path('../../lib', __FILE__) | |
| 2 2 | 
             
            require 'stackprof'
         | 
| 3 3 | 
             
            require 'minitest/autorun'
         | 
| 4 4 |  | 
| 5 | 
            -
            class ReportDumpTest <  | 
| 5 | 
            +
            class ReportDumpTest < Minitest::Test
         | 
| 6 6 | 
             
              require 'stringio'
         | 
| 7 7 |  | 
| 8 8 | 
             
              def test_dump_to_stdout
         | 
| @@ -32,3 +32,27 @@ class ReportDumpTest < MiniTest::Test | |
| 32 32 | 
             
                assert_equal expected, Marshal.load(marshal_data)
         | 
| 33 33 | 
             
              end
         | 
| 34 34 | 
             
            end
         | 
| 35 | 
            +
             | 
| 36 | 
            +
            class ReportReadTest < Minitest::Test
         | 
| 37 | 
            +
              require 'pathname'
         | 
| 38 | 
            +
             | 
| 39 | 
            +
              def test_from_file_read_json
         | 
| 40 | 
            +
                file = fixture("profile.json")
         | 
| 41 | 
            +
                report = StackProf::Report.from_file(file)
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                assert_equal({ mode: "cpu" }, report.data)
         | 
| 44 | 
            +
              end
         | 
| 45 | 
            +
             | 
| 46 | 
            +
              def test_from_file_read_marshal
         | 
| 47 | 
            +
                file = fixture("profile.dump")
         | 
| 48 | 
            +
                report = StackProf::Report.from_file(file)
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                assert_equal({ mode: "cpu" }, report.data)
         | 
| 51 | 
            +
              end
         | 
| 52 | 
            +
             | 
| 53 | 
            +
              private
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              def fixture(name)
         | 
| 56 | 
            +
                Pathname.new(__dir__).join("fixtures", name)
         | 
| 57 | 
            +
              end
         | 
| 58 | 
            +
            end
         |