# Copyright (C) 2010-2024 RhodeCode GmbH # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License, version 3 # (only), as published by the Free Software Foundation. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see . # # This program is dual-licensed. If you wish to learn more about the # RhodeCode Enterprise Edition, including its added features, Support services, # and proprietary license terms, please see https://rhodecode.com/licenses/ import mock import pytest from rhodecode.model.db import Session, UserLog from rhodecode.lib import hooks_base, utils2 def test_post_push_truncates_commits(user_regular, repo_stub): extras = { 'ip': '127.0.0.1', 'username': user_regular.username, 'user_id': user_regular.user_id, 'action': 'push_local', 'repository': repo_stub.repo_name, 'scm': 'git', 'config': '', 'server_url': 'http://example.com', 'make_lock': None, 'user_agent': 'some-client', 'locked_by': [None], 'commit_ids': ['abcde12345' * 4] * 30000, 'hook_type': 'large_push_test_type', 'is_shadow_repo': False, } extras = utils2.AttributeDict(extras) hooks_base.post_push(extras) # Calculate appropriate action string here commit_ids = extras.commit_ids[:400] entry = UserLog.query().order_by(UserLog.user_log_id.desc()).first() assert entry.action == 'user.push' assert entry.action_data['commit_ids'] == commit_ids Session().delete(entry) Session().commit() def assert_called_with_mock(callable_, expected_mock_name): mock_obj = callable_.call_args[0][0] mock_name = mock_obj._mock_new_parent._mock_new_name assert mock_name == expected_mock_name @pytest.fixture() def hook_extras(user_regular, repo_stub): extras = utils2.AttributeDict({ 'ip': '127.0.0.1', 'username': user_regular.username, 'user_id': user_regular.user_id, 'action': 'push', 'repository': repo_stub.repo_name, 'scm': '', 'config': '', 'repo_store': '', 'server_url': 'http://example.com', 'make_lock': None, 'user_agent': 'some-client', 'locked_by': [None], 'commit_ids': [], 'hook_type': 'test_type', 'is_shadow_repo': False, }) return extras class ExtensionMock(mock.Mock): def __repr__(self): return f'ExtensionMock({self._mock_name})' @property def output(self): return 'MOCK' @property def status(self): return 0 @pytest.mark.parametrize('func, extension, event', [ (hooks_base.pre_push, 'pre_push_extension', 'RepoPrePushEvent'), (hooks_base.post_push, 'post_push_extension', 'RepoPushEvent'), (hooks_base.pre_pull, 'pre_pull_extension', 'RepoPrePullEvent'), (hooks_base.post_pull, 'post_pull_extension', 'RepoPullEvent'), ]) def test_hooks_propagate(func, extension, event, hook_extras): """ Tests that our hook code propagates to rhodecode extensions and triggers the appropriate event. """ extension_mock = ExtensionMock() extension_mock._mock_name = extension events_mock = mock.Mock() patches = { 'Repository': mock.Mock(), 'events': events_mock, extension: extension_mock, } # Clear shadow repo flag. hook_extras.is_shadow_repo = False # Execute hook function. with mock.patch.multiple(hooks_base, **patches): func(hook_extras) # Assert that extensions are called and event was fired. extension_mock.assert_called_once() assert_called_with_mock(events_mock.trigger, event) @pytest.mark.parametrize('func, extension, event', [ (hooks_base.pre_push, 'pre_push_extension', 'RepoPrePushEvent'), (hooks_base.post_push, 'post_push_extension', 'RepoPushEvent'), (hooks_base.pre_pull, 'pre_pull_extension', 'RepoPrePullEvent'), (hooks_base.post_pull, 'post_pull_extension', 'RepoPullEvent'), ]) def test_hooks_propagates_not_on_shadow(func, extension, event, hook_extras): """ If hooks are called by a request to a shadow repo we only want to run our internal hooks code but not external ones like rhodecode extensions or trigger an event. """ extension_mock = ExtensionMock() extension_mock._mock_name = extension events_mock = mock.Mock() patches = { 'Repository': mock.Mock(), 'events': events_mock, extension: extension_mock, } # Set shadow repo flag. hook_extras.is_shadow_repo = True # Execute hook function. with mock.patch.multiple(hooks_base, **patches): func(hook_extras) # Assert that extensions are *not* called and event was *not* fired. assert not extension_mock.called assert not events_mock.trigger.called