rack-webprofiler 0.1.0.pre.alpha1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (57) hide show
  1. checksums.yaml +7 -0
  2. data/.editorconfig +27 -0
  3. data/.gitignore +11 -0
  4. data/.rspec +2 -0
  5. data/.rubocop.yml +34 -0
  6. data/.rubocop_todo.yml +0 -0
  7. data/.travis.yml +23 -0
  8. data/CHANGELOG.md +39 -0
  9. data/CODE_OF_CONDUCT.md +74 -0
  10. data/Gemfile +15 -0
  11. data/LICENSE +21 -0
  12. data/README.md +94 -0
  13. data/Rakefile +17 -0
  14. data/bin/ci-prepare +16 -0
  15. data/bin/console +10 -0
  16. data/bin/setup +8 -0
  17. data/examples/README.md +56 -0
  18. data/examples/rack/config.ru +52 -0
  19. data/examples/sinatra/app.rb +37 -0
  20. data/examples/sinatra/config.ru +3 -0
  21. data/lib/rack/templates/404.erb +1 -0
  22. data/lib/rack/templates/assets/rwpt.css +9 -0
  23. data/lib/rack/templates/assets/rwpt.js +0 -0
  24. data/lib/rack/templates/async.erb +120 -0
  25. data/lib/rack/templates/panel/index.erb +29 -0
  26. data/lib/rack/templates/panel/layout.erb +17 -0
  27. data/lib/rack/templates/panel/show.erb +19 -0
  28. data/lib/rack/templates/profiler.erb +70 -0
  29. data/lib/rack/web_profiler/auto_configure/rails.rb +12 -0
  30. data/lib/rack/web_profiler/collector/debug_collector.rb +31 -0
  31. data/lib/rack/web_profiler/collector/erb_collector.rb +0 -0
  32. data/lib/rack/web_profiler/collector/performance_collector.rb +1 -0
  33. data/lib/rack/web_profiler/collector/rack/rack_collector.rb +23 -0
  34. data/lib/rack/web_profiler/collector/rack/request_collector.rb +33 -0
  35. data/lib/rack/web_profiler/collector/rails/active_record_collector.rb +25 -0
  36. data/lib/rack/web_profiler/collector/rails/logger_collector.rb +22 -0
  37. data/lib/rack/web_profiler/collector/rails/rails_collector.rb +25 -0
  38. data/lib/rack/web_profiler/collector/rails/request_collector.rb +50 -0
  39. data/lib/rack/web_profiler/collector/ruby_collector.rb +46 -0
  40. data/lib/rack/web_profiler/collector/sinatra/request_collector.rb +38 -0
  41. data/lib/rack/web_profiler/collector/sinatra/sinatra_collector.rb +25 -0
  42. data/lib/rack/web_profiler/collector/time_collector.rb +23 -0
  43. data/lib/rack/web_profiler/collector.rb +142 -0
  44. data/lib/rack/web_profiler/collectors.rb +88 -0
  45. data/lib/rack/web_profiler/config.rb +61 -0
  46. data/lib/rack/web_profiler/controller.rb +125 -0
  47. data/lib/rack/web_profiler/engine.rb +109 -0
  48. data/lib/rack/web_profiler/erb.rb +9 -0
  49. data/lib/rack/web_profiler/model/collection_record.rb +45 -0
  50. data/lib/rack/web_profiler/model.rb +35 -0
  51. data/lib/rack/web_profiler/request.rb +14 -0
  52. data/lib/rack/web_profiler/router.rb +71 -0
  53. data/lib/rack/web_profiler/version.rb +5 -0
  54. data/lib/rack/web_profiler.rb +70 -0
  55. data/lib/rack/webprofiler.rb +1 -0
  56. data/rack-webprofiler.gemspec +38 -0
  57. metadata +198 -0
