90 lines
2.6 KiB
Ruby
90 lines
2.6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Gitlab
|
|
module Middleware
|
|
class Speedscope
|
|
def initialize(app)
|
|
@app = app
|
|
end
|
|
|
|
def call(env)
|
|
request = ActionDispatch::Request.new(env)
|
|
|
|
return @app.call(env) unless rendering_flamegraph?(request)
|
|
|
|
body = nil
|
|
|
|
::Gitlab::SafeRequestStore[:capturing_flamegraph] = true
|
|
|
|
require 'stackprof'
|
|
|
|
begin
|
|
mode = stackprof_mode(request)
|
|
flamegraph = ::StackProf.run(
|
|
mode: mode,
|
|
raw: true,
|
|
aggregate: false,
|
|
interval: ::Gitlab::StackProf.interval(mode)
|
|
) do
|
|
_, _, body = @app.call(env)
|
|
end
|
|
ensure
|
|
body.close if body.respond_to?(:close)
|
|
end
|
|
|
|
render_flamegraph(flamegraph, request)
|
|
end
|
|
|
|
private
|
|
|
|
def rendering_flamegraph?(request)
|
|
request.params['performance_bar'] == 'flamegraph' && ::Gitlab::PerformanceBar.allowed_for_user?(request.env['warden']&.user)
|
|
end
|
|
|
|
def render_flamegraph(graph, request)
|
|
headers = { 'Content-Type' => 'text/html' }
|
|
path = request.env['PATH_INFO'].sub('//', '/')
|
|
|
|
speedscope_path = ::Gitlab::Utils.append_path(::Gitlab.config.gitlab.relative_url_root, '/-/speedscope/index.html')
|
|
|
|
html = <<~HTML
|
|
<!DOCTYPE html>
|
|
<html>
|
|
<head>
|
|
<style>
|
|
body { margin: 0; height: 100vh; }
|
|
#speedscope-iframe { width: 100%; height: 100%; border: none; }
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<script type="text/javascript" nonce="#{request.content_security_policy_nonce}">
|
|
var graph = #{Gitlab::Json.generate(graph)};
|
|
var json = JSON.stringify(graph);
|
|
var blob = new Blob([json], { type: 'text/plain' });
|
|
var objUrl = encodeURIComponent(URL.createObjectURL(blob));
|
|
var iframe = document.createElement('IFRAME');
|
|
iframe.setAttribute('id', 'speedscope-iframe');
|
|
document.body.appendChild(iframe);
|
|
var iframeUrl = '#{speedscope_path}#profileURL=' + objUrl + '&title=' + 'Flamegraph for #{CGI.escape(path)} in #{stackprof_mode(request)} mode';
|
|
iframe.setAttribute('src', iframeUrl);
|
|
</script>
|
|
</body>
|
|
</html>
|
|
HTML
|
|
|
|
[200, headers, [html]]
|
|
end
|
|
|
|
def stackprof_mode(request)
|
|
case request.params['stackprof_mode']&.to_sym
|
|
when :cpu
|
|
:cpu
|
|
when :object
|
|
:object
|
|
else
|
|
:wall
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|