debian-mirror-gitlab/lib/gitlab/metrics/subscribers/active_record.rb

208 lines
7.8 KiB
Ruby
Raw Normal View History

2019-02-15 15:39:39 +05:30
# frozen_string_literal: true
module Gitlab
module Metrics
module Subscribers
# Class for tracking the total query duration of a transaction.
class ActiveRecord < ActiveSupport::Subscriber
attach_to :active_record
2018-11-20 20:47:30 +05:30
IGNORABLE_SQL = %w{BEGIN COMMIT}.freeze
2021-10-27 15:23:28 +05:30
DB_COUNTERS = %i{count write_count cached_count}.freeze
SQL_COMMANDS_WITH_COMMENTS_REGEX = %r{\A(/\*.*\*/\s)?((?!(.*[^\w'"](DELETE|UPDATE|INSERT INTO)[^\w'"])))(WITH.*)?(SELECT)((?!(FOR UPDATE|FOR SHARE)).)*$}i.freeze
2021-04-17 20:07:23 +05:30
2021-04-29 21:17:54 +05:30
SQL_DURATION_BUCKET = [0.05, 0.1, 0.25].freeze
TRANSACTION_DURATION_BUCKET = [0.1, 0.25, 1].freeze
2021-04-17 20:07:23 +05:30
2021-10-27 15:23:28 +05:30
DB_LOAD_BALANCING_ROLES = %i{replica primary}.freeze
DB_LOAD_BALANCING_COUNTERS = %i{count cached_count wal_count wal_cached_count}.freeze
DB_LOAD_BALANCING_DURATIONS = %i{duration_s}.freeze
2021-09-04 01:27:46 +05:30
SQL_WAL_LOCATION_REGEX = /(pg_current_wal_insert_lsn\(\)::text|pg_last_wal_replay_lsn\(\)::text)/.freeze
2021-04-17 20:07:23 +05:30
# This event is published from ActiveRecordBaseTransactionMetrics and
# used to record a database transaction duration when calling
# ActiveRecord::Base.transaction {} block.
def transaction(event)
2021-04-29 21:17:54 +05:30
observe(:gitlab_database_transaction_seconds, event) do
buckets TRANSACTION_DURATION_BUCKET
end
2021-04-17 20:07:23 +05:30
end
2018-11-20 20:47:30 +05:30
def sql(event)
2020-10-24 23:57:45 +05:30
# Mark this thread as requiring a database connection. This is used
# by the Gitlab::Metrics::Samplers::ThreadsSampler to count threads
# using a connection.
Thread.current[:uses_db_connection] = true
2018-11-20 20:47:30 +05:30
payload = event.payload
2021-04-17 20:07:23 +05:30
return if ignored_query?(payload)
2018-11-20 20:47:30 +05:30
2021-10-27 15:23:28 +05:30
db_config_name = db_config_name(event.payload)
increment(:count, db_config_name: db_config_name)
increment(:cached_count, db_config_name: db_config_name) if cached_query?(payload)
increment(:write_count, db_config_name: db_config_name) unless select_sql_command?(payload)
2021-02-22 17:27:13 +05:30
2021-04-29 21:17:54 +05:30
observe(:gitlab_sql_duration_seconds, event) do
buckets SQL_DURATION_BUCKET
end
2021-09-04 01:27:46 +05:30
2021-11-18 22:05:49 +05:30
db_role = ::Gitlab::Database::LoadBalancing.db_role_for_connection(payload[:connection])
return if db_role.blank?
2021-09-04 01:27:46 +05:30
2021-11-18 22:05:49 +05:30
increment_db_role_counters(db_role, payload)
observe_db_role_duration(db_role, event)
end
2020-07-28 23:09:34 +05:30
def self.db_counter_payload
return {} unless Gitlab::SafeRequestStore.active?
2021-09-04 01:27:46 +05:30
{}.tap do |payload|
2021-10-27 15:23:28 +05:30
db_counter_keys.each do |key|
payload[key] = Gitlab::SafeRequestStore[key].to_i
2021-09-04 01:27:46 +05:30
end
2021-11-18 22:05:49 +05:30
if ::Gitlab::SafeRequestStore.active?
2021-10-27 15:23:28 +05:30
load_balancing_metric_counter_keys.each do |counter|
2021-09-04 01:27:46 +05:30
payload[counter] = ::Gitlab::SafeRequestStore[counter].to_i
end
2021-09-30 23:02:18 +05:30
2021-10-27 15:23:28 +05:30
load_balancing_metric_duration_keys.each do |duration|
payload[duration] = ::Gitlab::SafeRequestStore[duration].to_f.round(3)
2021-09-30 23:02:18 +05:30
end
2021-09-04 01:27:46 +05:30
end
2021-04-17 20:07:23 +05:30
end
2020-07-28 23:09:34 +05:30
end
2021-09-04 01:27:46 +05:30
private
def wal_command?(payload)
payload[:sql].match(SQL_WAL_LOCATION_REGEX)
end
def increment_db_role_counters(db_role, payload)
2021-09-30 23:02:18 +05:30
cached = cached_query?(payload)
2021-10-27 15:23:28 +05:30
db_config_name = db_config_name(payload)
increment(:count, db_role: db_role, db_config_name: db_config_name)
increment(:cached_count, db_role: db_role, db_config_name: db_config_name) if cached
2021-09-30 23:02:18 +05:30
if wal_command?(payload)
2021-10-27 15:23:28 +05:30
increment(:wal_count, db_role: db_role, db_config_name: db_config_name)
increment(:wal_cached_count, db_role: db_role, db_config_name: db_config_name) if cached
2021-09-30 23:02:18 +05:30
end
2021-04-29 21:17:54 +05:30
end
2021-09-04 01:27:46 +05:30
def observe_db_role_duration(db_role, event)
observe("gitlab_sql_#{db_role}_duration_seconds".to_sym, event) do
buckets ::Gitlab::Metrics::Subscribers::ActiveRecord::SQL_DURATION_BUCKET
end
2021-09-30 23:02:18 +05:30
return unless ::Gitlab::SafeRequestStore.active?
2021-09-04 01:27:46 +05:30
duration = event.duration / 1000.0
2021-10-27 15:23:28 +05:30
duration_key = compose_metric_key(:duration_s, db_role)
2021-09-04 01:27:46 +05:30
::Gitlab::SafeRequestStore[duration_key] = (::Gitlab::SafeRequestStore[duration_key].presence || 0) + duration
2021-09-30 23:02:18 +05:30
# Per database metrics
2021-10-27 15:23:28 +05:30
db_config_name = db_config_name(event.payload)
duration_key = compose_metric_key(:duration_s, db_role, db_config_name)
::Gitlab::SafeRequestStore[duration_key] = (::Gitlab::SafeRequestStore[duration_key].presence || 0) + duration
2021-09-04 01:27:46 +05:30
end
2021-04-17 20:07:23 +05:30
def ignored_query?(payload)
payload[:name] == 'SCHEMA' || IGNORABLE_SQL.include?(payload[:sql])
2020-06-23 00:09:42 +05:30
end
2021-04-17 20:07:23 +05:30
def cached_query?(payload)
payload.fetch(:cached, payload[:name] == 'CACHE')
end
2020-06-23 00:09:42 +05:30
2021-04-17 20:07:23 +05:30
def select_sql_command?(payload)
payload[:sql].match(SQL_COMMANDS_WITH_COMMENTS_REGEX)
2020-07-28 23:09:34 +05:30
end
2021-10-27 15:23:28 +05:30
def increment(counter, db_config_name:, db_role: nil)
log_key = compose_metric_key(counter, db_role)
2020-07-28 23:09:34 +05:30
2021-10-27 15:23:28 +05:30
prometheus_key = if db_role
:"gitlab_transaction_db_#{db_role}_#{counter}_total"
else
:"gitlab_transaction_db_#{counter}_total"
end
if ENV['GITLAB_MULTIPLE_DATABASE_METRICS']
current_transaction&.increment(prometheus_key, 1, { db_config_name: db_config_name })
else
current_transaction&.increment(prometheus_key, 1)
end
Gitlab::SafeRequestStore[log_key] = Gitlab::SafeRequestStore[log_key].to_i + 1
# To avoid confusing log keys we only log the db_config_name metrics
# when we are also logging the db_role. Otherwise it will be hard to
# tell if the log key is referring to a db_role OR a db_config_name.
if db_role.present? && db_config_name.present?
log_key = compose_metric_key(counter, db_role, db_config_name)
Gitlab::SafeRequestStore[log_key] = Gitlab::SafeRequestStore[log_key].to_i + 1
end
2021-04-17 20:07:23 +05:30
end
2021-04-29 21:17:54 +05:30
def observe(histogram, event, &block)
2021-10-27 15:23:28 +05:30
db_config_name = db_config_name(event.payload)
if ENV['GITLAB_MULTIPLE_DATABASE_METRICS']
current_transaction&.observe(histogram, event.duration / 1000.0, { db_config_name: db_config_name }, &block)
else
current_transaction&.observe(histogram, event.duration / 1000.0, &block)
end
2020-06-23 00:09:42 +05:30
end
def current_transaction
2021-04-17 20:07:23 +05:30
::Gitlab::Metrics::WebTransaction.current || ::Gitlab::Metrics::BackgroundTransaction.current
end
2021-10-27 15:23:28 +05:30
def db_config_name(payload)
::Gitlab::Database.db_config_name(payload[:connection])
end
def self.db_counter_keys
DB_COUNTERS.map { |c| compose_metric_key(c) }
end
def self.load_balancing_metric_counter_keys
load_balancing_metric_keys(DB_LOAD_BALANCING_COUNTERS)
end
def self.load_balancing_metric_duration_keys
load_balancing_metric_keys(DB_LOAD_BALANCING_DURATIONS)
end
def self.load_balancing_metric_keys(metrics)
[].tap do |counters|
DB_LOAD_BALANCING_ROLES.each do |role|
metrics.each do |metric|
counters << compose_metric_key(metric, role)
next unless ENV['GITLAB_MULTIPLE_DATABASE_METRICS']
::Gitlab::Database.db_config_names.each do |config_name|
counters << compose_metric_key(metric, role, config_name)
end
end
end
end
end
def compose_metric_key(metric, db_role = nil, db_config_name = nil)
self.class.compose_metric_key(metric, db_role, db_config_name)
end
def self.compose_metric_key(metric, db_role = nil, db_config_name = nil)
[:db, db_role, db_config_name, metric].compact.join("_").to_sym
end
end
end
end
end