Commit 4183392e authored by Andy Soiron's avatar Andy Soiron Committed by Stan Hu

Fix group hook triggering from subgroup project

This commit fixes a bug where group hooks on a
parent group where not triggered from a project
in a subgroup
parent 8cdfa402
# frozen_string_literal: true
class AddIndexWebHooksOnGroupId < ActiveRecord::Migration[5.2]
include Gitlab::Database::MigrationHelpers
DOWNTIME = false
disable_ddl_transaction!
def up
add_concurrent_index :web_hooks, :group_id, where: "type = 'GroupHook'"
end
def down
remove_concurrent_index :web_hooks, :group_id, where: "type = 'GroupHook'"
end
end
...@@ -4474,6 +4474,7 @@ ActiveRecord::Schema.define(version: 2020_02_04_131054) do ...@@ -4474,6 +4474,7 @@ ActiveRecord::Schema.define(version: 2020_02_04_131054) do
t.string "encrypted_token_iv" t.string "encrypted_token_iv"
t.string "encrypted_url" t.string "encrypted_url"
t.string "encrypted_url_iv" t.string "encrypted_url_iv"
t.index ["group_id"], name: "index_web_hooks_on_group_id", where: "((type)::text = 'GroupHook'::text)"
t.index ["project_id"], name: "index_web_hooks_on_project_id" t.index ["project_id"], name: "index_web_hooks_on_project_id"
t.index ["type"], name: "index_web_hooks_on_type" t.index ["type"], name: "index_web_hooks_on_type"
end end
......
...@@ -359,7 +359,7 @@ module EE ...@@ -359,7 +359,7 @@ module EE
def has_group_hooks?(hooks_scope = :push_hooks) def has_group_hooks?(hooks_scope = :push_hooks)
return unless group && feature_available?(:group_webhooks) return unless group && feature_available?(:group_webhooks)
group.hooks.hooks_for(hooks_scope).any? group_hooks.hooks_for(hooks_scope).any?
end end
def execute_hooks(data, hooks_scope = :push_hooks) def execute_hooks(data, hooks_scope = :push_hooks)
...@@ -367,7 +367,7 @@ module EE ...@@ -367,7 +367,7 @@ module EE
if group && feature_available?(:group_webhooks) if group && feature_available?(:group_webhooks)
run_after_commit_or_now do run_after_commit_or_now do
group.hooks.hooks_for(hooks_scope).each do |hook| group_hooks.hooks_for(hooks_scope).each do |hook|
hook.async_execute(data, hooks_scope.to_s) hook.async_execute(data, hooks_scope.to_s)
end end
end end
...@@ -738,6 +738,12 @@ module EE ...@@ -738,6 +738,12 @@ module EE
private private
def group_hooks
return group.hooks unless ::Feature.enabled?(:sub_group_webhooks, self)
GroupHook.where(group_id: group.self_and_ancestors)
end
def set_override_pull_mirror_available def set_override_pull_mirror_available
self.pull_mirror_available_overridden = read_attribute(:mirror) self.pull_mirror_available_overridden = read_attribute(:mirror)
true true
......
---
title: Fix group hook triggering from subgroup project
merge_request: 23333
author:
type: fixed
...@@ -506,23 +506,71 @@ describe Project do ...@@ -506,23 +506,71 @@ describe Project do
end end
end end
describe "#execute_hooks" do describe '#has_group_hooks?' do
context "group hooks" do subject { project.has_group_hooks? }
let(:project) { create(:project) }
it { is_expected.to eq(nil) }
context 'project is in a group' do
let(:group) { create(:group) } let(:group) { create(:group) }
let(:project) { create(:project, namespace: group) } let(:project) { create(:project, namespace: group) }
let(:group_hook) { create(:group_hook, group: group, push_events: true) }
it 'executes the hook when the feature is enabled' do shared_examples 'returns nil when the feature is not available' do
stub_licensed_features(group_webhooks: true) specify do
stub_licensed_features(group_webhooks: false)
fake_service = double expect(subject).to eq(nil)
expect(WebHookService).to receive(:new) end
.with(group_hook, { some: 'info' }, 'push_hooks') { fake_service } end
expect(fake_service).to receive(:async_execute)
project.execute_hooks(some: 'info') it_behaves_like 'returns nil when the feature is not available'
it { is_expected.to eq(false) }
context 'the group has hooks' do
let!(:group_hook) { create(:group_hook, group: group, push_events: true) }
it { is_expected.to eq(true) }
it_behaves_like 'returns nil when the feature is not available'
context 'but the hook is not in scope' do
subject { project.has_group_hooks?(:issue_hooks) }
it_behaves_like 'returns nil when the feature is not available'
it { is_expected.to eq(false) }
end
end end
context 'the group inherits a hook' do
let(:parent_group) { create(:group) }
let!(:group_hook) { create(:group_hook, group: parent_group) }
let(:group) { create(:group, parent: parent_group) }
it_behaves_like 'returns nil when the feature is not available'
it { is_expected.to eq(true) }
context 'when sub_group_webhooks feature flag is disabled' do
before do
stub_feature_flags(sub_group_webhooks: false)
end
it { is_expected.to eq(false) }
end
end
end
end
describe "#execute_hooks" do
context "group hooks" do
let(:group) { create(:group) }
let(:project) { create(:project, namespace: group) }
let(:group_hook) { create(:group_hook, group: group, push_events: true) }
it 'does not execute the hook when the feature is disabled' do it 'does not execute the hook when the feature is disabled' do
stub_licensed_features(group_webhooks: false) stub_licensed_features(group_webhooks: false)
...@@ -531,23 +579,54 @@ describe Project do ...@@ -531,23 +579,54 @@ describe Project do
project.execute_hooks(some: 'info') project.execute_hooks(some: 'info')
end end
end
end
describe '#execute_hooks' do context 'when group_webhooks frature is enabled' do
it "triggers project and group hooks" do before do
group = create :group, name: 'gitlab' stub_licensed_features(group_webhooks: true)
project = create(:project, name: 'gitlabhq', namespace: group) end
project_hook = create(:project_hook, push_events: true, project: project) let(:fake_service) { double }
group_hook = create(:group_hook, push_events: true, group: group)
shared_examples 'triggering group webhook' do
it 'executes the hook' do
expect(fake_service).to receive(:async_execute).once
stub_request(:post, project_hook.url) expect(WebHookService).to receive(:new)
stub_request(:post, group_hook.url) .with(group_hook, { some: 'info' }, 'push_hooks') { fake_service }
project.execute_hooks(some: 'info')
end
end
expect_any_instance_of(GroupHook).to receive(:async_execute).and_return(true) it_behaves_like 'triggering group webhook'
expect_any_instance_of(ProjectHook).to receive(:async_execute).and_return(true)
project.execute_hooks({}, :push_hooks) context 'when sub_group_webhooks feature flag is disabled' do
before do
stub_feature_flags(sub_group_webhooks: false)
end
it_behaves_like 'triggering group webhook'
end
context 'in sub group' do
let(:sub_group) { create :group, parent: group }
let(:sub_sub_group) { create :group, parent: sub_group }
let(:project) { create(:project, namespace: sub_sub_group) }
it_behaves_like 'triggering group webhook'
context 'when sub_group_webhooks feature flag is disabled' do
before do
stub_feature_flags(sub_group_webhooks: false)
end
it 'does not execute the hook' do
expect(WebHookService).not_to receive(:new).with(group_hook, { some: 'info' }, 'push_hooks')
project.execute_hooks(some: 'info')
end
end
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