Commit 5d1d912b authored by Mikolaj Wawrzyniak's avatar Mikolaj Wawrzyniak

Apply JSON schema based validation to dashboards

To provide exhusitve list of errors, we should start using
JSON schema based validators in blob viewer for metrics dashboards YAML
files
parent 510e6ce6
...@@ -14,7 +14,8 @@ module Resolvers ...@@ -14,7 +14,8 @@ module Resolvers
def resolve(**args) def resolve(**args)
return unless environment return unless environment
::PerformanceMonitoring::PrometheusDashboard.find_for(project: environment.project, user: context[:current_user], path: args[:path], options: { environment: environment }) ::PerformanceMonitoring::PrometheusDashboard
.find_for(project: environment.project, user: context[:current_user], path: args[:path], options: { environment: environment })
end end
end end
end end
......
...@@ -16,6 +16,13 @@ module Types ...@@ -16,6 +16,13 @@ module Types
field :annotations, Types::Metrics::Dashboards::AnnotationType.connection_type, null: true, field :annotations, Types::Metrics::Dashboards::AnnotationType.connection_type, null: true,
description: 'Annotations added to the dashboard', description: 'Annotations added to the dashboard',
resolver: Resolvers::Metrics::Dashboards::AnnotationResolver resolver: Resolvers::Metrics::Dashboards::AnnotationResolver
# In order to maintain backward compatibility we need to return NULL when there are no warnings
# and dashboard validation returns an empty array when there are no issues.
def schema_validation_warnings
warnings = object.schema_validation_warnings
warnings unless warnings.empty?
end
end end
# rubocop: enable Graphql/AuthorizeTypes # rubocop: enable Graphql/AuthorizeTypes
end end
......
...@@ -26,19 +26,10 @@ module BlobViewer ...@@ -26,19 +26,10 @@ module BlobViewer
def parse_blob_data def parse_blob_data
yaml = ::Gitlab::Config::Loader::Yaml.new(blob.data).load_raw! yaml = ::Gitlab::Config::Loader::Yaml.new(blob.data).load_raw!
Gitlab::Metrics::Dashboard::Validator
::PerformanceMonitoring::PrometheusDashboard.from_json(yaml) .errors(yaml, dashboard_path: blob.path, project: project)
nil
rescue Gitlab::Config::Loader::FormatError => error rescue Gitlab::Config::Loader::FormatError => error
wrap_yml_syntax_error(error) [error]
rescue ActiveModel::ValidationError => invalid
invalid.model.errors
end
def wrap_yml_syntax_error(error)
::PerformanceMonitoring::PrometheusDashboard.new.errors.tap do |errors|
errors.add(:'YAML syntax', error.message)
end
end end
end end
end end
...@@ -53,14 +53,18 @@ module PerformanceMonitoring ...@@ -53,14 +53,18 @@ module PerformanceMonitoring
# This method is planned to be refactored as a part of https://gitlab.com/gitlab-org/gitlab/-/issues/219398 # This method is planned to be refactored as a part of https://gitlab.com/gitlab-org/gitlab/-/issues/219398
# implementation. For new existing logic was reused to faster deliver MVC # implementation. For new existing logic was reused to faster deliver MVC
def schema_validation_warnings def schema_validation_warnings
self.class.from_json(reload_schema) run_custom_validation.map(&:message)
nil
rescue ActiveModel::ValidationError => exception
exception.model.errors.map { |attr, error| "#{attr}: #{error}" }
end end
private private
def run_custom_validation
Gitlab::Metrics::Dashboard::Validator
.errors(reload_schema, dashboard_path: path, project: environment&.project)
rescue Gitlab::Config::Loader::FormatError => error
[error.message]
end
# dashboard finder methods are somehow limited, #find includes checking if # dashboard finder methods are somehow limited, #find includes checking if
# user is authorised to view selected dashboard, but modifies schema, which in some cases may # user is authorised to view selected dashboard, but modifies schema, which in some cases may
# cause false positives returned from validation, and #find_raw does not authorise users # cause false positives returned from validation, and #find_raw does not authorise users
......
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
= icon('warning fw') = icon('warning fw')
= _('Metrics Dashboard YAML definition is invalid:') = _('Metrics Dashboard YAML definition is invalid:')
%ul %ul
- viewer.errors.messages.each do |error| - viewer.errors.each do |error|
%li= error.join(': ') %li= error
= link_to _('Learn more'), help_page_path('operations/metrics/dashboards/index.md') = link_to _('Learn more'), help_page_path('operations/metrics/dashboards/index.md')
---
title: Change metrics dashboard schema validation messages into exhaustive list of
all encountered errors.
merge_request: 38925
author:
type: changed
...@@ -156,21 +156,46 @@ and files with invalid syntax display **Metrics Dashboard YAML definition is inv ...@@ -156,21 +156,46 @@ and files with invalid syntax display **Metrics Dashboard YAML definition is inv
When **Metrics Dashboard YAML definition is invalid** at least one of the following messages is displayed: When **Metrics Dashboard YAML definition is invalid** at least one of the following messages is displayed:
1. `dashboard: can't be blank` [learn more](#dashboard-top-level-properties) 1. `[location] is missing required keys: [list of missing keys]` - The entry at
1. `panel_groups: should be an array of panel_groups objects` [learn more](#dashboard-top-level-properties) `[location]` is missing a key, or a key has been mistyped. This
1. `group: can't be blank` [learn more](#panel-group-panel_groups-properties) example returns the error `root is missing required keys: panel_groups`:
1. `panels: should be an array of panels objects` [learn more](#panel-group-panel_groups-properties)
1. `title: can't be blank` [learn more](#panel-panels-properties) ```yaml
1. `metrics: should be an array of metrics objects` [learn more](#panel-panels-properties) dashboard: Important metrics
1. `query: can't be blank` [learn more](#metrics-metrics-properties) group_panels:
1. `query_range: can't be blank` [learn more](#metrics-metrics-properties) - ...
1. `unit: can't be blank` [learn more](#metrics-metrics-properties) ```
1. `YAML syntax: The parsed YAML is too big`
1. `[data] at [location] is not of type: [type]` - The entry at `[location]` contains
This is displayed when the YAML file is larger than 1 MB. `[data]` which type does not adhere to required types. This example returns the
error `'123' at /panel_groups/0/group is not of type: string`:
1. `YAML syntax: Invalid configuration format`
```yaml
This is displayed when the YAML file is empty or does not contain valid YAML. dashboard: Environment metrics
panel_groups:
- group: 123
panels:
...
```
1. `[data] at [location] is not one of: [types]` - The entry at `[location]` contains
`[data]` which is not included in the list of required values. This example returns
the error `'scatterplot-chart' at /panel_groups/0/panels/0/type is not one of: ["area-chart", "line-chart", "anomaly-chart", "bar", "column", "stacked-column", "single-stat", "heatmap"]`:
```yaml
dashboard: Environment metrics
panel_groups:
- group: Network
panels:
- title: Throughput
type: scatterplot-chart
y_label: Requests / Sec
...
```
1. `metric_id must be unique across a project` - At least two metrics entries have
the same `id` attribute, which [must be unique](#metrics-metrics-properties).
1. `The parsed YAML is too big` - The YAML file is larger than 1 MB.
1. `Invalid configuration format` - The YAML file is empty or does not contain valid YAML.
Metrics Dashboard YAML definition validation information is also available as a [GraphQL API field](../../../api/graphql/reference/index.md#metricsdashboard) Metrics Dashboard YAML definition validation information is also available as a [GraphQL API field](../../../api/graphql/reference/index.md#metricsdashboard)
...@@ -8,20 +8,18 @@ module Gitlab ...@@ -8,20 +8,18 @@ module Gitlab
class << self class << self
def validate(content, schema_path = DASHBOARD_SCHEMA_PATH, dashboard_path: nil, project: nil) def validate(content, schema_path = DASHBOARD_SCHEMA_PATH, dashboard_path: nil, project: nil)
errors = _validate(content, schema_path, dashboard_path: dashboard_path, project: project) errors(content, schema_path, dashboard_path: dashboard_path, project: project).empty?
errors.empty?
end end
def validate!(content, schema_path = DASHBOARD_SCHEMA_PATH, dashboard_path: nil, project: nil) def validate!(content, schema_path = DASHBOARD_SCHEMA_PATH, dashboard_path: nil, project: nil)
errors = _validate(content, schema_path, dashboard_path: dashboard_path, project: project) errors = errors(content, schema_path, dashboard_path: dashboard_path, project: project)
errors.empty? || raise(errors.first) errors.empty? || raise(errors.first)
end end
private def errors(content, schema_path = DASHBOARD_SCHEMA_PATH, dashboard_path: nil, project: nil)
Validator::Client
def _validate(content, schema_path, dashboard_path: nil, project: nil) .new(content, schema_path, dashboard_path: dashboard_path, project: project)
client = Validator::Client.new(content, schema_path, dashboard_path: dashboard_path, project: project) .execute
client.execute
end end
end end
end end
......
...@@ -46,7 +46,7 @@ module Gitlab ...@@ -46,7 +46,7 @@ module Gitlab
def validate_against_schema def validate_against_schema
schemer.validate(content).map do |error| schemer.validate(content).map do |error|
Errors::SchemaValidationError.new(error) ::Gitlab::Metrics::Dashboard::Validator::Errors::SchemaValidationError.new(error)
end end
end end
end end
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
"properties": { "properties": {
"type": { "type": {
"type": "string", "type": "string",
"enum": ["area-chart", "anomaly-chart", "bar", "column", "stacked-column", "single-stat", "heatmap"], "enum": ["area-chart", "line-chart", "anomaly-chart", "bar", "column", "stacked-column", "single-stat", "heatmap", "gauge"],
"default": "area-chart" "default": "area-chart"
}, },
"title": { "type": "string" }, "title": { "type": "string" },
......
...@@ -589,7 +589,7 @@ RSpec.describe 'File blob', :js do ...@@ -589,7 +589,7 @@ RSpec.describe 'File blob', :js do
aggregate_failures do aggregate_failures do
# shows that dashboard yaml is invalid # shows that dashboard yaml is invalid
expect(page).to have_content('Metrics Dashboard YAML definition is invalid:') expect(page).to have_content('Metrics Dashboard YAML definition is invalid:')
expect(page).to have_content("panel_groups: should be an array of panel_groups objects") expect(page).to have_content("root is missing required keys: panel_groups")
# shows a learn more link # shows a learn more link
expect(page).to have_link('Learn more') expect(page).to have_link('Learn more')
......
...@@ -34,6 +34,19 @@ RSpec.describe Gitlab::Metrics::Dashboard::Validator::Errors do ...@@ -34,6 +34,19 @@ RSpec.describe Gitlab::Metrics::Dashboard::Validator::Errors do
it { is_expected.to eq 'root is missing required keys: one' } it { is_expected.to eq 'root is missing required keys: one' }
end end
context 'when there is type mismatch' do
%w(null string boolean integer number array object).each do |expected_type|
context "on type: #{expected_type}" do
let(:type) { expected_type }
let(:details) { nil }
subject { described_class.new(error_hash).message }
it { is_expected.to eq "'property_name' at root is not of type: #{expected_type}" }
end
end
end
end end
context 'for nested object' do context 'for nested object' do
......
...@@ -143,4 +143,56 @@ RSpec.describe Gitlab::Metrics::Dashboard::Validator do ...@@ -143,4 +143,56 @@ RSpec.describe Gitlab::Metrics::Dashboard::Validator do
end end
end end
end end
describe '#errors' do
context 'valid dashboard schema' do
it 'returns no errors' do
expect(described_class.errors(valid_dashboard)).to eq []
end
context 'with duplicate metric_ids' do
it 'returns errors' do
expect(described_class.errors(duplicate_id_dashboard)).to eq [Gitlab::Metrics::Dashboard::Validator::Errors::DuplicateMetricIds.new]
end
end
context 'with dashboard_path and project' do
subject { described_class.errors(valid_dashboard, dashboard_path: 'test/path.yml', project: project) }
context 'with no conflicting metric identifiers in db' do
it { is_expected.to eq [] }
end
context 'with metric identifier present in current dashboard' do
before do
create(:prometheus_metric,
identifier: 'metric_a1',
dashboard_path: 'test/path.yml',
project: project
)
end
it { is_expected.to eq [] }
end
context 'with metric identifier present in another dashboard' do
before do
create(:prometheus_metric,
identifier: 'metric_a1',
dashboard_path: 'some/other/dashboard/path.yml',
project: project
)
end
it { is_expected.to eq [Gitlab::Metrics::Dashboard::Validator::Errors::DuplicateMetricIds.new] }
end
end
end
context 'invalid dashboard schema' do
it 'returns collection of validation errors' do
expect(described_class.errors(invalid_dashboard)).to all be_kind_of(Gitlab::Metrics::Dashboard::Validator::Errors::SchemaValidationError)
end
end
end
end end
...@@ -13,11 +13,11 @@ RSpec.describe BlobViewer::MetricsDashboardYml do ...@@ -13,11 +13,11 @@ RSpec.describe BlobViewer::MetricsDashboardYml do
subject(:viewer) { described_class.new(blob) } subject(:viewer) { described_class.new(blob) }
context 'when the definition is valid' do context 'when the definition is valid' do
let(:data) { File.read(Rails.root.join('config/prometheus/common_metrics.yml')) } let(:data) { fixture_file('lib/gitlab/metrics/dashboard/sample_dashboard.yml')}
describe '#valid?' do describe '#valid?' do
it 'calls prepare! on the viewer' do it 'calls prepare! on the viewer' do
allow(PerformanceMonitoring::PrometheusDashboard).to receive(:from_json) allow(Gitlab::Metrics::Dashboard::Validator).to receive(:errors)
expect(viewer).to receive(:prepare!) expect(viewer).to receive(:prepare!)
...@@ -30,46 +30,44 @@ RSpec.describe BlobViewer::MetricsDashboardYml do ...@@ -30,46 +30,44 @@ RSpec.describe BlobViewer::MetricsDashboardYml do
expect_next_instance_of(::Gitlab::Config::Loader::Yaml, data) do |loader| expect_next_instance_of(::Gitlab::Config::Loader::Yaml, data) do |loader|
expect(loader).to receive(:load_raw!).and_call_original expect(loader).to receive(:load_raw!).and_call_original
end end
expect(PerformanceMonitoring::PrometheusDashboard) expect(Gitlab::Metrics::Dashboard::Validator)
.to receive(:from_json) .to receive(:errors)
.with(yml) .with(yml, dashboard_path: '.gitlab/dashboards/custom-dashboard.yml', project: project)
.and_call_original .and_call_original
expect(viewer.valid?).to be_truthy expect(viewer.valid?).to be_truthy
end end
end end
describe '#errors' do describe '#errors' do
it 'returns nil' do it 'returns empty array' do
allow(PerformanceMonitoring::PrometheusDashboard).to receive(:from_json) allow(Gitlab::Metrics::Dashboard::Validator).to receive(:errors).and_return([])
expect(viewer.errors).to be nil expect(viewer.errors).to eq []
end end
end end
end end
context 'when definition is invalid' do context 'when definition is invalid' do
let(:error) { ActiveModel::ValidationError.new(PerformanceMonitoring::PrometheusDashboard.new.tap(&:validate)) } let(:error) { ::Gitlab::Metrics::Dashboard::Validator::Errors::SchemaValidationError.new }
let(:data) do let(:data) do
<<~YAML <<~YAML
dashboard: dashboard:
YAML YAML
end end
before do
allow(Gitlab::Metrics::Dashboard::Validator).to receive(:errors).and_return([error])
end
describe '#valid?' do describe '#valid?' do
it 'returns false' do it 'returns false' do
expect(PerformanceMonitoring::PrometheusDashboard) expect(viewer.valid?).to be false
.to receive(:from_json).and_raise(error)
expect(viewer.valid?).to be_falsey
end end
end end
describe '#errors' do describe '#errors' do
it 'returns validation errors' do it 'returns validation errors' do
allow(PerformanceMonitoring::PrometheusDashboard) expect(viewer.errors).to eq [error]
.to receive(:from_json).and_raise(error)
expect(viewer.errors).to be error.model.errors
end end
end end
end end
...@@ -85,17 +83,14 @@ RSpec.describe BlobViewer::MetricsDashboardYml do ...@@ -85,17 +83,14 @@ RSpec.describe BlobViewer::MetricsDashboardYml do
describe '#valid?' do describe '#valid?' do
it 'returns false' do it 'returns false' do
expect(PerformanceMonitoring::PrometheusDashboard).not_to receive(:from_json) expect(Gitlab::Metrics::Dashboard::Validator).not_to receive(:errors)
expect(viewer.valid?).to be_falsey expect(viewer.valid?).to be false
end end
end end
describe '#errors' do describe '#errors' do
it 'returns validation errors' do it 'returns validation errors' do
yaml_wrapped_errors = { 'YAML syntax': ["(<unknown>): did not find expected key while parsing a block mapping at line 1 column 1"] } expect(viewer.errors).to all be_kind_of Gitlab::Config::Loader::FormatError
expect(viewer.errors).to be_kind_of ActiveModel::Errors
expect(viewer.errors.messages).to eql(yaml_wrapped_errors)
end end
end end
end end
...@@ -113,15 +108,12 @@ RSpec.describe BlobViewer::MetricsDashboardYml do ...@@ -113,15 +108,12 @@ RSpec.describe BlobViewer::MetricsDashboardYml do
end end
it 'is invalid' do it 'is invalid' do
expect(PerformanceMonitoring::PrometheusDashboard).not_to receive(:from_json) expect(Gitlab::Metrics::Dashboard::Validator).not_to receive(:errors)
expect(viewer.valid?).to be(false) expect(viewer.valid?).to be false
end end
it 'returns validation errors' do it 'returns validation errors' do
yaml_wrapped_errors = { 'YAML syntax': ["The parsed YAML is too big"] } expect(viewer.errors).to all be_kind_of Gitlab::Config::Loader::FormatError
expect(viewer.errors).to be_kind_of(ActiveModel::Errors)
expect(viewer.errors.messages).to eq(yaml_wrapped_errors)
end end
end end
end end
...@@ -219,20 +219,31 @@ RSpec.describe PerformanceMonitoring::PrometheusDashboard do ...@@ -219,20 +219,31 @@ RSpec.describe PerformanceMonitoring::PrometheusDashboard do
end end
describe '#schema_validation_warnings' do describe '#schema_validation_warnings' do
let_it_be(:project) { create(:project) }
let_it_be(:environment) { create(:environment, project: project) }
let(:path) { '.gitlab/dashboards/test.yml' }
subject(:schema_validation_warnings) { described_class.new(json_content.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_return(json_content)
end
context 'when schema is valid' do context 'when schema is valid' do
it 'returns nil' do it 'returns nil' do
expect(described_class).to receive(:from_json) expect(Gitlab::Metrics::Dashboard::Validator).to receive(:errors).with(json_content, dashboard_path: path, project: project).and_return([])
expect(described_class.new.schema_validation_warnings).to be_nil
expect(schema_validation_warnings).to eq []
end end
end end
context 'when schema is invalid' do context 'when schema is invalid' do
it 'returns array with errors messages' do it 'returns array with errors messages' do
instance = described_class.new error = ::Gitlab::Metrics::Dashboard::Validator::Errors::SchemaValidationError.new
instance.errors.add(:test, 'test error')
expect(Gitlab::Metrics::Dashboard::Validator).to receive(:errors).with(json_content, dashboard_path: path, project: project).and_return([error])
expect(described_class).to receive(:from_json).and_raise(ActiveModel::ValidationError.new(instance)) expect(schema_validation_warnings).to eq [error.message]
expect(described_class.new.schema_validation_warnings).to eq ['test: test error']
end end
end end
end end
......
...@@ -7,7 +7,7 @@ RSpec.describe 'Getting Metrics Dashboard' do ...@@ -7,7 +7,7 @@ RSpec.describe 'Getting Metrics Dashboard' do
let_it_be(:current_user) { create(:user) } let_it_be(:current_user) { create(:user) }
let(:project) { create(:project) } let(:project) { create(:project) }
let!(:environment) { create(:environment, project: project) } let(:environment) { create(:environment, project: project) }
let(:query) do let(:query) do
graphql_query_for( graphql_query_for(
...@@ -67,7 +67,7 @@ RSpec.describe 'Getting Metrics Dashboard' do ...@@ -67,7 +67,7 @@ RSpec.describe 'Getting Metrics Dashboard' do
it 'returns metrics dashboard' do it 'returns metrics dashboard' do
dashboard = graphql_data.dig('project', 'environments', 'nodes', 0, 'metricsDashboard') dashboard = graphql_data.dig('project', 'environments', 'nodes', 0, 'metricsDashboard')
expect(dashboard).to eql("path" => path, "schemaValidationWarnings" => ["panel_groups: should be an array of panel_groups objects"]) expect(dashboard).to eql("path" => path, "schemaValidationWarnings" => ["root is missing required keys: panel_groups"])
end end
end end
...@@ -78,7 +78,7 @@ RSpec.describe 'Getting Metrics Dashboard' do ...@@ -78,7 +78,7 @@ RSpec.describe 'Getting Metrics Dashboard' do
it 'returns metrics dashboard' do it 'returns metrics dashboard' do
dashboard = graphql_data.dig('project', 'environments', 'nodes', 0, 'metricsDashboard') dashboard = graphql_data.dig('project', 'environments', 'nodes', 0, 'metricsDashboard')
expect(dashboard).to eql("path" => path, "schemaValidationWarnings" => ["dashboard: can't be blank", "panel_groups: should be an array of panel_groups objects"]) expect(dashboard).to eql("path" => path, "schemaValidationWarnings" => ["root is missing required keys: dashboard, panel_groups"])
end end
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment