Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
407bdc95
Commit
407bdc95
authored
Jul 13, 2020
by
Jacob Vosmaer
Committed by
Sean McGivern
Jul 13, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Measure Redis command latency per instance
parent
c597b7e0
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
81 additions
and
9 deletions
+81
-9
doc/administration/monitoring/prometheus/gitlab_metrics.md
doc/administration/monitoring/prometheus/gitlab_metrics.md
+1
-0
lib/gitlab/instrumentation/redis_base.rb
lib/gitlab/instrumentation/redis_base.rb
+13
-2
lib/gitlab/instrumentation/redis_interceptor.rb
lib/gitlab/instrumentation/redis_interceptor.rb
+17
-4
spec/lib/gitlab/instrumentation/redis_interceptor_spec.rb
spec/lib/gitlab/instrumentation/redis_interceptor_spec.rb
+50
-3
No files found.
doc/administration/monitoring/prometheus/gitlab_metrics.md
View file @
407bdc95
...
@@ -263,6 +263,7 @@ instance (`cache`, `shared_state` etc.).
...
@@ -263,6 +263,7 @@ instance (`cache`, `shared_state` etc.).
|:--------------------------------- |:------- |:----- |:----------- |
|:--------------------------------- |:------- |:----- |:----------- |
|
`gitlab_redis_client_exceptions_total`
| Counter | 13.2 | Number of Redis client exceptions, broken down by exception class |
|
`gitlab_redis_client_exceptions_total`
| Counter | 13.2 | Number of Redis client exceptions, broken down by exception class |
|
`gitlab_redis_client_requests_total`
| Counter | 13.2 | Number of Redis client requests |
|
`gitlab_redis_client_requests_total`
| Counter | 13.2 | Number of Redis client requests |
|
`gitlab_redis_client_requests_duration_seconds`
| Histogram | 13.2 | Redis request latency, excluding blocking commands |
## Metrics shared directory
## Metrics shared directory
...
...
lib/gitlab/instrumentation/redis_base.rb
View file @
407bdc95
...
@@ -81,12 +81,12 @@ module Gitlab
...
@@ -81,12 +81,12 @@ module Gitlab
self
self
end
end
def
count_request
def
instance_
count_request
@request_counter
||=
Gitlab
::
Metrics
.
counter
(
:gitlab_redis_client_requests_total
,
'Client side Redis request count, per Redis server'
)
@request_counter
||=
Gitlab
::
Metrics
.
counter
(
:gitlab_redis_client_requests_total
,
'Client side Redis request count, per Redis server'
)
@request_counter
.
increment
({
storage:
storage_key
})
@request_counter
.
increment
({
storage:
storage_key
})
end
end
def
count_exception
(
ex
)
def
instance_
count_exception
(
ex
)
# This metric is meant to give a client side view of how the Redis
# This metric is meant to give a client side view of how the Redis
# server is doing. Redis itself does not expose error counts. This
# server is doing. Redis itself does not expose error counts. This
# metric can be used for Redis alerting and service health monitoring.
# metric can be used for Redis alerting and service health monitoring.
...
@@ -94,6 +94,17 @@ module Gitlab
...
@@ -94,6 +94,17 @@ module Gitlab
@exception_counter
.
increment
({
storage:
storage_key
,
exception:
ex
.
class
.
to_s
})
@exception_counter
.
increment
({
storage:
storage_key
,
exception:
ex
.
class
.
to_s
})
end
end
def
instance_observe_duration
(
duration
)
@request_latency_histogram
||=
Gitlab
::
Metrics
.
histogram
(
:gitlab_redis_client_requests_duration_seconds
,
'Client side Redis request latency, per Redis server, excluding blocking commands'
,
{},
[
0.001
,
0.005
,
0.01
]
)
@request_latency_histogram
.
observe
({
storage:
storage_key
},
duration
)
end
private
private
def
request_count_key
def
request_count_key
...
...
lib/gitlab/instrumentation/redis_interceptor.rb
View file @
407bdc95
...
@@ -5,19 +5,26 @@ require 'redis'
...
@@ -5,19 +5,26 @@ require 'redis'
module
Gitlab
module
Gitlab
module
Instrumentation
module
Instrumentation
module
RedisInterceptor
module
RedisInterceptor
APDEX_EXCLUDE
=
%w[brpop blpop brpoplpush bzpopmin bzpopmax xread xreadgroup]
.
freeze
def
call
(
*
args
,
&
block
)
def
call
(
*
args
,
&
block
)
instrumentation_class
.
count_request
start
=
Time
.
now
# must come first so that 'start' is always defined
instrumentation_class
.
instance_count_request
instrumentation_class
.
redis_cluster_validate!
(
args
.
first
)
instrumentation_class
.
redis_cluster_validate!
(
args
.
first
)
start
=
Time
.
now
super
(
*
args
,
&
block
)
super
(
*
args
,
&
block
)
rescue
::
Redis
::
BaseError
=>
ex
rescue
::
Redis
::
BaseError
=>
ex
instrumentation_class
.
count_exception
(
ex
)
instrumentation_class
.
instance_
count_exception
(
ex
)
raise
ex
raise
ex
ensure
ensure
duration
=
(
Time
.
now
-
start
)
duration
=
Time
.
now
-
start
unless
APDEX_EXCLUDE
.
include?
(
command_from_args
(
args
))
instrumentation_class
.
instance_observe_duration
(
duration
)
end
if
::
RequestStore
.
active?
if
::
RequestStore
.
active?
# These metrics measure total Redis usage per Rails request / job.
instrumentation_class
.
increment_request_count
instrumentation_class
.
increment_request_count
instrumentation_class
.
add_duration
(
duration
)
instrumentation_class
.
add_duration
(
duration
)
instrumentation_class
.
add_call_details
(
duration
,
args
)
instrumentation_class
.
add_call_details
(
duration
,
args
)
...
@@ -83,6 +90,12 @@ module Gitlab
...
@@ -83,6 +90,12 @@ module Gitlab
def
instrumentation_class
def
instrumentation_class
@options
[
:instrumentation_class
]
# rubocop:disable Gitlab/ModuleWithInstanceVariables
@options
[
:instrumentation_class
]
# rubocop:disable Gitlab/ModuleWithInstanceVariables
end
end
def
command_from_args
(
args
)
command
=
args
[
0
]
command
=
command
[
0
]
if
command
.
is_a?
(
Array
)
command
.
to_s
.
downcase
end
end
end
end
end
end
end
...
...
spec/lib/gitlab/instrumentation/redis_interceptor_spec.rb
View file @
407bdc95
...
@@ -47,15 +47,15 @@ RSpec.describe Gitlab::Instrumentation::RedisInterceptor, :clean_gitlab_redis_sh
...
@@ -47,15 +47,15 @@ RSpec.describe Gitlab::Instrumentation::RedisInterceptor, :clean_gitlab_redis_sh
let
(
:instrumentation_class
)
{
Gitlab
::
Redis
::
SharedState
.
instrumentation_class
}
let
(
:instrumentation_class
)
{
Gitlab
::
Redis
::
SharedState
.
instrumentation_class
}
it
'counts successful requests'
do
it
'counts successful requests'
do
expect
(
instrumentation_class
).
to
receive
(
:count_request
).
and_call_original
expect
(
instrumentation_class
).
to
receive
(
:
instance_
count_request
).
and_call_original
Gitlab
::
Redis
::
SharedState
.
with
{
|
redis
|
redis
.
call
(
:get
,
'foobar'
)
}
Gitlab
::
Redis
::
SharedState
.
with
{
|
redis
|
redis
.
call
(
:get
,
'foobar'
)
}
end
end
it
'counts exceptions'
do
it
'counts exceptions'
do
expect
(
instrumentation_class
).
to
receive
(
:count_exception
)
expect
(
instrumentation_class
).
to
receive
(
:
instance_
count_exception
)
.
with
(
instance_of
(
Redis
::
CommandError
)).
and_call_original
.
with
(
instance_of
(
Redis
::
CommandError
)).
and_call_original
expect
(
instrumentation_class
).
to
receive
(
:count_request
).
and_call_original
expect
(
instrumentation_class
).
to
receive
(
:
instance_
count_request
).
and_call_original
expect
do
expect
do
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
Gitlab
::
Redis
::
SharedState
.
with
do
|
redis
|
...
@@ -64,4 +64,51 @@ RSpec.describe Gitlab::Instrumentation::RedisInterceptor, :clean_gitlab_redis_sh
...
@@ -64,4 +64,51 @@ RSpec.describe Gitlab::Instrumentation::RedisInterceptor, :clean_gitlab_redis_sh
end
.
to
raise_exception
(
Redis
::
CommandError
)
end
.
to
raise_exception
(
Redis
::
CommandError
)
end
end
end
end
describe
'latency'
do
let
(
:instrumentation_class
)
{
Gitlab
::
Redis
::
SharedState
.
instrumentation_class
}
describe
'commands in the apdex'
do
where
(
:command
)
do
[
[[
:get
,
'foobar'
]],
[
%w[GET foobar]
]
]
end
with_them
do
it
'measures requests we want in the apdex'
do
expect
(
instrumentation_class
).
to
receive
(
:instance_observe_duration
).
with
(
a_value
>
0
)
.
and_call_original
Gitlab
::
Redis
::
SharedState
.
with
{
|
redis
|
redis
.
call
(
*
command
)
}
end
end
end
describe
'commands not in the apdex'
do
where
(
:command
)
do
[
[
%w[brpop foobar 0.01]
],
[
%w[blpop foobar 0.01]
],
[
%w[brpoplpush foobar bazqux 0.01]
],
[
%w[bzpopmin foobar 0.01]
],
[
%w[bzpopmax foobar 0.01]
],
[
%w[xread block 1 streams mystream 0-0]
],
[
%w[xreadgroup group mygroup myconsumer block 1 streams foobar 0-0]
]
]
end
with_them
do
it
'skips requests we do not want in the apdex'
do
expect
(
instrumentation_class
).
not_to
receive
(
:instance_observe_duration
)
begin
Gitlab
::
Redis
::
SharedState
.
with
{
|
redis
|
redis
.
call
(
*
command
)
}
rescue
Gitlab
::
Instrumentation
::
RedisClusterValidator
::
CrossSlotError
,
::
Redis
::
CommandError
end
end
end
end
end
end
end
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment