mirror of
https://github.com/puppetlabs/vmpooler.git
synced 2026-01-26 01:58:41 -05:00
(MAINT) Clarity refactor of Prom Stats code
Introducing the Prometheus Stats code into ABS showed that the Clarity could be improved a bit with better variable naming, some refactoring to reduce repitition and documenting the Metrics table itself. Filtering these changes back to the vmpooler code base.
This commit is contained in:
parent
2556ed6105
commit
0a6ad896f5
5 changed files with 173 additions and 56 deletions
|
|
@ -38,8 +38,8 @@ module Vmpooler
|
||||||
# Using customised collector that filters out hostnames on API paths
|
# Using customised collector that filters out hostnames on API paths
|
||||||
require 'vmpooler/metrics/promstats/collector_middleware'
|
require 'vmpooler/metrics/promstats/collector_middleware'
|
||||||
require 'prometheus/middleware/exporter'
|
require 'prometheus/middleware/exporter'
|
||||||
use Vmpooler::Metrics::Promstats::CollectorMiddleware, metrics_prefix: "#{metrics.metrics_prefix}_http"
|
use Vmpooler::Metrics::Promstats::CollectorMiddleware, metrics_prefix: "#{metrics.prometheus_prefix}_http"
|
||||||
use Prometheus::Middleware::Exporter, path: metrics.endpoint
|
use Prometheus::Middleware::Exporter, path: metrics.prometheus_endpoint
|
||||||
end
|
end
|
||||||
|
|
||||||
if torun.include? :api
|
if torun.include? :api
|
||||||
|
|
|
||||||
|
|
@ -809,7 +809,7 @@ module Vmpooler
|
||||||
|
|
||||||
post "#{api_prefix}/ondemandvm/?" do
|
post "#{api_prefix}/ondemandvm/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.post.ondemand.requestid')
|
metrics.increment('http_requests_vm_total.post.ondemand.requestid')
|
||||||
|
|
||||||
need_token! if Vmpooler::API.settings.config[:auth]
|
need_token! if Vmpooler::API.settings.config[:auth]
|
||||||
|
|
||||||
|
|
@ -847,7 +847,7 @@ module Vmpooler
|
||||||
post "#{api_prefix}/ondemandvm/:template/?" do
|
post "#{api_prefix}/ondemandvm/:template/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
result = { 'ok' => false }
|
result = { 'ok' => false }
|
||||||
metrics.increment('api_vm.delete.ondemand.template')
|
metrics.increment('http_requests_vm_total.delete.ondemand.template')
|
||||||
|
|
||||||
need_token! if Vmpooler::API.settings.config[:auth]
|
need_token! if Vmpooler::API.settings.config[:auth]
|
||||||
|
|
||||||
|
|
@ -874,7 +874,7 @@ module Vmpooler
|
||||||
|
|
||||||
get "#{api_prefix}/ondemandvm/:requestid/?" do
|
get "#{api_prefix}/ondemandvm/:requestid/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.get.ondemand.request')
|
metrics.increment('http_requests_vm_total.get.ondemand.request')
|
||||||
|
|
||||||
status 404
|
status 404
|
||||||
result = check_ondemand_request(params[:requestid])
|
result = check_ondemand_request(params[:requestid])
|
||||||
|
|
@ -885,7 +885,7 @@ module Vmpooler
|
||||||
delete "#{api_prefix}/ondemandvm/:requestid/?" do
|
delete "#{api_prefix}/ondemandvm/:requestid/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
need_token! if Vmpooler::API.settings.config[:auth]
|
need_token! if Vmpooler::API.settings.config[:auth]
|
||||||
metrics.increment('api_vm.delete.ondemand.request')
|
metrics.increment('http_requests_vm_total.delete.ondemand.request')
|
||||||
|
|
||||||
status 404
|
status 404
|
||||||
result = delete_ondemand_request(params[:requestid])
|
result = delete_ondemand_request(params[:requestid])
|
||||||
|
|
@ -896,7 +896,7 @@ module Vmpooler
|
||||||
post "#{api_prefix}/vm/?" do
|
post "#{api_prefix}/vm/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
result = { 'ok' => false }
|
result = { 'ok' => false }
|
||||||
metrics.increment('api_vm.post.vm.checkout')
|
metrics.increment('http_requests_vm_total.post.vm.checkout')
|
||||||
|
|
||||||
payload = JSON.parse(request.body.read)
|
payload = JSON.parse(request.body.read)
|
||||||
|
|
||||||
|
|
@ -1051,7 +1051,7 @@ module Vmpooler
|
||||||
post "#{api_prefix}/vm/:template/?" do
|
post "#{api_prefix}/vm/:template/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
result = { 'ok' => false }
|
result = { 'ok' => false }
|
||||||
metrics.increment('api_vm.get.vm.template')
|
metrics.increment('http_requests_vm_total.get.vm.template')
|
||||||
|
|
||||||
payload = extract_templates_from_query_params(params[:template])
|
payload = extract_templates_from_query_params(params[:template])
|
||||||
|
|
||||||
|
|
@ -1075,7 +1075,7 @@ module Vmpooler
|
||||||
|
|
||||||
get "#{api_prefix}/vm/:hostname/?" do
|
get "#{api_prefix}/vm/:hostname/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.get.vm.hostname')
|
metrics.increment('http_requests_vm_total.get.vm.hostname')
|
||||||
|
|
||||||
result = {}
|
result = {}
|
||||||
|
|
||||||
|
|
@ -1148,7 +1148,7 @@ module Vmpooler
|
||||||
|
|
||||||
delete "#{api_prefix}/vm/:hostname/?" do
|
delete "#{api_prefix}/vm/:hostname/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.delete.vm.hostname')
|
metrics.increment('http_requests_vm_total.delete.vm.hostname')
|
||||||
|
|
||||||
result = {}
|
result = {}
|
||||||
|
|
||||||
|
|
@ -1177,7 +1177,7 @@ module Vmpooler
|
||||||
|
|
||||||
put "#{api_prefix}/vm/:hostname/?" do
|
put "#{api_prefix}/vm/:hostname/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.put.vm.modify')
|
metrics.increment('http_requests_vm_total.put.vm.modify')
|
||||||
|
|
||||||
status 404
|
status 404
|
||||||
result = { 'ok' => false }
|
result = { 'ok' => false }
|
||||||
|
|
@ -1254,7 +1254,7 @@ module Vmpooler
|
||||||
|
|
||||||
post "#{api_prefix}/vm/:hostname/disk/:size/?" do
|
post "#{api_prefix}/vm/:hostname/disk/:size/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.post.vm.disksize')
|
metrics.increment('http_requests_vm_total.post.vm.disksize')
|
||||||
|
|
||||||
need_token! if Vmpooler::API.settings.config[:auth]
|
need_token! if Vmpooler::API.settings.config[:auth]
|
||||||
|
|
||||||
|
|
@ -1278,7 +1278,7 @@ module Vmpooler
|
||||||
|
|
||||||
post "#{api_prefix}/vm/:hostname/snapshot/?" do
|
post "#{api_prefix}/vm/:hostname/snapshot/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.post.vm.snapshot')
|
metrics.increment('http_requests_vm_total.post.vm.snapshot')
|
||||||
|
|
||||||
need_token! if Vmpooler::API.settings.config[:auth]
|
need_token! if Vmpooler::API.settings.config[:auth]
|
||||||
|
|
||||||
|
|
@ -1304,7 +1304,7 @@ module Vmpooler
|
||||||
|
|
||||||
post "#{api_prefix}/vm/:hostname/snapshot/:snapshot/?" do
|
post "#{api_prefix}/vm/:hostname/snapshot/:snapshot/?" do
|
||||||
content_type :json
|
content_type :json
|
||||||
metrics.increment('api_vm.post.vm.disksize')
|
metrics.increment('http_requests_vm_total.post.vm.disksize')
|
||||||
|
|
||||||
need_token! if Vmpooler::API.settings.config[:auth]
|
need_token! if Vmpooler::API.settings.config[:auth]
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -5,7 +5,7 @@ require 'prometheus/client'
|
||||||
module Vmpooler
|
module Vmpooler
|
||||||
class Metrics
|
class Metrics
|
||||||
class Promstats < Metrics
|
class Promstats < Metrics
|
||||||
attr_reader :prefix, :endpoint, :metrics_prefix
|
attr_reader :prefix, :prometheus_endpoint, :prometheus_prefix
|
||||||
|
|
||||||
# Constants for Metric Types
|
# Constants for Metric Types
|
||||||
M_COUNTER = 1
|
M_COUNTER = 1
|
||||||
|
|
@ -24,22 +24,135 @@ module Vmpooler
|
||||||
|
|
||||||
def initialize(logger, params = {})
|
def initialize(logger, params = {})
|
||||||
@prefix = params['prefix'] || 'vmpooler'
|
@prefix = params['prefix'] || 'vmpooler'
|
||||||
@metrics_prefix = params['metrics_prefix'] || 'vmpooler'
|
@prometheus_prefix = params['prometheus_prefix'] || 'vmpooler'
|
||||||
@endpoint = params['endpoint'] || '/prometheus'
|
@prometheus_endpoint = params['prometheus_endpoint'] || '/prometheus'
|
||||||
@logger = logger
|
@logger = logger
|
||||||
|
|
||||||
# Setup up prometheus registry and data structures
|
# Setup up prometheus registry and data structures
|
||||||
@prometheus = Prometheus::Client.registry
|
@prometheus = Prometheus::Client.registry
|
||||||
end
|
end
|
||||||
|
|
||||||
# Metrics structure used to register the metrics and also translate/interpret the incoming metrics.
|
=begin # rubocop:disable Style/BlockComments
|
||||||
|
The Metrics table is used to register metrics and translate/interpret the incoming metrics.
|
||||||
|
|
||||||
|
This table describes all of the prometheus metrics that are recognised by the application.
|
||||||
|
The background documentation for defining metrics is at: https://prometheus.io/docs/introduction/
|
||||||
|
In particular, the naming practices should be adhered to: https://prometheus.io/docs/practices/naming/
|
||||||
|
The Ruby Client docs are also useful: https://github.com/prometheus/client_ruby
|
||||||
|
|
||||||
|
The table here allows the currently used stats definitions to be translated correctly for Prometheus.
|
||||||
|
The current format is of the form A.B.C, where the final fields may be actual values (e.g. poolname).
|
||||||
|
Prometheus metrics cannot use the '.' as a character, so this is either translated into '_' or
|
||||||
|
variable parameters are expressed as labels accompanying the metric.
|
||||||
|
|
||||||
|
Sample statistics are:
|
||||||
|
# Example showing hostnames (FQDN)
|
||||||
|
migrate_from.pix-jj26-chassis1-2.ops.puppetlabs.net
|
||||||
|
migrate_to.pix-jj26-chassis1-8.ops.puppetlabs.net
|
||||||
|
|
||||||
|
# Example showing poolname as a parameter
|
||||||
|
poolreset.invalid.centos-8-x86_64
|
||||||
|
|
||||||
|
# Examples showing similar sub-typed checkout stats
|
||||||
|
checkout.empty.centos-8-x86_64
|
||||||
|
checkout.invalid.centos-8-x86_64
|
||||||
|
checkout.invalid.unknown
|
||||||
|
checkout.success.centos-8-x86_64
|
||||||
|
|
||||||
|
# Stats without any final parameter.
|
||||||
|
connect.fail
|
||||||
|
connect.open
|
||||||
|
delete.failed
|
||||||
|
delete.success
|
||||||
|
|
||||||
|
# Stats with multiple param_labels
|
||||||
|
vmpooler_user.debian-8-x86_64-pixa4.john
|
||||||
|
|
||||||
|
The metrics implementation here preserves the existing framework which will continue to support
|
||||||
|
graphite and statsd (since vmpooler is used outside of puppet). Some rationalisation and renaming
|
||||||
|
of the actual metrics was done to get a more usable model to fit within the prometheus framework.
|
||||||
|
This particularly applies to the user stats collected once individual machines are terminated as
|
||||||
|
this would have challenged prometheus' ability due to multiple (8) parameters being collected
|
||||||
|
in a single measure (which has a very high cardinality).
|
||||||
|
|
||||||
|
Prometheus requires all metrics to be pre-registered (which is the primary reason for this
|
||||||
|
table) and also uses labels to differentiate the characteristics of the measurement. This
|
||||||
|
is used throughout to capture information such as poolnames. So for example, this is a sample
|
||||||
|
of the prometheus metrics generated for the "vmpooler_ready" measurement:
|
||||||
|
|
||||||
|
# TYPE vmpooler_ready gauge
|
||||||
|
# HELP vmpooler_ready vmpooler number of machines in ready State
|
||||||
|
vmpooler_ready{vmpooler_instance="vmpooler",poolname="win-10-ent-x86_64-pixa4"} 2.0
|
||||||
|
vmpooler_ready{vmpooler_instance="vmpooler",poolname="debian-8-x86_64-pixa4"} 2.0
|
||||||
|
vmpooler_ready{vmpooler_instance="vmpooler",poolname="centos-8-x86_64-pixa4"} 2.0
|
||||||
|
|
||||||
|
Prometheus supports the following metric types:
|
||||||
|
(see https://prometheus.io/docs/concepts/metric_types/)
|
||||||
|
|
||||||
|
Counter (increment):
|
||||||
|
A counter is a cumulative metric that represents a single monotonically increasing counter whose
|
||||||
|
value can only increase or be reset to zero on restart
|
||||||
|
|
||||||
|
Gauge:
|
||||||
|
A gauge is a metric that represents a single numerical value that can arbitrarily go up and down.
|
||||||
|
|
||||||
|
Histogram:
|
||||||
|
A histogram samples observations (usually things like request durations or response sizes) and
|
||||||
|
counts them in configurable buckets. It also provides a sum of all observed values.
|
||||||
|
This replaces the timer metric supported by statsd
|
||||||
|
|
||||||
|
Summary :
|
||||||
|
Summary provides a total count of observations and a sum of all observed values, it calculates
|
||||||
|
configurable quantiles over a sliding time window.
|
||||||
|
(Summary is not used in vmpooler)
|
||||||
|
|
||||||
|
vmpooler_metrics_table is a table of hashes, where the hash key represents the first part of the
|
||||||
|
metric name, e.g. for the metric 'delete.*' (see above) the key would be 'delete:'. "Sub-metrics",
|
||||||
|
are supported, again for the 'delete.*' example, this can be subbed into '.failed' and '.success'
|
||||||
|
|
||||||
|
The entries within the hash as are follows:
|
||||||
|
|
||||||
|
mtype:
|
||||||
|
Metric type, which is one of the following constants:
|
||||||
|
M_COUNTER = 1
|
||||||
|
M_GAUGE = 2
|
||||||
|
M_SUMMARY = 3
|
||||||
|
M_HISTOGRAM = 4
|
||||||
|
|
||||||
|
torun:
|
||||||
|
Indicates which process the metric is for - within vmpooler this is either ':api' or ':manager'
|
||||||
|
(there is a suggestion that we change this to two separate tables).
|
||||||
|
|
||||||
|
docstring:
|
||||||
|
Documentation string for the metric - this is displayed as HELP text by the endpoint.
|
||||||
|
|
||||||
|
metric_suffixes:
|
||||||
|
Array of sub-metrics of the form 'sub-metric: "doc-string for sub-metric"'. This supports
|
||||||
|
the generation of individual sub-metrics for all elements in the array.
|
||||||
|
|
||||||
|
param_labels:
|
||||||
|
This is an optional array of symbols for the final labels in a metric. It should not be
|
||||||
|
specified if there are no additional parameters.
|
||||||
|
|
||||||
|
If it specified, it can either be a single symbol, or two or more symbols. The treatment
|
||||||
|
differs if there is only one symbol given as all of the remainder of the metric string
|
||||||
|
supplied is collected into a label with the symbol name. This allows the handling of
|
||||||
|
node names (FQDN).
|
||||||
|
|
||||||
|
To illustrate:
|
||||||
|
1. In the 'connect.*' or 'delete.*' example above, it should not be specified.
|
||||||
|
2. For the 'migrate_from.*' example above, the remainder of the measure is collected
|
||||||
|
as the 'host_name' label.
|
||||||
|
3. For the 'vmpooler_user' example above, the first parameter is treated as the pool
|
||||||
|
name, and the second as the username.
|
||||||
|
|
||||||
|
=end
|
||||||
def vmpooler_metrics_table
|
def vmpooler_metrics_table
|
||||||
{
|
{
|
||||||
errors: {
|
errors: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'Count of errors for pool',
|
docstring: 'Count of errors for pool',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_errors",
|
|
||||||
metric_suffixes: {
|
metric_suffixes: {
|
||||||
markedasfailed: 'timeout waiting for instance to initialise',
|
markedasfailed: 'timeout waiting for instance to initialise',
|
||||||
duplicatehostname: 'unable to create instance due to duplicate hostname',
|
duplicatehostname: 'unable to create instance due to duplicate hostname',
|
||||||
|
|
@ -51,42 +164,36 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'Number of pool instances this user created created',
|
docstring: 'Number of pool instances this user created created',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_user",
|
|
||||||
param_labels: %i[user poolname]
|
param_labels: %i[user poolname]
|
||||||
},
|
},
|
||||||
usage_litmus: {
|
usage_litmus: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'Pools by Litmus job usage',
|
docstring: 'Pools by Litmus job usage',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_usage_litmus",
|
|
||||||
param_labels: %i[user poolname]
|
param_labels: %i[user poolname]
|
||||||
},
|
},
|
||||||
usage_jenkins_instance: {
|
usage_jenkins_instance: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'Pools by Jenkins instance usage',
|
docstring: 'Pools by Jenkins instance usage',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_usage_jenkins_instance",
|
|
||||||
param_labels: %i[jenkins_instance value_stream poolname]
|
param_labels: %i[jenkins_instance value_stream poolname]
|
||||||
},
|
},
|
||||||
usage_branch_project: {
|
usage_branch_project: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'Pools by branch/project usage',
|
docstring: 'Pools by branch/project usage',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_usage_branch_project",
|
|
||||||
param_labels: %i[branch project poolname]
|
param_labels: %i[branch project poolname]
|
||||||
},
|
},
|
||||||
usage_job_component: {
|
usage_job_component: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'Pools by job/component usage',
|
docstring: 'Pools by job/component usage',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_usage_job_component",
|
|
||||||
param_labels: %i[job_name component_to_test poolname]
|
param_labels: %i[job_name component_to_test poolname]
|
||||||
},
|
},
|
||||||
checkout: {
|
checkout: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'Pool checkout counts',
|
docstring: 'Pool checkout counts',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_checkout",
|
|
||||||
metric_suffixes: {
|
metric_suffixes: {
|
||||||
nonresponsive: 'checkout failed - non responsive machine',
|
nonresponsive: 'checkout failed - non responsive machine',
|
||||||
empty: 'checkout failed - no machine',
|
empty: 'checkout failed - no machine',
|
||||||
|
|
@ -99,7 +206,6 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'Delete machine',
|
docstring: 'Delete machine',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_delete",
|
|
||||||
metric_suffixes: {
|
metric_suffixes: {
|
||||||
success: 'succeeded',
|
success: 'succeeded',
|
||||||
failed: 'failed'
|
failed: 'failed'
|
||||||
|
|
@ -110,7 +216,6 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'Ondemand request',
|
docstring: 'Ondemand request',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_ondemandrequest_generate",
|
|
||||||
metric_suffixes: {
|
metric_suffixes: {
|
||||||
duplicaterequests: 'failed duplicate request',
|
duplicaterequests: 'failed duplicate request',
|
||||||
success: 'succeeded'
|
success: 'succeeded'
|
||||||
|
|
@ -121,7 +226,6 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'Ondemand request failure',
|
docstring: 'Ondemand request failure',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_ondemandrequest_fail",
|
|
||||||
metric_suffixes: {
|
metric_suffixes: {
|
||||||
toomanyrequests: 'too many requests',
|
toomanyrequests: 'too many requests',
|
||||||
invalid: 'invalid poolname'
|
invalid: 'invalid poolname'
|
||||||
|
|
@ -132,7 +236,6 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'vmpooler pool configuration request',
|
docstring: 'vmpooler pool configuration request',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_config",
|
|
||||||
metric_suffixes: { invalid: 'Invalid' },
|
metric_suffixes: { invalid: 'Invalid' },
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
|
|
@ -140,7 +243,6 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'Pool reset counter',
|
docstring: 'Pool reset counter',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_poolreset",
|
|
||||||
metric_suffixes: { invalid: 'Invalid Pool' },
|
metric_suffixes: { invalid: 'Invalid Pool' },
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
|
|
@ -148,7 +250,6 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'vmpooler connect (to vSphere)',
|
docstring: 'vmpooler connect (to vSphere)',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_connect",
|
|
||||||
metric_suffixes: {
|
metric_suffixes: {
|
||||||
open: 'Connect Succeeded',
|
open: 'Connect Succeeded',
|
||||||
fail: 'Connect Failed'
|
fail: 'Connect Failed'
|
||||||
|
|
@ -159,42 +260,36 @@ module Vmpooler
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'vmpooler machine migrated from',
|
docstring: 'vmpooler machine migrated from',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_migrate_from",
|
|
||||||
param_labels: %i[host_name]
|
param_labels: %i[host_name]
|
||||||
},
|
},
|
||||||
migrate_to: {
|
migrate_to: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'vmpooler machine migrated to',
|
docstring: 'vmpooler machine migrated to',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_migrate_to",
|
|
||||||
param_labels: %i[host_name]
|
param_labels: %i[host_name]
|
||||||
},
|
},
|
||||||
api_vm: {
|
http_requests_vm_total: {
|
||||||
mtype: M_COUNTER,
|
mtype: M_COUNTER,
|
||||||
torun: %i[api],
|
torun: %i[api],
|
||||||
docstring: 'Total number of HTTP request/sub-operations handled by the Rack application under the /vm endpoint',
|
docstring: 'Total number of HTTP request/sub-operations handled by the Rack application under the /vm endpoint',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_http_requests_vm_total",
|
|
||||||
param_labels: %i[method subpath operation]
|
param_labels: %i[method subpath operation]
|
||||||
},
|
},
|
||||||
ready: {
|
ready: {
|
||||||
mtype: M_GAUGE,
|
mtype: M_GAUGE,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'vmpooler number of machines in ready State',
|
docstring: 'vmpooler number of machines in ready State',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_ready",
|
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
running: {
|
running: {
|
||||||
mtype: M_GAUGE,
|
mtype: M_GAUGE,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'vmpooler number of machines running',
|
docstring: 'vmpooler number of machines running',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_running",
|
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
connection_available: {
|
connection_available: {
|
||||||
mtype: M_GAUGE,
|
mtype: M_GAUGE,
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
docstring: 'vmpooler redis connections available',
|
docstring: 'vmpooler redis connections available',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_connection_available",
|
|
||||||
param_labels: %i[type provider]
|
param_labels: %i[type provider]
|
||||||
},
|
},
|
||||||
time_to_ready_state: {
|
time_to_ready_state: {
|
||||||
|
|
@ -202,7 +297,6 @@ module Vmpooler
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
buckets: POOLER_READY_TIME_BUCKETS,
|
buckets: POOLER_READY_TIME_BUCKETS,
|
||||||
docstring: 'Time taken for machine to read ready state for pool',
|
docstring: 'Time taken for machine to read ready state for pool',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_time_to_ready_state",
|
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
migrate: {
|
migrate: {
|
||||||
|
|
@ -210,7 +304,6 @@ module Vmpooler
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
buckets: POOLER_CLONE_TIME_BUCKETS,
|
buckets: POOLER_CLONE_TIME_BUCKETS,
|
||||||
docstring: 'vmpooler time taken to migrate machine for pool',
|
docstring: 'vmpooler time taken to migrate machine for pool',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_migrate",
|
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
clone: {
|
clone: {
|
||||||
|
|
@ -218,7 +311,6 @@ module Vmpooler
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
buckets: POOLER_CLONE_TIME_BUCKETS,
|
buckets: POOLER_CLONE_TIME_BUCKETS,
|
||||||
docstring: 'vmpooler time taken to clone machine',
|
docstring: 'vmpooler time taken to clone machine',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_clone",
|
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
destroy: {
|
destroy: {
|
||||||
|
|
@ -226,7 +318,6 @@ module Vmpooler
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
buckets: POOLER_CLONE_TIME_BUCKETS,
|
buckets: POOLER_CLONE_TIME_BUCKETS,
|
||||||
docstring: 'vmpooler time taken to destroy machine',
|
docstring: 'vmpooler time taken to destroy machine',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_destroy",
|
|
||||||
param_labels: %i[poolname]
|
param_labels: %i[poolname]
|
||||||
},
|
},
|
||||||
connection_waited: {
|
connection_waited: {
|
||||||
|
|
@ -234,7 +325,6 @@ module Vmpooler
|
||||||
torun: %i[manager],
|
torun: %i[manager],
|
||||||
buckets: REDIS_CONNECT_BUCKETS,
|
buckets: REDIS_CONNECT_BUCKETS,
|
||||||
docstring: 'vmpooler redis connection wait time',
|
docstring: 'vmpooler redis connection wait time',
|
||||||
prom_metric_prefix: "#{@metrics_prefix}_connection_waited",
|
|
||||||
param_labels: %i[type provider]
|
param_labels: %i[type provider]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -279,7 +369,7 @@ module Vmpooler
|
||||||
|
|
||||||
def setup_prometheus_metrics(torun)
|
def setup_prometheus_metrics(torun)
|
||||||
@p_metrics = vmpooler_metrics_table
|
@p_metrics = vmpooler_metrics_table
|
||||||
@p_metrics.each do |_name, metric_spec|
|
@p_metrics.each do |name, metric_spec|
|
||||||
# Only register metrics appropriate to api or manager
|
# Only register metrics appropriate to api or manager
|
||||||
next if (torun & metric_spec[:torun]).empty?
|
next if (torun & metric_spec[:torun]).empty?
|
||||||
|
|
||||||
|
|
@ -288,7 +378,7 @@ module Vmpooler
|
||||||
metric_spec[:metric_suffixes].each do |metric_suffix|
|
metric_spec[:metric_suffixes].each do |metric_suffix|
|
||||||
add_prometheus_metric(
|
add_prometheus_metric(
|
||||||
metric_spec,
|
metric_spec,
|
||||||
"#{metric_spec[:prom_metric_prefix]}_#{metric_suffix[0]}",
|
"#{@prometheus_prefix}_#{name}_#{metric_suffix[0]}",
|
||||||
"#{metric_spec[:docstring]} #{metric_suffix[1]}"
|
"#{metric_spec[:docstring]} #{metric_suffix[1]}"
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
@ -296,7 +386,7 @@ module Vmpooler
|
||||||
# No Additional counter suffixes so register this as metric.
|
# No Additional counter suffixes so register this as metric.
|
||||||
add_prometheus_metric(
|
add_prometheus_metric(
|
||||||
metric_spec,
|
metric_spec,
|
||||||
metric_spec[:prom_metric_prefix],
|
"#{@prometheus_prefix}_#{name}",
|
||||||
metric_spec[:docstring]
|
metric_spec[:docstring]
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
@ -315,9 +405,9 @@ module Vmpooler
|
||||||
metric_subkey = sublabels.shift.to_sym
|
metric_subkey = sublabels.shift.to_sym
|
||||||
raise("Invalid Metric #{metric_key}_#{metric_subkey} for #{label}") unless metric[:metric_suffixes].key? metric_subkey.to_sym
|
raise("Invalid Metric #{metric_key}_#{metric_subkey} for #{label}") unless metric[:metric_suffixes].key? metric_subkey.to_sym
|
||||||
|
|
||||||
metric[:metric_name] = "#{metric[:prom_metric_prefix]}_#{metric_subkey}"
|
metric[:metric_name] = "#{@prometheus_prefix}_#{metric_key}_#{metric_subkey}"
|
||||||
else
|
else
|
||||||
metric[:metric_name] = metric[:prom_metric_prefix]
|
metric[:metric_name] = "#{@prometheus_prefix}_#{metric_key}"
|
||||||
end
|
end
|
||||||
|
|
||||||
# Check if we are looking for a parameter value at last element.
|
# Check if we are looking for a parameter value at last element.
|
||||||
|
|
|
||||||
|
|
@ -4,7 +4,7 @@ require 'spec_helper'
|
||||||
|
|
||||||
describe 'prometheus' do
|
describe 'prometheus' do
|
||||||
logger = MockLogger.new
|
logger = MockLogger.new
|
||||||
params = { 'prefix': 'test', 'metrics_prefix': 'mtest', 'endpoint': 'eptest' }
|
params = { 'prefix' => 'test', 'prometheus_prefix' => 'mtest', 'prometheus_endpoint' => 'eptest' }
|
||||||
subject = Vmpooler::Metrics::Promstats.new(logger, params)
|
subject = Vmpooler::Metrics::Promstats.new(logger, params)
|
||||||
let(:logger) { MockLogger.new }
|
let(:logger) { MockLogger.new }
|
||||||
|
|
||||||
|
|
@ -24,7 +24,7 @@ describe 'prometheus' do
|
||||||
before { subject.instance_variable_set(:@p_metrics, { foo: foo_metrics }) }
|
before { subject.instance_variable_set(:@p_metrics, { foo: foo_metrics }) }
|
||||||
|
|
||||||
it 'returns the metric for a given label including parsed labels' do
|
it 'returns the metric for a given label including parsed labels' do
|
||||||
expect(subject.find_metric('foo.bar')).to include(metric_name: '_bar')
|
expect(subject.find_metric('foo.bar')).to include(metric_name: 'mtest_foo_bar')
|
||||||
expect(subject.find_metric('foo.bar')).to include(foo_metrics)
|
expect(subject.find_metric('foo.bar')).to include(foo_metrics)
|
||||||
expect(subject.find_metric('foo.bar')).to include(labels_hash)
|
expect(subject.find_metric('foo.bar')).to include(labels_hash)
|
||||||
end
|
end
|
||||||
|
|
@ -57,7 +57,6 @@ describe 'prometheus' do
|
||||||
Prometheus::Client.config.data_store = Prometheus::Client::DataStores::Synchronized.new
|
Prometheus::Client.config.data_store = Prometheus::Client::DataStores::Synchronized.new
|
||||||
subject.setup_prometheus_metrics(%i[api manager])
|
subject.setup_prometheus_metrics(%i[api manager])
|
||||||
end
|
end
|
||||||
let(:MCOUNTER) { 1 }
|
|
||||||
|
|
||||||
describe '#setup_prometheus_metrics' do
|
describe '#setup_prometheus_metrics' do
|
||||||
it 'calls add_prometheus_metric for each item in list' do
|
it 'calls add_prometheus_metric for each item in list' do
|
||||||
|
|
@ -260,12 +259,12 @@ describe 'prometheus' do
|
||||||
po.get(labels: metric[:labels])
|
po.get(labels: metric[:labels])
|
||||||
}.by(1)
|
}.by(1)
|
||||||
end
|
end
|
||||||
it 'Increments label api_vm.#{method}.#{subpath}.#{operation}' do
|
it 'Increments label http_requests_vm_total.#{method}.#{subpath}.#{operation}' do
|
||||||
method = 'get'
|
method = 'get'
|
||||||
subpath = 'template'
|
subpath = 'template'
|
||||||
operation = 'something'
|
operation = 'something'
|
||||||
expect { subject.increment("api_vm.#{method}.#{subpath}.#{operation}") }.to change {
|
expect { subject.increment("http_requests_vm_total.#{method}.#{subpath}.#{operation}") }.to change {
|
||||||
metric, po = subject.get("api_vm.#{method}.#{subpath}.#{operation}")
|
metric, po = subject.get("http_requests_vm_total.#{method}.#{subpath}.#{operation}")
|
||||||
po.get(labels: metric[:labels])
|
po.get(labels: metric[:labels])
|
||||||
}.by(1)
|
}.by(1)
|
||||||
end
|
end
|
||||||
|
|
|
||||||
|
|
@ -240,7 +240,7 @@
|
||||||
#
|
#
|
||||||
# This section contains the connection information required to store
|
# This section contains the connection information required to store
|
||||||
# historical data via statsd. This is mutually exclusive with graphite
|
# historical data via statsd. This is mutually exclusive with graphite
|
||||||
# and takes precedence.
|
# and prometheus and takes precedence.
|
||||||
#
|
#
|
||||||
# Available configuration parameters:
|
# Available configuration parameters:
|
||||||
#
|
#
|
||||||
|
|
@ -267,7 +267,7 @@
|
||||||
#
|
#
|
||||||
# This section contains the connection information required to store
|
# This section contains the connection information required to store
|
||||||
# historical data in an external Graphite database. This is mutually exclusive
|
# historical data in an external Graphite database. This is mutually exclusive
|
||||||
# with statsd.
|
# with statsd and prometheus - i.e. only one can be selected.
|
||||||
#
|
#
|
||||||
# Available configuration parameters:
|
# Available configuration parameters:
|
||||||
#
|
#
|
||||||
|
|
@ -288,6 +288,34 @@
|
||||||
:graphite:
|
:graphite:
|
||||||
server: 'graphite.example.com'
|
server: 'graphite.example.com'
|
||||||
|
|
||||||
|
# :prometheus
|
||||||
|
#
|
||||||
|
# This section contains the connection information required to store
|
||||||
|
# historical data in an external Graphite database. This is mutually exclusive
|
||||||
|
# with statsd and graphite - i.e. only one can be selected.
|
||||||
|
#
|
||||||
|
# Available configuration parameters:
|
||||||
|
#
|
||||||
|
# - prefix
|
||||||
|
# The prefix for this vmpooler instance.
|
||||||
|
# (optional; default: 'vmpooler')
|
||||||
|
#
|
||||||
|
# - prometheus_prefix
|
||||||
|
# The prefix to use while storing prometheus data.
|
||||||
|
# (optional; default: 'vmpooler')
|
||||||
|
#
|
||||||
|
# - prometheus_endpoint
|
||||||
|
# The metrics endpoint on the vmpooler server
|
||||||
|
# (optional; default: '/prometheus')
|
||||||
|
|
||||||
|
# Example:
|
||||||
|
|
||||||
|
:prometheus:
|
||||||
|
prefix: 'staging'
|
||||||
|
prometheus_prefix: 'vmpooler'
|
||||||
|
prometheus_endpoint: '/prometheus'
|
||||||
|
|
||||||
|
|
||||||
# :auth:
|
# :auth:
|
||||||
#
|
#
|
||||||
# This section contains information related to authenticating users
|
# This section contains information related to authenticating users
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue