summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJames Lopez <james@jameslopez.es>2017-05-03 17:20:12 +0200
committerJames Lopez <james@jameslopez.es>2017-05-03 17:20:12 +0200
commit264bf229277caf1c1eaca4e83921ca1b305d5401 (patch)
tree610c4b0427f8303a5484c06e6f4779b7c9e1669d
parente81ea165ba738fedab07d5e20423856e004e2594 (diff)
downloadgitlab-ce-264bf229277caf1c1eaca4e83921ca1b305d5401.tar.gz
add propagate service worker and updated spec and controller
-rw-r--r--app/controllers/admin/services_controller.rb2
-rw-r--r--app/models/service.rb10
-rw-r--r--app/workers/propagate_project_service_worker.rb49
-rw-r--r--spec/workers/propagate_project_service_worker_spec.rb19
4 files changed, 73 insertions, 7 deletions
diff --git a/app/controllers/admin/services_controller.rb b/app/controllers/admin/services_controller.rb
index 37a1a23178e..2b6f335cb2b 100644
--- a/app/controllers/admin/services_controller.rb
+++ b/app/controllers/admin/services_controller.rb
@@ -15,7 +15,7 @@ class Admin::ServicesController < Admin::ApplicationController
end
def update
- if service.update_attributes(service_params[:service])
+ if service.update_and_propagate(service_params[:service])
redirect_to admin_application_settings_services_path,
notice: 'Application settings saved successfully'
else
diff --git a/app/models/service.rb b/app/models/service.rb
index c71a7d169ec..dea22fd96a7 100644
--- a/app/models/service.rb
+++ b/app/models/service.rb
@@ -254,6 +254,16 @@ class Service < ActiveRecord::Base
service
end
+ def update_and_propagate(service_params)
+ return false unless update_attributes(service_params)
+
+ if service_params[:active] == 1
+ PropagateProjectServiceWorker.perform_async(service_params[:id])
+ end
+
+ true
+ end
+
private
def cache_project_has_external_issue_tracker
diff --git a/app/workers/propagate_project_service_worker.rb b/app/workers/propagate_project_service_worker.rb
new file mode 100644
index 00000000000..53551770968
--- /dev/null
+++ b/app/workers/propagate_project_service_worker.rb
@@ -0,0 +1,49 @@
+# Worker for updating any project specific caches.
+class PropagateProjectServiceWorker
+ include Sidekiq::Worker
+ include DedicatedSidekiqQueue
+
+ LEASE_TIMEOUT = 30.minutes.to_i
+
+ def perform(template_id)
+ template = Service.find_by(id: template_id)
+
+ return unless template&.active
+ return unless try_obtain_lease_for(template.id)
+
+ Rails.logger.info("Propagating services for template #{template.id}")
+
+ project_ids_for_template(template) do |project_id|
+ Service.build_from_template(project_id, template).save!
+ end
+ end
+
+ private
+
+ def project_ids_for_template(template)
+ limit = 100
+ offset = 0
+
+ loop do
+ batch = project_ids_batch(limit, offset, template.type)
+
+ batch.each { |project_id| yield(project_id) }
+
+ break if batch.count < limit
+
+ offset += limit
+ end
+ end
+
+ def project_ids_batch(limit, offset, template_type)
+ Project.joins('LEFT JOIN services ON services.project_id = projects.id').
+ where('services.type != ? OR services.id IS NULL', template_type).
+ limit(limit).offset(offset).pluck(:id)
+ end
+
+ def try_obtain_lease_for(template_id)
+ Gitlab::ExclusiveLease.
+ new("propagate_project_service_worker:#{template_id}", timeout: LEASE_TIMEOUT).
+ try_obtain
+ end
+end
diff --git a/spec/workers/propagate_project_service_worker_spec.rb b/spec/workers/propagate_project_service_worker_spec.rb
index ce01a663a8f..d525a8b4a23 100644
--- a/spec/workers/propagate_project_service_worker_spec.rb
+++ b/spec/workers/propagate_project_service_worker_spec.rb
@@ -1,36 +1,43 @@
require 'spec_helper'
-describe PruneOldEventsWorker do
+describe PropagateProjectServiceWorker do
describe '#perform' do
let!(:service_template) do
PushoverService.create(
template: true,
+ active: true,
properties: {
device: 'MyDevice',
sound: 'mic',
priority: 4,
+ user_key: 'asdf',
api_key: '123456789'
})
end
let!(:project) { create(:empty_project) }
+ before do
+ allow_any_instance_of(Gitlab::ExclusiveLease).to receive(:try_obtain).
+ and_return(true)
+ end
+
it 'creates services for projects' do
- expect { subject.perform }.to change { Service.count }.by(1)
+ expect { subject.perform(service_template.id) }.to change { Service.count }.by(1)
end
it 'does not create the service if it exists already' do
Service.build_from_template(project.id, service_template).save!
- expect { subject.perform }.not_to change { Service.count }
+ expect { subject.perform(service_template.id) }.not_to change { Service.count }
end
it 'creates the service containing the template attributes' do
- subject.perform
+ subject.perform(service_template.id)
- service = Service.find_by(service_template.merge(project_id: project.id, template: false))
+ service = Service.find_by(type: service_template.type, template: false)
- expect(service).not_to be_nil
+ expect(service.properties).to eq(service_template.properties)
end
end
end