2020-05-24 23:13:21 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
RSpec.describe PerformanceMonitoring::PrometheusDashboard do
|
2020-05-24 23:13:21 +05:30
|
|
|
let(:json_content) do
|
|
|
|
{
|
|
|
|
"dashboard" => "Dashboard Title",
|
|
|
|
"templating" => {
|
|
|
|
"variables" => {
|
|
|
|
"variable1" => %w(value1 value2 value3)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"panel_groups" => [{
|
|
|
|
"group" => "Group Title",
|
|
|
|
"panels" => [{
|
|
|
|
"type" => "area-chart",
|
|
|
|
"title" => "Chart Title",
|
|
|
|
"y_label" => "Y-Axis",
|
|
|
|
"metrics" => [{
|
|
|
|
"id" => "metric_of_ages",
|
|
|
|
"unit" => "count",
|
|
|
|
"label" => "Metric of Ages",
|
|
|
|
"query_range" => "http_requests_total"
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.from_json' do
|
|
|
|
subject { described_class.from_json(json_content) }
|
|
|
|
|
|
|
|
it 'creates a PrometheusDashboard object' do
|
|
|
|
expect(subject).to be_a PerformanceMonitoring::PrometheusDashboard
|
|
|
|
expect(subject.dashboard).to eq(json_content['dashboard'])
|
|
|
|
expect(subject.panel_groups).to all(be_a PerformanceMonitoring::PrometheusPanelGroup)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'validations' do
|
2020-06-23 00:09:42 +05:30
|
|
|
shared_examples 'validation failed' do |errors_messages|
|
|
|
|
it 'raises error with corresponding messages', :aggregate_failures do
|
|
|
|
expect { subject }.to raise_error do |error|
|
|
|
|
expect(error).to be_kind_of(ActiveModel::ValidationError)
|
|
|
|
expect(error.model.errors.messages).to eq(errors_messages)
|
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'dashboard content is missing' do
|
|
|
|
let(:json_content) { nil }
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it_behaves_like 'validation failed', panel_groups: ["should be an array of panel_groups objects"], dashboard: ["can't be blank"]
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'dashboard content is NOT a hash' do
|
|
|
|
let(:json_content) { YAML.safe_load("'test'") }
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it_behaves_like 'validation failed', panel_groups: ["should be an array of panel_groups objects"], dashboard: ["can't be blank"]
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'content is an array' do
|
|
|
|
let(:json_content) { [{ "dashboard" => "Dashboard Title" }] }
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it_behaves_like 'validation failed', panel_groups: ["should be an array of panel_groups objects"], dashboard: ["can't be blank"]
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
context 'dashboard definition is missing panels_groups and dashboard keys' do
|
|
|
|
let(:json_content) do
|
|
|
|
{
|
|
|
|
"dashboard" => nil
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it_behaves_like 'validation failed', panel_groups: ["should be an array of panel_groups objects"], dashboard: ["can't be blank"]
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'group definition is missing panels and group keys' do
|
|
|
|
let(:json_content) do
|
|
|
|
{
|
|
|
|
"dashboard" => "Dashboard Title",
|
|
|
|
"templating" => {
|
|
|
|
"variables" => {
|
|
|
|
"variable1" => %w(value1 value2 value3)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"panel_groups" => [{ "group" => nil }]
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it_behaves_like 'validation failed', panels: ["should be an array of panels objects"], group: ["can't be blank"]
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'panel definition is missing metrics and title keys' do
|
|
|
|
let(:json_content) do
|
|
|
|
{
|
|
|
|
"dashboard" => "Dashboard Title",
|
|
|
|
"templating" => {
|
|
|
|
"variables" => {
|
|
|
|
"variable1" => %w(value1 value2 value3)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"panel_groups" => [{
|
|
|
|
"group" => "Group Title",
|
|
|
|
"panels" => [{
|
|
|
|
"type" => "area-chart",
|
|
|
|
"y_label" => "Y-Axis"
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
it_behaves_like 'validation failed', metrics: ["should be an array of metrics objects"], title: ["can't be blank"]
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
context 'metrics definition is missing unit, query and query_range keys' do
|
|
|
|
let(:json_content) do
|
|
|
|
{
|
|
|
|
"dashboard" => "Dashboard Title",
|
|
|
|
"templating" => {
|
|
|
|
"variables" => {
|
|
|
|
"variable1" => %w(value1 value2 value3)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
"panel_groups" => [{
|
|
|
|
"group" => "Group Title",
|
|
|
|
"panels" => [{
|
|
|
|
"type" => "area-chart",
|
|
|
|
"title" => "Chart Title",
|
|
|
|
"y_label" => "Y-Axis",
|
|
|
|
"metrics" => [{
|
|
|
|
"id" => "metric_of_ages",
|
|
|
|
"label" => "Metric of Ages",
|
|
|
|
"query_range" => nil
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
it_behaves_like 'validation failed', unit: ["can't be blank"], query_range: ["can't be blank"], query: ["can't be blank"]
|
|
|
|
end
|
|
|
|
|
|
|
|
# for each parent entry validation first is done to its children,
|
|
|
|
# whole execution is stopped on first encountered error
|
|
|
|
# which is the one that is reported
|
|
|
|
context 'multiple offences on different levels' do
|
|
|
|
let(:json_content) do
|
|
|
|
{
|
|
|
|
"dashboard" => nil,
|
|
|
|
"panel_groups" => [{
|
|
|
|
"group" => nil,
|
|
|
|
"panels" => [{
|
|
|
|
"type" => "area-chart",
|
|
|
|
"title" => nil,
|
|
|
|
"y_label" => "Y-Axis",
|
|
|
|
"metrics" => [{
|
|
|
|
"id" => "metric_of_ages",
|
|
|
|
"label" => "Metric of Ages",
|
|
|
|
"query_range" => 'query'
|
|
|
|
}, {
|
|
|
|
"id" => "metric_of_ages",
|
|
|
|
"unit" => "count",
|
|
|
|
"label" => "Metric of Ages",
|
|
|
|
"query_range" => nil
|
|
|
|
}]
|
|
|
|
}]
|
|
|
|
}, {
|
|
|
|
"group" => 'group',
|
|
|
|
"panels" => nil
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
|
2020-06-23 00:09:42 +05:30
|
|
|
it_behaves_like 'validation failed', unit: ["can't be blank"]
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.find_for' do
|
|
|
|
let(:project) { build_stubbed(:project) }
|
|
|
|
let(:user) { build_stubbed(:user) }
|
2020-07-28 23:09:34 +05:30
|
|
|
let(:environment) { build_stubbed(:environment, project: project) }
|
2020-05-24 23:13:21 +05:30
|
|
|
let(:path) { ::Metrics::Dashboard::SystemDashboardService::DASHBOARD_PATH }
|
|
|
|
|
|
|
|
context 'dashboard has been found' do
|
|
|
|
it 'uses dashboard finder to find and load dashboard data and returns dashboard instance', :aggregate_failures do
|
2022-07-16 23:28:13 +05:30
|
|
|
expect(Gitlab::Metrics::Dashboard::Finder).to receive(:find).with(project, user, { environment: environment, dashboard_path: path }).and_return(status: :success, dashboard: json_content)
|
2020-05-24 23:13:21 +05:30
|
|
|
|
|
|
|
dashboard_instance = described_class.find_for(project: project, user: user, path: path, options: { environment: environment })
|
|
|
|
|
|
|
|
expect(dashboard_instance).to be_instance_of described_class
|
2020-06-23 00:09:42 +05:30
|
|
|
expect(dashboard_instance.environment).to eq environment
|
|
|
|
expect(dashboard_instance.path).to eq path
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'dashboard has NOT been found' do
|
|
|
|
it 'returns nil' do
|
2020-06-23 00:09:42 +05:30
|
|
|
allow(Gitlab::Metrics::Dashboard::Finder).to receive(:find).and_return(http_status: :not_found)
|
2020-05-24 23:13:21 +05:30
|
|
|
|
|
|
|
dashboard_instance = described_class.find_for(project: project, user: user, path: path, options: { environment: environment })
|
|
|
|
|
|
|
|
expect(dashboard_instance).to be_nil
|
|
|
|
end
|
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
|
|
|
|
context 'dashboard has invalid schema', :aggregate_failures do
|
|
|
|
it 'still returns dashboard object' do
|
|
|
|
expect(Gitlab::Metrics::Dashboard::Finder).to receive(:find).and_return(http_status: :unprocessable_entity)
|
|
|
|
|
|
|
|
dashboard_instance = described_class.find_for(project: project, user: user, path: path, options: { environment: environment })
|
|
|
|
|
|
|
|
expect(dashboard_instance).to be_instance_of described_class
|
|
|
|
expect(dashboard_instance.environment).to eq environment
|
|
|
|
expect(dashboard_instance.path).to eq path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#schema_validation_warnings' do
|
2020-11-24 15:15:51 +05:30
|
|
|
let(:environment) { create(:environment, project: project) }
|
|
|
|
let(:path) { '.gitlab/dashboards/test.yml' }
|
|
|
|
let(:project) { create(:project, :repository, :custom_repo, files: { path => dashboard_schema.to_yaml }) }
|
|
|
|
|
|
|
|
subject(:schema_validation_warnings) { described_class.new(dashboard_schema.merge(path: path, environment: environment)).schema_validation_warnings }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Gitlab::Metrics::Dashboard::Finder).to receive(:find_raw).with(project, dashboard_path: path).and_call_original
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'metrics_dashboard_exhaustive_validations is on' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(metrics_dashboard_exhaustive_validations: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when schema is valid' do
|
|
|
|
let(:dashboard_schema) { YAML.safe_load(fixture_file('lib/gitlab/metrics/dashboard/sample_dashboard.yml')) }
|
|
|
|
|
|
|
|
it 'returns empty array' do
|
|
|
|
expect(Gitlab::Metrics::Dashboard::Validator).to receive(:errors).with(dashboard_schema, dashboard_path: path, project: project).and_return([])
|
|
|
|
|
|
|
|
expect(schema_validation_warnings).to eq []
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when schema is invalid' do
|
|
|
|
let(:dashboard_schema) { YAML.safe_load(fixture_file('lib/gitlab/metrics/dashboard/dashboard_missing_panel_groups.yml')) }
|
|
|
|
|
|
|
|
it 'returns array with errors messages' do
|
|
|
|
error = ::Gitlab::Metrics::Dashboard::Validator::Errors::SchemaValidationError.new
|
|
|
|
|
|
|
|
expect(Gitlab::Metrics::Dashboard::Validator).to receive(:errors).with(dashboard_schema, dashboard_path: path, project: project).and_return([error])
|
|
|
|
|
|
|
|
expect(schema_validation_warnings).to eq [error.message]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when YAML has wrong syntax' do
|
|
|
|
let(:project) { create(:project, :repository, :custom_repo, files: { path => fixture_file('lib/gitlab/metrics/dashboard/broken_yml_syntax.yml') }) }
|
|
|
|
|
|
|
|
subject(:schema_validation_warnings) { described_class.new(path: path, environment: environment).schema_validation_warnings }
|
|
|
|
|
|
|
|
it 'returns array with errors messages' do
|
|
|
|
expect(Gitlab::Metrics::Dashboard::Validator).not_to receive(:errors)
|
|
|
|
|
|
|
|
expect(schema_validation_warnings).to eq ['Invalid yaml']
|
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
context 'metrics_dashboard_exhaustive_validations is off' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(metrics_dashboard_exhaustive_validations: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when schema is valid' do
|
|
|
|
let(:dashboard_schema) { YAML.safe_load(fixture_file('lib/gitlab/metrics/dashboard/sample_dashboard.yml')) }
|
|
|
|
|
|
|
|
it 'returns empty array' do
|
|
|
|
expect(described_class).to receive(:from_json).with(dashboard_schema)
|
|
|
|
|
|
|
|
expect(schema_validation_warnings).to eq []
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when schema is invalid' do
|
|
|
|
let(:dashboard_schema) { YAML.safe_load(fixture_file('lib/gitlab/metrics/dashboard/dashboard_missing_panel_groups.yml')) }
|
|
|
|
|
|
|
|
it 'returns array with errors messages' do
|
|
|
|
instance = described_class.new
|
|
|
|
instance.errors.add(:test, 'test error')
|
|
|
|
|
|
|
|
expect(described_class).to receive(:from_json).and_raise(ActiveModel::ValidationError.new(instance))
|
|
|
|
expect(described_class.new.schema_validation_warnings).to eq ['test: test error']
|
|
|
|
end
|
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
context 'when YAML has wrong syntax' do
|
|
|
|
let(:project) { create(:project, :repository, :custom_repo, files: { path => fixture_file('lib/gitlab/metrics/dashboard/broken_yml_syntax.yml') }) }
|
|
|
|
|
|
|
|
subject(:schema_validation_warnings) { described_class.new(path: path, environment: environment).schema_validation_warnings }
|
|
|
|
|
|
|
|
it 'returns array with errors messages' do
|
|
|
|
expect(described_class).not_to receive(:from_json)
|
|
|
|
|
|
|
|
expect(schema_validation_warnings).to eq ['Invalid yaml']
|
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
end
|
|
|
|
end
|
2020-05-24 23:13:21 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '#to_yaml' do
|
|
|
|
subject { prometheus_dashboard.to_yaml }
|
|
|
|
|
|
|
|
let(:prometheus_dashboard) { described_class.from_json(json_content) }
|
|
|
|
let(:expected_yaml) do
|
|
|
|
"---\npanel_groups:\n- panels:\n - metrics:\n - id: metric_of_ages\n unit: count\n label: Metric of Ages\n query: \n query_range: http_requests_total\n type: area-chart\n title: Chart Title\n y_label: Y-Axis\n weight: \n group: Group Title\n priority: \ndashboard: Dashboard Title\n"
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to eq(expected_yaml) }
|
|
|
|
end
|
|
|
|
end
|