|
|
# 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 <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 os
|
|
|
import pytest
|
|
|
|
|
|
from rhodecode.apps._base import ADMIN_PREFIX
|
|
|
from rhodecode.lib import helpers as h
|
|
|
from rhodecode.model.db import Repository, UserRepoToPerm, User, RepoGroup
|
|
|
from rhodecode.model.meta import Session
|
|
|
from rhodecode.model.repo_group import RepoGroupModel
|
|
|
from rhodecode.tests import (
|
|
|
assert_session_flash, TEST_USER_REGULAR_LOGIN, TESTS_TMP_PATH)
|
|
|
from rhodecode.tests.fixtures.rc_fixture import Fixture
|
|
|
from rhodecode.tests.routes import route_path
|
|
|
|
|
|
|
|
|
fixture = Fixture()
|
|
|
|
|
|
|
|
|
def _get_permission_for_user(user, repo):
|
|
|
perm = UserRepoToPerm.query()\
|
|
|
.filter(UserRepoToPerm.repository ==
|
|
|
Repository.get_by_repo_name(repo))\
|
|
|
.filter(UserRepoToPerm.user == User.get_by_username(user))\
|
|
|
.all()
|
|
|
return perm
|
|
|
|
|
|
|
|
|
@pytest.mark.usefixtures("app")
|
|
|
class TestAdminRepositoryGroups(object):
|
|
|
|
|
|
def test_show_repo_groups(self, autologin_user):
|
|
|
self.app.get(route_path('repo_groups'))
|
|
|
|
|
|
def test_show_repo_groups_data(self, autologin_user, xhr_header):
|
|
|
response = self.app.get(route_path(
|
|
|
'repo_groups_data'), extra_environ=xhr_header)
|
|
|
|
|
|
all_repo_groups = RepoGroup.query().count()
|
|
|
assert response.json['recordsTotal'] == all_repo_groups
|
|
|
|
|
|
def test_show_repo_groups_data_filtered(self, autologin_user, xhr_header):
|
|
|
response = self.app.get(route_path(
|
|
|
'repo_groups_data', params={'search[value]': 'empty_search'}),
|
|
|
extra_environ=xhr_header)
|
|
|
|
|
|
all_repo_groups = RepoGroup.query().count()
|
|
|
assert response.json['recordsTotal'] == all_repo_groups
|
|
|
assert response.json['recordsFiltered'] == 0
|
|
|
|
|
|
def test_show_repo_groups_after_creating_group(self, autologin_user, xhr_header):
|
|
|
fixture.create_repo_group('test_repo_group')
|
|
|
response = self.app.get(route_path(
|
|
|
'repo_groups_data'), extra_environ=xhr_header)
|
|
|
response.mustcontain('<a href=\\"/{}/_edit\\" title=\\"Edit\\">Edit</a>'.format('test_repo_group'))
|
|
|
fixture.destroy_repo_group('test_repo_group')
|
|
|
|
|
|
def test_new(self, autologin_user):
|
|
|
self.app.get(route_path('repo_group_new'))
|
|
|
|
|
|
def test_new_with_parent_group(self, autologin_user, user_util):
|
|
|
gr = user_util.create_repo_group()
|
|
|
|
|
|
self.app.get(route_path('repo_group_new'),
|
|
|
params=dict(parent_group=gr.group_name))
|
|
|
|
|
|
def test_new_by_regular_user_no_permission(self, autologin_regular_user):
|
|
|
self.app.get(route_path('repo_group_new'), status=403)
|
|
|
|
|
|
@pytest.mark.parametrize('repo_group_name', [
|
|
|
'git_repo',
|
|
|
'git_repo_ąć',
|
|
|
'hg_repo',
|
|
|
'12345',
|
|
|
'hg_repo_ąć',
|
|
|
])
|
|
|
def test_create(self, autologin_user, repo_group_name, csrf_token):
|
|
|
repo_group_name_non_ascii = repo_group_name
|
|
|
description = 'description for newly created repo group'
|
|
|
|
|
|
response = self.app.post(
|
|
|
route_path('repo_group_create'),
|
|
|
fixture._get_group_create_params(
|
|
|
group_name=repo_group_name,
|
|
|
group_description=description,
|
|
|
csrf_token=csrf_token))
|
|
|
|
|
|
# run the check page that triggers the flash message
|
|
|
repo_gr_url = h.route_path(
|
|
|
'repo_group_home', repo_group_name=repo_group_name)
|
|
|
|
|
|
assert_session_flash(
|
|
|
response,
|
|
|
'Created repository group <a href="%s">%s</a>' % (
|
|
|
repo_gr_url, repo_group_name_non_ascii))
|
|
|
|
|
|
# # test if the repo group was created in the database
|
|
|
new_repo_group = RepoGroupModel()._get_repo_group(
|
|
|
repo_group_name_non_ascii)
|
|
|
assert new_repo_group is not None
|
|
|
|
|
|
assert new_repo_group.group_name == repo_group_name_non_ascii
|
|
|
assert new_repo_group.group_description == description
|
|
|
|
|
|
# test if the repository is visible in the list ?
|
|
|
response = self.app.get(repo_gr_url)
|
|
|
response.mustcontain(repo_group_name)
|
|
|
|
|
|
# test if the repository group was created on filesystem
|
|
|
is_on_filesystem = os.path.isdir(
|
|
|
os.path.join(TESTS_TMP_PATH, repo_group_name))
|
|
|
if not is_on_filesystem:
|
|
|
self.fail('no repo group %s in filesystem' % repo_group_name)
|
|
|
|
|
|
RepoGroupModel().delete(repo_group_name_non_ascii)
|
|
|
Session().commit()
|
|
|
|
|
|
@pytest.mark.parametrize('repo_group_name', [
|
|
|
'git_repo',
|
|
|
'git_repo_ąć',
|
|
|
'hg_repo',
|
|
|
'12345',
|
|
|
'hg_repo_ąć',
|
|
|
])
|
|
|
def test_create_subgroup(self, autologin_user, user_util, repo_group_name, csrf_token):
|
|
|
parent_group = user_util.create_repo_group()
|
|
|
parent_group_name = parent_group.group_name
|
|
|
|
|
|
expected_group_name = '{}/{}'.format(
|
|
|
parent_group_name, repo_group_name)
|
|
|
expected_group_name_non_ascii = expected_group_name
|
|
|
|
|
|
try:
|
|
|
response = self.app.post(
|
|
|
route_path('repo_group_create'),
|
|
|
fixture._get_group_create_params(
|
|
|
group_name=repo_group_name,
|
|
|
group_parent_id=parent_group.group_id,
|
|
|
group_description='Test desciption',
|
|
|
csrf_token=csrf_token))
|
|
|
|
|
|
assert_session_flash(
|
|
|
response,
|
|
|
u'Created repository group <a href="%s">%s</a>' % (
|
|
|
h.route_path('repo_group_home',
|
|
|
repo_group_name=expected_group_name),
|
|
|
expected_group_name_non_ascii))
|
|
|
finally:
|
|
|
RepoGroupModel().delete(expected_group_name_non_ascii)
|
|
|
Session().commit()
|
|
|
|
|
|
def test_user_with_creation_permissions_cannot_create_subgroups(
|
|
|
self, autologin_regular_user, user_util):
|
|
|
|
|
|
user_util.grant_user_permission(
|
|
|
TEST_USER_REGULAR_LOGIN, 'hg.repogroup.create.true')
|
|
|
parent_group = user_util.create_repo_group()
|
|
|
parent_group_id = parent_group.group_id
|
|
|
self.app.get(
|
|
|
route_path('repo_group_new',
|
|
|
params=dict(parent_group=parent_group_id), ),
|
|
|
status=403)
|
|
|
|