##// END OF EJS Templates
Added tag v5.2.0 for changeset 50cf7822bd80
Added tag v5.2.0 for changeset 50cf7822bd80

File last commit:

r5516:3496180b default
r5547:aedaaf1e stable
Show More
repo_settings_advanced.py
309 lines | 12.1 KiB | text/x-python | PythonLexer
/ rhodecode / apps / repository / views / repo_settings_advanced.py
copyrights: updated for 2023
r5088 # Copyright (C) 2011-2023 RhodeCode GmbH
repo-settings: moved advanced setion into pyramid views....
r1751 #
# 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 logging
application: not use config.scan(), and replace all @add_view decorator into a explicit add_view call for faster app start.
r4610
repo-settings: moved advanced setion into pyramid views....
r1751 from pyramid.httpexceptions import HTTPFound
repository: show recomendation for updating hooks if they are outdated.
r4290 from packaging.version import Version
repo-settings: moved advanced setion into pyramid views....
r1751
repositories: added option to archive repositories instead of deleting them....
r3090 from rhodecode import events
repo-settings: moved advanced setion into pyramid views....
r1751 from rhodecode.apps._base import RepoAppView
from rhodecode.lib import helpers as h
from rhodecode.lib import audit_logger
from rhodecode.lib.auth import (
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 LoginRequired, HasRepoPermissionAnyDecorator, CSRFRequired,
HasRepoPermissionAny)
feat(artifacts): new artifact storage engines allowing an s3 based uploads
r5516 from rhodecode.lib.exceptions import AttachedForksError, AttachedPullRequestsError, AttachedArtifactsError
repo-settings: moved advanced setion into pyramid views....
r1751 from rhodecode.lib.utils2 import safe_int
from rhodecode.lib.vcs import RepositoryError
from rhodecode.model.db import Session, UserFollowing, User, Repository
permissions: properly flush user cache permissions in more cases of permission changes....
r3824 from rhodecode.model.permission import PermissionModel
repo-settings: moved advanced setion into pyramid views....
r1751 from rhodecode.model.repo import RepoModel
from rhodecode.model.scm import ScmModel
log = logging.getLogger(__name__)
application: not use config.scan(), and replace all @add_view decorator into a explicit add_view call for faster app start.
r4610 class RepoSettingsAdvancedView(RepoAppView):
repo-settings: moved advanced setion into pyramid views....
r1751
def load_default_context(self):
c = self._get_local_tmpl_context()
return c
repositories: added option to archive repositories instead of deleting them....
r3090 def _get_users_with_permissions(self):
user_permissions = {}
for perm in self.db_repo.permissions():
user_permissions[perm.user_id] = perm
return user_permissions
repo-settings: moved advanced setion into pyramid views....
r1751 @LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
def edit_advanced(self):
dan
cache-commits: add option to refresh caches manually from advanced pages.
r4159 _ = self.request.translate
repo-settings: moved advanced setion into pyramid views....
r1751 c = self.load_default_context()
c.active = 'advanced'
app: use simpler way to extract default_user_id, this will be now registered at server...
r4332 c.default_user_id = User.get_default_user_id()
repo-settings: moved advanced setion into pyramid views....
r1751 c.in_public_journal = UserFollowing.query() \
.filter(UserFollowing.user_id == c.default_user_id) \
repositories: rewrote whole admin section to pyramid....
r2014 .filter(UserFollowing.follows_repository == self.db_repo).scalar()
repo-settings: moved advanced setion into pyramid views....
r1751
repositories: handle vcs backend unavailable correctly in advanced settings.
r3382 c.ver_info_dict = self.rhodecode_vcs_repo.get_hooks_info()
repository: show recomendation for updating hooks if they are outdated.
r4290 c.hooks_outdated = False
try:
if Version(c.ver_info_dict['pre_version']) < Version(c.rhodecode_version):
c.hooks_outdated = True
except Exception:
pass
repositories: handle vcs backend unavailable correctly in advanced settings.
r3382
dan
cache-commits: add option to refresh caches manually from advanced pages.
r4159 # update commit cache if GET flag is present
if self.request.GET.get('update_commit_cache'):
self.db_repo.update_commit_cache()
h.flash(_('updated commit cache'), category='success')
repo-settings: moved advanced setion into pyramid views....
r1751 return self._get_template_context(c)
@LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
security: added missing csrf checks in few missing views.
r1811 @CSRFRequired()
repositories: added option to archive repositories instead of deleting them....
r3090 def edit_advanced_archive(self):
"""
Archives the repository. It will become read-only, and not visible in search
or other queries. But still visible for super-admins.
"""
_ = self.request.translate
try:
old_data = self.db_repo.get_api_data()
RepoModel().archive(self.db_repo)
repo = audit_logger.RepoWrap(repo_id=None, repo_name=self.db_repo.repo_name)
audit_logger.store_web(
'repo.archive', action_data={'old_data': old_data},
user=self._rhodecode_user, repo=repo)
ScmModel().mark_for_invalidation(self.db_repo_name, delete=True)
h.flash(
_('Archived repository `%s`') % self.db_repo_name,
category='success')
Session().commit()
except Exception:
log.exception("Exception during archiving of repository")
h.flash(_('An error occurred during archiving of `%s`')
% self.db_repo_name, category='error')
# redirect to advanced for more deletion options
raise HTTPFound(
h.route_path('edit_repo_advanced', repo_name=self.db_repo_name,
_anchor='advanced-archive'))
# flush permissions for all users defined in permissions
affected_user_ids = self._get_users_with_permissions().keys()
permissions: properly flush user cache permissions in more cases of permission changes....
r3824 PermissionModel().trigger_permission_flush(affected_user_ids)
repositories: added option to archive repositories instead of deleting them....
r3090
raise HTTPFound(h.route_path('home'))
@LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
@CSRFRequired()
repo-settings: moved advanced setion into pyramid views....
r1751 def edit_advanced_delete(self):
"""
Deletes the repository, or shows warnings if deletion is not possible
because of attached forks or other errors.
"""
_ = self.request.translate
handle_forks = self.request.POST.get('forks', None)
repository: add check preventing of removal of repo with attached pull requests....
r3089 if handle_forks == 'detach_forks':
handle_forks = 'detach'
elif handle_forks == 'delete_forks':
handle_forks = 'delete'
repo-settings: moved advanced setion into pyramid views....
r1751
feat(artifacts): new artifact storage engines allowing an s3 based uploads
r5516 repo_advanced_url = h.route_path(
'edit_repo_advanced', repo_name=self.db_repo_name,
_anchor='advanced-delete')
repo-settings: moved advanced setion into pyramid views....
r1751 try:
repository: add check preventing of removal of repo with attached pull requests....
r3089 old_data = self.db_repo.get_api_data()
RepoModel().delete(self.db_repo, forks=handle_forks)
repo-settings: moved advanced setion into pyramid views....
r1751 _forks = self.db_repo.forks.count()
if _forks and handle_forks:
if handle_forks == 'detach_forks':
h.flash(_('Detached %s forks') % _forks, category='success')
elif handle_forks == 'delete_forks':
h.flash(_('Deleted %s forks') % _forks, category='success')
repository: add check preventing of removal of repo with attached pull requests....
r3089 repo = audit_logger.RepoWrap(repo_id=None, repo_name=self.db_repo.repo_name)
audit-logs: use specific web/api calls....
r1806 audit_logger.store_web(
audit-logs: implemented full audit logs across application....
r1829 'repo.delete', action_data={'old_data': old_data},
audit-logs: use specific web/api calls....
r1806 user=self._rhodecode_user, repo=repo)
repo-settings: moved advanced setion into pyramid views....
r1751
audit-logger: unify calls to repo.delete and also store source of call, api/web.
r1752 ScmModel().mark_for_invalidation(self.db_repo_name, delete=True)
repo-settings: moved advanced setion into pyramid views....
r1751 h.flash(
_('Deleted repository `%s`') % self.db_repo_name,
category='success')
Session().commit()
except AttachedForksError:
delete_anchor = h.link_to(_('detach or delete'), repo_advanced_url)
h.flash(_('Cannot delete `{repo}` it still contains attached forks. '
'Try using {delete_or_detach} option.')
.format(repo=self.db_repo_name, delete_or_detach=delete_anchor),
category='warning')
# redirect to advanced for forks handle action ?
raise HTTPFound(repo_advanced_url)
repository: add check preventing of removal of repo with attached pull requests....
r3089 except AttachedPullRequestsError:
attached_prs = len(self.db_repo.pull_requests_source +
self.db_repo.pull_requests_target)
h.flash(
_('Cannot delete `{repo}` it still contains {num} attached pull requests. '
'Consider archiving the repository instead.').format(
repo=self.db_repo_name, num=attached_prs), category='warning')
# redirect to advanced for forks handle action ?
raise HTTPFound(repo_advanced_url)
feat(artifacts): new artifact storage engines allowing an s3 based uploads
r5516 except AttachedArtifactsError:
attached_artifacts = len(self.db_repo.artifacts)
h.flash(
_('Cannot delete `{repo}` it still contains {num} attached artifacts. '
'Consider archiving the repository instead.').format(
repo=self.db_repo_name, num=attached_artifacts), category='warning')
# redirect to advanced for forks handle action ?
raise HTTPFound(repo_advanced_url)
repo-settings: moved advanced setion into pyramid views....
r1751 except Exception:
log.exception("Exception during deletion of repository")
h.flash(_('An error occurred during deletion of `%s`')
% self.db_repo_name, category='error')
# redirect to advanced for more deletion options
raise HTTPFound(
repo-settings: fix error on passing in _anchor into wrong call....
r2171 h.route_path('edit_repo_advanced', repo_name=self.db_repo_name,
_anchor='advanced-delete'))
repo-settings: moved advanced setion into pyramid views....
r1751
raise HTTPFound(h.route_path('home'))
@LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
@CSRFRequired()
def edit_advanced_journal(self):
"""
Set's this repository to be visible in public journal,
in other words making default user to follow this repo
"""
_ = self.request.translate
try:
app: use simpler way to extract default_user_id, this will be now registered at server...
r4332 user_id = User.get_default_user_id()
repo-settings: moved advanced setion into pyramid views....
r1751 ScmModel().toggle_following_repo(self.db_repo.repo_id, user_id)
h.flash(_('Updated repository visibility in public journal'),
category='success')
Session().commit()
except Exception:
h.flash(_('An error occurred during setting this '
'repository in public journal'),
category='error')
raise HTTPFound(
h.route_path('edit_repo_advanced', repo_name=self.db_repo_name))
@LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
@CSRFRequired()
def edit_advanced_fork(self):
"""
Mark given repository as a fork of another
"""
_ = self.request.translate
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 new_fork_id = safe_int(self.request.POST.get('id_fork_of'))
# valid repo, re-check permissions
if new_fork_id:
repo = Repository.get(new_fork_id)
# ensure we have at least read access to the repo we mark
perm_check = HasRepoPermissionAny(
'repository.read', 'repository.write', 'repository.admin')
repo-settings: moved advanced setion into pyramid views....
r1751
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 if repo and perm_check(repo_name=repo.repo_name):
new_fork_id = repo.repo_id
else:
new_fork_id = None
repo-settings: moved advanced setion into pyramid views....
r1751
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 try:
repo-settings: moved advanced setion into pyramid views....
r1751 repo = ScmModel().mark_as_fork(
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 self.db_repo_name, new_fork_id, self._rhodecode_user.user_id)
repo-settings: moved advanced setion into pyramid views....
r1751 fork = repo.fork.repo_name if repo.fork else _('Nothing')
Session().commit()
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 h.flash(
_('Marked repo %s as fork of %s') % (self.db_repo_name, fork),
category='success')
repo-settings: moved advanced setion into pyramid views....
r1751 except RepositoryError as e:
log.exception("Repository Error occurred")
h.flash(str(e), category='error')
repo-forks: security, check for access to fork_id parameter to prevent...
r2173 except Exception:
repo-settings: moved advanced setion into pyramid views....
r1751 log.exception("Exception while editing fork")
h.flash(_('An error occurred during this operation'),
category='error')
raise HTTPFound(
h.route_path('edit_repo_advanced', repo_name=self.db_repo_name))
@LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
@CSRFRequired()
application: not use config.scan(), and replace all @add_view decorator into a explicit add_view call for faster app start.
r4610 def edit_advanced_toggle_locking(self):
repo-settings: moved advanced setion into pyramid views....
r1751 """
Toggle locking of repository
"""
_ = self.request.translate
set_lock = self.request.POST.get('set_lock')
set_unlock = self.request.POST.get('set_unlock')
try:
if set_lock:
Repository.lock(self.db_repo, self._rhodecode_user.user_id,
lock_reason=Repository.LOCK_WEB)
h.flash(_('Locked repository'), category='success')
elif set_unlock:
Repository.unlock(self.db_repo)
h.flash(_('Unlocked repository'), category='success')
except Exception as e:
log.exception("Exception during unlocking")
h.flash(_('An error occurred during unlocking'), category='error')
raise HTTPFound(
h.route_path('edit_repo_advanced', repo_name=self.db_repo_name))
repo-settings: add hidden view to force re-install hooks....
r2678
@LoginRequired()
@HasRepoPermissionAnyDecorator('repository.admin')
def edit_advanced_install_hooks(self):
"""
Install Hooks for repository
"""
_ = self.request.translate
self.load_default_context()
self.rhodecode_vcs_repo.install_hooks(force=True)
code: small cleanups of code
r2746 h.flash(_('installed updated hooks into this repository'),
category='success')
repo-settings: add hidden view to force re-install hooks....
r2678 raise HTTPFound(
h.route_path('edit_repo_advanced', repo_name=self.db_repo_name))