Commit 98f81303 authored by Peter Leitzen's avatar Peter Leitzen Committed by Jan Provaznik

Improve default title and description of incidents

Parse and expose alert's `startsAt`

Parse strictly RFC3339

Present alert's `starts_at` as RFC3339

Expose alert's `starts_at` in the incident summary

Parse `labels/alertname` as alert title

Add missing spec for Alert#description

Add missing specs for Alert#annotations

Add missing spec for Alert#starts_at

Add missing specs for Alert#valid?

Tweak specs for Alert#{gitlab_alert,environment}

Parse full_query from generatorURL

Move creation of incident summary into presenter

Expose alert's full query if available

Adjust specs for alert mails

`startsAt` is now mandatory

Add changelog entry

Remove implicit `nil` in rescue

Define `let` before the `before` block

Inline key-value split

Define value inside the shared example

Improve example naming

Apply review suggestions
parent 3eed257d
...@@ -27,6 +27,19 @@ module Projects ...@@ -27,6 +27,19 @@ module Projects
end end
end end
def starts_at
super&.rfc3339
end
def issue_summary_markdown
<<~MARKDOWN.chomp
## Summary
#{metadata_list}
#{annotation_list}
MARKDOWN
end
private private
def alert_title def alert_title
...@@ -38,6 +51,31 @@ module Projects ...@@ -38,6 +51,31 @@ module Projects
"#{gitlab_alert.title} #{gitlab_alert.computed_operator} #{gitlab_alert.threshold} for 5 minutes" "#{gitlab_alert.title} #{gitlab_alert.computed_operator} #{gitlab_alert.threshold} for 5 minutes"
end end
def metadata_list
metadata = []
metadata << bullet('starts_at', starts_at) if starts_at
metadata << bullet('full_query', backtick(full_query)) if full_query
metadata.join("\n")
end
def annotation_list
strong_memoize(:annotation_list) do
annotations
.map { |annotation| bullet(annotation.label, annotation.value) }
.join("\n")
end
end
def bullet(key, value)
"* #{key}: #{value}"
end
def backtick(value)
"`#{value}`"
end
end end
end end
end end
...@@ -41,24 +41,12 @@ module IncidentManagement ...@@ -41,24 +41,12 @@ module IncidentManagement
end end
def alert_summary def alert_summary
<<~MARKDOWN alert.issue_summary_markdown
## Summary
#{annotation_list}
MARKDOWN
end
def annotation_list
strong_memoize(:annotation_list) do
alert.annotations
.map { |annotation| "* #{annotation.label}: #{annotation.value}" }
.join("\n")
end
end end
def alert def alert
strong_memoize(:alert) do strong_memoize(:alert) do
Gitlab::Alerting::Alert.new(project: project, payload: params) Gitlab::Alerting::Alert.new(project: project, payload: params).present
end end
end end
......
...@@ -60,7 +60,7 @@ module Projects ...@@ -60,7 +60,7 @@ module Projects
def validate_date(date) def validate_date(date)
return unless date return unless date
Time.parse(date) Time.rfc3339(date)
date date
rescue ArgumentError rescue ArgumentError
end end
......
---
title: Improve default title and description of issues opened from managed Prometheus alerts
merge_request: 14614
author:
type: changed
...@@ -17,7 +17,7 @@ module Gitlab ...@@ -17,7 +17,7 @@ module Gitlab
def title def title
strong_memoize(:title) do strong_memoize(:title) do
parse_title_from_payload gitlab_alert&.title || parse_title_from_payload
end end
end end
...@@ -37,8 +37,20 @@ module Gitlab ...@@ -37,8 +37,20 @@ module Gitlab
end end
end end
def starts_at
strong_memoize(:starts_at) do
parse_datetime_from_payload('startsAt')
end
end
def full_query
strong_memoize(:full_query) do
gitlab_alert&.full_query || parse_expr_from_payload
end
end
def valid? def valid?
project && title project && title && starts_at
end end
def present def present
...@@ -58,9 +70,9 @@ module Gitlab ...@@ -58,9 +70,9 @@ module Gitlab
end end
def parse_title_from_payload def parse_title_from_payload
gitlab_alert&.title || payload&.dig('annotations', 'title') ||
payload&.dig('annotations', 'title') || payload&.dig('annotations', 'summary') ||
payload&.dig('annotations', 'summary') payload&.dig('labels', 'alertname')
end end
def parse_description_from_payload def parse_description_from_payload
...@@ -72,6 +84,27 @@ module Gitlab ...@@ -72,6 +84,27 @@ module Gitlab
Alerting::AlertAnnotation.new(label: label, value: value) Alerting::AlertAnnotation.new(label: label, value: value)
end end
end end
def parse_datetime_from_payload(field)
value = payload&.dig(field)
return unless value
Time.rfc3339(value)
rescue ArgumentError
end
# Parses `g0.expr` from `generatorURL`.
#
# Example: http://localhost:9090/graph?g0.expr=vector%281%29&g0.tab=1
def parse_expr_from_payload
url = payload&.dig('generatorURL')
return unless url
uri = URI(url)
Rack::Utils.parse_query(uri.query).fetch('g0.expr')
rescue URI::InvalidURIError, KeyError
end
end end
end end
end end
...@@ -9,6 +9,10 @@ FactoryBot.define do ...@@ -9,6 +9,10 @@ FactoryBot.define do
metric_id nil metric_id nil
after(:build) do |alert, evaluator| after(:build) do |alert, evaluator|
unless alert.payload.key?('startsAt')
alert.payload['startsAt'] = Time.now.rfc3339
end
if metric_id = evaluator.metric_id if metric_id = evaluator.metric_id
alert.payload['labels'] ||= {} alert.payload['labels'] ||= {}
alert.payload['labels']['gitlab_alert_id'] = metric_id.to_s alert.payload['labels']['gitlab_alert_id'] = metric_id.to_s
......
...@@ -8,74 +8,213 @@ describe Gitlab::Alerting::Alert do ...@@ -8,74 +8,213 @@ describe Gitlab::Alerting::Alert do
let(:alert) { build(:alerting_alert, project: project, payload: payload) } let(:alert) { build(:alerting_alert, project: project, payload: payload) }
let(:payload) { {} } let(:payload) { {} }
context 'with gitlab alert' do shared_context 'gitlab alert' do
let(:gitlab_alert_id) { gitlab_alert.prometheus_metric_id.to_s }
let!(:gitlab_alert) { create(:prometheus_alert, project: project) } let!(:gitlab_alert) { create(:prometheus_alert, project: project) }
before do before do
payload['labels'] = { payload['labels'] = { 'gitlab_alert_id' => gitlab_alert_id }
'gitlab_alert_id' => gitlab_alert_id
}
end end
end
context 'with matching gitlab_alert_id' do shared_examples 'invalid alert' do
let(:gitlab_alert_id) { gitlab_alert.prometheus_metric_id.to_s } it 'is invalid' do
expect(alert).not_to be_valid
end
end
it 'loads gitlab_alert' do shared_examples 'parse payload' do |*pairs|
expect(alert.gitlab_alert).to eq(gitlab_alert) context 'without payload' do
end it { is_expected.to be_nil }
end
it 'delegates environment to gitlab_alert' do pairs.each do |pair|
expect(alert.environment).to eq(gitlab_alert.environment) context "with #{pair}" do
let(:value) { 'some value' }
before do
section, name = pair.split('/')
payload[section] = { name => value }
end
it { is_expected.to eq(value) }
end end
end
end
it 'prefers gitlab_alert\'s title over annotated title' do describe '#gitlab_alert' do
payload['annontations'] = { 'title' => 'other title' } subject { alert.gitlab_alert }
expect(alert.title).to eq(gitlab_alert.title) context 'without payload' do
it { is_expected.to be_nil }
end
context 'with gitlab alert' do
include_context 'gitlab alert'
it { is_expected.to eq(gitlab_alert) }
end
context 'with unknown gitlab alert' do
include_context 'gitlab alert' do
let(:gitlab_alert_id) { 'unknown' }
end end
it 'is valid' do it { is_expected.to be_nil }
expect(alert).to be_valid end
end
describe '#title' do
subject { alert.title }
it_behaves_like 'parse payload',
'annotations/title',
'annotations/summary',
'labels/alertname'
context 'with gitlab alert' do
include_context 'gitlab alert'
context 'with annotations/title' do
let(:value) { 'annotation title' }
before do
payload['annotations'] = { 'title' => value }
end
it { is_expected.to eq(gitlab_alert.title) }
end end
end end
end
describe '#description' do
subject { alert.description }
it_behaves_like 'parse payload', 'annotations/description'
end
context 'with unknown gitlab_alert_id' do describe '#annotations' do
let(:gitlab_alert_id) { 'unknown' } subject { alert.annotations }
it 'cannot load gitlab_alert' do context 'without payload' do
expect(alert.gitlab_alert).to be_nil it { is_expected.to eq([]) }
end
context 'with payload' do
before do
payload['annotations'] = { 'foo' => 'value1', 'bar' => 'value2' }
end end
it 'is invalid' do it 'parses annotations' do
expect(alert).not_to be_valid expect(subject.size).to eq(2)
expect(subject.map(&:label)).to eq(%w[foo bar])
expect(subject.map(&:value)).to eq(%w[value1 value2])
end end
end end
end end
context 'with annotations' do describe '#environment' do
before do subject { alert.environment }
payload['annotations'] = {
'label' => 'value', context 'without gitlab_alert' do
'another' => 'value2' it { is_expected.to be_nil }
}
end end
it 'parses annotations' do context 'with gitlab alert' do
expect(alert.annotations.size).to eq(2) include_context 'gitlab alert'
expect(alert.annotations.map(&:label)).to eq(%w(label another))
expect(alert.annotations.map(&:value)).to eq(%w(value value2)) it { is_expected.to eq(gitlab_alert.environment) }
end end
end end
context 'without annotations' do describe '#starts_at' do
it 'has no annotations' do subject { alert.starts_at }
expect(alert.annotations).to be_empty
context 'with empty startsAt' do
before do
payload['startsAt'] = nil
end
it { is_expected.to be_nil }
end
context 'with invalid startsAt' do
before do
payload['startsAt'] = 'invalid'
end
it { is_expected.to be_nil }
end
context 'with payload' do
let(:time) { Time.now.change(usec: 0) }
before do
payload['startsAt'] = time.rfc3339
end
it { is_expected.to eq(time) }
end end
end end
context 'with empty payload' do describe '#full_query' do
it 'cannot load gitlab_alert' do using RSpec::Parameterized::TableSyntax
expect(alert.gitlab_alert).to be_nil
subject { alert.full_query }
where(:generator_url, :expected_query) do
nil | nil
'http://localhost' | nil
'invalid url' | nil
'http://localhost:9090/graph?g1.expr=vector%281%29' | nil
'http://localhost:9090/graph?g0.expr=vector%281%29' | 'vector(1)'
end
with_them do
before do
payload['generatorURL'] = generator_url
end
it { is_expected.to eq(expected_query) }
end
context 'with gitlab alert' do
include_context 'gitlab alert'
before do
payload['generatorURL'] = 'http://localhost:9090/graph?g0.expr=vector%281%29'
end
it { is_expected.to eq(gitlab_alert.full_query) }
end
end
describe '#valid?' do
before do
payload.update(
'annotations' => { 'title' => 'some title' },
'startsAt' => Time.now.rfc3339
)
end
subject { alert }
it { is_expected.to be_valid }
context 'without project' do
# Redefine to prevent:
# project is a NilClass - rspec-set works with ActiveRecord models only
let(:alert) { build(:alerting_alert, project: nil, payload: payload) }
it { is_expected.not_to be_valid }
end
context 'without starts_at' do
before do
payload['startsAt'] = nil
end
it { is_expected.not_to be_valid }
end end
end end
end end
...@@ -20,13 +20,13 @@ describe EE::Emails::Projects do ...@@ -20,13 +20,13 @@ describe EE::Emails::Projects do
Notify.prometheus_alert_fired_email(project.id, user.id, alert_params) Notify.prometheus_alert_fired_email(project.id, user.id, alert_params)
end end
context 'with an alert' do let(:alert_params) do
let(:alert_params) do { 'startsAt' => Time.now.rfc3339 }
{ end
'labels' => {
'gitlab_alert_id' => alert.prometheus_metric_id.to_s context 'with a gitlab alert' do
} before do
} alert_params['labels'] = { 'gitlab_alert_id' => alert.prometheus_metric_id.to_s }
end end
let(:title) do let(:title) do
...@@ -60,19 +60,15 @@ describe EE::Emails::Projects do ...@@ -60,19 +60,15 @@ describe EE::Emails::Projects do
end end
end end
context 'without an alert' do context 'with no payload' do
let(:alert_params) { {} } let(:alert_params) { {} }
it_behaves_like 'no email' it_behaves_like 'no email'
end end
context 'with an unknown alert' do context 'with an unknown alert' do
let(:alert_params) do before do
{ alert_params['labels'] = { 'gitlab_alert_id' => 'unknown' }
'labels' => {
'gitlab_alert_id' => 'unknown'
}
}
end end
it_behaves_like 'no email' it_behaves_like 'no email'
...@@ -85,12 +81,8 @@ describe EE::Emails::Projects do ...@@ -85,12 +81,8 @@ describe EE::Emails::Projects do
metrics_project_environments_url(project) metrics_project_environments_url(project)
end end
let(:alert_params) do before do
{ alert_params['annotations'] = { 'title' => title }
'annotations' => {
'title' => title
}
}
end end
it_behaves_like 'an email sent from GitLab' it_behaves_like 'an email sent from GitLab'
......
...@@ -6,7 +6,8 @@ describe Projects::Prometheus::AlertPresenter do ...@@ -6,7 +6,8 @@ describe Projects::Prometheus::AlertPresenter do
set(:project) { create(:project) } set(:project) { create(:project) }
let(:presenter) { described_class.new(alert) } let(:presenter) { described_class.new(alert) }
let(:alert) { create(:alerting_alert, project: project) } let(:payload) { {} }
let(:alert) { create(:alerting_alert, project: project, payload: payload) }
describe '#project_full_path' do describe '#project_full_path' do
subject { presenter.project_full_path } subject { presenter.project_full_path }
...@@ -14,6 +15,58 @@ describe Projects::Prometheus::AlertPresenter do ...@@ -14,6 +15,58 @@ describe Projects::Prometheus::AlertPresenter do
it { is_expected.to eq(project.full_path) } it { is_expected.to eq(project.full_path) }
end end
describe '#starts_at' do
subject { presenter.starts_at }
before do
payload['startsAt'] = starts_at
end
context 'with valid datetime' do
let(:datetime) { Time.now }
let(:starts_at) { datetime.rfc3339 }
it { is_expected.to eq(datetime.rfc3339) }
end
context 'with invalid datetime' do
let(:starts_at) { 'invalid' }
it { is_expected.to be_nil }
end
end
describe '#issue_summary_markdown' do
subject { presenter.issue_summary_markdown }
context 'without default payload' do
it do
is_expected.to include('## Summary')
is_expected.to include('* starts_at:')
is_expected.not_to include('* full_query:')
end
end
context 'with annotations' do
before do
payload['annotations'] = { 'foo' => 'value1', 'bar' => 'value2' }
end
it do
is_expected.to include('* foo: value1')
is_expected.to include('* bar: value2')
end
end
context 'with full query' do
before do
payload['generatorURL'] = 'http://host?g0.expr=query'
end
it { is_expected.to include('* full_query: `query`') }
end
end
context 'with gitlab alert' do context 'with gitlab alert' do
let(:gitlab_alert) { create(:prometheus_alert, project: project) } let(:gitlab_alert) { create(:prometheus_alert, project: project) }
let(:metric_id) { gitlab_alert.prometheus_metric_id } let(:metric_id) { gitlab_alert.prometheus_metric_id }
...@@ -49,10 +102,6 @@ describe Projects::Prometheus::AlertPresenter do ...@@ -49,10 +102,6 @@ describe Projects::Prometheus::AlertPresenter do
end end
describe '#performance_dashboard_link' do describe '#performance_dashboard_link' do
before do
gitlab_alert.save!
end
let(:expected_link) do let(:expected_link) do
Gitlab::Routing.url_helpers Gitlab::Routing.url_helpers
.metrics_project_environment_url(project, alert.environment) .metrics_project_environment_url(project, alert.environment)
......
...@@ -5,9 +5,19 @@ require 'spec_helper' ...@@ -5,9 +5,19 @@ require 'spec_helper'
describe IncidentManagement::CreateIssueService do describe IncidentManagement::CreateIssueService do
let(:project) { create(:project, :repository) } let(:project) { create(:project, :repository) }
let(:service) { described_class.new(project, nil, alert_payload) } let(:service) { described_class.new(project, nil, alert_payload) }
let(:alert_starts_at) { Time.now }
let(:alert_title) { 'TITLE' } let(:alert_title) { 'TITLE' }
let(:alert_annotations) { { title: alert_title } }
let(:alert_payload) do let(:alert_payload) do
build_alert_payload(annotations: { title: alert_title }) build_alert_payload(
annotations: alert_annotations,
starts_at: alert_starts_at
)
end
let(:alert_presenter) do
Gitlab::Alerting::Alert.new(project: project, payload: alert_payload).present
end end
let!(:setting) do let!(:setting) do
...@@ -30,8 +40,7 @@ describe IncidentManagement::CreateIssueService do ...@@ -30,8 +40,7 @@ describe IncidentManagement::CreateIssueService do
expect(issue.author).to eq(User.alert_bot) expect(issue.author).to eq(User.alert_bot)
expect(issue.title).to eq(alert_title) expect(issue.title).to eq(alert_title)
expect(issue.description).to include('Summary') expect(issue.description).to include(alert_presenter.issue_summary_markdown)
expect(issue.description).to include(alert_title)
expect(issue.description).not_to include(summary_separator) expect(issue.description).not_to include(summary_separator)
end end
end end
...@@ -48,8 +57,7 @@ describe IncidentManagement::CreateIssueService do ...@@ -48,8 +57,7 @@ describe IncidentManagement::CreateIssueService do
it 'creates an issue appending issue template' do it 'creates an issue appending issue template' do
expect(subject).to include(status: :success) expect(subject).to include(status: :success)
expect(issue.description).to include('Summary') expect(issue.description).to include(alert_presenter.issue_summary_markdown)
expect(issue.description).to include(alert_title)
expect(issue.description).to include(summary_separator) expect(issue.description).to include(summary_separator)
expect(issue.description).to include(issue_template_content) expect(issue.description).to include(issue_template_content)
end end
...@@ -93,15 +101,27 @@ describe IncidentManagement::CreateIssueService do ...@@ -93,15 +101,27 @@ describe IncidentManagement::CreateIssueService do
end end
end end
context 'with an invalid alert payload' do describe 'with invalid alert payload' do
let(:alert_payload) { build_alert_payload(annotations: {}) } shared_examples 'invalid alert' do
it 'does not create an issue' do
expect(service)
.to receive(:log_error)
.with(error_message('invalid alert'))
expect(subject).to eq(status: :error, message: 'invalid alert')
end
end
context 'without title' do
let(:alert_annotations) { {} }
it 'does not create an issue' do it_behaves_like 'invalid alert'
expect(service) end
.to receive(:log_error)
.with(error_message('invalid alert'))
expect(subject).to eq(status: :error, message: 'invalid alert') context 'without startsAt' do
let(:alert_starts_at) { nil }
it_behaves_like 'invalid alert'
end end
end end
end end
...@@ -122,8 +142,12 @@ describe IncidentManagement::CreateIssueService do ...@@ -122,8 +142,12 @@ describe IncidentManagement::CreateIssueService do
private private
def build_alert_payload(annotations: {}) def build_alert_payload(annotations: {}, starts_at: Time.now)
{ 'annotations' => annotations.stringify_keys } {
'annotations' => annotations.stringify_keys
}.tap do |payload|
payload['startsAt'] = starts_at.rfc3339 if starts_at
end
end end
def error_message(message) def error_message(message)
......
...@@ -267,7 +267,7 @@ describe Projects::Prometheus::Alerts::CreateEventsService do ...@@ -267,7 +267,7 @@ describe Projects::Prometheus::Alerts::CreateEventsService do
# Example: 2018-09-27T18:25:31.079079416Z # Example: 2018-09-27T18:25:31.079079416Z
def utc_rfc3339(date) def utc_rfc3339(date)
date.utc.strftime("%FT%T.%9NZ") date.utc.rfc3339
rescue rescue
date date
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