repos_group.py
314 lines
| 11.0 KiB
| text/x-python
|
PythonLexer
r1345 | # -*- coding: utf-8 -*- | |||
""" | ||||
rhodecode.model.user_group | ||||
~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
users groups model for RhodeCode | ||||
:created_on: Jan 25, 2011 | ||||
:author: marcink | ||||
r1824 | :copyright: (C) 2011-2012 Marcin Kuzminski <marcin@python-works.com> | |||
r1345 | :license: GPLv3, see COPYING for more details. | |||
""" | ||||
# This program is free software: you can redistribute it and/or modify | ||||
# it under the terms of the GNU General Public License as published by | ||||
# the Free Software Foundation, either version 3 of the License, or | ||||
# (at your option) any later version. | ||||
# | ||||
# 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 General Public License | ||||
# along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||
import os | ||||
import logging | ||||
import traceback | ||||
r1347 | import shutil | |||
r1345 | ||||
r2109 | from rhodecode.lib.utils2 import LazyProperty | |||
r1345 | ||||
from rhodecode.model import BaseModel | ||||
r1982 | from rhodecode.model.db import RepoGroup, RhodeCodeUi, UserRepoGroupToPerm, \ | |||
User, Permission, UsersGroupRepoGroupToPerm, UsersGroup | ||||
r1345 | ||||
Nicolas VINOT
|
r1593 | log = logging.getLogger(__name__) | ||
r1345 | ||||
Nicolas VINOT
|
r1593 | class ReposGroupModel(BaseModel): | ||
r1345 | ||||
r2522 | cls = RepoGroup | |||
r1982 | def __get_users_group(self, users_group): | |||
return self._get_instance(UsersGroup, users_group, | ||||
callback=UsersGroup.get_by_group_name) | ||||
r2631 | def _get_repos_group(self, repos_group): | |||
r1982 | return self._get_instance(RepoGroup, repos_group, | |||
callback=RepoGroup.get_by_group_name) | ||||
r1345 | @LazyProperty | |||
Nicolas VINOT
|
r1593 | def repos_path(self): | ||
r1345 | """ | |||
Get's the repositories root path from database | ||||
""" | ||||
r2708 | q = RhodeCodeUi.get_by_key('/') | |||
r1345 | return q.ui_value | |||
r1982 | def _create_default_perms(self, new_group): | |||
# create default permission | ||||
repo_group_to_perm = UserRepoGroupToPerm() | ||||
default_perm = 'group.read' | ||||
for p in User.get_by_username('default').user_perms: | ||||
if p.permission.permission_name.startswith('group.'): | ||||
default_perm = p.permission.permission_name | ||||
break | ||||
repo_group_to_perm.permission_id = self.sa.query(Permission)\ | ||||
.filter(Permission.permission_name == default_perm)\ | ||||
.one().permission_id | ||||
repo_group_to_perm.group = new_group | ||||
repo_group_to_perm.user_id = User.get_by_username('default').user_id | ||||
self.sa.add(repo_group_to_perm) | ||||
r1594 | def __create_group(self, group_name): | |||
r1345 | """ | |||
makes repositories group on filesystem | ||||
:param repo_name: | ||||
:param parent_id: | ||||
""" | ||||
r1594 | create_path = os.path.join(self.repos_path, group_name) | |||
r1976 | log.debug('creating new group in %s' % create_path) | |||
r1345 | ||||
Nicolas VINOT
|
r1593 | if os.path.isdir(create_path): | ||
raise Exception('That directory already exists !') | ||||
r1345 | ||||
Nicolas VINOT
|
r1593 | os.makedirs(create_path) | ||
r1345 | ||||
r1594 | def __rename_group(self, old, new): | |||
r1345 | """ | |||
Renames a group on filesystem | ||||
r1818 | ||||
r1345 | :param group_name: | |||
""" | ||||
r1594 | ||||
if old == new: | ||||
log.debug('skipping group rename') | ||||
return | ||||
r1976 | log.debug('renaming repos group from %s to %s' % (old, new)) | |||
r1347 | ||||
r1594 | old_path = os.path.join(self.repos_path, old) | |||
new_path = os.path.join(self.repos_path, new) | ||||
r1349 | ||||
r1976 | log.debug('renaming repos paths from %s to %s' % (old_path, new_path)) | |||
r1349 | ||||
Nicolas VINOT
|
r1593 | if os.path.isdir(new_path): | ||
raise Exception('Was trying to rename to already ' | ||||
'existing dir %s' % new_path) | ||||
shutil.move(old_path, new_path) | ||||
r1345 | ||||
Nicolas VINOT
|
r1593 | def __delete_group(self, group): | ||
r1345 | """ | |||
Deletes a group from a filesystem | ||||
r1818 | ||||
r1346 | :param group: instance of group from database | |||
r1345 | """ | |||
r1633 | paths = group.full_path.split(RepoGroup.url_sep()) | |||
Nicolas VINOT
|
r1593 | paths = os.sep.join(paths) | ||
r1346 | ||||
Nicolas VINOT
|
r1593 | rm_path = os.path.join(self.repos_path, paths) | ||
r1594 | if os.path.isdir(rm_path): | |||
# delete only if that path really exists | ||||
os.rmdir(rm_path) | ||||
r1345 | ||||
r2529 | def create(self, group_name, group_description, parent=None, just_db=False): | |||
r1345 | try: | |||
r1633 | new_repos_group = RepoGroup() | |||
r1982 | new_repos_group.group_description = group_description | |||
r2631 | new_repos_group.parent_group = self._get_repos_group(parent) | |||
r1982 | new_repos_group.group_name = new_repos_group.get_new_name(group_name) | |||
r1345 | ||||
Nicolas VINOT
|
r1593 | self.sa.add(new_repos_group) | ||
r1982 | self._create_default_perms(new_repos_group) | |||
if not just_db: | ||||
# we need to flush here, in order to check if database won't | ||||
# throw any exceptions, create filesystem dirs at the very end | ||||
self.sa.flush() | ||||
self.__create_group(new_repos_group.group_name) | ||||
r1345 | ||||
Nicolas VINOT
|
r1589 | return new_repos_group | ||
r1345 | except: | |||
Nicolas VINOT
|
r1593 | log.error(traceback.format_exc()) | ||
r1345 | raise | |||
Nicolas VINOT
|
r1593 | def update(self, repos_group_id, form_data): | ||
r1345 | ||||
try: | ||||
r1633 | repos_group = RepoGroup.get(repos_group_id) | |||
r1982 | ||||
# update permissions | ||||
for member, perm, member_type in form_data['perms_updates']: | ||||
if member_type == 'user': | ||||
# this updates also current one if found | ||||
ReposGroupModel().grant_user_permission( | ||||
repos_group=repos_group, user=member, perm=perm | ||||
) | ||||
else: | ||||
ReposGroupModel().grant_users_group_permission( | ||||
repos_group=repos_group, group_name=member, perm=perm | ||||
) | ||||
# set new permissions | ||||
for member, perm, member_type in form_data['perms_new']: | ||||
if member_type == 'user': | ||||
ReposGroupModel().grant_user_permission( | ||||
repos_group=repos_group, user=member, perm=perm | ||||
) | ||||
else: | ||||
ReposGroupModel().grant_users_group_permission( | ||||
repos_group=repos_group, group_name=member, perm=perm | ||||
) | ||||
r1594 | old_path = repos_group.full_path | |||
r1734 | ||||
r1594 | # change properties | |||
repos_group.group_description = form_data['group_description'] | ||||
r1633 | repos_group.parent_group = RepoGroup.get(form_data['group_parent_id']) | |||
r2059 | repos_group.group_parent_id = form_data['group_parent_id'] | |||
r2749 | repos_group.enable_locking = form_data['enable_locking'] | |||
r1594 | repos_group.group_name = repos_group.get_new_name(form_data['group_name']) | |||
new_path = repos_group.full_path | ||||
r1345 | ||||
Nicolas VINOT
|
r1593 | self.sa.add(repos_group) | ||
r1347 | ||||
r2749 | # iterate over all members of this groups and set the locking ! | |||
# this can be potentially heavy operation | ||||
for obj in repos_group.recursive_groups_and_repos(): | ||||
#set the value from it's parent | ||||
obj.enable_locking = repos_group.enable_locking | ||||
self.sa.add(obj) | ||||
r1818 | # we need to get all repositories from this new group and | |||
r1594 | # rename them accordingly to new group path | |||
for r in repos_group.repositories: | ||||
r.repo_name = r.get_new_name(r.just_name) | ||||
self.sa.add(r) | ||||
r1347 | ||||
r2059 | self.__rename_group(old_path, new_path) | |||
r1594 | return repos_group | |||
r1345 | except: | |||
Nicolas VINOT
|
r1593 | log.error(traceback.format_exc()) | ||
r1345 | raise | |||
r2467 | def delete(self, repos_group): | |||
r2631 | repos_group = self._get_repos_group(repos_group) | |||
r1345 | try: | |||
r2467 | self.sa.delete(repos_group) | |||
self.__delete_group(repos_group) | ||||
r1345 | except: | |||
r2467 | log.exception('Error removing repos_group %s' % repos_group) | |||
r1345 | raise | |||
r1982 | ||||
def grant_user_permission(self, repos_group, user, perm): | ||||
""" | ||||
Grant permission for user on given repositories group, or update | ||||
existing one if found | ||||
:param repos_group: Instance of ReposGroup, repositories_group_id, | ||||
or repositories_group name | ||||
:param user: Instance of User, user_id or username | ||||
:param perm: Instance of Permission, or permission_name | ||||
""" | ||||
r2631 | repos_group = self._get_repos_group(repos_group) | |||
r2432 | user = self._get_user(user) | |||
permission = self._get_perm(perm) | ||||
r1982 | ||||
# check if we have that permission already | ||||
obj = self.sa.query(UserRepoGroupToPerm)\ | ||||
.filter(UserRepoGroupToPerm.user == user)\ | ||||
.filter(UserRepoGroupToPerm.group == repos_group)\ | ||||
.scalar() | ||||
if obj is None: | ||||
# create new ! | ||||
obj = UserRepoGroupToPerm() | ||||
obj.group = repos_group | ||||
obj.user = user | ||||
obj.permission = permission | ||||
self.sa.add(obj) | ||||
def revoke_user_permission(self, repos_group, user): | ||||
""" | ||||
Revoke permission for user on given repositories group | ||||
:param repos_group: Instance of ReposGroup, repositories_group_id, | ||||
or repositories_group name | ||||
:param user: Instance of User, user_id or username | ||||
""" | ||||
r2631 | repos_group = self._get_repos_group(repos_group) | |||
r2432 | user = self._get_user(user) | |||
r1982 | ||||
obj = self.sa.query(UserRepoGroupToPerm)\ | ||||
.filter(UserRepoGroupToPerm.user == user)\ | ||||
.filter(UserRepoGroupToPerm.group == repos_group)\ | ||||
.one() | ||||
self.sa.delete(obj) | ||||
def grant_users_group_permission(self, repos_group, group_name, perm): | ||||
""" | ||||
Grant permission for users group on given repositories group, or update | ||||
existing one if found | ||||
:param repos_group: Instance of ReposGroup, repositories_group_id, | ||||
or repositories_group name | ||||
:param group_name: Instance of UserGroup, users_group_id, | ||||
or users group name | ||||
:param perm: Instance of Permission, or permission_name | ||||
""" | ||||
r2631 | repos_group = self._get_repos_group(repos_group) | |||
r1982 | group_name = self.__get_users_group(group_name) | |||
r2432 | permission = self._get_perm(perm) | |||
r1982 | ||||
# check if we have that permission already | ||||
obj = self.sa.query(UsersGroupRepoGroupToPerm)\ | ||||
.filter(UsersGroupRepoGroupToPerm.group == repos_group)\ | ||||
.filter(UsersGroupRepoGroupToPerm.users_group == group_name)\ | ||||
.scalar() | ||||
if obj is None: | ||||
# create new | ||||
obj = UsersGroupRepoGroupToPerm() | ||||
obj.group = repos_group | ||||
obj.users_group = group_name | ||||
obj.permission = permission | ||||
self.sa.add(obj) | ||||
def revoke_users_group_permission(self, repos_group, group_name): | ||||
""" | ||||
Revoke permission for users group on given repositories group | ||||
:param repos_group: Instance of ReposGroup, repositories_group_id, | ||||
or repositories_group name | ||||
:param group_name: Instance of UserGroup, users_group_id, | ||||
or users group name | ||||
""" | ||||
r2631 | repos_group = self._get_repos_group(repos_group) | |||
r1982 | group_name = self.__get_users_group(group_name) | |||
obj = self.sa.query(UsersGroupRepoGroupToPerm)\ | ||||
.filter(UsersGroupRepoGroupToPerm.group == repos_group)\ | ||||
.filter(UsersGroupRepoGroupToPerm.users_group == group_name)\ | ||||
.one() | ||||
self.sa.delete(obj) | ||||