2019-09-30 21:07:59 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'redis'
|
|
|
|
|
|
|
|
module Gitlab
|
|
|
|
module Peek
|
|
|
|
module RedisInstrumented
|
|
|
|
def call(*args, &block)
|
|
|
|
start = Time.now
|
|
|
|
super(*args, &block)
|
|
|
|
ensure
|
|
|
|
duration = (Time.now - start)
|
|
|
|
add_call_details(duration, args)
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def add_call_details(duration, args)
|
2019-10-12 21:52:04 +05:30
|
|
|
return unless peek_enabled?
|
2019-09-30 21:07:59 +05:30
|
|
|
# redis-rb passes an array (e.g. [:get, key])
|
|
|
|
return unless args.length == 1
|
|
|
|
|
|
|
|
detail_store << {
|
|
|
|
cmd: args.first,
|
|
|
|
duration: duration,
|
2019-10-12 21:52:04 +05:30
|
|
|
backtrace: ::Gitlab::Profiler.clean_backtrace(caller)
|
2019-09-30 21:07:59 +05:30
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
def peek_enabled?
|
|
|
|
Gitlab::SafeRequestStore.store[:peek_enabled]
|
|
|
|
end
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
def detail_store
|
|
|
|
::Gitlab::SafeRequestStore['redis_call_details'] ||= []
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
module Peek
|
|
|
|
module Views
|
2019-10-12 21:52:04 +05:30
|
|
|
class RedisDetailed < DetailedView
|
2019-09-30 21:07:59 +05:30
|
|
|
REDACTED_MARKER = "<redacted>"
|
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
def key
|
|
|
|
'redis'
|
2019-09-30 21:07:59 +05:30
|
|
|
end
|
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
private
|
2019-09-30 21:07:59 +05:30
|
|
|
|
|
|
|
def format_call_details(call)
|
2019-10-12 21:52:04 +05:30
|
|
|
super.merge(cmd: format_command(call[:cmd]))
|
2019-09-30 21:07:59 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def format_command(cmd)
|
|
|
|
if cmd.length >= 2 && cmd.first =~ /^auth$/i
|
|
|
|
cmd[-1] = REDACTED_MARKER
|
|
|
|
# Scrub out the value of the SET calls to avoid binary
|
|
|
|
# data or large data from spilling into the view
|
|
|
|
elsif cmd.length >= 3 && cmd.first =~ /set/i
|
|
|
|
cmd[2..-1] = REDACTED_MARKER
|
|
|
|
end
|
|
|
|
|
|
|
|
cmd.join(' ')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class Redis::Client
|
|
|
|
prepend Gitlab::Peek::RedisInstrumented
|
|
|
|
end
|