@@ -0,0 +1,120 @@
1
+ <div id="rwpt<%= @token %>" class="rwpt-toolbar" style="display: none"></div>
2
+ <script>
3
+ /*<![CDATA[*/
4
+ APjs = (function() {
5
+ "use strict";
6
+ var noop = function() {},
7
+ profilerStorageKey = 'rack-webprofiler/',
8
+ request = function(url, onSuccess, onError, payload, options) {
9
+ var xhr = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP');
10
+ options = options || {};
11
+ options.maxTries = options.maxTries || 0;
12
+ xhr.open(options.method || 'GET', url, true);
13
+ xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
14
+ xhr.onreadystatechange = function(state) {
15
+ if (4 !== xhr.readyState) {
16
+ return null;
17
+ }
18
+ if (xhr.status == 404 && options.maxTries > 1) {
19
+ setTimeout(function() {
20
+ options.maxTries--;
21
+ request(url, onSuccess, onError, payload, options);
22
+ }, 500);
23
+ return null;
24
+ }
25
+ if (200 === xhr.status) {
26
+ (onSuccess || noop)(xhr);
27
+ } else {
28
+ (onError || noop)(xhr);
29
+ }
30
+ };
31
+ xhr.send(payload || '');
32
+ },
33
+ hasClass = function(el, klass) {
34
+ return el.className && el.className.match(new RegExp('\\b' + klass + '\\b'));
35
+ },
36
+ removeClass = function(el, klass) {
37
+ if (el.className) {
38
+ el.className = el.className.replace(new RegExp('\\b' + klass + '\\b'), ' ');
39
+ }
40
+ },
41
+ addClass = function(el, klass) {
42
+ if (!hasClass(el, klass)) {
43
+ el.className += " " + klass;
44
+ }
45
+ },
46
+ getPreference = function(name) {
47
+ if (!window.localStorage) {
48
+ return null;
49
+ }
50
+ return localStorage.getItem(profilerStorageKey + name);
51
+ },
52
+ setPreference = function(name, value) {
53
+ if (!window.localStorage) {
54
+ return null;
55
+ }
56
+ localStorage.setItem(profilerStorageKey + name, value);
57
+ };
58
+ return {
59
+ hasClass: hasClass,
60
+ removeClass: removeClass,
61
+ addClass: addClass,
62
+ getPreference: getPreference,
63
+ setPreference: setPreference,
64
+ request: request,
65
+ load: function(selector, url, onSuccess, onError, options) {
66
+ var el = document.getElementById(selector);
67
+ if (el && el.getAttribute('data-rwpturl') !== url) {
68
+ request(url, function(xhr) {
69
+ el.innerHTML = xhr.responseText;
70
+ el.setAttribute('data-rwpturl', url);
71
+ removeClass(el, 'loading');
72
+ (onSuccess || noop)(xhr, el);
73
+ }, function(xhr) {
74
+ (onError || noop)(xhr, el);
75
+ }, '', options);
76
+ }
77
+ return this;
78
+ },
79
+ toggle: function(selector, elOn, elOff) {
80
+ var i, style, tmp = elOn.style.display,
81
+ el = document.getElementById(selector);
82
+ elOn.style.display = elOff.style.display;
83
+ elOff.style.display = tmp;
84
+ if (el) {
85
+ el.style.display = 'none' === tmp ? 'none' : 'block';
86
+ }
87
+ return this;
88
+ }
89
+ }
90
+ })(); /*]]>*/
91
+ </script>
92
+ <script>
93
+ /*<![CDATA[*/
94
+ (function() {
95
+ APjs.load('rwpt<%= @token %>', '<%= @url %>', function(xhr, el) {
96
+ el.style.display = 'block';
97
+ return;
98
+
99
+ el.style.display = -1 !== xhr.responseText.indexOf('rwpt-toolbarreset') ? 'block' : 'none';
100
+ if (el.style.display == 'none') {
101
+ return;
102
+ }
103
+ if (APjs.getPreference('toolbar/displayState') == 'none') {
104
+ document.getElementById('rwptToolbarMainContent-<%= @token %>').style.display = 'none';
105
+ document.getElementById('rwptToolbarClearer-<%= @token %>').style.display = 'none';
106
+ document.getElementById('rwptMiniToolbar-<%= @token %>').style.display = 'block';
107
+ } else {
108
+ document.getElementById('rwptToolbarMainContent-<%= @token %>').style.display = 'block';
109
+ document.getElementById('rwptToolbarClearer-<%= @token %>').style.display = 'block';
110
+ document.getElementById('rwptMiniToolbar-<%= @token %>').style.display = 'none';
111
+ }
112
+ }, function(xhr) {
113
+ if (xhr.status !== 0) {
114
+ confirm('An error occurred while loading the web debug toolbar (' + xhr.status + ': ' + xhr.statusText + ').\\n\\nDo you want to open the profiler?') && (window.location = '<%= @url %>');
115
+ }
116
+ }, {
117
+ 'maxTries': 5
118
+ });
119
+ })(); /*]]>*/
120
+ </script>
@@ -0,0 +1,29 @@
1
+ # TODO list the 20 lastest requests
2
+
3
+ <table>
4
+ <thead>
5
+ <tr>
6
+ <th>token</th>
7
+ <th>method</th>
8
+ <th>status</th>
9
+ <th>url</th>
10
+ <th>ip</th>
11
+ <th>date</th>
12
+ </tr>
13
+ </thead>
14
+ <tbody>
15
+ <% @collections.each do |r| %>
16
+ <tr>
17
+ <td><a href="<%= WebProfiler::Router.url_for_profiler(r.token) %>"><%= r.token %></a></td>
18
+ <td><%= r.http_method %></td>
19
+ <td><%= r.http_status %></td>
20
+ <td><%= r.url %></td>
21
+ <td><%= r.ip %></td>
22
+ <td><span title="<%= r.created_at %>"><%= r.created_at.strftime('%Y-%m-%d %H:%M:%S') %></span></td>
23
+ </tr>
24
+ <% end %>
25
+ </tbody>
26
+ </table>
27
+
28
+
29
+ <a href="<%= WebProfiler::Router.url_for_clean_profiler %>">Cleanup database</a>
@@ -0,0 +1,17 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Rack WebProfiler</title>
5
+ <style>
6
+ <%= partial 'assets/rwpt.css' %>
7
+ </style>
8
+ <script type="text/javascript">
9
+ <%= partial 'assets/rwpt.js' %>
10
+ </script>
11
+ </head>
12
+ <body>
13
+
14
+ <%= yield %>
15
+
16
+ </body>
17
+ </html>
@@ -0,0 +1,19 @@
1
+ <%= partial 'profiler.erb' %>
2
+
3
+ <% @collection.datas.each do |name, datas| %>
4
+ <div class="">
5
+ <h2><%= name %></h2>
6
+
7
+ <table>
8
+ <% datas[:datas].each do |k, v| %>
9
+ <tr>
10
+ <th><%= k %></th>
11
+ <td><%= v.inspect %></td>
12
+ </tr>
13
+ <% end %>
14
+ </table>
15
+ </div>
16
+ <% end %>
17
+
18
+
19
+ <a href="<%= WebProfiler::Router.url_for_profiler %>">All requests</a>
@@ -0,0 +1,70 @@
1
+ <style>
2
+ #rack-webprofiler {
3
+ position: fixed;
4
+ bottom: 0;
5
+ left: 0;
6
+ width: 100%;
7
+ height: 40px;
8
+
9
+ background-color: #1c1f33;
10
+ color: #f3fcf0;
11
+ font-family: "Helvetica Neue", Arial, sans-serif;
12
+ font-size: 14px;
13
+ }
14
+
15
+ #rack-webprofiler * {
16
+ margin: 0;
17
+ padding: 0;
18
+ }
19
+
20
+ #rack-webprofiler .rack-webprofiler_collectors {
21
+ height: 40px;
22
+ }
23
+
24
+ #rack-webprofiler .rack-webprofiler_collectors > .rack-webprofiler_collector {
25
+ float: left;
26
+ padding: 0 10px;
27
+ border: 1px solid #282f4c;
28
+ height: 38px;
29
+ line-height: 40px;
30
+ }
31
+
32
+ #rack-webprofiler .rack-webprofiler_collectors .rack-webprofiler_collector img {
33
+ width: 20px;
34
+ height: 20px;
35
+ position: relative;
36
+ top: 4px;
37
+ }
38
+ #rack-webprofiler .rack-webprofiler_collectors .rack-webprofiler_collector.error,
39
+ #rack-webprofiler .rack-webprofiler_collectors .rack-webprofiler_collector.success {
40
+ height: 35px;
41
+ border-bottom-style: solid;
42
+ border-bottom-width: 4px;
43
+ }
44
+ #rack-webprofiler .rack-webprofiler_collectors .rack-webprofiler_collector.error {
45
+ border-bottom-color: #ff2851;
46
+ }
47
+ #rack-webprofiler .rack-webprofiler_collectors .rack-webprofiler_collector.success {
48
+ border-bottom-color: #44db5e;
49
+ }
50
+ </style>
51
+ <%
52
+ datas = @collection.datas
53
+ token = @collection.token
54
+ %>
55
+ <div id="rack-webprofiler">
56
+ <div class="rack-webprofiler_collectors">
57
+
58
+ <% @collectors.each do |name, collector| %>
59
+ <div class="rack-webprofiler_collector rack-webprofiler_collectors_<%= name %> <%= datas[name.to_sym][:status] %>">
60
+ <% unless collector.icon.nil? %>
61
+ <img alt="" src="<%= collector.icon %>" />
62
+ <% end %>
63
+
64
+ <%= render_collector(collector.template, datas[name.to_sym][:datas]) %>
65
+ <%#= yield :tab %>
66
+ <a href="<%= WebProfiler::Router.url_for_profiler(token) %>">+</a>
67
+ </div>
68
+ <% end %>
69
+ </div>
70
+ </div>
@@ -0,0 +1,12 @@
1
+ module Rack
2
+ # AutoConfigure::Rails
3
+ class WebProfiler::AutoConfigure::Rails
4
+ class Engine < ::Rails::Engine # :nodoc:
5
+ initializer "rack-web_profiler.configure_middleware" do |app|
6
+ app.middleware.use Rack::WebProfiler do |c|
7
+ c.tmp_dir = File.expand_path(File.join(Rails.root, "tmp"), __FILE__)
8
+ end
9
+ end
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,31 @@
1
+ module Rack
2
+ class WebProfiler::Collector::DebugCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "debug"
8
+ position 1
9
+
10
+ collect do |request, response|
11
+ store :key, "value"
12
+ store :backtrace, "value"
13
+ end
14
+
15
+ template __FILE__, type: :DATA
16
+ end
17
+ end
18
+
19
+ # # @see also https://gist.github.com/ubermajestix/3644301
20
+ # module Object
21
+ # def inspect
22
+ # res = super
23
+ # # save the res on collector storage
24
+ # res
25
+ # end
26
+ # end
27
+
28
+ __END__
29
+ <% content_for :tab do %>
30
+ <%= data[:ruby_version] %>
31
+ <% end %>
File without changes
@@ -0,0 +1 @@
1
+ # See: https://github.com/noahd1/oink/blob/master/lib/oink/middleware.rb#L39
@@ -0,0 +1,23 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Rack::RackCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "rack"
8
+ position 1
9
+
10
+ collect do |_request, _response|
11
+ store :rack_version, Rack.release
12
+ end
13
+
14
+ template __FILE__, type: :DATA
15
+
16
+ is_enabled? -> { !defined?(Rails) && !defined?(Sinatra) }
17
+ end
18
+ end
19
+
20
+ __END__
21
+ <%# content_for :tab do %>
22
+ <%= data[:rack_version] %>
23
+ <%# end %>
@@ -0,0 +1,33 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Rack::RequestCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "rack_request"
8
+ position 2
9
+
10
+ collect do |request, response|
11
+ store :request_path, request.path
12
+ store :request_method, request.request_method
13
+ store :response_status, response.status
14
+
15
+ if response.successful?
16
+ status :success
17
+ elsif response.redirection?
18
+ status :warning
19
+ else
20
+ status :error
21
+ end
22
+ end
23
+
24
+ template __FILE__, type: :DATA
25
+
26
+ is_enabled? -> { !defined?(Rails) && !defined?(Sinatra) }
27
+ end
28
+ end
29
+
30
+ __END__
31
+ <%# content_for :tab do %>
32
+ <%= data[:response_status] %> | <%= data[:request_method] %> <%= data[:request_path] %>
33
+ <%# end %>
@@ -0,0 +1,25 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Rails::ActiveRecordCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "rails_activerecord"
8
+ position 1
9
+
10
+ collect do |_request, _response|
11
+ store :sql_requests, []
12
+ end
13
+
14
+ template __FILE__, type: :DATA
15
+
16
+ is_enabled? -> { defined? ActiveRecord }
17
+ end
18
+ end
19
+
20
+ # See: https://github.com/noahd1/oink/blob/master/lib/oink/middleware.rb#L46
21
+
22
+ __END__
23
+ <%# content_for :tab do %>
24
+
25
+ <%# end %>
@@ -0,0 +1,22 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Rails::LoggerCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "rails_logger"
8
+ position 1
9
+
10
+ collect do |_request, _response|
11
+ end
12
+
13
+ template __FILE__, type: :DATA
14
+
15
+ is_enabled? -> { defined? Rails }
16
+ end
17
+ end
18
+
19
+ __END__
20
+ <%# content_for :tab do %>
21
+
22
+ <%# end %>
@@ -0,0 +1,25 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Rails::RailsCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "rails"
8
+ position 1
9
+
10
+ collect do |_request, _response|
11
+ store :rails_version, Rails.version
12
+ store :rails_env, Rails.env
13
+ store :rails_doc_url, "http://api.rubyonrails.org/v#{Rails.version}/"
14
+ end
15
+
16
+ template __FILE__, type: :DATA
17
+
18
+ is_enabled? -> { defined? Rails }
19
+ end
20
+ end
21
+
22
+ __END__
23
+ <%# content_for :tab do %>
24
+ <%= data[:rails_version] %> | <%= data[:rails_env] %>
25
+ <%# end %>
@@ -0,0 +1,50 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Rails::RequestCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "rails_request"
8
+ position 1
9
+
10
+ collect do |request, response|
11
+ route, _matches, request_params = find_route(request)
12
+
13
+ store :request_path, request.path
14
+ store :request_method, request.request_method
15
+ store :request_params, request_params || {}
16
+ store :request_cookies, request.cookies
17
+ store :request_get, request.GET
18
+ store :request_post, request.POST
19
+ # store :rack_env, request.env.each { |k, v| v.to_s }
20
+ # puts request.env.map{ |k, v| k => v.to_s }
21
+ store :response_status, response.status
22
+ store :route_name, route.nil? ? nil : route.name
23
+
24
+ if response.successful?
25
+ status :success
26
+ elsif response.redirection?
27
+ status :warning
28
+ else
29
+ status :error
30
+ end
31
+ end
32
+
33
+ template __FILE__, type: :DATA
34
+
35
+ is_enabled? -> { defined? Rails }
36
+
37
+ class << self
38
+ def find_route(request)
39
+ Rails.application.routes.router.recognize(request) do |route, matches, params|
40
+ return [route, matches, params]
41
+ end
42
+ end
43
+ end
44
+ end
45
+ end
46
+
47
+ __END__
48
+ <%# content_for :tab do %>
49
+ <%= data[:response_status] %> | <%= data[:request_method] %> <%= data[:request_path] %>
50
+ <%# end %>
@@ -0,0 +1,46 @@
1
+ module Rack
2
+ class WebProfiler::Collector::RubyCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon <<-'ICON'
6
+ 
7
+ ICON
8
+
9
+ collector_name "ruby"
10
+ position 0
11
+
12
+ collect do |_request, _response|
13
+ store :ruby_version, RUBY_VERSION
14
+ store :ruby_patchlevel, RUBY_PATCHLEVEL
15
+ store :ruby_release_date, RUBY_RELEASE_DATE
16
+ store :ruby_platform, RUBY_PLATFORM
17
+ store :ruby_revision, RUBY_REVISION
18
+ store :gems_list, gems_list
19
+ store :ruby_doc_url, "http://www.ruby-doc.org/core-#{RUBY_VERSION}/"
20
+ end
21
+
22
+ template __FILE__, type: :DATA
23
+
24
+ class << self
25
+ def gems_list
26
+ gems = []
27
+
28
+ Gem.loaded_specs.values.each do |g|
29
+ gems << {
30
+ name: g.name,
31
+ version: g.version.to_s,
32
+ homepage: g.homepage,
33
+ summary: g.summary,
34
+ }
35
+ end
36
+
37
+ gems
38
+ end
39
+ end
40
+ end
41
+ end
42
+
43
+ __END__
44
+ <%# content_for :tab do %>
45
+ <%= data[:ruby_version] %>
46
+ <%# end %>
@@ -0,0 +1,38 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Sinatra::RequestCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "sinatra_request"
8
+ position 2
9
+
10
+ collect do |request, response|
11
+ store :request_path, request.path
12
+ store :request_method, request.request_method
13
+ store :request_cookies, request.cookies
14
+ store :request_get, request.GET
15
+ store :request_post, request.POST
16
+ # store :rack_env, request.env.each { |k, v| v.to_s }
17
+ # puts request.env.map{ |k, v| k => v.to_s }
18
+ store :response_status, response.status
19
+
20
+ if response.successful?
21
+ status :success
22
+ elsif response.redirection?
23
+ status :warning
24
+ else
25
+ status :error
26
+ end
27
+ end
28
+
29
+ template __FILE__, type: :DATA
30
+
31
+ is_enabled? -> { defined? Sinatra }
32
+ end
33
+ end
34
+
35
+ __END__
36
+ <%# content_for :tab do %>
37
+ <%= data[:response_status] %> | <%= data[:request_method] %> <%= data[:request_path] %>
38
+ <%# end %>
@@ -0,0 +1,25 @@
1
+ module Rack
2
+ class WebProfiler::Collector::Sinatra::SinatraCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "sinatra"
8
+ position 1
9
+
10
+ collect do |_request, _response|
11
+ store :sinatra_version, Sinatra::VERSION
12
+ store :sinatra_env, Sinatra::Base.environment
13
+ store :sinatra_doc_url, "http://www.sinatrarb.com/documentation.html"
14
+ end
15
+
16
+ template __FILE__, type: :DATA
17
+
18
+ is_enabled? -> { defined? Sinatra }
19
+ end
20
+ end
21
+
22
+ __END__
23
+ <%# content_for :tab do %>
24
+ <%= data[:sinatra_version] %> | <%= data[:sinatra_env] %>
25
+ <%# end %>
@@ -0,0 +1,23 @@
1
+ module Rack
2
+ class WebProfiler::Collector::TimeCollector
3
+ include Rack::WebProfiler::Collector::DSL
4
+
5
+ icon nil
6
+
7
+ collector_name "time"
8
+ position 3
9
+
10
+ collect do |request, _response|
11
+ store :runtime, request.runtime
12
+
13
+ status :warning if request.runtime >= 500
14
+ end
15
+
16
+ template __FILE__, type: :DATA
17
+ end
18
+ end
19
+
20
+ __END__
21
+ <%# content_for :tab do %>
22
+ <%= (data[:runtime] * 1000.0).round(2) %> ms
23
+ <%# end %>