repos.py
431 lines
| 16.1 KiB
| text/x-python
|
PythonLexer
r824 | # -*- coding: utf-8 -*- | |||
""" | ||||
rhodecode.controllers.admin.repos | ||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
r1203 | ||||
r824 | Admin controller for RhodeCode | |||
r1203 | ||||
r824 | :created_on: Apr 7, 2010 | |||
:author: marcink | ||||
r1203 | :copyright: (C) 2009-2011 Marcin Kuzminski <marcin@python-works.com> | |||
r824 | :license: GPLv3, see COPYING for more details. | |||
""" | ||||
r1206 | # 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. | ||||
r1203 | # | |||
r547 | # 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. | ||||
r1203 | # | |||
r547 | # You should have received a copy of the GNU General Public License | |||
r1206 | # along with this program. If not, see <http://www.gnu.org/licenses/>. | |||
r824 | ||||
import logging | ||||
import traceback | ||||
import formencode | ||||
from operator import itemgetter | ||||
r547 | from formencode import htmlfill | |||
r824 | ||||
r547 | from paste.httpexceptions import HTTPInternalServerError | |||
from pylons import request, response, session, tmpl_context as c, url | ||||
from pylons.controllers.util import abort, redirect | ||||
from pylons.i18n.translation import _ | ||||
r824 | ||||
r547 | from rhodecode.lib import helpers as h | |||
from rhodecode.lib.auth import LoginRequired, HasPermissionAllDecorator, \ | ||||
HasPermissionAnyDecorator | ||||
from rhodecode.lib.base import BaseController, render | ||||
r1022 | from rhodecode.lib.utils import invalidate_cache, action_logger, repo_name_slug | |||
r1085 | from rhodecode.lib.helpers import get_token | |||
r1112 | from rhodecode.model.db import User, Repository, UserFollowing, Group | |||
r547 | from rhodecode.model.forms import RepoForm | |||
r691 | from rhodecode.model.scm import ScmModel | |||
r629 | from rhodecode.model.repo import RepoModel | |||
r1361 | from sqlalchemy.exc import IntegrityError | |||
r824 | ||||
r547 | log = logging.getLogger(__name__) | |||
r1245 | ||||
r547 | class ReposController(BaseController): | |||
r1085 | """ | |||
REST Controller styled on the Atom Publishing Protocol""" | ||||
r547 | # To properly map this controller, ensure your config/routing.py | |||
# file has a resource setup: | ||||
# map.resource('repo', 'repos') | ||||
r636 | ||||
r547 | @LoginRequired() | |||
@HasPermissionAnyDecorator('hg.admin', 'hg.create.repository') | ||||
def __before__(self): | ||||
c.admin_user = session.get('admin_user') | ||||
c.admin_username = session.get('admin_username') | ||||
super(ReposController, self).__before__() | ||||
r636 | ||||
r1159 | def __load_defaults(self): | |||
repo_model = RepoModel() | ||||
r1112 | ||||
r1159 | c.repo_groups = [('', '')] | |||
r1245 | parents_link = lambda k: h.literal('»'.join( | |||
map(lambda k: k.group_name, | ||||
r1159 | k.parents + [k]) | |||
) | ||||
) | ||||
c.repo_groups.extend([(x.group_id, parents_link(x)) for \ | ||||
x in self.sa.query(Group).all()]) | ||||
r1345 | c.repo_groups = sorted(c.repo_groups, | |||
key=lambda t: t[1].split('»')[0]) | ||||
r1159 | c.repo_groups_choices = map(lambda k: unicode(k[0]), c.repo_groups) | |||
c.users_array = repo_model.get_users_js() | ||||
c.users_groups_array = repo_model.get_users_groups_js() | ||||
def __load_data(self, repo_name=None): | ||||
r1112 | """ | |||
Load defaults settings for edit, and update | ||||
r1203 | ||||
r1112 | :param repo_name: | |||
""" | ||||
r1159 | self.__load_defaults() | |||
r1366 | c.repo_info = db_repo = Repository.by_repo_name(repo_name) | |||
repo = scm_repo = db_repo.scm_instance | ||||
r1112 | ||||
if c.repo_info is None: | ||||
h.flash(_('%s repository is not mapped to db perhaps' | ||||
' it was created or renamed from the filesystem' | ||||
' please run the application again' | ||||
' in order to rescan repositories') % repo_name, | ||||
category='error') | ||||
return redirect(url('repos')) | ||||
c.default_user_id = User.by_username('default').user_id | ||||
c.in_public_journal = self.sa.query(UserFollowing)\ | ||||
.filter(UserFollowing.user_id == c.default_user_id)\ | ||||
.filter(UserFollowing.follows_repository == c.repo_info).scalar() | ||||
if c.repo_info.stats: | ||||
last_rev = c.repo_info.stats.stat_on_revision | ||||
else: | ||||
last_rev = 0 | ||||
c.stats_revision = last_rev | ||||
c.repo_last_rev = repo.count() - 1 if repo.revisions else 0 | ||||
if last_rev == 0 or c.repo_last_rev == 0: | ||||
c.stats_percentage = 0 | ||||
else: | ||||
c.stats_percentage = '%.2f' % ((float((last_rev)) / | ||||
c.repo_last_rev) * 100) | ||||
defaults = c.repo_info.get_dict() | ||||
group, repo_name = c.repo_info.groups_and_repo | ||||
defaults['repo_name'] = repo_name | ||||
r1161 | defaults['repo_group'] = getattr(group[-1] if group else None, | |||
'group_id', None) | ||||
r1159 | ||||
r1112 | #fill owner | |||
if c.repo_info.user: | ||||
r1245 | defaults.update({'user': c.repo_info.user.username}) | |||
r1112 | else: | |||
replacement_user = self.sa.query(User)\ | ||||
.filter(User.admin == True).first().username | ||||
r1245 | defaults.update({'user': replacement_user}) | |||
r1112 | ||||
#fill repository users | ||||
for p in c.repo_info.repo_to_perm: | ||||
defaults.update({'u_perm_%s' % p.user.username: | ||||
p.permission.permission_name}) | ||||
#fill repository groups | ||||
for p in c.repo_info.users_group_to_perm: | ||||
defaults.update({'g_perm_%s' % p.users_group.users_group_name: | ||||
p.permission.permission_name}) | ||||
return defaults | ||||
r636 | @HasPermissionAllDecorator('hg.admin') | |||
r547 | def index(self, format='html'): | |||
"""GET /repos: All items in the collection""" | ||||
# url('repos') | ||||
r1344 | ||||
r1366 | c.repos_list = ScmModel().get_repos(Repository.query() | |||
.order_by(Repository.repo_name) | ||||
.all(), sort_key='name_sort') | ||||
r547 | return render('admin/repos/repos.html') | |||
r636 | ||||
r547 | @HasPermissionAnyDecorator('hg.admin', 'hg.create.repository') | |||
def create(self): | ||||
r1085 | """ | |||
POST /repos: Create a new item""" | ||||
r547 | # url('repos') | |||
repo_model = RepoModel() | ||||
r1159 | self.__load_defaults() | |||
r547 | form_result = {} | |||
try: | ||||
r1159 | form_result = RepoForm(repo_groups=c.repo_groups_choices)()\ | |||
.to_python(dict(request.POST)) | ||||
r1121 | repo_model.create(form_result, self.rhodecode_user) | |||
r1112 | if form_result['clone_uri']: | |||
h.flash(_('created repository %s from %s') \ | ||||
% (form_result['repo_name'], form_result['clone_uri']), | ||||
category='success') | ||||
else: | ||||
h.flash(_('created repository %s') % form_result['repo_name'], | ||||
r547 | category='success') | |||
if request.POST.get('user_created'): | ||||
r1361 | #created by regular non admin user | |||
r564 | action_logger(self.rhodecode_user, 'user_created_repo', | |||
r1361 | form_result['repo_name_full'], '', self.sa) | |||
r547 | else: | |||
r564 | action_logger(self.rhodecode_user, 'admin_created_repo', | |||
r1361 | form_result['repo_name_full'], '', self.sa) | |||
r636 | ||||
r564 | except formencode.Invalid, errors: | |||
r1112 | ||||
r547 | c.new_repo = errors.value['repo_name'] | |||
r636 | ||||
r547 | if request.POST.get('user_created'): | |||
r = render('admin/repos/repo_add_create_repository.html') | ||||
r636 | else: | |||
r547 | r = render('admin/repos/repo_add.html') | |||
r636 | ||||
r547 | return htmlfill.render( | |||
r, | ||||
defaults=errors.value, | ||||
errors=errors.error_dict or {}, | ||||
prefix_error=False, | ||||
r636 | encoding="UTF-8") | |||
r547 | ||||
except Exception: | ||||
log.error(traceback.format_exc()) | ||||
r860 | msg = _('error occurred during creation of repository %s') \ | |||
r547 | % form_result.get('repo_name') | |||
h.flash(msg, category='error') | ||||
if request.POST.get('user_created'): | ||||
r636 | return redirect(url('home')) | |||
r547 | return redirect(url('repos')) | |||
r636 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def new(self, format='html'): | ||||
"""GET /repos/new: Form to create a new item""" | ||||
new_repo = request.GET.get('repo', '') | ||||
r1022 | c.new_repo = repo_name_slug(new_repo) | |||
r1159 | self.__load_defaults() | |||
r547 | return render('admin/repos/repo_add.html') | |||
r636 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def update(self, repo_name): | ||||
r1085 | """ | |||
PUT /repos/repo_name: Update an existing item""" | ||||
r547 | # Forms posted to this method should contain a hidden field: | |||
# <input type="hidden" name="_method" value="PUT" /> | ||||
# Or using helpers: | ||||
# h.form(url('repo', repo_name=ID), | ||||
# method='put') | ||||
# url('repo', repo_name=ID) | ||||
r1159 | self.__load_defaults() | |||
r547 | repo_model = RepoModel() | |||
changed_name = repo_name | ||||
r1245 | _form = RepoForm(edit=True, old_data={'repo_name': repo_name}, | |||
r1159 | repo_groups=c.repo_groups_choices)() | |||
r547 | try: | |||
form_result = _form.to_python(dict(request.POST)) | ||||
repo_model.update(repo_name, form_result) | ||||
r665 | invalidate_cache('get_repo_cached_%s' % repo_name) | |||
r692 | h.flash(_('Repository %s updated successfully' % repo_name), | |||
r547 | category='success') | |||
r1323 | changed_name = form_result['repo_name_full'] | |||
r660 | action_logger(self.rhodecode_user, 'admin_updated_repo', | |||
changed_name, '', self.sa) | ||||
r564 | except formencode.Invalid, errors: | |||
r1112 | defaults = self.__load_data(repo_name) | |||
defaults.update(errors.value) | ||||
r547 | return htmlfill.render( | |||
render('admin/repos/repo_edit.html'), | ||||
r1112 | defaults=defaults, | |||
r547 | errors=errors.error_dict or {}, | |||
prefix_error=False, | ||||
encoding="UTF-8") | ||||
r636 | ||||
r547 | except Exception: | |||
log.error(traceback.format_exc()) | ||||
r692 | h.flash(_('error occurred during update of repository %s') \ | |||
r547 | % repo_name, category='error') | |||
return redirect(url('edit_repo', repo_name=changed_name)) | ||||
r636 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def delete(self, repo_name): | ||||
r1085 | """ | |||
DELETE /repos/repo_name: Delete an existing item""" | ||||
r547 | # Forms posted to this method should contain a hidden field: | |||
# <input type="hidden" name="_method" value="DELETE" /> | ||||
# Or using helpers: | ||||
# h.form(url('repo', repo_name=ID), | ||||
# method='delete') | ||||
# url('repo', repo_name=ID) | ||||
r636 | ||||
r547 | repo_model = RepoModel() | |||
r735 | repo = repo_model.get_by_repo_name(repo_name) | |||
r547 | if not repo: | |||
r636 | h.flash(_('%s repository is not mapped to db perhaps' | |||
r547 | ' it was moved or renamed from the filesystem' | |||
' please run the application again' | ||||
' in order to rescan repositories') % repo_name, | ||||
category='error') | ||||
r636 | ||||
r547 | return redirect(url('repos')) | |||
try: | ||||
r564 | action_logger(self.rhodecode_user, 'admin_deleted_repo', | |||
r547 | repo_name, '', self.sa) | |||
r636 | repo_model.delete(repo) | |||
r665 | invalidate_cache('get_repo_cached_%s' % repo_name) | |||
r547 | h.flash(_('deleted repository %s') % repo_name, category='success') | |||
r636 | ||||
r1361 | except IntegrityError, e: | |||
if e.message.find('repositories_fork_id_fkey'): | ||||
log.error(traceback.format_exc()) | ||||
h.flash(_('Cannot delete %s it still contains attached ' | ||||
'forks') % repo_name, | ||||
category='warning') | ||||
else: | ||||
log.error(traceback.format_exc()) | ||||
h.flash(_('An error occurred during ' | ||||
'deletion of %s') % repo_name, | ||||
category='error') | ||||
r547 | except Exception, e: | |||
log.error(traceback.format_exc()) | ||||
r860 | h.flash(_('An error occurred during deletion of %s') % repo_name, | |||
r547 | category='error') | |||
r636 | ||||
r547 | return redirect(url('repos')) | |||
r636 | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r547 | def delete_perm_user(self, repo_name): | |||
r1085 | """ | |||
DELETE an existing repository permission user | ||||
r1203 | ||||
r604 | :param repo_name: | |||
r547 | """ | |||
r636 | ||||
r547 | try: | |||
repo_model = RepoModel() | ||||
r636 | repo_model.delete_perm_user(request.POST, repo_name) | |||
r564 | except Exception, e: | |||
r860 | h.flash(_('An error occurred during deletion of repository user'), | |||
r547 | category='error') | |||
raise HTTPInternalServerError() | ||||
r636 | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r1015 | def delete_perm_users_group(self, repo_name): | |||
r1085 | """ | |||
DELETE an existing repository permission users group | ||||
r1203 | ||||
r1015 | :param repo_name: | |||
r708 | """ | |||
r1015 | try: | |||
repo_model = RepoModel() | ||||
repo_model.delete_perm_users_group(request.POST, repo_name) | ||||
except Exception, e: | ||||
h.flash(_('An error occurred during deletion of repository' | ||||
' users groups'), | ||||
category='error') | ||||
raise HTTPInternalServerError() | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
def repo_stats(self, repo_name): | ||||
r1085 | """ | |||
DELETE an existing repository statistics | ||||
r1203 | ||||
r708 | :param repo_name: | |||
""" | ||||
try: | ||||
repo_model = RepoModel() | ||||
repo_model.delete_stats(repo_name) | ||||
except Exception, e: | ||||
r860 | h.flash(_('An error occurred during deletion of repository stats'), | |||
r708 | category='error') | |||
return redirect(url('edit_repo', repo_name=repo_name)) | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
def repo_cache(self, repo_name): | ||||
r1085 | """ | |||
INVALIDATE existing repository cache | ||||
r1203 | ||||
r708 | :param repo_name: | |||
""" | ||||
try: | ||||
ScmModel().mark_for_invalidation(repo_name) | ||||
except Exception, e: | ||||
r711 | h.flash(_('An error occurred during cache invalidation'), | |||
r708 | category='error') | |||
return redirect(url('edit_repo', repo_name=repo_name)) | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r1085 | def repo_public_journal(self, repo_name): | |||
""" | ||||
Set's this repository to be visible in public journal, | ||||
in other words assing default user to follow this repo | ||||
r1203 | ||||
r1085 | :param repo_name: | |||
""" | ||||
cur_token = request.POST.get('auth_token') | ||||
token = get_token() | ||||
if cur_token == token: | ||||
try: | ||||
repo_id = Repository.by_repo_name(repo_name).repo_id | ||||
user_id = User.by_username('default').user_id | ||||
self.scm_model.toggle_following_repo(repo_id, user_id) | ||||
h.flash(_('Updated repository visibility in public journal'), | ||||
category='success') | ||||
except: | ||||
h.flash(_('An error occurred during setting this' | ||||
' repository in public journal'), | ||||
category='error') | ||||
else: | ||||
h.flash(_('Token mismatch'), category='error') | ||||
return redirect(url('edit_repo', repo_name=repo_name)) | ||||
r1114 | @HasPermissionAllDecorator('hg.admin') | |||
def repo_pull(self, repo_name): | ||||
""" | ||||
Runs task to update given repository with remote changes, | ||||
ie. make pull on remote location | ||||
r1203 | ||||
r1114 | :param repo_name: | |||
""" | ||||
try: | ||||
r1121 | ScmModel().pull_changes(repo_name, self.rhodecode_user.username) | |||
r1114 | h.flash(_('Pulled from remote location'), category='success') | |||
except Exception, e: | ||||
h.flash(_('An error occurred during pull from remote location'), | ||||
category='error') | ||||
r1085 | ||||
r1114 | return redirect(url('edit_repo', repo_name=repo_name)) | |||
r1085 | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r547 | def show(self, repo_name, format='html'): | |||
"""GET /repos/repo_name: Show a specific item""" | ||||
# url('repo', repo_name=ID) | ||||
r636 | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r547 | def edit(self, repo_name, format='html'): | |||
"""GET /repos/repo_name/edit: Form to edit an existing item""" | ||||
# url('edit_repo', repo_name=ID) | ||||
r1112 | defaults = self.__load_data(repo_name) | |||
r636 | ||||
r547 | return htmlfill.render( | |||
render('admin/repos/repo_edit.html'), | ||||
defaults=defaults, | ||||
encoding="UTF-8", | ||||
force_defaults=False | ||||
r636 | ) | |||