Merge branch 'instrument-infra' into 'master'
Add Prometheus metrics endpoint and basic infrastructure to meter code See merge request !11553
This commit is contained in:
commit
37dd19935b
28 changed files with 707 additions and 239 deletions
3
Gemfile
3
Gemfile
|
@ -268,6 +268,9 @@ group :metrics do
|
||||||
gem 'allocations', '~> 1.0', require: false, platform: :mri
|
gem 'allocations', '~> 1.0', require: false, platform: :mri
|
||||||
gem 'method_source', '~> 0.8', require: false
|
gem 'method_source', '~> 0.8', require: false
|
||||||
gem 'influxdb', '~> 0.2', require: false
|
gem 'influxdb', '~> 0.2', require: false
|
||||||
|
|
||||||
|
# Prometheus
|
||||||
|
gem 'prometheus-client-mmap', '~>0.7.0.beta5'
|
||||||
end
|
end
|
||||||
|
|
||||||
group :development do
|
group :development do
|
||||||
|
|
|
@ -457,6 +457,7 @@ GEM
|
||||||
mimemagic (0.3.0)
|
mimemagic (0.3.0)
|
||||||
mini_portile2 (2.1.0)
|
mini_portile2 (2.1.0)
|
||||||
minitest (5.7.0)
|
minitest (5.7.0)
|
||||||
|
mmap2 (2.2.6)
|
||||||
mousetrap-rails (1.4.6)
|
mousetrap-rails (1.4.6)
|
||||||
multi_json (1.12.1)
|
multi_json (1.12.1)
|
||||||
multi_xml (0.6.0)
|
multi_xml (0.6.0)
|
||||||
|
@ -560,6 +561,8 @@ GEM
|
||||||
premailer-rails (1.9.2)
|
premailer-rails (1.9.2)
|
||||||
actionmailer (>= 3, < 6)
|
actionmailer (>= 3, < 6)
|
||||||
premailer (~> 1.7, >= 1.7.9)
|
premailer (~> 1.7, >= 1.7.9)
|
||||||
|
prometheus-client-mmap (0.7.0.beta5)
|
||||||
|
mmap2 (~> 2.2.6)
|
||||||
pry (0.10.4)
|
pry (0.10.4)
|
||||||
coderay (~> 1.1.0)
|
coderay (~> 1.1.0)
|
||||||
method_source (~> 0.8.1)
|
method_source (~> 0.8.1)
|
||||||
|
@ -995,6 +998,7 @@ DEPENDENCIES
|
||||||
pg (~> 0.18.2)
|
pg (~> 0.18.2)
|
||||||
poltergeist (~> 1.9.0)
|
poltergeist (~> 1.9.0)
|
||||||
premailer-rails (~> 1.9.0)
|
premailer-rails (~> 1.9.0)
|
||||||
|
prometheus-client-mmap (~> 0.7.0.beta5)
|
||||||
pry-byebug (~> 3.4.1)
|
pry-byebug (~> 3.4.1)
|
||||||
pry-rails (~> 0.3.4)
|
pry-rails (~> 0.3.4)
|
||||||
rack-attack (~> 4.4.1)
|
rack-attack (~> 4.4.1)
|
||||||
|
|
|
@ -149,6 +149,7 @@ class Admin::ApplicationSettingsController < Admin::ApplicationController
|
||||||
:version_check_enabled,
|
:version_check_enabled,
|
||||||
:terminal_max_session_time,
|
:terminal_max_session_time,
|
||||||
:polling_interval_multiplier,
|
:polling_interval_multiplier,
|
||||||
|
:prometheus_metrics_enabled,
|
||||||
:usage_ping_enabled,
|
:usage_ping_enabled,
|
||||||
|
|
||||||
disabled_oauth_sign_in_sources: [],
|
disabled_oauth_sign_in_sources: [],
|
||||||
|
|
|
@ -20,25 +20,8 @@ class HealthController < ActionController::Base
|
||||||
render_check_results(results)
|
render_check_results(results)
|
||||||
end
|
end
|
||||||
|
|
||||||
def metrics
|
|
||||||
results = CHECKS.flat_map(&:metrics)
|
|
||||||
|
|
||||||
response = results.map(&method(:metric_to_prom_line)).join("\n")
|
|
||||||
|
|
||||||
render text: response, content_type: 'text/plain; version=0.0.4'
|
|
||||||
end
|
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def metric_to_prom_line(metric)
|
|
||||||
labels = metric.labels&.map { |key, value| "#{key}=\"#{value}\"" }&.join(',') || ''
|
|
||||||
if labels.empty?
|
|
||||||
"#{metric.name} #{metric.value}"
|
|
||||||
else
|
|
||||||
"#{metric.name}{#{labels}} #{metric.value}"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def render_check_results(results)
|
def render_check_results(results)
|
||||||
flattened = results.flat_map do |name, result|
|
flattened = results.flat_map do |name, result|
|
||||||
if result.is_a?(Gitlab::HealthChecks::Result)
|
if result.is_a?(Gitlab::HealthChecks::Result)
|
||||||
|
|
21
app/controllers/metrics_controller.rb
Normal file
21
app/controllers/metrics_controller.rb
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
class MetricsController < ActionController::Base
|
||||||
|
include RequiresHealthToken
|
||||||
|
|
||||||
|
protect_from_forgery with: :exception
|
||||||
|
|
||||||
|
before_action :validate_prometheus_metrics
|
||||||
|
|
||||||
|
def index
|
||||||
|
render text: metrics_service.metrics_text, content_type: 'text/plain; verssion=0.0.4'
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def metrics_service
|
||||||
|
@metrics_service ||= MetricsService.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def validate_prometheus_metrics
|
||||||
|
render_404 unless Gitlab::Metrics.prometheus_metrics_enabled?
|
||||||
|
end
|
||||||
|
end
|
|
@ -47,6 +47,10 @@ class SessionsController < Devise::SessionsController
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def login_counter
|
||||||
|
@login_counter ||= Gitlab::Metrics.counter(:user_session_logins, 'User sign in count')
|
||||||
|
end
|
||||||
|
|
||||||
# Handle an "initial setup" state, where there's only one user, it's an admin,
|
# Handle an "initial setup" state, where there's only one user, it's an admin,
|
||||||
# and they require a password change.
|
# and they require a password change.
|
||||||
def check_initial_setup
|
def check_initial_setup
|
||||||
|
@ -129,6 +133,7 @@ class SessionsController < Devise::SessionsController
|
||||||
end
|
end
|
||||||
|
|
||||||
def log_user_activity(user)
|
def log_user_activity(user)
|
||||||
|
login_counter.increment
|
||||||
Users::ActivityService.new(user, 'login').execute
|
Users::ActivityService.new(user, 'login').execute
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
33
app/services/metrics_service.rb
Normal file
33
app/services/metrics_service.rb
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
require 'prometheus/client/formats/text'
|
||||||
|
|
||||||
|
class MetricsService
|
||||||
|
CHECKS = [
|
||||||
|
Gitlab::HealthChecks::DbCheck,
|
||||||
|
Gitlab::HealthChecks::RedisCheck,
|
||||||
|
Gitlab::HealthChecks::FsShardsCheck
|
||||||
|
].freeze
|
||||||
|
|
||||||
|
def prometheus_metrics_text
|
||||||
|
Prometheus::Client::Formats::Text.marshal_multiprocess(multiprocess_metrics_path)
|
||||||
|
end
|
||||||
|
|
||||||
|
def health_metrics_text
|
||||||
|
metrics = CHECKS.flat_map(&:metrics)
|
||||||
|
|
||||||
|
formatter.marshal(metrics)
|
||||||
|
end
|
||||||
|
|
||||||
|
def metrics_text
|
||||||
|
"#{health_metrics_text}#{prometheus_metrics_text}"
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def formatter
|
||||||
|
@formatter ||= Gitlab::HealthChecks::PrometheusTextFormat.new
|
||||||
|
end
|
||||||
|
|
||||||
|
def multiprocess_metrics_path
|
||||||
|
@multiprocess_metrics_path ||= Rails.root.join(ENV['prometheus_multiproc_dir']).freeze
|
||||||
|
end
|
||||||
|
end
|
|
@ -232,7 +232,7 @@
|
||||||
= f.number_field :container_registry_token_expire_delay, class: 'form-control'
|
= f.number_field :container_registry_token_expire_delay, class: 'form-control'
|
||||||
|
|
||||||
%fieldset
|
%fieldset
|
||||||
%legend Metrics
|
%legend Metrics - Influx
|
||||||
%p
|
%p
|
||||||
Setup InfluxDB to measure a wide variety of statistics like the time spent
|
Setup InfluxDB to measure a wide variety of statistics like the time spent
|
||||||
in running SQL queries. These settings require a
|
in running SQL queries. These settings require a
|
||||||
|
@ -296,6 +296,21 @@
|
||||||
The amount of points to store in a single UDP packet. More points
|
The amount of points to store in a single UDP packet. More points
|
||||||
results in fewer but larger UDP packets being sent.
|
results in fewer but larger UDP packets being sent.
|
||||||
|
|
||||||
|
%fieldset
|
||||||
|
%legend Metrics - Prometheus
|
||||||
|
%p
|
||||||
|
Setup Prometheus to measure a variety of statistics that partially overlap and complement Influx based metrics.
|
||||||
|
This setting requires a
|
||||||
|
= link_to 'restart', help_page_path('administration/restart_gitlab')
|
||||||
|
to take effect.
|
||||||
|
= link_to icon('question-circle'), help_page_path('administration/monitoring/performance/introduction')
|
||||||
|
.form-group
|
||||||
|
.col-sm-offset-2.col-sm-10
|
||||||
|
.checkbox
|
||||||
|
= f.label :prometheus_metrics_enabled do
|
||||||
|
= f.check_box :prometheus_metrics_enabled
|
||||||
|
Enable Prometheus Metrics
|
||||||
|
|
||||||
%fieldset
|
%fieldset
|
||||||
%legend Background Jobs
|
%legend Background Jobs
|
||||||
%p
|
%p
|
||||||
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
---
|
||||||
|
title: Add prometheus based metrics collection to gitlab webapp
|
||||||
|
merge_request:
|
||||||
|
author:
|
|
@ -15,6 +15,9 @@ if defined?(Unicorn)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# set default directory for multiproces metrics gathering
|
||||||
|
ENV['prometheus_multiproc_dir'] ||= 'tmp/prometheus_multiproc_dir'
|
||||||
|
|
||||||
require ::File.expand_path('../config/environment', __FILE__)
|
require ::File.expand_path('../config/environment', __FILE__)
|
||||||
|
|
||||||
map ENV['RAILS_RELATIVE_URL_ROOT'] || "/" do
|
map ENV['RAILS_RELATIVE_URL_ROOT'] || "/" do
|
||||||
|
|
|
@ -38,10 +38,10 @@ Rails.application.routes.draw do
|
||||||
# Health check
|
# Health check
|
||||||
get 'health_check(/:checks)' => 'health_check#index', as: :health_check
|
get 'health_check(/:checks)' => 'health_check#index', as: :health_check
|
||||||
|
|
||||||
scope path: '-', controller: 'health' do
|
scope path: '-' do
|
||||||
get :liveness
|
get 'liveness' => 'health#liveness'
|
||||||
get :readiness
|
get 'readiness' => 'health#readiness'
|
||||||
get :metrics
|
resources :metrics, only: [:index]
|
||||||
end
|
end
|
||||||
|
|
||||||
# Koding route
|
# Koding route
|
||||||
|
|
|
@ -1,16 +1,26 @@
|
||||||
if ENV['GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN'].present?
|
def save(settings, topic)
|
||||||
settings = ApplicationSetting.current || ApplicationSetting.create_from_defaults
|
|
||||||
settings.set_runners_registration_token(ENV['GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN'])
|
|
||||||
|
|
||||||
if settings.save
|
if settings.save
|
||||||
puts "Saved Runner Registration Token".color(:green)
|
puts "Saved #{topic}".color(:green)
|
||||||
else
|
else
|
||||||
puts "Could not save Runner Registration Token".color(:red)
|
puts "Could not save #{topic}".color(:red)
|
||||||
puts
|
puts
|
||||||
settings.errors.full_messages.map do |message|
|
settings.errors.full_messages.map do |message|
|
||||||
puts "--> #{message}".color(:red)
|
puts "--> #{message}".color(:red)
|
||||||
end
|
end
|
||||||
puts
|
puts
|
||||||
exit 1
|
exit(1)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if ENV['GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN'].present?
|
||||||
|
settings = Gitlab::CurrentSettings.current_application_settings
|
||||||
|
settings.set_runners_registration_token(ENV['GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN'])
|
||||||
|
save(settings, 'Runner Registration Token')
|
||||||
|
end
|
||||||
|
|
||||||
|
if ENV['GITLAB_PROMETHEUS_METRICS_ENABLED'].present?
|
||||||
|
settings = Gitlab::CurrentSettings.current_application_settings
|
||||||
|
value = Gitlab::Utils.to_boolean(ENV['GITLAB_PROMETHEUS_METRICS_ENABLED']) || false
|
||||||
|
settings.prometheus_metrics_enabled = value
|
||||||
|
save(settings, 'Prometheus metrics enabled flag')
|
||||||
|
end
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
class AddPrometheusSettingsToMetricsSettings < ActiveRecord::Migration
|
||||||
|
include Gitlab::Database::MigrationHelpers
|
||||||
|
|
||||||
|
disable_ddl_transaction!
|
||||||
|
|
||||||
|
DOWNTIME = false
|
||||||
|
|
||||||
|
def up
|
||||||
|
add_column_with_default(:application_settings, :prometheus_metrics_enabled, :boolean,
|
||||||
|
default: false, allow_null: false)
|
||||||
|
end
|
||||||
|
|
||||||
|
def down
|
||||||
|
remove_column(:application_settings, :prometheus_metrics_enabled)
|
||||||
|
end
|
||||||
|
end
|
|
@ -123,6 +123,7 @@ ActiveRecord::Schema.define(version: 20170525174156) do
|
||||||
t.integer "cached_markdown_version"
|
t.integer "cached_markdown_version"
|
||||||
t.boolean "clientside_sentry_enabled", default: false, null: false
|
t.boolean "clientside_sentry_enabled", default: false, null: false
|
||||||
t.string "clientside_sentry_dsn"
|
t.string "clientside_sentry_dsn"
|
||||||
|
t.boolean "prometheus_metrics_enabled", default: false, null: false
|
||||||
end
|
end
|
||||||
|
|
||||||
create_table "audit_events", force: :cascade do |t|
|
create_table "audit_events", force: :cascade do |t|
|
||||||
|
|
|
@ -110,6 +110,7 @@ module API
|
||||||
optional :default_artifacts_expire_in, type: String, desc: "Set the default expiration time for each job's artifacts"
|
optional :default_artifacts_expire_in, type: String, desc: "Set the default expiration time for each job's artifacts"
|
||||||
optional :max_pages_size, type: Integer, desc: 'Maximum size of pages in MB'
|
optional :max_pages_size, type: Integer, desc: 'Maximum size of pages in MB'
|
||||||
optional :container_registry_token_expire_delay, type: Integer, desc: 'Authorization token duration (minutes)'
|
optional :container_registry_token_expire_delay, type: Integer, desc: 'Authorization token duration (minutes)'
|
||||||
|
optional :prometheus_metrics_enabled, type: Boolean, desc: 'Enable Prometheus metrics'
|
||||||
optional :metrics_enabled, type: Boolean, desc: 'Enable the InfluxDB metrics'
|
optional :metrics_enabled, type: Boolean, desc: 'Enable the InfluxDB metrics'
|
||||||
given metrics_enabled: ->(val) { val } do
|
given metrics_enabled: ->(val) { val } do
|
||||||
requires :metrics_host, type: String, desc: 'The InfluxDB host'
|
requires :metrics_host, type: String, desc: 'The InfluxDB host'
|
||||||
|
|
40
lib/gitlab/health_checks/prometheus_text_format.rb
Normal file
40
lib/gitlab/health_checks/prometheus_text_format.rb
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
module Gitlab
|
||||||
|
module HealthChecks
|
||||||
|
class PrometheusTextFormat
|
||||||
|
def marshal(metrics)
|
||||||
|
"#{metrics_with_type_declarations(metrics).join("\n")}\n"
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def metrics_with_type_declarations(metrics)
|
||||||
|
type_declaration_added = {}
|
||||||
|
|
||||||
|
metrics.flat_map do |metric|
|
||||||
|
metric_lines = []
|
||||||
|
|
||||||
|
unless type_declaration_added.has_key?(metric.name)
|
||||||
|
type_declaration_added[metric.name] = true
|
||||||
|
metric_lines << metric_type_declaration(metric)
|
||||||
|
end
|
||||||
|
|
||||||
|
metric_lines << metric_text(metric)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def metric_type_declaration(metric)
|
||||||
|
"# TYPE #{metric.name} gauge"
|
||||||
|
end
|
||||||
|
|
||||||
|
def metric_text(metric)
|
||||||
|
labels = metric.labels&.map { |key, value| "#{key}=\"#{value}\"" }&.join(',') || ''
|
||||||
|
|
||||||
|
if labels.empty?
|
||||||
|
"#{metric.name} #{metric.value}"
|
||||||
|
else
|
||||||
|
"#{metric.name}{#{labels}} #{metric.value}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,161 +1,10 @@
|
||||||
module Gitlab
|
module Gitlab
|
||||||
module Metrics
|
module Metrics
|
||||||
extend Gitlab::CurrentSettings
|
extend Gitlab::Metrics::InfluxDb
|
||||||
|
extend Gitlab::Metrics::Prometheus
|
||||||
RAILS_ROOT = Rails.root.to_s
|
|
||||||
METRICS_ROOT = Rails.root.join('lib', 'gitlab', 'metrics').to_s
|
|
||||||
PATH_REGEX = /^#{RAILS_ROOT}\/?/
|
|
||||||
|
|
||||||
def self.settings
|
|
||||||
@settings ||= {
|
|
||||||
enabled: current_application_settings[:metrics_enabled],
|
|
||||||
pool_size: current_application_settings[:metrics_pool_size],
|
|
||||||
timeout: current_application_settings[:metrics_timeout],
|
|
||||||
method_call_threshold: current_application_settings[:metrics_method_call_threshold],
|
|
||||||
host: current_application_settings[:metrics_host],
|
|
||||||
port: current_application_settings[:metrics_port],
|
|
||||||
sample_interval: current_application_settings[:metrics_sample_interval] || 15,
|
|
||||||
packet_size: current_application_settings[:metrics_packet_size] || 1
|
|
||||||
}
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.enabled?
|
def self.enabled?
|
||||||
settings[:enabled] || false
|
influx_metrics_enabled? || prometheus_metrics_enabled?
|
||||||
end
|
|
||||||
|
|
||||||
def self.mri?
|
|
||||||
RUBY_ENGINE == 'ruby'
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.method_call_threshold
|
|
||||||
# This is memoized since this method is called for every instrumented
|
|
||||||
# method. Loading data from an external cache on every method call slows
|
|
||||||
# things down too much.
|
|
||||||
@method_call_threshold ||= settings[:method_call_threshold]
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.pool
|
|
||||||
@pool
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.submit_metrics(metrics)
|
|
||||||
prepared = prepare_metrics(metrics)
|
|
||||||
|
|
||||||
pool.with do |connection|
|
|
||||||
prepared.each_slice(settings[:packet_size]) do |slice|
|
|
||||||
begin
|
|
||||||
connection.write_points(slice)
|
|
||||||
rescue StandardError
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
rescue Errno::EADDRNOTAVAIL, SocketError => ex
|
|
||||||
Gitlab::EnvironmentLogger.error('Cannot resolve InfluxDB address. GitLab Performance Monitoring will not work.')
|
|
||||||
Gitlab::EnvironmentLogger.error(ex)
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.prepare_metrics(metrics)
|
|
||||||
metrics.map do |hash|
|
|
||||||
new_hash = hash.symbolize_keys
|
|
||||||
|
|
||||||
new_hash[:tags].each do |key, value|
|
|
||||||
if value.blank?
|
|
||||||
new_hash[:tags].delete(key)
|
|
||||||
else
|
|
||||||
new_hash[:tags][key] = escape_value(value)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
new_hash
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def self.escape_value(value)
|
|
||||||
value.to_s.gsub('=', '\\=')
|
|
||||||
end
|
|
||||||
|
|
||||||
# Measures the execution time of a block.
|
|
||||||
#
|
|
||||||
# Example:
|
|
||||||
#
|
|
||||||
# Gitlab::Metrics.measure(:find_by_username_duration) do
|
|
||||||
# User.find_by_username(some_username)
|
|
||||||
# end
|
|
||||||
#
|
|
||||||
# name - The name of the field to store the execution time in.
|
|
||||||
#
|
|
||||||
# Returns the value yielded by the supplied block.
|
|
||||||
def self.measure(name)
|
|
||||||
trans = current_transaction
|
|
||||||
|
|
||||||
return yield unless trans
|
|
||||||
|
|
||||||
real_start = Time.now.to_f
|
|
||||||
cpu_start = System.cpu_time
|
|
||||||
|
|
||||||
retval = yield
|
|
||||||
|
|
||||||
cpu_stop = System.cpu_time
|
|
||||||
real_stop = Time.now.to_f
|
|
||||||
|
|
||||||
real_time = (real_stop - real_start) * 1000.0
|
|
||||||
cpu_time = cpu_stop - cpu_start
|
|
||||||
|
|
||||||
trans.increment("#{name}_real_time", real_time)
|
|
||||||
trans.increment("#{name}_cpu_time", cpu_time)
|
|
||||||
trans.increment("#{name}_call_count", 1)
|
|
||||||
|
|
||||||
retval
|
|
||||||
end
|
|
||||||
|
|
||||||
# Adds a tag to the current transaction (if any)
|
|
||||||
#
|
|
||||||
# name - The name of the tag to add.
|
|
||||||
# value - The value of the tag.
|
|
||||||
def self.tag_transaction(name, value)
|
|
||||||
trans = current_transaction
|
|
||||||
|
|
||||||
trans&.add_tag(name, value)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Sets the action of the current transaction (if any)
|
|
||||||
#
|
|
||||||
# action - The name of the action.
|
|
||||||
def self.action=(action)
|
|
||||||
trans = current_transaction
|
|
||||||
|
|
||||||
trans&.action = action
|
|
||||||
end
|
|
||||||
|
|
||||||
# Tracks an event.
|
|
||||||
#
|
|
||||||
# See `Gitlab::Metrics::Transaction#add_event` for more details.
|
|
||||||
def self.add_event(*args)
|
|
||||||
trans = current_transaction
|
|
||||||
|
|
||||||
trans&.add_event(*args)
|
|
||||||
end
|
|
||||||
|
|
||||||
# Returns the prefix to use for the name of a series.
|
|
||||||
def self.series_prefix
|
|
||||||
@series_prefix ||= Sidekiq.server? ? 'sidekiq_' : 'rails_'
|
|
||||||
end
|
|
||||||
|
|
||||||
# Allow access from other metrics related middlewares
|
|
||||||
def self.current_transaction
|
|
||||||
Transaction.current
|
|
||||||
end
|
|
||||||
|
|
||||||
# When enabled this should be set before being used as the usual pattern
|
|
||||||
# "@foo ||= bar" is _not_ thread-safe.
|
|
||||||
if enabled?
|
|
||||||
@pool = ConnectionPool.new(size: settings[:pool_size], timeout: settings[:timeout]) do
|
|
||||||
host = settings[:host]
|
|
||||||
port = settings[:port]
|
|
||||||
|
|
||||||
InfluxDB::Client.
|
|
||||||
new(udp: { host: host, port: port })
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
170
lib/gitlab/metrics/influx_db.rb
Normal file
170
lib/gitlab/metrics/influx_db.rb
Normal file
|
@ -0,0 +1,170 @@
|
||||||
|
module Gitlab
|
||||||
|
module Metrics
|
||||||
|
module InfluxDb
|
||||||
|
extend Gitlab::CurrentSettings
|
||||||
|
extend self
|
||||||
|
|
||||||
|
MUTEX = Mutex.new
|
||||||
|
private_constant :MUTEX
|
||||||
|
|
||||||
|
def influx_metrics_enabled?
|
||||||
|
settings[:enabled] || false
|
||||||
|
end
|
||||||
|
|
||||||
|
RAILS_ROOT = Rails.root.to_s
|
||||||
|
METRICS_ROOT = Rails.root.join('lib', 'gitlab', 'metrics').to_s
|
||||||
|
PATH_REGEX = /^#{RAILS_ROOT}\/?/
|
||||||
|
|
||||||
|
def settings
|
||||||
|
@settings ||= {
|
||||||
|
enabled: current_application_settings[:metrics_enabled],
|
||||||
|
pool_size: current_application_settings[:metrics_pool_size],
|
||||||
|
timeout: current_application_settings[:metrics_timeout],
|
||||||
|
method_call_threshold: current_application_settings[:metrics_method_call_threshold],
|
||||||
|
host: current_application_settings[:metrics_host],
|
||||||
|
port: current_application_settings[:metrics_port],
|
||||||
|
sample_interval: current_application_settings[:metrics_sample_interval] || 15,
|
||||||
|
packet_size: current_application_settings[:metrics_packet_size] || 1
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def mri?
|
||||||
|
RUBY_ENGINE == 'ruby'
|
||||||
|
end
|
||||||
|
|
||||||
|
def method_call_threshold
|
||||||
|
# This is memoized since this method is called for every instrumented
|
||||||
|
# method. Loading data from an external cache on every method call slows
|
||||||
|
# things down too much.
|
||||||
|
@method_call_threshold ||= settings[:method_call_threshold]
|
||||||
|
end
|
||||||
|
|
||||||
|
def submit_metrics(metrics)
|
||||||
|
prepared = prepare_metrics(metrics)
|
||||||
|
|
||||||
|
pool&.with do |connection|
|
||||||
|
prepared.each_slice(settings[:packet_size]) do |slice|
|
||||||
|
begin
|
||||||
|
connection.write_points(slice)
|
||||||
|
rescue StandardError
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
rescue Errno::EADDRNOTAVAIL, SocketError => ex
|
||||||
|
Gitlab::EnvironmentLogger.error('Cannot resolve InfluxDB address. GitLab Performance Monitoring will not work.')
|
||||||
|
Gitlab::EnvironmentLogger.error(ex)
|
||||||
|
end
|
||||||
|
|
||||||
|
def prepare_metrics(metrics)
|
||||||
|
metrics.map do |hash|
|
||||||
|
new_hash = hash.symbolize_keys
|
||||||
|
|
||||||
|
new_hash[:tags].each do |key, value|
|
||||||
|
if value.blank?
|
||||||
|
new_hash[:tags].delete(key)
|
||||||
|
else
|
||||||
|
new_hash[:tags][key] = escape_value(value)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
new_hash
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def escape_value(value)
|
||||||
|
value.to_s.gsub('=', '\\=')
|
||||||
|
end
|
||||||
|
|
||||||
|
# Measures the execution time of a block.
|
||||||
|
#
|
||||||
|
# Example:
|
||||||
|
#
|
||||||
|
# Gitlab::Metrics.measure(:find_by_username_duration) do
|
||||||
|
# User.find_by_username(some_username)
|
||||||
|
# end
|
||||||
|
#
|
||||||
|
# name - The name of the field to store the execution time in.
|
||||||
|
#
|
||||||
|
# Returns the value yielded by the supplied block.
|
||||||
|
def measure(name)
|
||||||
|
trans = current_transaction
|
||||||
|
|
||||||
|
return yield unless trans
|
||||||
|
|
||||||
|
real_start = Time.now.to_f
|
||||||
|
cpu_start = System.cpu_time
|
||||||
|
|
||||||
|
retval = yield
|
||||||
|
|
||||||
|
cpu_stop = System.cpu_time
|
||||||
|
real_stop = Time.now.to_f
|
||||||
|
|
||||||
|
real_time = (real_stop - real_start) * 1000.0
|
||||||
|
cpu_time = cpu_stop - cpu_start
|
||||||
|
|
||||||
|
trans.increment("#{name}_real_time", real_time)
|
||||||
|
trans.increment("#{name}_cpu_time", cpu_time)
|
||||||
|
trans.increment("#{name}_call_count", 1)
|
||||||
|
|
||||||
|
retval
|
||||||
|
end
|
||||||
|
|
||||||
|
# Adds a tag to the current transaction (if any)
|
||||||
|
#
|
||||||
|
# name - The name of the tag to add.
|
||||||
|
# value - The value of the tag.
|
||||||
|
def tag_transaction(name, value)
|
||||||
|
trans = current_transaction
|
||||||
|
|
||||||
|
trans&.add_tag(name, value)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Sets the action of the current transaction (if any)
|
||||||
|
#
|
||||||
|
# action - The name of the action.
|
||||||
|
def action=(action)
|
||||||
|
trans = current_transaction
|
||||||
|
|
||||||
|
trans&.action = action
|
||||||
|
end
|
||||||
|
|
||||||
|
# Tracks an event.
|
||||||
|
#
|
||||||
|
# See `Gitlab::Metrics::Transaction#add_event` for more details.
|
||||||
|
def add_event(*args)
|
||||||
|
trans = current_transaction
|
||||||
|
|
||||||
|
trans&.add_event(*args)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Returns the prefix to use for the name of a series.
|
||||||
|
def series_prefix
|
||||||
|
@series_prefix ||= Sidekiq.server? ? 'sidekiq_' : 'rails_'
|
||||||
|
end
|
||||||
|
|
||||||
|
# Allow access from other metrics related middlewares
|
||||||
|
def current_transaction
|
||||||
|
Transaction.current
|
||||||
|
end
|
||||||
|
|
||||||
|
# When enabled this should be set before being used as the usual pattern
|
||||||
|
# "@foo ||= bar" is _not_ thread-safe.
|
||||||
|
def pool
|
||||||
|
if influx_metrics_enabled?
|
||||||
|
if @pool.nil?
|
||||||
|
MUTEX.synchronize do
|
||||||
|
@pool ||= ConnectionPool.new(size: settings[:pool_size], timeout: settings[:timeout]) do
|
||||||
|
host = settings[:host]
|
||||||
|
port = settings[:port]
|
||||||
|
|
||||||
|
InfluxDB::Client.
|
||||||
|
new(udp: { host: host, port: port })
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
@pool
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
10
lib/gitlab/metrics/null_metric.rb
Normal file
10
lib/gitlab/metrics/null_metric.rb
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
module Gitlab
|
||||||
|
module Metrics
|
||||||
|
# Mocks ::Prometheus::Client::Metric and all derived metrics
|
||||||
|
class NullMetric
|
||||||
|
def method_missing(name, *args, &block)
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
41
lib/gitlab/metrics/prometheus.rb
Normal file
41
lib/gitlab/metrics/prometheus.rb
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
require 'prometheus/client'
|
||||||
|
|
||||||
|
module Gitlab
|
||||||
|
module Metrics
|
||||||
|
module Prometheus
|
||||||
|
include Gitlab::CurrentSettings
|
||||||
|
|
||||||
|
def prometheus_metrics_enabled?
|
||||||
|
@prometheus_metrics_enabled ||= current_application_settings[:prometheus_metrics_enabled] || false
|
||||||
|
end
|
||||||
|
|
||||||
|
def registry
|
||||||
|
@registry ||= ::Prometheus::Client.registry
|
||||||
|
end
|
||||||
|
|
||||||
|
def counter(name, docstring, base_labels = {})
|
||||||
|
provide_metric(name) || registry.counter(name, docstring, base_labels)
|
||||||
|
end
|
||||||
|
|
||||||
|
def summary(name, docstring, base_labels = {})
|
||||||
|
provide_metric(name) || registry.summary(name, docstring, base_labels)
|
||||||
|
end
|
||||||
|
|
||||||
|
def gauge(name, docstring, base_labels = {})
|
||||||
|
provide_metric(name) || registry.gauge(name, docstring, base_labels)
|
||||||
|
end
|
||||||
|
|
||||||
|
def histogram(name, docstring, base_labels = {}, buckets = ::Prometheus::Client::Histogram::DEFAULT_BUCKETS)
|
||||||
|
provide_metric(name) || registry.histogram(name, docstring, base_labels, buckets)
|
||||||
|
end
|
||||||
|
|
||||||
|
def provide_metric(name)
|
||||||
|
if prometheus_metrics_enabled?
|
||||||
|
registry.get(name)
|
||||||
|
else
|
||||||
|
NullMetric.new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -54,43 +54,4 @@ describe HealthController do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe '#metrics' do
|
|
||||||
context 'authorization token provided' do
|
|
||||||
before do
|
|
||||||
request.headers['TOKEN'] = token
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'returns DB ping metrics' do
|
|
||||||
get :metrics
|
|
||||||
expect(response.body).to match(/^db_ping_timeout 0$/)
|
|
||||||
expect(response.body).to match(/^db_ping_success 1$/)
|
|
||||||
expect(response.body).to match(/^db_ping_latency [0-9\.]+$/)
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'returns Redis ping metrics' do
|
|
||||||
get :metrics
|
|
||||||
expect(response.body).to match(/^redis_ping_timeout 0$/)
|
|
||||||
expect(response.body).to match(/^redis_ping_success 1$/)
|
|
||||||
expect(response.body).to match(/^redis_ping_latency [0-9\.]+$/)
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'returns file system check metrics' do
|
|
||||||
get :metrics
|
|
||||||
expect(response.body).to match(/^filesystem_access_latency{shard="default"} [0-9\.]+$/)
|
|
||||||
expect(response.body).to match(/^filesystem_accessible{shard="default"} 1$/)
|
|
||||||
expect(response.body).to match(/^filesystem_write_latency{shard="default"} [0-9\.]+$/)
|
|
||||||
expect(response.body).to match(/^filesystem_writable{shard="default"} 1$/)
|
|
||||||
expect(response.body).to match(/^filesystem_read_latency{shard="default"} [0-9\.]+$/)
|
|
||||||
expect(response.body).to match(/^filesystem_readable{shard="default"} 1$/)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
context 'without authorization token' do
|
|
||||||
it 'returns proper response' do
|
|
||||||
get :metrics
|
|
||||||
expect(response.status).to eq(404)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|
70
spec/controllers/metrics_controller_spec.rb
Normal file
70
spec/controllers/metrics_controller_spec.rb
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
require 'spec_helper'
|
||||||
|
|
||||||
|
describe MetricsController do
|
||||||
|
include StubENV
|
||||||
|
|
||||||
|
let(:token) { current_application_settings.health_check_access_token }
|
||||||
|
let(:json_response) { JSON.parse(response.body) }
|
||||||
|
let(:metrics_multiproc_dir) { Dir.mktmpdir }
|
||||||
|
|
||||||
|
before do
|
||||||
|
stub_env('IN_MEMORY_APPLICATION_SETTINGS', 'false')
|
||||||
|
stub_env('prometheus_multiproc_dir', metrics_multiproc_dir)
|
||||||
|
allow(Gitlab::Metrics).to receive(:prometheus_metrics_enabled?).and_return(true)
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#index' do
|
||||||
|
context 'authorization token provided' do
|
||||||
|
before do
|
||||||
|
request.headers['TOKEN'] = token
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns DB ping metrics' do
|
||||||
|
get :index
|
||||||
|
|
||||||
|
expect(response.body).to match(/^db_ping_timeout 0$/)
|
||||||
|
expect(response.body).to match(/^db_ping_success 1$/)
|
||||||
|
expect(response.body).to match(/^db_ping_latency [0-9\.]+$/)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns Redis ping metrics' do
|
||||||
|
get :index
|
||||||
|
|
||||||
|
expect(response.body).to match(/^redis_ping_timeout 0$/)
|
||||||
|
expect(response.body).to match(/^redis_ping_success 1$/)
|
||||||
|
expect(response.body).to match(/^redis_ping_latency [0-9\.]+$/)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns file system check metrics' do
|
||||||
|
get :index
|
||||||
|
|
||||||
|
expect(response.body).to match(/^filesystem_access_latency{shard="default"} [0-9\.]+$/)
|
||||||
|
expect(response.body).to match(/^filesystem_accessible{shard="default"} 1$/)
|
||||||
|
expect(response.body).to match(/^filesystem_write_latency{shard="default"} [0-9\.]+$/)
|
||||||
|
expect(response.body).to match(/^filesystem_writable{shard="default"} 1$/)
|
||||||
|
expect(response.body).to match(/^filesystem_read_latency{shard="default"} [0-9\.]+$/)
|
||||||
|
expect(response.body).to match(/^filesystem_readable{shard="default"} 1$/)
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'prometheus metrics are disabled' do
|
||||||
|
before do
|
||||||
|
allow(Gitlab::Metrics).to receive(:prometheus_metrics_enabled?).and_return(false)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns proper response' do
|
||||||
|
get :index
|
||||||
|
|
||||||
|
expect(response.status).to eq(404)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'without authorization token' do
|
||||||
|
it 'returns proper response' do
|
||||||
|
get :index
|
||||||
|
|
||||||
|
expect(response.status).to eq(404)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,16 +0,0 @@
|
||||||
require 'spec_helper'
|
|
||||||
|
|
||||||
describe 'seed production settings', lib: true do
|
|
||||||
include StubENV
|
|
||||||
|
|
||||||
context 'GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN is set in the environment' do
|
|
||||||
before do
|
|
||||||
stub_env('GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN', '013456789')
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'writes the token to the database' do
|
|
||||||
load(File.join(__dir__, '../../../db/fixtures/production/010_settings.rb'))
|
|
||||||
expect(ApplicationSetting.current.runners_registration_token).to eq('013456789')
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
58
spec/db/production/settings_spec.rb
Normal file
58
spec/db/production/settings_spec.rb
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
require 'spec_helper'
|
||||||
|
require 'rainbow/ext/string'
|
||||||
|
|
||||||
|
describe 'seed production settings', lib: true do
|
||||||
|
include StubENV
|
||||||
|
let(:settings_file) { Rails.root.join('db/fixtures/production/010_settings.rb') }
|
||||||
|
let(:settings) { Gitlab::CurrentSettings.current_application_settings }
|
||||||
|
|
||||||
|
context 'GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN is set in the environment' do
|
||||||
|
before do
|
||||||
|
stub_env('GITLAB_SHARED_RUNNERS_REGISTRATION_TOKEN', '013456789')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'writes the token to the database' do
|
||||||
|
load(settings_file)
|
||||||
|
|
||||||
|
expect(settings.runners_registration_token).to eq('013456789')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'GITLAB_PROMETHEUS_METRICS_ENABLED is set in the environment' do
|
||||||
|
context 'GITLAB_PROMETHEUS_METRICS_ENABLED is true' do
|
||||||
|
before do
|
||||||
|
stub_env('GITLAB_PROMETHEUS_METRICS_ENABLED', 'true')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'prometheus_metrics_enabled is set to true ' do
|
||||||
|
load(settings_file)
|
||||||
|
|
||||||
|
expect(settings.prometheus_metrics_enabled).to eq(true)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'GITLAB_PROMETHEUS_METRICS_ENABLED is false' do
|
||||||
|
before do
|
||||||
|
stub_env('GITLAB_PROMETHEUS_METRICS_ENABLED', 'false')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'prometheus_metrics_enabled is set to false' do
|
||||||
|
load(settings_file)
|
||||||
|
|
||||||
|
expect(settings.prometheus_metrics_enabled).to eq(false)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'GITLAB_PROMETHEUS_METRICS_ENABLED is false' do
|
||||||
|
before do
|
||||||
|
stub_env('GITLAB_PROMETHEUS_METRICS_ENABLED', '')
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'prometheus_metrics_enabled is set to false' do
|
||||||
|
load(settings_file)
|
||||||
|
|
||||||
|
expect(settings.prometheus_metrics_enabled).to eq(false)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
41
spec/lib/gitlab/health_checks/prometheus_text_format_spec.rb
Normal file
41
spec/lib/gitlab/health_checks/prometheus_text_format_spec.rb
Normal file
|
@ -0,0 +1,41 @@
|
||||||
|
describe Gitlab::HealthChecks::PrometheusTextFormat do
|
||||||
|
let(:metric_class) { Gitlab::HealthChecks::Metric }
|
||||||
|
subject { described_class.new }
|
||||||
|
|
||||||
|
describe '#marshal' do
|
||||||
|
let(:sample_metrics) do
|
||||||
|
[metric_class.new('metric1', 1),
|
||||||
|
metric_class.new('metric2', 2)]
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'marshal to text with non repeating type definition' do
|
||||||
|
expected = <<-EXPECTED.strip_heredoc
|
||||||
|
# TYPE metric1 gauge
|
||||||
|
metric1 1
|
||||||
|
# TYPE metric2 gauge
|
||||||
|
metric2 2
|
||||||
|
EXPECTED
|
||||||
|
|
||||||
|
expect(subject.marshal(sample_metrics)).to eq(expected)
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'metrics where name repeats' do
|
||||||
|
let(:sample_metrics) do
|
||||||
|
[metric_class.new('metric1', 1),
|
||||||
|
metric_class.new('metric1', 2),
|
||||||
|
metric_class.new('metric2', 3)]
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'marshal to text with non repeating type definition' do
|
||||||
|
expected = <<-EXPECTED.strip_heredoc
|
||||||
|
# TYPE metric1 gauge
|
||||||
|
metric1 1
|
||||||
|
metric1 2
|
||||||
|
# TYPE metric2 gauge
|
||||||
|
metric2 3
|
||||||
|
EXPECTED
|
||||||
|
expect(subject.marshal(sample_metrics)).to eq(expected)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -1,6 +1,8 @@
|
||||||
require 'spec_helper'
|
require 'spec_helper'
|
||||||
|
|
||||||
describe Gitlab::Metrics do
|
describe Gitlab::Metrics do
|
||||||
|
include StubENV
|
||||||
|
|
||||||
describe '.settings' do
|
describe '.settings' do
|
||||||
it 'returns a Hash' do
|
it 'returns a Hash' do
|
||||||
expect(described_class.settings).to be_an_instance_of(Hash)
|
expect(described_class.settings).to be_an_instance_of(Hash)
|
||||||
|
@ -9,7 +11,19 @@ describe Gitlab::Metrics do
|
||||||
|
|
||||||
describe '.enabled?' do
|
describe '.enabled?' do
|
||||||
it 'returns a boolean' do
|
it 'returns a boolean' do
|
||||||
expect([true, false].include?(described_class.enabled?)).to eq(true)
|
expect(described_class.enabled?).to be_in([true, false])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '.prometheus_metrics_enabled?' do
|
||||||
|
it 'returns a boolean' do
|
||||||
|
expect(described_class.prometheus_metrics_enabled?).to be_in([true, false])
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '.influx_metrics_enabled?' do
|
||||||
|
it 'returns a boolean' do
|
||||||
|
expect(described_class.influx_metrics_enabled?).to be_in([true, false])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -177,4 +191,133 @@ describe Gitlab::Metrics do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
shared_examples 'prometheus metrics API' do
|
||||||
|
describe '#counter' do
|
||||||
|
subject { described_class.counter(:couter, 'doc') }
|
||||||
|
|
||||||
|
describe '#increment' do
|
||||||
|
it 'successfully calls #increment without arguments' do
|
||||||
|
expect { subject.increment }.not_to raise_exception
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'successfully calls #increment with 1 argument' do
|
||||||
|
expect { subject.increment({}) }.not_to raise_exception
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'successfully calls #increment with 2 arguments' do
|
||||||
|
expect { subject.increment({}, 1) }.not_to raise_exception
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#summary' do
|
||||||
|
subject { described_class.summary(:summary, 'doc') }
|
||||||
|
|
||||||
|
describe '#observe' do
|
||||||
|
it 'successfully calls #observe with 2 arguments' do
|
||||||
|
expect { subject.observe({}, 2) }.not_to raise_exception
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#gauge' do
|
||||||
|
subject { described_class.gauge(:gauge, 'doc') }
|
||||||
|
|
||||||
|
describe '#set' do
|
||||||
|
it 'successfully calls #set with 2 arguments' do
|
||||||
|
expect { subject.set({}, 1) }.not_to raise_exception
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#histogram' do
|
||||||
|
subject { described_class.histogram(:histogram, 'doc') }
|
||||||
|
|
||||||
|
describe '#observe' do
|
||||||
|
it 'successfully calls #observe with 2 arguments' do
|
||||||
|
expect { subject.observe({}, 2) }.not_to raise_exception
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'prometheus metrics disabled' do
|
||||||
|
before do
|
||||||
|
allow(described_class).to receive(:prometheus_metrics_enabled?).and_return(false)
|
||||||
|
end
|
||||||
|
|
||||||
|
it_behaves_like 'prometheus metrics API'
|
||||||
|
|
||||||
|
describe '#null_metric' do
|
||||||
|
subject { described_class.provide_metric(:test) }
|
||||||
|
|
||||||
|
it { is_expected.to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#counter' do
|
||||||
|
subject { described_class.counter(:counter, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#summary' do
|
||||||
|
subject { described_class.summary(:summary, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#gauge' do
|
||||||
|
subject { described_class.gauge(:gauge, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#histogram' do
|
||||||
|
subject { described_class.histogram(:histogram, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'prometheus metrics enabled' do
|
||||||
|
let(:metrics_multiproc_dir) { Dir.mktmpdir }
|
||||||
|
|
||||||
|
before do
|
||||||
|
stub_const('Prometheus::Client::Multiprocdir', metrics_multiproc_dir)
|
||||||
|
allow(described_class).to receive(:prometheus_metrics_enabled?).and_return(true)
|
||||||
|
end
|
||||||
|
|
||||||
|
it_behaves_like 'prometheus metrics API'
|
||||||
|
|
||||||
|
describe '#null_metric' do
|
||||||
|
subject { described_class.provide_metric(:test) }
|
||||||
|
|
||||||
|
it { is_expected.to be_nil }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#counter' do
|
||||||
|
subject { described_class.counter(:name, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.not_to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#summary' do
|
||||||
|
subject { described_class.summary(:name, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.not_to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#gauge' do
|
||||||
|
subject { described_class.gauge(:name, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.not_to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#histogram' do
|
||||||
|
subject { described_class.histogram(:name, 'doc') }
|
||||||
|
|
||||||
|
it { is_expected.not_to be_a(Gitlab::Metrics::NullMetric) }
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -3,6 +3,7 @@ SimpleCovEnv.start!
|
||||||
|
|
||||||
ENV["RAILS_ENV"] ||= 'test'
|
ENV["RAILS_ENV"] ||= 'test'
|
||||||
ENV["IN_MEMORY_APPLICATION_SETTINGS"] = 'true'
|
ENV["IN_MEMORY_APPLICATION_SETTINGS"] = 'true'
|
||||||
|
# ENV['prometheus_multiproc_dir'] = 'tmp/prometheus_multiproc_dir_test'
|
||||||
|
|
||||||
require File.expand_path("../../config/environment", __FILE__)
|
require File.expand_path("../../config/environment", __FILE__)
|
||||||
require 'rspec/rails'
|
require 'rspec/rails'
|
||||||
|
|
0
tmp/prometheus_multiproc_dir/.gitkeep
Normal file
0
tmp/prometheus_multiproc_dir/.gitkeep
Normal file
Loading…
Reference in a new issue