BigW Consortium Gitlab

middleware.rb 1.16 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
require 'ruby-prof'

module Gitlab
  module RequestProfiler
    class Middleware
      def initialize(app)
        @app = app
      end

      def call(env)
        if profile?(env)
          call_with_profiling(env)
        else
          @app.call(env)
        end
      end

      def profile?(env)
        header_token = env['HTTP_X_PROFILE_TOKEN']
        return unless header_token.present?

22
        profile_token = Gitlab::RequestProfiler.profile_token
23 24 25 26 27 28 29 30
        return unless profile_token.present?

        header_token == profile_token
      end

      def call_with_profiling(env)
        ret = nil
        result = RubyProf::Profile.profile do
31 32 33
          ret = catch(:warden) do
            @app.call(env)
          end
34 35 36 37 38 39 40 41 42 43 44
        end

        printer   = RubyProf::CallStackPrinter.new(result)
        file_name = "#{env['PATH_INFO'].tr('/', '|')}_#{Time.current.to_i}.html"
        file_path = "#{PROFILES_DIR}/#{file_name}"

        FileUtils.mkdir_p(PROFILES_DIR)
        File.open(file_path, 'wb') do |file|
          printer.print(file)
        end

45 46 47 48 49
        if ret.is_a?(Array)
          ret
        else
          throw(:warden, ret)
        end
50 51 52 53
      end
    end
  end
end