Commit 2cc9f641 authored by Luke Duncalfe's avatar Luke Duncalfe Committed by Arturo Herrero

Add PG trigger for has_external_issue_tracker

We cache whether a project has an External Issue Tracker integration
enabled in the `has_external_issue_tracker` column on `projects`. It
will be `TRUE` if the project has any integration (a record in
`services`) with a category of "issue_tracker" enabled.

This was previously maintained with application code.

This cache is easy to fall out of consistency when we fail to maintain
it during bulk operations (including PostgreSQL cascading deletes
https://gitlab.com/gitlab-org/gitlab/-/merge_requests/48163/) as
mentioned in:

- https://gitlab.com/gitlab-org/gitlab/-/issues/273574#note_457006686
- https://gitlab.com/gitlab-org/gitlab/-/issues/289798

As Ecosystem is increasingly changing integration data using bulk
operations this has been changed to be maintained through a PostgreSQL
trigger.

https://gitlab.com/gitlab-org/gitlab/-/issues/290715
https://gitlab.com/gitlab-org/gitlab/-/issues/289798
parent a0e3d861
...@@ -1314,21 +1314,11 @@ class Project < ApplicationRecord ...@@ -1314,21 +1314,11 @@ class Project < ApplicationRecord
end end
def external_issue_tracker def external_issue_tracker
if has_external_issue_tracker.nil? cache_has_external_issue_tracker if has_external_issue_tracker.nil?
cache_has_external_issue_tracker
end
if has_external_issue_tracker? return unless has_external_issue_tracker?
strong_memoize(:external_issue_tracker) do
services.external_issue_trackers.first
end
else
nil
end
end
def cache_has_external_issue_tracker @external_issue_tracker ||= services.external_issue_trackers.first
update_column(:has_external_issue_tracker, services.external_issue_trackers.any?) if Gitlab::Database.read_write?
end end
def external_references_supported? def external_references_supported?
...@@ -2690,6 +2680,10 @@ class Project < ApplicationRecord ...@@ -2690,6 +2680,10 @@ class Project < ApplicationRecord
def cache_has_external_wiki def cache_has_external_wiki
update_column(:has_external_wiki, services.external_wikis.any?) if Gitlab::Database.read_write? update_column(:has_external_wiki, services.external_wikis.any?) if Gitlab::Database.read_write?
end end
def cache_has_external_issue_tracker
update_column(:has_external_issue_tracker, services.external_issue_trackers.any?) if Gitlab::Database.read_write?
end
end end
Project.prepend_if_ee('EE::Project') Project.prepend_if_ee('EE::Project')
...@@ -46,7 +46,6 @@ class Service < ApplicationRecord ...@@ -46,7 +46,6 @@ class Service < ApplicationRecord
after_initialize :initialize_properties after_initialize :initialize_properties
after_commit :reset_updated_properties after_commit :reset_updated_properties
after_commit :cache_project_has_external_issue_tracker
belongs_to :project, inverse_of: :services belongs_to :project, inverse_of: :services
belongs_to :group, inverse_of: :services belongs_to :group, inverse_of: :services
...@@ -438,10 +437,6 @@ class Service < ApplicationRecord ...@@ -438,10 +437,6 @@ class Service < ApplicationRecord
ProjectServiceWorker.perform_async(id, data) ProjectServiceWorker.perform_async(id, data)
end end
def external_issue_tracker?
category == :issue_tracker && active?
end
def external_wiki? def external_wiki?
type == 'ExternalWikiService' && active? type == 'ExternalWikiService' && active?
end end
...@@ -461,12 +456,6 @@ class Service < ApplicationRecord ...@@ -461,12 +456,6 @@ class Service < ApplicationRecord
errors.add(:project_id, 'The service cannot belong to both a project and a group') if project_id && group_id errors.add(:project_id, 'The service cannot belong to both a project and a group') if project_id && group_id
end end
def cache_project_has_external_issue_tracker
if project && !project.destroyed?
project.cache_has_external_issue_tracker
end
end
def valid_recipients? def valid_recipients?
activated? && !importing? activated? && !importing?
end end
......
...@@ -11,8 +11,6 @@ class BulkCreateIntegrationService ...@@ -11,8 +11,6 @@ class BulkCreateIntegrationService
service_list = ServiceList.new(batch, service_hash, association).to_array service_list = ServiceList.new(batch, service_hash, association).to_array
Service.transaction do Service.transaction do
run_callbacks(batch) if association == 'project'
results = bulk_insert(*service_list) results = bulk_insert(*service_list)
if integration.data_fields_present? if integration.data_fields_present?
...@@ -33,14 +31,6 @@ class BulkCreateIntegrationService ...@@ -33,14 +31,6 @@ class BulkCreateIntegrationService
klass.insert_all(items_to_insert, returning: [:id]) klass.insert_all(items_to_insert, returning: [:id])
end end
# rubocop: disable CodeReuse/ActiveRecord
def run_callbacks(batch)
if integration.external_issue_tracker?
Project.where(id: batch.select(:id)).update_all(has_external_issue_tracker: true)
end
end
# rubocop: enable CodeReuse/ActiveRecord
def service_hash def service_hash
if integration.template? if integration.template?
integration.to_service_hash integration.to_service_hash
......
---
title: Add PostgreSQL trigger to maintain projects.has_external_issue_tracker
merge_request: 51852
author:
type: changed
# frozen_string_literal: true
class AddHasExternalIssueTrackerTrigger < ActiveRecord::Migration[6.0]
include Gitlab::Database::SchemaHelpers
DOWNTIME = false
FUNCTION_NAME = 'set_has_external_issue_tracker'
TRIGGER_ON_INSERT_NAME = 'trigger_has_external_issue_tracker_on_insert'
TRIGGER_ON_UPDATE_NAME = 'trigger_has_external_issue_tracker_on_update'
TRIGGER_ON_DELETE_NAME = 'trigger_has_external_issue_tracker_on_delete'
def up
create_trigger_function(FUNCTION_NAME, replace: true) do
<<~SQL
UPDATE projects SET has_external_issue_tracker = (
EXISTS
(
SELECT 1
FROM services
WHERE project_id = COALESCE(NEW.project_id, OLD.project_id)
AND active = TRUE
AND category = 'issue_tracker'
)
)
WHERE projects.id = COALESCE(NEW.project_id, OLD.project_id);
RETURN NULL;
SQL
end
execute(<<~SQL)
CREATE TRIGGER #{TRIGGER_ON_INSERT_NAME}
AFTER INSERT ON services
FOR EACH ROW
WHEN (NEW.category = 'issue_tracker' AND NEW.active = TRUE AND NEW.project_id IS NOT NULL)
EXECUTE FUNCTION #{FUNCTION_NAME}();
SQL
execute(<<~SQL)
CREATE TRIGGER #{TRIGGER_ON_UPDATE_NAME}
AFTER UPDATE ON services
FOR EACH ROW
WHEN (NEW.category = 'issue_tracker' AND OLD.active != NEW.active AND NEW.project_id IS NOT NULL)
EXECUTE FUNCTION #{FUNCTION_NAME}();
SQL
execute(<<~SQL)
CREATE TRIGGER #{TRIGGER_ON_DELETE_NAME}
AFTER DELETE ON services
FOR EACH ROW
WHEN (OLD.category = 'issue_tracker' AND OLD.active = TRUE AND OLD.project_id IS NOT NULL)
EXECUTE FUNCTION #{FUNCTION_NAME}();
SQL
end
def down
drop_trigger(:services, TRIGGER_ON_INSERT_NAME)
drop_trigger(:services, TRIGGER_ON_UPDATE_NAME)
drop_trigger(:services, TRIGGER_ON_DELETE_NAME)
drop_function(FUNCTION_NAME)
end
end
c0d22d00d52a516347930e1a36f350113c0949214925176f08ceed81999746bd
\ No newline at end of file
...@@ -10,6 +10,26 @@ CREATE EXTENSION IF NOT EXISTS btree_gist; ...@@ -10,6 +10,26 @@ CREATE EXTENSION IF NOT EXISTS btree_gist;
CREATE EXTENSION IF NOT EXISTS pg_trgm; CREATE EXTENSION IF NOT EXISTS pg_trgm;
CREATE FUNCTION set_has_external_issue_tracker() RETURNS trigger
LANGUAGE plpgsql
AS $$
BEGIN
UPDATE projects SET has_external_issue_tracker = (
EXISTS
(
SELECT 1
FROM services
WHERE project_id = COALESCE(NEW.project_id, OLD.project_id)
AND active = TRUE
AND category = 'issue_tracker'
)
)
WHERE projects.id = COALESCE(NEW.project_id, OLD.project_id);
RETURN NULL;
END
$$;
CREATE FUNCTION set_has_external_wiki() RETURNS trigger CREATE FUNCTION set_has_external_wiki() RETURNS trigger
LANGUAGE plpgsql LANGUAGE plpgsql
AS $$ AS $$
...@@ -23637,6 +23657,12 @@ ALTER INDEX product_analytics_events_experimental_pkey ATTACH PARTITION gitlab_p ...@@ -23637,6 +23657,12 @@ ALTER INDEX product_analytics_events_experimental_pkey ATTACH PARTITION gitlab_p
CREATE TRIGGER table_sync_trigger_ee39a25f9d AFTER INSERT OR DELETE OR UPDATE ON audit_events FOR EACH ROW EXECUTE PROCEDURE table_sync_function_2be879775d(); CREATE TRIGGER table_sync_trigger_ee39a25f9d AFTER INSERT OR DELETE OR UPDATE ON audit_events FOR EACH ROW EXECUTE PROCEDURE table_sync_function_2be879775d();
CREATE TRIGGER trigger_has_external_issue_tracker_on_delete AFTER DELETE ON services FOR EACH ROW WHEN ((((old.category)::text = 'issue_tracker'::text) AND (old.active = true) AND (old.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_issue_tracker();
CREATE TRIGGER trigger_has_external_issue_tracker_on_insert AFTER INSERT ON services FOR EACH ROW WHEN ((((new.category)::text = 'issue_tracker'::text) AND (new.active = true) AND (new.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_issue_tracker();
CREATE TRIGGER trigger_has_external_issue_tracker_on_update AFTER UPDATE ON services FOR EACH ROW WHEN ((((new.category)::text = 'issue_tracker'::text) AND (old.active <> new.active) AND (new.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_issue_tracker();
CREATE TRIGGER trigger_has_external_wiki_on_delete AFTER DELETE ON services FOR EACH ROW WHEN ((((old.type)::text = 'ExternalWikiService'::text) AND (old.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_wiki(); CREATE TRIGGER trigger_has_external_wiki_on_delete AFTER DELETE ON services FOR EACH ROW WHEN ((((old.type)::text = 'ExternalWikiService'::text) AND (old.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_wiki();
CREATE TRIGGER trigger_has_external_wiki_on_insert AFTER INSERT ON services FOR EACH ROW WHEN (((new.active = true) AND ((new.type)::text = 'ExternalWikiService'::text) AND (new.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_wiki(); CREATE TRIGGER trigger_has_external_wiki_on_insert AFTER INSERT ON services FOR EACH ROW WHEN (((new.active = true) AND ((new.type)::text = 'ExternalWikiService'::text) AND (new.project_id IS NOT NULL))) EXECUTE PROCEDURE set_has_external_wiki();
......
# frozen_string_literal: true
require 'spec_helper'
require_migration!
RSpec.describe AddHasExternalIssueTrackerTrigger do
let(:migration) { described_class.new }
let(:namespaces) { table(:namespaces) }
let(:projects) { table(:projects) }
let(:services) { table(:services) }
before do
@namespace = namespaces.create!(name: 'foo', path: 'foo')
@project = projects.create!(namespace_id: @namespace.id)
end
describe '#up' do
before do
migrate!
end
describe 'INSERT trigger' do
it 'sets `has_external_issue_tracker` to true when active `issue_tracker` is inserted' do
expect do
services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
end.to change { @project.reload.has_external_issue_tracker }.to(true)
end
it 'does not set `has_external_issue_tracker` to true when service is for a different project' do
different_project = projects.create!(namespace_id: @namespace.id)
expect do
services.create!(category: 'issue_tracker', active: true, project_id: different_project.id)
end.not_to change { @project.reload.has_external_issue_tracker }
end
it 'does not set `has_external_issue_tracker` to true when inactive `issue_tracker` is inserted' do
expect do
services.create!(category: 'issue_tracker', active: false, project_id: @project.id)
end.not_to change { @project.reload.has_external_issue_tracker }
end
it 'does not set `has_external_issue_tracker` to true when a non-`issue tracker` active service is inserted' do
expect do
services.create!(category: 'my_type', active: true, project_id: @project.id)
end.not_to change { @project.reload.has_external_issue_tracker }
end
end
describe 'UPDATE trigger' do
it 'sets `has_external_issue_tracker` to true when `issue_tracker` is made active' do
service = services.create!(category: 'issue_tracker', active: false, project_id: @project.id)
expect do
service.update!(active: true)
end.to change { @project.reload.has_external_issue_tracker }.to(true)
end
it 'sets `has_external_issue_tracker` to false when `issue_tracker` is made inactive' do
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
expect do
service.update!(active: false)
end.to change { @project.reload.has_external_issue_tracker }.to(false)
end
it 'sets `has_external_issue_tracker` to false when `issue_tracker` is made inactive, and an inactive `issue_tracker` exists' do
services.create!(category: 'issue_tracker', active: false, project_id: @project.id)
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
expect do
service.update!(active: false)
end.to change { @project.reload.has_external_issue_tracker }.to(false)
end
it 'does not change `has_external_issue_tracker` when `issue_tracker` is made inactive, if an active `issue_tracker` exists' do
services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
expect do
service.update!(active: false)
end.not_to change { @project.reload.has_external_issue_tracker }
end
it 'does not change `has_external_issue_tracker` when service is for a different project' do
different_project = projects.create!(namespace_id: @namespace.id)
service = services.create!(category: 'issue_tracker', active: false, project_id: different_project.id)
expect do
service.update!(active: true)
end.not_to change { @project.reload.has_external_issue_tracker }
end
end
describe 'DELETE trigger' do
it 'sets `has_external_issue_tracker` to false when `issue_tracker` is deleted' do
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
expect do
service.delete
end.to change { @project.reload.has_external_issue_tracker }.to(false)
end
it 'sets `has_external_issue_tracker` to false when `issue_tracker` is deleted, if an inactive `issue_tracker` still exists' do
services.create!(category: 'issue_tracker', active: false, project_id: @project.id)
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
expect do
service.delete
end.to change { @project.reload.has_external_issue_tracker }.to(false)
end
it 'does not change `has_external_issue_tracker` when `issue_tracker` is deleted, if an active `issue_tracker` still exists' do
services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
expect do
service.delete
end.not_to change { @project.reload.has_external_issue_tracker }
end
it 'does not change `has_external_issue_tracker` when service is for a different project' do
different_project = projects.create!(namespace_id: @namespace.id)
service = services.create!(category: 'issue_tracker', active: true, project_id: different_project.id)
expect do
service.delete
end.not_to change { @project.reload.has_external_issue_tracker }
end
end
end
describe '#down' do
before do
migration.up
migration.down
end
it 'drops the INSERT trigger' do
expect do
services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
end.not_to change { @project.reload.has_external_issue_tracker }
end
it 'drops the UPDATE trigger' do
service = services.create!(category: 'issue_tracker', active: false, project_id: @project.id)
@project.update!(has_external_issue_tracker: false)
expect do
service.update!(active: true)
end.not_to change { @project.reload.has_external_issue_tracker }
end
it 'drops the DELETE trigger' do
service = services.create!(category: 'issue_tracker', active: true, project_id: @project.id)
@project.update!(has_external_issue_tracker: true)
expect do
service.delete
end.not_to change { @project.reload.has_external_issue_tracker }
end
end
end
...@@ -754,9 +754,8 @@ RSpec.describe MergeRequest, factory_default: :keep do ...@@ -754,9 +754,8 @@ RSpec.describe MergeRequest, factory_default: :keep do
context 'when both internal and external issue trackers are enabled' do context 'when both internal and external issue trackers are enabled' do
before do before do
subject.project.has_external_issue_tracker = true
subject.project.save!
create(:jira_service, project: subject.project) create(:jira_service, project: subject.project)
subject.project.reload
end end
it 'does not cache issues from external trackers' do it 'does not cache issues from external trackers' do
...@@ -1263,9 +1262,10 @@ RSpec.describe MergeRequest, factory_default: :keep do ...@@ -1263,9 +1262,10 @@ RSpec.describe MergeRequest, factory_default: :keep do
describe '#issues_mentioned_but_not_closing' do describe '#issues_mentioned_but_not_closing' do
let(:closing_issue) { create :issue, project: subject.project } let(:closing_issue) { create :issue, project: subject.project }
let(:mentioned_issue) { create :issue, project: subject.project } let(:mentioned_issue) { create :issue, project: subject.project }
let(:commit) { double('commit', safe_message: "Fixes #{closing_issue.to_reference}") } let(:commit) { double('commit', safe_message: "Fixes #{closing_issue.to_reference}") }
subject { create(:merge_request, source_project: create(:project)) }
it 'detects issues mentioned in description but not closed' do it 'detects issues mentioned in description but not closed' do
subject.project.add_developer(subject.author) subject.project.add_developer(subject.author)
subject.description = "Is related to #{mentioned_issue.to_reference} and #{closing_issue.to_reference}" subject.description = "Is related to #{mentioned_issue.to_reference} and #{closing_issue.to_reference}"
...@@ -1279,13 +1279,12 @@ RSpec.describe MergeRequest, factory_default: :keep do ...@@ -1279,13 +1279,12 @@ RSpec.describe MergeRequest, factory_default: :keep do
end end
context 'when the project has an external issue tracker' do context 'when the project has an external issue tracker' do
subject { create(:merge_request, source_project: create(:project, :repository)) }
before do before do
subject.project.add_developer(subject.author) subject.project.add_developer(subject.author)
commit = double(:commit, safe_message: 'Fixes TEST-3') commit = double(:commit, safe_message: 'Fixes TEST-3')
create(:jira_service, project: subject.project) create(:jira_service, project: subject.project)
subject.project.reload
allow(subject).to receive(:commits).and_return([commit]) allow(subject).to receive(:commits).and_return([commit])
allow(subject).to receive(:description).and_return('Is related to TEST-2 and TEST-3') allow(subject).to receive(:description).and_return('Is related to TEST-2 and TEST-3')
...@@ -1469,6 +1468,8 @@ RSpec.describe MergeRequest, factory_default: :keep do ...@@ -1469,6 +1468,8 @@ RSpec.describe MergeRequest, factory_default: :keep do
end end
describe '#default_merge_commit_message' do describe '#default_merge_commit_message' do
subject { create(:merge_request, source_project: create(:project)) }
it 'includes merge information as the title' do it 'includes merge information as the title' do
request = build(:merge_request, source_branch: 'source', target_branch: 'target') request = build(:merge_request, source_branch: 'source', target_branch: 'target')
...@@ -1645,7 +1646,7 @@ RSpec.describe MergeRequest, factory_default: :keep do ...@@ -1645,7 +1646,7 @@ RSpec.describe MergeRequest, factory_default: :keep do
end end
it_behaves_like 'an editable mentionable' do it_behaves_like 'an editable mentionable' do
subject { create(:merge_request, :simple) } subject { create(:merge_request, :simple, source_project: create(:project, :repository)) }
let(:backref_text) { "merge request #{subject.to_reference}" } let(:backref_text) { "merge request #{subject.to_reference}" }
let(:set_mentionable_text) { ->(txt) { subject.description = txt } } let(:set_mentionable_text) { ->(txt) { subject.description = txt } }
......
...@@ -1002,103 +1002,125 @@ RSpec.describe Project, factory_default: :keep do ...@@ -1002,103 +1002,125 @@ RSpec.describe Project, factory_default: :keep do
end end
end end
describe '#external_issue_tracker' do describe '#has_wiki?' do
let(:project) { create(:project) } let(:no_wiki_project) { create(:project, :wiki_disabled, has_external_wiki: false) }
let(:ext_project) { create(:redmine_project) } let(:wiki_enabled_project) { create(:project) }
let(:external_wiki_project) { create(:project, has_external_wiki: true) }
context 'on existing projects with no value for has_external_issue_tracker' do it 'returns true if project is wiki enabled or has external wiki' do
before do expect(wiki_enabled_project).to have_wiki
project.update_column(:has_external_issue_tracker, nil) expect(external_wiki_project).to have_wiki
ext_project.update_column(:has_external_issue_tracker, nil) expect(no_wiki_project).not_to have_wiki
end
end
describe '#default_owner' do
let_it_be(:owner) { create(:user) }
let_it_be(:namespace) { create(:namespace, owner: owner) }
context 'the project does not have a group' do
let(:project) { build(:project, namespace: namespace) }
it 'is the namespace owner' do
expect(project.default_owner).to eq(owner)
end end
end
it 'updates the has_external_issue_tracker boolean' do context 'the project is in a group' do
expect do let(:group) { build(:group) }
project.external_issue_tracker let(:project) { build(:project, group: group, namespace: namespace) }
end.to change { project.reload.has_external_issue_tracker }.to(false)
expect do it 'is the group owner' do
ext_project.external_issue_tracker allow(group).to receive(:default_owner).and_return(Object.new)
end.to change { ext_project.reload.has_external_issue_tracker }.to(true)
expect(project.default_owner).to eq(group.default_owner)
end end
end end
end
describe '#external_issue_tracker' do
it 'sets Project#has_external_issue_tracker when it is nil' do
project_with_no_tracker = create(:project, has_external_issue_tracker: nil)
project_with_tracker = create(:redmine_project, has_external_issue_tracker: nil)
expect do
project_with_no_tracker.external_issue_tracker
end.to change { project_with_no_tracker.reload.has_external_issue_tracker }.from(nil).to(false)
expect do
project_with_tracker.external_issue_tracker
end.to change { project_with_tracker.reload.has_external_issue_tracker }.from(nil).to(true)
end
it 'returns nil and does not query services when there is no external issue tracker' do it 'returns nil and does not query services when there is no external issue tracker' do
expect(project).not_to receive(:services) project = create(:project)
expect(project).not_to receive(:services)
expect(project.external_issue_tracker).to eq(nil) expect(project.external_issue_tracker).to eq(nil)
end end
it 'retrieves external_issue_tracker querying services and cache it when there is external issue tracker' do it 'retrieves external_issue_tracker querying services and cache it when there is external issue tracker' do
ext_project.reload # Factory returns a project with changed attributes project = create(:redmine_project)
expect(ext_project).to receive(:services).once.and_call_original
2.times { expect(ext_project.external_issue_tracker).to be_a_kind_of(RedmineService) } expect(project).to receive(:services).once.and_call_original
2.times { expect(project.external_issue_tracker).to be_a_kind_of(RedmineService) }
end end
end end
describe '#cache_has_external_issue_tracker' do describe '#has_external_issue_tracker' do
let_it_be(:project) { create(:project, has_external_issue_tracker: nil) } let_it_be(:project) { create(:project) }
it 'stores true if there is any external_issue_tracker' do
services = double(:service, external_issue_trackers: [RedmineService.new])
expect(project).to receive(:services).and_return(services)
expect do def subject
project.cache_has_external_issue_tracker project.reload.has_external_issue_tracker
end.to change { project.has_external_issue_tracker}.to(true)
end end
it 'stores false if there is no external_issue_tracker' do it 'is false when external issue tracker service is not active' do
services = double(:service, external_issue_trackers: []) create(:service, project: project, category: 'issue_tracker', active: false)
expect(project).to receive(:services).and_return(services)
expect do is_expected.to eq(false)
project.cache_has_external_issue_tracker
end.to change { project.has_external_issue_tracker}.to(false)
end end
it 'does not cache data when in a read-only GitLab instance' do it 'is false when other service is active' do
allow(Gitlab::Database).to receive(:read_only?) { true } create(:service, project: project, category: 'not_issue_tracker', active: true)
expect do is_expected.to eq(false)
project.cache_has_external_issue_tracker
end.not_to change { project.has_external_issue_tracker }
end end
end
describe '#has_wiki?' do context 'when there is an active external issue tracker service' do
let(:no_wiki_project) { create(:project, :wiki_disabled, has_external_wiki: false) } let!(:service) do
let(:wiki_enabled_project) { create(:project) } create(:service, project: project, type: 'JiraService', category: 'issue_tracker', active: true)
let(:external_wiki_project) { create(:project, has_external_wiki: true) } end
it 'returns true if project is wiki enabled or has external wiki' do
expect(wiki_enabled_project).to have_wiki
expect(external_wiki_project).to have_wiki
expect(no_wiki_project).not_to have_wiki
end
end
describe '#default_owner' do specify { is_expected.to eq(true) }
let_it_be(:owner) { create(:user) }
let_it_be(:namespace) { create(:namespace, owner: owner) }
context 'the project does not have a group' do it 'becomes false when external issue tracker service is destroyed' do
let(:project) { build(:project, namespace: namespace) } expect do
Service.find(service.id).delete
end.to change { subject }.to(false)
end
it 'is the namespace owner' do it 'becomes false when external issue tracker service becomes inactive' do
expect(project.default_owner).to eq(owner) expect do
service.update_column(:active, false)
end.to change { subject }.to(false)
end end
end
context 'the project is in a group' do context 'when there are two active external issue tracker services' do
let(:group) { build(:group) } let_it_be(:second_service) do
let(:project) { build(:project, group: group, namespace: namespace) } create(:service, project: project, type: 'CustomIssueTracker', category: 'issue_tracker', active: true)
end
it 'is the group owner' do it 'does not become false when external issue tracker service is destroyed' do
allow(group).to receive(:default_owner).and_return(Object.new) expect do
Service.find(service.id).delete
end.not_to change { subject }
end
expect(project.default_owner).to eq(group.default_owner) it 'does not become false when external issue tracker service becomes inactive' do
expect do
service.update_column(:active, false)
end.not_to change { subject }
end
end end
end end
end end
......
...@@ -753,38 +753,6 @@ RSpec.describe Service do ...@@ -753,38 +753,6 @@ RSpec.describe Service do
end end
end end
describe "callbacks" do
let!(:service) do
RedmineService.new(
project: project,
active: true,
properties: {
project_url: 'http://redmine/projects/project_name_in_redmine',
issues_url: "http://redmine/#{project.id}/project_name_in_redmine/:id",
new_issue_url: 'http://redmine/projects/project_name_in_redmine/issues/new'
}
)
end
describe "on create" do
it "updates the has_external_issue_tracker boolean" do
expect do
service.save!
end.to change { service.project.has_external_issue_tracker }.from(false).to(true)
end
end
describe "on update" do
it "updates the has_external_issue_tracker boolean" do
service.save!
expect do
service.update(active: false)
end.to change { service.project.has_external_issue_tracker }.from(true).to(false)
end
end
end
describe '#api_field_names' do describe '#api_field_names' do
let(:fake_service) do let(:fake_service) do
Class.new(Service) do Class.new(Service) do
...@@ -864,20 +832,6 @@ RSpec.describe Service do ...@@ -864,20 +832,6 @@ RSpec.describe Service do
end end
end end
describe '#external_issue_tracker?' do
where(:category, :active, :result) do
:issue_tracker | true | true
:issue_tracker | false | false
:common | true | false
end
with_them do
it 'returns the right result' do
expect(build(:service, category: category, active: active).external_issue_tracker?).to eq(result)
end
end
end
describe '#external_wiki?' do describe '#external_wiki?' do
where(:type, :active, :result) do where(:type, :active, :result) do
'ExternalWikiService' | true | true 'ExternalWikiService' | true | true
......
...@@ -43,46 +43,6 @@ RSpec.describe BulkCreateIntegrationService do ...@@ -43,46 +43,6 @@ RSpec.describe BulkCreateIntegrationService do
end end
end end
shared_examples 'updates project callbacks' do
it 'updates projects#has_external_issue_tracker for issue tracker services' do
described_class.new(integration, batch, association).execute
expect(project.reload.has_external_issue_tracker).to eq(true)
expect(excluded_project.reload.has_external_issue_tracker).to eq(false)
end
context 'with an external wiki integration' do
before do
integration.update!(category: 'common', type: 'ExternalWikiService')
end
it 'updates projects#has_external_wiki for external wiki services' do
described_class.new(integration, batch, association).execute
expect(project.reload.has_external_wiki).to eq(true)
expect(excluded_project.reload.has_external_wiki).to eq(false)
end
end
end
shared_examples 'does not update project callbacks' do
it 'does not update projects#has_external_issue_tracker for issue tracker services' do
described_class.new(integration, batch, association).execute
expect(project.reload.has_external_issue_tracker).to eq(false)
end
context 'with an inactive external wiki integration' do
let(:integration) { create(:external_wiki_service, :instance, active: false) }
it 'does not update projects#has_external_wiki for external wiki services' do
described_class.new(integration, batch, association).execute
expect(project.reload.has_external_wiki).to eq(false)
end
end
end
context 'passing an instance-level integration' do context 'passing an instance-level integration' do
let(:integration) { instance_integration } let(:integration) { instance_integration }
let(:inherit_from_id) { integration.id } let(:inherit_from_id) { integration.id }
...@@ -95,15 +55,6 @@ RSpec.describe BulkCreateIntegrationService do ...@@ -95,15 +55,6 @@ RSpec.describe BulkCreateIntegrationService do
it_behaves_like 'creates integration from batch ids' it_behaves_like 'creates integration from batch ids'
it_behaves_like 'updates inherit_from_id' it_behaves_like 'updates inherit_from_id'
it_behaves_like 'updates project callbacks'
context 'when integration is not active' do
before do
integration.update!(active: false)
end
it_behaves_like 'does not update project callbacks'
end
end end
context 'with a group association' do context 'with a group association' do
...@@ -130,7 +81,6 @@ RSpec.describe BulkCreateIntegrationService do ...@@ -130,7 +81,6 @@ RSpec.describe BulkCreateIntegrationService do
it_behaves_like 'creates integration from batch ids' it_behaves_like 'creates integration from batch ids'
it_behaves_like 'updates inherit_from_id' it_behaves_like 'updates inherit_from_id'
it_behaves_like 'updates project callbacks'
end end
context 'with a group association' do context 'with a group association' do
...@@ -157,7 +107,6 @@ RSpec.describe BulkCreateIntegrationService do ...@@ -157,7 +107,6 @@ RSpec.describe BulkCreateIntegrationService do
let(:inherit_from_id) { integration.id } let(:inherit_from_id) { integration.id }
it_behaves_like 'creates integration from batch ids' it_behaves_like 'creates integration from batch ids'
it_behaves_like 'updates project callbacks'
end end
end end
end end
...@@ -84,6 +84,7 @@ RSpec.describe Issues::CloseService do ...@@ -84,6 +84,7 @@ RSpec.describe Issues::CloseService do
let!(:external_issue_tracker) { create(:jira_service, project: project) } let!(:external_issue_tracker) { create(:jira_service, project: project) }
it 'closes the issue on the external issue tracker' do it 'closes the issue on the external issue tracker' do
project.reload
expect(project.external_issue_tracker).to receive(:close_issue) expect(project.external_issue_tracker).to receive(:close_issue)
described_class.new(project, user).close_issue(external_issue) described_class.new(project, user).close_issue(external_issue)
...@@ -94,6 +95,7 @@ RSpec.describe Issues::CloseService do ...@@ -94,6 +95,7 @@ RSpec.describe Issues::CloseService do
let!(:external_issue_tracker) { create(:jira_service, project: project, active: false) } let!(:external_issue_tracker) { create(:jira_service, project: project, active: false) }
it 'does not close the issue on the external issue tracker' do it 'does not close the issue on the external issue tracker' do
project.reload
expect(project.external_issue_tracker).not_to receive(:close_issue) expect(project.external_issue_tracker).not_to receive(:close_issue)
described_class.new(project, user).close_issue(external_issue) described_class.new(project, user).close_issue(external_issue)
...@@ -104,6 +106,7 @@ RSpec.describe Issues::CloseService do ...@@ -104,6 +106,7 @@ RSpec.describe Issues::CloseService do
let!(:external_issue_tracker) { create(:bugzilla_service, project: project) } let!(:external_issue_tracker) { create(:bugzilla_service, project: project) }
it 'does not close the issue on the external issue tracker' do it 'does not close the issue on the external issue tracker' do
project.reload
expect(project.external_issue_tracker).not_to receive(:close_issue) expect(project.external_issue_tracker).not_to receive(:close_issue)
described_class.new(project, user).close_issue(external_issue) described_class.new(project, user).close_issue(external_issue)
......
...@@ -252,6 +252,7 @@ RSpec.describe MergeRequests::BuildService do ...@@ -252,6 +252,7 @@ RSpec.describe MergeRequests::BuildService do
issue.update!(iid: 123) issue.update!(iid: 123)
else else
create(:"#{issue_tracker}_service", project: project) create(:"#{issue_tracker}_service", project: project)
project.reload
end end
end end
...@@ -351,6 +352,7 @@ RSpec.describe MergeRequests::BuildService do ...@@ -351,6 +352,7 @@ RSpec.describe MergeRequests::BuildService do
issue.update!(iid: 123) issue.update!(iid: 123)
else else
create(:"#{issue_tracker}_service", project: project) create(:"#{issue_tracker}_service", project: project)
project.reload
end end
end end
......
...@@ -729,12 +729,14 @@ RSpec.describe ::SystemNotes::IssuablesService do ...@@ -729,12 +729,14 @@ RSpec.describe ::SystemNotes::IssuablesService do
it 'is false with issue tracker supporting referencing' do it 'is false with issue tracker supporting referencing' do
create(:jira_service, project: project) create(:jira_service, project: project)
project.reload
expect(service.cross_reference_disallowed?(noteable)).to be_falsey expect(service.cross_reference_disallowed?(noteable)).to be_falsey
end end
it 'is true with issue tracker not supporting referencing' do it 'is true with issue tracker not supporting referencing' do
create(:bugzilla_service, project: project) create(:bugzilla_service, project: project)
project.reload
expect(service.cross_reference_disallowed?(noteable)).to be_truthy expect(service.cross_reference_disallowed?(noteable)).to be_truthy
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