settings.py
421 lines
| 16.1 KiB
| text/x-python
|
PythonLexer
r779 | # -*- coding: utf-8 -*- | |||
""" | ||||
r860 | rhodecode.controllers.admin.settings | |||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
r1203 | ||||
r779 | settings controller for rhodecode admin | |||
r1203 | ||||
r779 | :created_on: Jul 14, 2010 | |||
:author: marcink | ||||
r1824 | :copyright: (C) 2010-2012 Marcin Kuzminski <marcin@python-works.com> | |||
r779 | :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/>. | |||
r779 | ||||
r890 | import logging | |||
import traceback | ||||
import formencode | ||||
r2192 | import pkg_resources | |||
import platform | ||||
r890 | ||||
r1022 | from sqlalchemy import func | |||
r547 | from formencode import htmlfill | |||
r1036 | from pylons import request, session, tmpl_context as c, url, config | |||
r547 | from pylons.controllers.util import abort, redirect | |||
from pylons.i18n.translation import _ | ||||
r890 | ||||
r547 | from rhodecode.lib import helpers as h | |||
from rhodecode.lib.auth import LoginRequired, HasPermissionAllDecorator, \ | ||||
r779 | HasPermissionAnyDecorator, NotAnonymous | |||
r547 | from rhodecode.lib.base import BaseController, render | |||
r705 | from rhodecode.lib.celerylib import tasks, run_task | |||
r547 | from rhodecode.lib.utils import repo2db_mapper, invalidate_cache, \ | |||
r1027 | set_rhodecode_config, repo_name_slug | |||
r1633 | from rhodecode.model.db import RhodeCodeUi, Repository, RepoGroup, \ | |||
RhodeCodeSetting | ||||
r547 | from rhodecode.model.forms import UserForm, ApplicationSettingsForm, \ | |||
ApplicationUiSettingsForm | ||||
r691 | from rhodecode.model.scm import ScmModel | |||
r629 | from rhodecode.model.user import UserModel | |||
r1501 | from rhodecode.model.db import User | |||
r1718 | from rhodecode.model.notification import EmailNotificationModel | |||
r1749 | from rhodecode.model.meta import Session | |||
r629 | ||||
r547 | log = logging.getLogger(__name__) | |||
class SettingsController(BaseController): | ||||
"""REST Controller styled on the Atom Publishing Protocol""" | ||||
# To properly map this controller, ensure your config/routing.py | ||||
# file has a resource setup: | ||||
r1203 | # map.resource('setting', 'settings', controller='admin/settings', | |||
r547 | # path_prefix='/admin', name_prefix='admin_') | |||
@LoginRequired() | ||||
def __before__(self): | ||||
c.admin_user = session.get('admin_user') | ||||
c.admin_username = session.get('admin_username') | ||||
r2192 | c.modules = sorted([(p.project_name, p.version) | |||
for p in pkg_resources.working_set]) | ||||
c.py_version = platform.python_version() | ||||
c.platform = platform.platform() | ||||
r547 | super(SettingsController, self).__before__() | |||
r629 | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r547 | def index(self, format='html'): | |||
"""GET /admin/settings: All items in the collection""" | ||||
# url('admin_settings') | ||||
r1633 | defaults = RhodeCodeSetting.get_app_settings() | |||
r756 | defaults.update(self.get_hg_ui_settings()) | |||
r2192 | ||||
r547 | return htmlfill.render( | |||
render('admin/settings/settings.html'), | ||||
defaults=defaults, | ||||
encoding="UTF-8", | ||||
force_defaults=False | ||||
r629 | ) | |||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def create(self): | ||||
"""POST /admin/settings: Create a new item""" | ||||
# url('admin_settings') | ||||
r629 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def new(self, format='html'): | ||||
"""GET /admin/settings/new: Form to create a new item""" | ||||
# url('admin_new_setting') | ||||
r629 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def update(self, setting_id): | ||||
"""PUT /admin/settings/setting_id: Update an existing item""" | ||||
# Forms posted to this method should contain a hidden field: | ||||
# <input type="hidden" name="_method" value="PUT" /> | ||||
# Or using helpers: | ||||
# h.form(url('admin_setting', setting_id=ID), | ||||
# method='put') | ||||
# url('admin_setting', setting_id=ID) | ||||
if setting_id == 'mapping': | ||||
rm_obsolete = request.POST.get('destroy', False) | ||||
r1976 | log.debug('Rescanning directories with destroy=%s' % rm_obsolete) | |||
r1036 | initial = ScmModel().repo_scan() | |||
r1039 | log.debug('invalidating all repositories') | |||
r665 | for repo_name in initial.keys(): | |||
invalidate_cache('get_repo_cached_%s' % repo_name) | ||||
r1039 | added, removed = repo2db_mapper(initial, rm_obsolete) | |||
r665 | ||||
r1039 | h.flash(_('Repositories successfully' | |||
r1245 | ' rescanned added: %s,removed: %s') % (added, removed), | |||
category='success') | ||||
r629 | ||||
r547 | if setting_id == 'whoosh': | |||
r756 | repo_location = self.get_hg_ui_settings()['paths_root_path'] | |||
r547 | full_index = request.POST.get('full_index', False) | |||
r1164 | run_task(tasks.whoosh_index, repo_location, full_index) | |||
r629 | ||||
r547 | h.flash(_('Whoosh reindex task scheduled'), category='success') | |||
if setting_id == 'global': | ||||
r629 | ||||
r547 | application_form = ApplicationSettingsForm()() | |||
try: | ||||
form_result = application_form.to_python(dict(request.POST)) | ||||
r890 | ||||
r547 | try: | |||
r1633 | hgsettings1 = RhodeCodeSetting.get_by_name('title') | |||
r1245 | hgsettings1.app_settings_value = \ | |||
form_result['rhodecode_title'] | ||||
r629 | ||||
r1633 | hgsettings2 = RhodeCodeSetting.get_by_name('realm') | |||
r1245 | hgsettings2.app_settings_value = \ | |||
form_result['rhodecode_realm'] | ||||
r629 | ||||
r1633 | hgsettings3 = RhodeCodeSetting.get_by_name('ga_code') | |||
r1245 | hgsettings3.app_settings_value = \ | |||
form_result['rhodecode_ga_code'] | ||||
r629 | ||||
r547 | self.sa.add(hgsettings1) | |||
self.sa.add(hgsettings2) | ||||
r890 | self.sa.add(hgsettings3) | |||
r547 | self.sa.commit() | |||
r548 | set_rhodecode_config(config) | |||
r547 | h.flash(_('Updated application settings'), | |||
category='success') | ||||
r629 | ||||
r1164 | except Exception: | |||
r547 | log.error(traceback.format_exc()) | |||
r1245 | h.flash(_('error occurred during updating ' | |||
'application settings'), | ||||
r547 | category='error') | |||
r629 | ||||
r547 | self.sa.rollback() | |||
r629 | ||||
r564 | except formencode.Invalid, errors: | |||
r547 | return htmlfill.render( | |||
render('admin/settings/settings.html'), | ||||
defaults=errors.value, | ||||
errors=errors.error_dict or {}, | ||||
prefix_error=False, | ||||
r629 | encoding="UTF-8") | |||
r547 | if setting_id == 'mercurial': | |||
application_form = ApplicationUiSettingsForm()() | ||||
try: | ||||
form_result = application_form.to_python(dict(request.POST)) | ||||
r629 | ||||
r547 | try: | |||
r629 | ||||
r549 | hgsettings1 = self.sa.query(RhodeCodeUi)\ | |||
.filter(RhodeCodeUi.ui_key == 'push_ssl').one() | ||||
r547 | hgsettings1.ui_value = form_result['web_push_ssl'] | |||
r629 | ||||
r549 | hgsettings2 = self.sa.query(RhodeCodeUi)\ | |||
.filter(RhodeCodeUi.ui_key == '/').one() | ||||
r629 | hgsettings2.ui_value = form_result['paths_root_path'] | |||
r547 | #HOOKS | |||
r549 | hgsettings3 = self.sa.query(RhodeCodeUi)\ | |||
.filter(RhodeCodeUi.ui_key == 'changegroup.update').one() | ||||
r1245 | hgsettings3.ui_active = \ | |||
bool(form_result['hooks_changegroup_update']) | ||||
r629 | ||||
r549 | hgsettings4 = self.sa.query(RhodeCodeUi)\ | |||
r1245 | .filter(RhodeCodeUi.ui_key == | |||
'changegroup.repo_size').one() | ||||
hgsettings4.ui_active = \ | ||||
bool(form_result['hooks_changegroup_repo_size']) | ||||
r629 | ||||
r661 | hgsettings5 = self.sa.query(RhodeCodeUi)\ | |||
r1245 | .filter(RhodeCodeUi.ui_key == | |||
'pretxnchangegroup.push_logger').one() | ||||
hgsettings5.ui_active = \ | ||||
bool(form_result['hooks_pretxnchangegroup' | ||||
'_push_logger']) | ||||
r629 | ||||
r661 | hgsettings6 = self.sa.query(RhodeCodeUi)\ | |||
r1245 | .filter(RhodeCodeUi.ui_key == | |||
'preoutgoing.pull_logger').one() | ||||
hgsettings6.ui_active = \ | ||||
bool(form_result['hooks_preoutgoing_pull_logger']) | ||||
r629 | ||||
r547 | self.sa.add(hgsettings1) | |||
self.sa.add(hgsettings2) | ||||
self.sa.add(hgsettings3) | ||||
self.sa.add(hgsettings4) | ||||
r661 | self.sa.add(hgsettings5) | |||
self.sa.add(hgsettings6) | ||||
r547 | self.sa.commit() | |||
r629 | ||||
r547 | h.flash(_('Updated mercurial settings'), | |||
category='success') | ||||
r629 | ||||
r547 | except: | |||
log.error(traceback.format_exc()) | ||||
r1245 | h.flash(_('error occurred during updating ' | |||
'application settings'), category='error') | ||||
r629 | ||||
r547 | self.sa.rollback() | |||
r629 | ||||
r564 | except formencode.Invalid, errors: | |||
r547 | return htmlfill.render( | |||
render('admin/settings/settings.html'), | ||||
defaults=errors.value, | ||||
errors=errors.error_dict or {}, | ||||
prefix_error=False, | ||||
r629 | encoding="UTF-8") | |||
r1460 | if setting_id == 'hooks': | |||
ui_key = request.POST.get('new_hook_ui_key') | ||||
ui_value = request.POST.get('new_hook_ui_value') | ||||
try: | ||||
r1673 | ||||
r1460 | if ui_value and ui_key: | |||
RhodeCodeUi.create_or_update_hook(ui_key, ui_value) | ||||
h.flash(_('Added new hook'), | ||||
category='success') | ||||
# check for edits | ||||
update = False | ||||
_d = request.POST.dict_of_lists() | ||||
r1798 | for k, v in zip(_d.get('hook_ui_key', []), | |||
_d.get('hook_ui_value_new', [])): | ||||
r1460 | RhodeCodeUi.create_or_update_hook(k, v) | |||
update = True | ||||
if update: | ||||
h.flash(_('Updated hooks'), category='success') | ||||
r2086 | self.sa.commit() | |||
r1460 | except: | |||
log.error(traceback.format_exc()) | ||||
h.flash(_('error occurred during hook creation'), | ||||
category='error') | ||||
return redirect(url('admin_edit_setting', setting_id='hooks')) | ||||
r1673 | if setting_id == 'email': | |||
test_email = request.POST.get('test_email') | ||||
test_email_subj = 'RhodeCode TestEmail' | ||||
test_email_body = 'RhodeCode Email test' | ||||
r1798 | ||||
r1717 | test_email_html_body = EmailNotificationModel()\ | |||
r1798 | .get_email_tmpl(EmailNotificationModel.TYPE_DEFAULT, | |||
body=test_email_body) | ||||
r1673 | ||||
r1798 | recipients = [test_email] if [test_email] else None | |||
run_task(tasks.send_email, recipients, test_email_subj, | ||||
r1717 | test_email_body, test_email_html_body) | |||
r1673 | h.flash(_('Email task created'), category='success') | |||
r547 | return redirect(url('admin_settings')) | |||
r629 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def delete(self, setting_id): | ||||
"""DELETE /admin/settings/setting_id: Delete an existing item""" | ||||
# Forms posted to this method should contain a hidden field: | ||||
# <input type="hidden" name="_method" value="DELETE" /> | ||||
# Or using helpers: | ||||
# h.form(url('admin_setting', setting_id=ID), | ||||
# method='delete') | ||||
# url('admin_setting', setting_id=ID) | ||||
r1460 | if setting_id == 'hooks': | |||
hook_id = request.POST.get('hook_id') | ||||
RhodeCodeUi.delete(hook_id) | ||||
r2086 | self.sa.commit() | |||
r1673 | ||||
r547 | @HasPermissionAllDecorator('hg.admin') | |||
def show(self, setting_id, format='html'): | ||||
r1245 | """ | |||
GET /admin/settings/setting_id: Show a specific item""" | ||||
r547 | # url('admin_setting', setting_id=ID) | |||
r629 | ||||
@HasPermissionAllDecorator('hg.admin') | ||||
r547 | def edit(self, setting_id, format='html'): | |||
r1245 | """ | |||
GET /admin/settings/setting_id/edit: Form to | ||||
edit an existing item""" | ||||
r547 | # url('admin_edit_setting', setting_id=ID) | |||
r1460 | c.hooks = RhodeCodeUi.get_builtin_hooks() | |||
c.custom_hooks = RhodeCodeUi.get_custom_hooks() | ||||
return htmlfill.render( | ||||
render('admin/settings/hooks.html'), | ||||
defaults={}, | ||||
encoding="UTF-8", | ||||
force_defaults=False | ||||
) | ||||
r779 | @NotAnonymous() | |||
r547 | def my_account(self): | |||
""" | ||||
r1203 | GET /_admin/my_account Displays info about my account | |||
r547 | """ | |||
r779 | # url('admin_settings_my_account') | |||
r665 | ||||
r1501 | c.user = User.get(self.rhodecode_user.user_id) | |||
r1366 | all_repos = self.sa.query(Repository)\ | |||
r1045 | .filter(Repository.user_id == c.user.user_id)\ | |||
r1366 | .order_by(func.lower(Repository.repo_name)).all() | |||
r691 | c.user_repos = ScmModel().get_repos(all_repos) | |||
r629 | ||||
r547 | if c.user.username == 'default': | |||
r629 | h.flash(_("You can't edit this user since it's" | |||
r547 | " crucial for entire application"), category='warning') | |||
return redirect(url('users')) | ||||
r629 | ||||
r832 | defaults = c.user.get_dict() | |||
r547 | return htmlfill.render( | |||
render('admin/users/user_edit_my_account.html'), | ||||
defaults=defaults, | ||||
encoding="UTF-8", | ||||
force_defaults=False | ||||
r629 | ) | |||
r547 | ||||
def my_account_update(self): | ||||
"""PUT /_admin/my_account_update: Update an existing item""" | ||||
# Forms posted to this method should contain a hidden field: | ||||
# <input type="hidden" name="_method" value="PUT" /> | ||||
# Or using helpers: | ||||
# h.form(url('admin_settings_my_account_update'), | ||||
# method='put') | ||||
# url('admin_settings_my_account_update', id=ID) | ||||
user_model = UserModel() | ||||
r1121 | uid = self.rhodecode_user.user_id | |||
r1245 | _form = UserForm(edit=True, | |||
old_data={'user_id': uid, | ||||
'email': self.rhodecode_user.email})() | ||||
r547 | form_result = {} | |||
try: | ||||
form_result = _form.to_python(dict(request.POST)) | ||||
user_model.update_my_account(uid, form_result) | ||||
r779 | h.flash(_('Your account was updated successfully'), | |||
r547 | category='success') | |||
r1749 | Session.commit() | |||
r564 | except formencode.Invalid, errors: | |||
r1501 | c.user = User.get(self.rhodecode_user.user_id) | |||
r687 | all_repos = self.sa.query(Repository)\ | |||
.filter(Repository.user_id == c.user.user_id)\ | ||||
.order_by(func.lower(Repository.repo_name))\ | ||||
.all() | ||||
r691 | c.user_repos = ScmModel().get_repos(all_repos) | |||
r687 | ||||
r547 | return htmlfill.render( | |||
render('admin/users/user_edit_my_account.html'), | ||||
defaults=errors.value, | ||||
errors=errors.error_dict or {}, | ||||
prefix_error=False, | ||||
encoding="UTF-8") | ||||
except Exception: | ||||
log.error(traceback.format_exc()) | ||||
r779 | h.flash(_('error occurred during update of user %s') \ | |||
r547 | % form_result.get('username'), category='error') | |||
r629 | ||||
r547 | return redirect(url('my_account')) | |||
r629 | ||||
r779 | @NotAnonymous() | |||
r547 | @HasPermissionAnyDecorator('hg.admin', 'hg.create.repository') | |||
def create_repository(self): | ||||
"""GET /_admin/create_repository: Form to create a new item""" | ||||
r1161 | ||||
r1633 | c.repo_groups = RepoGroup.groups_choices() | |||
r1161 | c.repo_groups_choices = map(lambda k: unicode(k[0]), c.repo_groups) | |||
r547 | new_repo = request.GET.get('repo', '') | |||
r1022 | c.new_repo = repo_name_slug(new_repo) | |||
r1161 | ||||
r547 | return render('admin/repos/repo_add_create_repository.html') | |||
r629 | ||||
r756 | def get_hg_ui_settings(self): | |||
ret = self.sa.query(RhodeCodeUi).all() | ||||
if not ret: | ||||
raise Exception('Could not get application ui settings !') | ||||
settings = {} | ||||
for each in ret: | ||||
k = each.ui_key | ||||
v = each.ui_value | ||||
if k == '/': | ||||
k = 'root_path' | ||||
if k.find('.') != -1: | ||||
k = k.replace('.', '_') | ||||
if each.ui_section == 'hooks': | ||||
v = each.ui_active | ||||
settings[each.ui_section + '_' + k] = v | ||||
return settings | ||||