2020-01-01 13:55:28 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module Prometheus
|
|
|
|
class ProxyVariableSubstitutionService < BaseService
|
|
|
|
include Stepable
|
|
|
|
|
2020-05-28 22:55:37 +05:30
|
|
|
VARIABLE_INTERPOLATION_REGEX = /
|
|
|
|
%{ # Variable needs to be wrapped in these chars.
|
|
|
|
\s* # Allow whitespace before and after the variable name.
|
|
|
|
(?<variable> # Named capture.
|
|
|
|
\w+ # Match one or more word characters.
|
|
|
|
)
|
|
|
|
\s*
|
|
|
|
}
|
|
|
|
/x.freeze
|
|
|
|
|
2020-03-13 15:44:24 +05:30
|
|
|
steps :validate_variables,
|
|
|
|
:add_params_to_result,
|
2020-04-22 19:07:51 +05:30
|
|
|
:substitute_params,
|
2020-03-13 15:44:24 +05:30
|
|
|
:substitute_ruby_variables,
|
|
|
|
:substitute_liquid_variables
|
2020-01-01 13:55:28 +05:30
|
|
|
|
|
|
|
def initialize(environment, params = {})
|
|
|
|
@environment, @params = environment, params.deep_dup
|
|
|
|
end
|
|
|
|
|
|
|
|
def execute
|
|
|
|
execute_steps
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2020-03-13 15:44:24 +05:30
|
|
|
def validate_variables(_result)
|
|
|
|
return success unless variables
|
|
|
|
|
|
|
|
unless variables.is_a?(Array) && variables.size.even?
|
|
|
|
return error(_('Optional parameter "variables" must be an array of keys and values. Ex: [key1, value1, key2, value2]'))
|
|
|
|
end
|
|
|
|
|
|
|
|
success
|
|
|
|
end
|
|
|
|
|
2020-01-01 13:55:28 +05:30
|
|
|
def add_params_to_result(result)
|
|
|
|
result[:params] = params
|
|
|
|
|
|
|
|
success(result)
|
|
|
|
end
|
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
def substitute_params(result)
|
|
|
|
start_time = result[:params][:start_time]
|
|
|
|
end_time = result[:params][:end_time]
|
|
|
|
|
|
|
|
result[:params][:start] = start_time if start_time
|
|
|
|
result[:params][:end] = end_time if end_time
|
|
|
|
|
|
|
|
success(result)
|
|
|
|
end
|
|
|
|
|
2020-05-28 22:55:37 +05:30
|
|
|
def substitute_ruby_variables(result)
|
|
|
|
return success(result) unless query(result)
|
|
|
|
|
|
|
|
result[:params][:query] = gsub(query(result), full_context)
|
|
|
|
|
|
|
|
success(result)
|
|
|
|
end
|
|
|
|
|
2020-03-13 15:44:24 +05:30
|
|
|
def substitute_liquid_variables(result)
|
|
|
|
return success(result) unless query(result)
|
|
|
|
|
|
|
|
result[:params][:query] =
|
|
|
|
TemplateEngines::LiquidService.new(query(result)).render(full_context)
|
|
|
|
|
|
|
|
success(result)
|
|
|
|
rescue TemplateEngines::LiquidService::RenderError => e
|
|
|
|
error(e.message)
|
|
|
|
end
|
|
|
|
|
2020-05-28 22:55:37 +05:30
|
|
|
def gsub(string, context)
|
|
|
|
# Search for variables of the form `%{variable}` in the string and replace
|
|
|
|
# them with their value.
|
|
|
|
string.gsub(VARIABLE_INTERPOLATION_REGEX) do |match|
|
|
|
|
# Replace with the value of the variable, or if there is no such variable,
|
|
|
|
# replace the invalid variable with itself. So,
|
|
|
|
# `up{instance="%{invalid_variable}"}` will remain
|
|
|
|
# `up{instance="%{invalid_variable}"}` after substitution.
|
|
|
|
context.fetch($~[:variable], match)
|
|
|
|
end
|
2020-01-01 13:55:28 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def predefined_context
|
2020-05-28 22:55:37 +05:30
|
|
|
Gitlab::Prometheus::QueryVariables.call(@environment).stringify_keys
|
2020-01-01 13:55:28 +05:30
|
|
|
end
|
|
|
|
|
2020-03-13 15:44:24 +05:30
|
|
|
def full_context
|
|
|
|
@full_context ||= predefined_context.reverse_merge(variables_hash)
|
|
|
|
end
|
|
|
|
|
|
|
|
def variables
|
|
|
|
params[:variables]
|
|
|
|
end
|
|
|
|
|
|
|
|
def variables_hash
|
|
|
|
# .each_slice(2) converts ['key1', 'value1', 'key2', 'value2'] into
|
|
|
|
# [['key1', 'value1'], ['key2', 'value2']] which is then converted into
|
|
|
|
# a hash by to_h: {'key1' => 'value1', 'key2' => 'value2'}
|
|
|
|
# to_h will raise an ArgumentError if the number of elements in the original
|
|
|
|
# array is not even.
|
|
|
|
variables&.each_slice(2).to_h
|
|
|
|
end
|
|
|
|
|
|
|
|
def query(result)
|
|
|
|
result[:params][:query]
|
2020-01-01 13:55:28 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|