Commit bb4d25a5 authored by Sean Arnold's avatar Sean Arnold

Add cronjob and schedule check workers

Add specs and config
parent bcfd9e32
...@@ -645,6 +645,9 @@ Gitlab.ee do ...@@ -645,6 +645,9 @@ Gitlab.ee do
Settings.cron_jobs['incident_management_persist_oncall_rotation_worker'] ||= Settingslogic.new({}) Settings.cron_jobs['incident_management_persist_oncall_rotation_worker'] ||= Settingslogic.new({})
Settings.cron_jobs['incident_management_persist_oncall_rotation_worker']['cron'] ||= '*/5 * * * *' Settings.cron_jobs['incident_management_persist_oncall_rotation_worker']['cron'] ||= '*/5 * * * *'
Settings.cron_jobs['incident_management_persist_oncall_rotation_worker']['job_class'] = 'IncidentManagement::OncallRotations::PersistAllRotationsShiftsJob' Settings.cron_jobs['incident_management_persist_oncall_rotation_worker']['job_class'] = 'IncidentManagement::OncallRotations::PersistAllRotationsShiftsJob'
Settings.cron_jobs['incident_management_schedule_escalation_check_worker'] ||= Settingslogic.new({})
Settings.cron_jobs['incident_management_schedule_escalation_check_worker']['cron'] ||= '*/1 * * * *'
Settings.cron_jobs['incident_management_schedule_escalation_check_worker']['job_class'] = 'IncidentManagement::Escalations::ScheduleEscalationCheckCronWorker'
Settings.cron_jobs['import_software_licenses_worker'] ||= Settingslogic.new({}) Settings.cron_jobs['import_software_licenses_worker'] ||= Settingslogic.new({})
Settings.cron_jobs['import_software_licenses_worker']['cron'] ||= '0 3 * * 0' Settings.cron_jobs['import_software_licenses_worker']['cron'] ||= '0 3 * * 0'
Settings.cron_jobs['import_software_licenses_worker']['job_class'] = 'ImportSoftwareLicensesWorker' Settings.cron_jobs['import_software_licenses_worker']['job_class'] = 'ImportSoftwareLicensesWorker'
......
...@@ -180,6 +180,8 @@ ...@@ -180,6 +180,8 @@
- 2 - 2
- - incident_management_apply_incident_sla_exceeded_label - - incident_management_apply_incident_sla_exceeded_label
- 1 - 1
- - incident_management_escalations_pending_alert_escalation_check
- 1
- - incident_management_oncall_rotations_persist_shifts_job - - incident_management_oncall_rotations_persist_shifts_job
- 1 - 1
- - invalid_gpg_signature_update - - invalid_gpg_signature_update
......
...@@ -4,11 +4,15 @@ module IncidentManagement ...@@ -4,11 +4,15 @@ module IncidentManagement
module PendingEscalations module PendingEscalations
class Alert < ApplicationRecord class Alert < ApplicationRecord
include PartitionedTable include PartitionedTable
include EachBatch
alias_attribute :target, :alert alias_attribute :target, :alert
self.primary_key = :id self.primary_key = :id
self.table_name = 'incident_management_pending_alert_escalations' self.table_name = 'incident_management_pending_alert_escalations'
ESCALATION_BUFFER = 1.month.freeze
partitioned_by :process_at, strategy: :monthly partitioned_by :process_at, strategy: :monthly
belongs_to :oncall_schedule, class_name: 'OncallSchedule', foreign_key: 'schedule_id' belongs_to :oncall_schedule, class_name: 'OncallSchedule', foreign_key: 'schedule_id'
......
...@@ -28,11 +28,13 @@ module IncidentManagement ...@@ -28,11 +28,13 @@ module IncidentManagement
end end
def create_escalations(rules) def create_escalations(rules)
rules.each do |rule| escalation_ids = rules.map do |rule|
escalaton = create_escalation(rule) escalaton = create_escalation(rule)
process_escalation(escalaton) if rule.elapsed_time_seconds == 0 escalaton.id
end end
process_escalations(escalation_ids)
rescue StandardError => e rescue StandardError => e
Gitlab::ErrorTracking.track_exception(e, target_type: target.class.to_s, target_id: target.id) Gitlab::ErrorTracking.track_exception(e, target_type: target.class.to_s, target_id: target.id)
end end
...@@ -47,8 +49,10 @@ module IncidentManagement ...@@ -47,8 +49,10 @@ module IncidentManagement
) )
end end
def process_escalation(escalation) def process_escalations(escalation_ids)
::IncidentManagement::PendingEscalations::ProcessService.new(escalation).execute args = escalation_ids.map { |id| [id] }
::IncidentManagement::Escalations::PendingAlertEscalationCheckWorker.bulk_perform_async(args) # rubocop:disable Scalability/BulkPerformWithContext
end end
end end
end end
......
...@@ -294,6 +294,15 @@ ...@@ -294,6 +294,15 @@
:weight: 1 :weight: 1
:idempotent: :idempotent:
:tags: [] :tags: []
- :name: cronjob:incident_management_escalations_schedule_escalation_check_cron
:worker_name: IncidentManagement::Escalations::ScheduleEscalationCheckCronWorker
:feature_category: :incident_management
:has_external_dependencies:
:urgency: :low
:resource_boundary: :unknown
:weight: 1
:idempotent: true
:tags: []
- :name: cronjob:incident_management_incident_sla_exceeded_check - :name: cronjob:incident_management_incident_sla_exceeded_check
:worker_name: IncidentManagement::IncidentSlaExceededCheckWorker :worker_name: IncidentManagement::IncidentSlaExceededCheckWorker
:feature_category: :incident_management :feature_category: :incident_management
...@@ -1044,6 +1053,15 @@ ...@@ -1044,6 +1053,15 @@
:idempotent: true :idempotent: true
:tags: :tags:
- :exclude_from_kubernetes - :exclude_from_kubernetes
- :name: incident_management_escalations_pending_alert_escalation_check
:worker_name: IncidentManagement::Escalations::PendingAlertEscalationCheckWorker
:feature_category: :incident_management
:has_external_dependencies:
:urgency: :high
:resource_boundary: :unknown
:weight: 1
:idempotent: true
:tags: []
- :name: incident_management_oncall_rotations_persist_shifts_job - :name: incident_management_oncall_rotations_persist_shifts_job
:worker_name: IncidentManagement::OncallRotations::PersistShiftsJob :worker_name: IncidentManagement::OncallRotations::PersistShiftsJob
:feature_category: :incident_management :feature_category: :incident_management
......
# frozen_string_literal: true
module IncidentManagement
module Escalations
class PendingAlertEscalationCheckWorker
include ApplicationWorker
urgency :high
idempotent!
feature_category :incident_management
def initialize(escalation_id)
@escalation_id = escalation_id
end
def perform
escalation = IncidentManagement::PendingEscalations::Alert.find(escalation_id)
IncidentManagement::PendingEscalations::ProcessService.new(escalation).execute
end
private
attr_reader :escalation_id
end
end
end
# frozen_string_literal: true
module IncidentManagement
module Escalations
class ScheduleEscalationCheckCronWorker
include ApplicationWorker
# This worker does not perform work scoped to a context
include CronjobQueue # rubocop:disable Scalability/CronWorkerContext
idempotent!
feature_category :incident_management
def perform
::IncidentManagement::PendingEscalations::Alert.processable.each_batch do |relation|
args = relation.pluck(:id).map { |id| [id] } # rubocop:disable CodeReuse/ActiveRecord
::IncidentManagement::Escalations::PendingAlertEscalationCheckWorker.bulk_perform_async(args) # rubocop:disable Scalability/BulkPerformWithContext
end
end
end
end
end
...@@ -53,13 +53,11 @@ RSpec.describe IncidentManagement::PendingEscalations::CreateService do ...@@ -53,13 +53,11 @@ RSpec.describe IncidentManagement::PendingEscalations::CreateService do
end end
end end
it 'processes the escalation' do it 'creates the escalations and queues the escalation process check' do
expect(IncidentManagement::PendingEscalations::ProcessService) expect(IncidentManagement::Escalations::PendingAlertEscalationCheckWorker)
.to receive(:new) .to receive(:bulk_perform_async)
.with(having_attributes(rule_id: first_escalation_rule.id)) .with([[a_kind_of(Integer)], [a_kind_of(Integer)]])
.and_return(process_service_spy)
expect(process_service_spy).to receive(:execute)
expect { execute }.to change { IncidentManagement::PendingEscalations::Alert.count }.by(rule_count) expect { execute }.to change { IncidentManagement::PendingEscalations::Alert.count }.by(rule_count)
end end
......
...@@ -8,6 +8,13 @@ RSpec.shared_examples 'creates an escalation' do |count| ...@@ -8,6 +8,13 @@ RSpec.shared_examples 'creates an escalation' do |count|
.exactly(count).times .exactly(count).times
.and_call_original .and_call_original
expected_args = []
count.times { expected_args << [a_kind_of(Integer)]}
expect(IncidentManagement::Escalations::PendingAlertEscalationCheckWorker)
.to receive(:bulk_perform_async)
.with(expected_args)
subject subject
end end
end end
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe IncidentManagement::Escalations::PendingAlertEscalationCheckWorker do
let(:worker) { described_class.new(escalation.id) }
let_it_be(:escalation) { create(:incident_management_pending_alert_escalation) }
describe '#perform' do
subject { worker.perform }
it 'processes the escalation' do
process_service = spy(IncidentManagement::PendingEscalations::ProcessService)
expect(IncidentManagement::PendingEscalations::ProcessService).to receive(:new).with(escalation).and_return(process_service)
subject
expect(process_service).to have_received(:execute)
end
end
end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe IncidentManagement::Escalations::ScheduleEscalationCheckCronWorker do
let(:worker) { described_class.new }
let_it_be(:escalation_1) { create(:incident_management_pending_alert_escalation, process_at: 5.minutes.ago) }
let_it_be(:escalation_2) { create(:incident_management_pending_alert_escalation, process_at: 2.days.ago) }
let_it_be(:escalation_not_ready_to_process) { create(:incident_management_pending_alert_escalation) }
describe '#perform' do
subject { worker.perform }
it 'schedules a job for each processable escalation' do
expect(IncidentManagement::Escalations::PendingAlertEscalationCheckWorker).to receive(:bulk_perform_async)
.with(array_including([escalation_2.id], [escalation_1.id]))
subject
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