test_repo_changelog.py
220 lines
| 8.4 KiB
| text/x-python
|
PythonLexer
r1931 | ||||
r5088 | # Copyright (C) 2010-2023 RhodeCode GmbH | |||
r1931 | # | |||
# 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 <http://www.gnu.org/licenses/>. | ||||
# | ||||
# 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 re | ||||
import pytest | ||||
from rhodecode.apps.repository.views.repo_changelog import DEFAULT_CHANGELOG_SIZE | ||||
from rhodecode.tests import TestController | ||||
MATCH_HASH = re.compile(r'<span class="commit_hash">r(\d+):[\da-f]+</span>') | ||||
def route_path(name, params=None, **kwargs): | ||||
r5087 | import urllib.request | |||
import urllib.parse | ||||
import urllib.error | ||||
r1931 | ||||
base_url = { | ||||
r3742 | 'repo_changelog': '/{repo_name}/changelog', | |||
'repo_commits': '/{repo_name}/commits', | ||||
'repo_commits_file': '/{repo_name}/commits/{commit_id}/{f_path}', | ||||
'repo_commits_elements': '/{repo_name}/commits_elements', | ||||
r1931 | }[name].format(**kwargs) | |||
if params: | ||||
r4914 | base_url = '{}?{}'.format(base_url, urllib.parse.urlencode(params)) | |||
r1931 | return base_url | |||
r3742 | def assert_commits_on_page(response, indexes): | |||
r5087 | found_indexes = [int(idx) for idx in MATCH_HASH.findall(response.text)] | |||
r3742 | assert found_indexes == indexes | |||
r1931 | class TestChangelogController(TestController): | |||
r3742 | def test_commits_page(self, backend): | |||
self.log_user() | ||||
response = self.app.get( | ||||
route_path('repo_commits', repo_name=backend.repo_name)) | ||||
first_idx = -1 | ||||
last_idx = -DEFAULT_CHANGELOG_SIZE | ||||
r3776 | self.assert_commit_range_on_page(response, first_idx, last_idx, backend) | |||
r3742 | ||||
r1931 | def test_changelog(self, backend): | |||
self.log_user() | ||||
response = self.app.get( | ||||
route_path('repo_changelog', repo_name=backend.repo_name)) | ||||
first_idx = -1 | ||||
last_idx = -DEFAULT_CHANGELOG_SIZE | ||||
self.assert_commit_range_on_page( | ||||
response, first_idx, last_idx, backend) | ||||
@pytest.mark.backends("hg", "git") | ||||
def test_changelog_filtered_by_branch(self, backend): | ||||
self.log_user() | ||||
self.app.get( | ||||
route_path('repo_changelog', repo_name=backend.repo_name, | ||||
params=dict(branch=backend.default_branch_name)), | ||||
status=200) | ||||
r3742 | @pytest.mark.backends("hg", "git") | |||
def test_commits_filtered_by_branch(self, backend): | ||||
self.log_user() | ||||
self.app.get( | ||||
route_path('repo_commits', repo_name=backend.repo_name, | ||||
params=dict(branch=backend.default_branch_name)), | ||||
status=200) | ||||
r1931 | @pytest.mark.backends("svn") | |||
def test_changelog_filtered_by_branch_svn(self, autologin_user, backend): | ||||
repo = backend['svn-simple-layout'] | ||||
response = self.app.get( | ||||
route_path('repo_changelog', repo_name=repo.repo_name, | ||||
params=dict(branch='trunk')), | ||||
status=200) | ||||
r3742 | assert_commits_on_page(response, indexes=[15, 12, 7, 3, 2, 1]) | |||
r1931 | ||||
r3742 | def test_commits_filtered_by_wrong_branch(self, backend): | |||
r1931 | self.log_user() | |||
branch = 'wrong-branch-name' | ||||
response = self.app.get( | ||||
r3742 | route_path('repo_commits', repo_name=backend.repo_name, | |||
r1931 | params=dict(branch=branch)), | |||
status=302) | ||||
r3742 | expected_url = '/{repo}/commits/{branch}'.format( | |||
r1931 | repo=backend.repo_name, branch=branch) | |||
assert expected_url in response.location | ||||
response = response.follow() | ||||
expected_warning = 'Branch {} is not found.'.format(branch) | ||||
r4994 | assert expected_warning in response.text | |||
r1931 | ||||
@pytest.mark.xfail_backends("svn", reason="Depends on branch support") | ||||
r5087 | def test_changelog_filtered_by_branch_with_merges(self, autologin_user, backend): | |||
r1931 | ||||
# Note: The changelog of branch "b" does not contain the commit "a1" | ||||
# although this is a parent of commit "b1". And branch "b" has commits | ||||
# which have a smaller index than commit "a1". | ||||
commits = [ | ||||
{'message': 'a'}, | ||||
{'message': 'b', 'branch': 'b'}, | ||||
{'message': 'a1', 'parents': ['a']}, | ||||
{'message': 'b1', 'branch': 'b', 'parents': ['b', 'a1']}, | ||||
] | ||||
backend.create_repo(commits) | ||||
self.app.get( | ||||
route_path('repo_changelog', repo_name=backend.repo_name, | ||||
params=dict(branch='b')), | ||||
status=200) | ||||
@pytest.mark.backends("hg") | ||||
r3742 | def test_commits_closed_branches(self, autologin_user, backend): | |||
r1931 | repo = backend['closed_branch'] | |||
response = self.app.get( | ||||
r3742 | route_path('repo_commits', repo_name=repo.repo_name, | |||
r1931 | params=dict(branch='experimental')), | |||
status=200) | ||||
r3742 | assert_commits_on_page(response, indexes=[3, 1]) | |||
r1931 | ||||
def test_changelog_pagination(self, backend): | ||||
self.log_user() | ||||
# pagination, walk up to page 6 | ||||
changelog_url = route_path( | ||||
r3742 | 'repo_commits', repo_name=backend.repo_name) | |||
r1931 | ||||
for page in range(1, 7): | ||||
response = self.app.get(changelog_url, {'page': page}) | ||||
first_idx = -DEFAULT_CHANGELOG_SIZE * (page - 1) - 1 | ||||
last_idx = -DEFAULT_CHANGELOG_SIZE * page | ||||
self.assert_commit_range_on_page(response, first_idx, last_idx, backend) | ||||
def assert_commit_range_on_page( | ||||
self, response, first_idx, last_idx, backend): | ||||
input_template = ( | ||||
r3772 | """<input class="commit-range" """ | |||
r3776 | """data-commit-id="%(raw_id)s" data-commit-idx="%(idx)s" """ | |||
"""data-short-id="%(short_id)s" id="%(raw_id)s" """ | ||||
r1931 | """name="%(raw_id)s" type="checkbox" value="1" />""" | |||
) | ||||
r3772 | ||||
r1931 | commit_span_template = """<span class="commit_hash">r%s:%s</span>""" | |||
repo = backend.repo | ||||
first_commit_on_page = repo.get_commit(commit_idx=first_idx) | ||||
response.mustcontain( | ||||
r3772 | input_template % {'raw_id': first_commit_on_page.raw_id, | |||
r3776 | 'idx': first_commit_on_page.idx, | |||
'short_id': first_commit_on_page.short_id}) | ||||
r3772 | ||||
r1931 | response.mustcontain(commit_span_template % ( | |||
first_commit_on_page.idx, first_commit_on_page.short_id) | ||||
) | ||||
last_commit_on_page = repo.get_commit(commit_idx=last_idx) | ||||
response.mustcontain( | ||||
r3772 | input_template % {'raw_id': last_commit_on_page.raw_id, | |||
r3776 | 'idx': last_commit_on_page.idx, | |||
'short_id': last_commit_on_page.short_id}) | ||||
r1931 | response.mustcontain(commit_span_template % ( | |||
last_commit_on_page.idx, last_commit_on_page.short_id) | ||||
) | ||||
first_commit_of_next_page = repo.get_commit(commit_idx=last_idx - 1) | ||||
first_span_of_next_page = commit_span_template % ( | ||||
first_commit_of_next_page.idx, first_commit_of_next_page.short_id) | ||||
assert first_span_of_next_page not in response | ||||
@pytest.mark.parametrize('test_path', [ | ||||
'vcs/exceptions.py', | ||||
'/vcs/exceptions.py', | ||||
'//vcs/exceptions.py' | ||||
]) | ||||
r3742 | def test_commits_with_filenode(self, backend, test_path): | |||
r1931 | self.log_user() | |||
response = self.app.get( | ||||
r3742 | route_path('repo_commits_file', repo_name=backend.repo_name, | |||
r1931 | commit_id='tip', f_path=test_path), | |||
) | ||||
# history commits messages | ||||
response.mustcontain('Added exceptions module, this time for real') | ||||
response.mustcontain('Added not implemented hg backend test case') | ||||
response.mustcontain('Added BaseChangeset class') | ||||
r3742 | def test_commits_with_filenode_that_is_dirnode(self, backend): | |||
r1931 | self.log_user() | |||
self.app.get( | ||||
r3742 | route_path('repo_commits_file', repo_name=backend.repo_name, | |||
r1931 | commit_id='tip', f_path='/tests'), | |||
status=302) | ||||
r3742 | def test_commits_with_filenode_not_existing(self, backend): | |||
r1931 | self.log_user() | |||
self.app.get( | ||||
r3742 | route_path('repo_commits_file', repo_name=backend.repo_name, | |||
r1931 | commit_id='tip', f_path='wrong_path'), | |||
status=302) | ||||