Show More
@@ -1,544 +1,548 b'' | |||||
1 | # -*- coding: utf-8 -*- |
|
1 | # -*- coding: utf-8 -*- | |
2 | """ |
|
2 | """ | |
3 | rhodecode.controllers.admin.settings |
|
3 | rhodecode.controllers.admin.settings | |
4 | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |
|
4 | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | |
5 |
|
5 | |||
6 | settings controller for rhodecode admin |
|
6 | settings controller for rhodecode admin | |
7 |
|
7 | |||
8 | :created_on: Jul 14, 2010 |
|
8 | :created_on: Jul 14, 2010 | |
9 | :author: marcink |
|
9 | :author: marcink | |
10 | :copyright: (C) 2010-2012 Marcin Kuzminski <marcin@python-works.com> |
|
10 | :copyright: (C) 2010-2012 Marcin Kuzminski <marcin@python-works.com> | |
11 | :license: GPLv3, see COPYING for more details. |
|
11 | :license: GPLv3, see COPYING for more details. | |
12 | """ |
|
12 | """ | |
13 | # This program is free software: you can redistribute it and/or modify |
|
13 | # This program is free software: you can redistribute it and/or modify | |
14 | # it under the terms of the GNU General Public License as published by |
|
14 | # it under the terms of the GNU General Public License as published by | |
15 | # the Free Software Foundation, either version 3 of the License, or |
|
15 | # the Free Software Foundation, either version 3 of the License, or | |
16 | # (at your option) any later version. |
|
16 | # (at your option) any later version. | |
17 | # |
|
17 | # | |
18 | # This program is distributed in the hope that it will be useful, |
|
18 | # This program is distributed in the hope that it will be useful, | |
19 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
19 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | |
20 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
20 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
21 | # GNU General Public License for more details. |
|
21 | # GNU General Public License for more details. | |
22 | # |
|
22 | # | |
23 | # You should have received a copy of the GNU General Public License |
|
23 | # You should have received a copy of the GNU General Public License | |
24 | # along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
24 | # along with this program. If not, see <http://www.gnu.org/licenses/>. | |
25 |
|
25 | |||
26 | import logging |
|
26 | import logging | |
27 | import traceback |
|
27 | import traceback | |
28 | import formencode |
|
28 | import formencode | |
29 | import pkg_resources |
|
29 | import pkg_resources | |
30 | import platform |
|
30 | import platform | |
31 |
|
31 | |||
32 | from sqlalchemy import func |
|
32 | from sqlalchemy import func | |
33 | from formencode import htmlfill |
|
33 | from formencode import htmlfill | |
34 | from pylons import request, session, tmpl_context as c, url, config |
|
34 | from pylons import request, session, tmpl_context as c, url, config | |
35 | from pylons.controllers.util import abort, redirect |
|
35 | from pylons.controllers.util import abort, redirect | |
36 | from pylons.i18n.translation import _ |
|
36 | from pylons.i18n.translation import _ | |
37 |
|
37 | |||
38 | from rhodecode.lib import helpers as h |
|
38 | from rhodecode.lib import helpers as h | |
39 | from rhodecode.lib.auth import LoginRequired, HasPermissionAllDecorator, \ |
|
39 | from rhodecode.lib.auth import LoginRequired, HasPermissionAllDecorator, \ | |
40 | HasPermissionAnyDecorator, NotAnonymous, HasPermissionAny,\ |
|
40 | HasPermissionAnyDecorator, NotAnonymous, HasPermissionAny,\ | |
41 | HasReposGroupPermissionAll, HasReposGroupPermissionAny |
|
41 | HasReposGroupPermissionAll, HasReposGroupPermissionAny | |
42 | from rhodecode.lib.base import BaseController, render |
|
42 | from rhodecode.lib.base import BaseController, render | |
43 | from rhodecode.lib.celerylib import tasks, run_task |
|
43 | from rhodecode.lib.celerylib import tasks, run_task | |
44 | from rhodecode.lib.utils import repo2db_mapper, invalidate_cache, \ |
|
44 | from rhodecode.lib.utils import repo2db_mapper, invalidate_cache, \ | |
45 | set_rhodecode_config, repo_name_slug, check_git_version |
|
45 | set_rhodecode_config, repo_name_slug, check_git_version | |
46 | from rhodecode.model.db import RhodeCodeUi, Repository, RepoGroup, \ |
|
46 | from rhodecode.model.db import RhodeCodeUi, Repository, RepoGroup, \ | |
47 | RhodeCodeSetting, PullRequest, PullRequestReviewers |
|
47 | RhodeCodeSetting, PullRequest, PullRequestReviewers | |
48 | from rhodecode.model.forms import UserForm, ApplicationSettingsForm, \ |
|
48 | from rhodecode.model.forms import UserForm, ApplicationSettingsForm, \ | |
49 | ApplicationUiSettingsForm, ApplicationVisualisationForm |
|
49 | ApplicationUiSettingsForm, ApplicationVisualisationForm | |
50 | from rhodecode.model.scm import ScmModel, GroupList |
|
50 | from rhodecode.model.scm import ScmModel, GroupList | |
51 | from rhodecode.model.user import UserModel |
|
51 | from rhodecode.model.user import UserModel | |
52 | from rhodecode.model.repo import RepoModel |
|
52 | from rhodecode.model.repo import RepoModel | |
53 | from rhodecode.model.db import User |
|
53 | from rhodecode.model.db import User | |
54 | from rhodecode.model.notification import EmailNotificationModel |
|
54 | from rhodecode.model.notification import EmailNotificationModel | |
55 | from rhodecode.model.meta import Session |
|
55 | from rhodecode.model.meta import Session | |
56 | from rhodecode.lib.utils2 import str2bool, safe_unicode |
|
56 | from rhodecode.lib.utils2 import str2bool, safe_unicode | |
57 | from rhodecode.lib.compat import json |
|
57 | from rhodecode.lib.compat import json | |
58 | from webob.exc import HTTPForbidden |
|
58 | from webob.exc import HTTPForbidden | |
59 | log = logging.getLogger(__name__) |
|
59 | log = logging.getLogger(__name__) | |
60 |
|
60 | |||
61 |
|
61 | |||
62 | class SettingsController(BaseController): |
|
62 | class SettingsController(BaseController): | |
63 | """REST Controller styled on the Atom Publishing Protocol""" |
|
63 | """REST Controller styled on the Atom Publishing Protocol""" | |
64 | # To properly map this controller, ensure your config/routing.py |
|
64 | # To properly map this controller, ensure your config/routing.py | |
65 | # file has a resource setup: |
|
65 | # file has a resource setup: | |
66 | # map.resource('setting', 'settings', controller='admin/settings', |
|
66 | # map.resource('setting', 'settings', controller='admin/settings', | |
67 | # path_prefix='/admin', name_prefix='admin_') |
|
67 | # path_prefix='/admin', name_prefix='admin_') | |
68 |
|
68 | |||
69 | @LoginRequired() |
|
69 | @LoginRequired() | |
70 | def __before__(self): |
|
70 | def __before__(self): | |
71 | c.admin_user = session.get('admin_user') |
|
71 | c.admin_user = session.get('admin_user') | |
72 | c.admin_username = session.get('admin_username') |
|
72 | c.admin_username = session.get('admin_username') | |
73 | c.modules = sorted([(p.project_name, p.version) |
|
73 | c.modules = sorted([(p.project_name, p.version) | |
74 | for p in pkg_resources.working_set] |
|
74 | for p in pkg_resources.working_set] | |
75 | + [('git', check_git_version())], |
|
75 | + [('git', check_git_version())], | |
76 | key=lambda k: k[0].lower()) |
|
76 | key=lambda k: k[0].lower()) | |
77 | c.py_version = platform.python_version() |
|
77 | c.py_version = platform.python_version() | |
78 | c.platform = platform.platform() |
|
78 | c.platform = platform.platform() | |
79 | super(SettingsController, self).__before__() |
|
79 | super(SettingsController, self).__before__() | |
80 |
|
80 | |||
81 | @HasPermissionAllDecorator('hg.admin') |
|
81 | @HasPermissionAllDecorator('hg.admin') | |
82 | def index(self, format='html'): |
|
82 | def index(self, format='html'): | |
83 | """GET /admin/settings: All items in the collection""" |
|
83 | """GET /admin/settings: All items in the collection""" | |
84 | # url('admin_settings') |
|
84 | # url('admin_settings') | |
85 |
|
85 | |||
86 | defaults = RhodeCodeSetting.get_app_settings() |
|
86 | defaults = RhodeCodeSetting.get_app_settings() | |
87 | defaults.update(self._get_hg_ui_settings()) |
|
87 | defaults.update(self._get_hg_ui_settings()) | |
88 |
|
88 | |||
89 | return htmlfill.render( |
|
89 | return htmlfill.render( | |
90 | render('admin/settings/settings.html'), |
|
90 | render('admin/settings/settings.html'), | |
91 | defaults=defaults, |
|
91 | defaults=defaults, | |
92 | encoding="UTF-8", |
|
92 | encoding="UTF-8", | |
93 | force_defaults=False |
|
93 | force_defaults=False | |
94 | ) |
|
94 | ) | |
95 |
|
95 | |||
96 | @HasPermissionAllDecorator('hg.admin') |
|
96 | @HasPermissionAllDecorator('hg.admin') | |
97 | def create(self): |
|
97 | def create(self): | |
98 | """POST /admin/settings: Create a new item""" |
|
98 | """POST /admin/settings: Create a new item""" | |
99 | # url('admin_settings') |
|
99 | # url('admin_settings') | |
100 |
|
100 | |||
101 | @HasPermissionAllDecorator('hg.admin') |
|
101 | @HasPermissionAllDecorator('hg.admin') | |
102 | def new(self, format='html'): |
|
102 | def new(self, format='html'): | |
103 | """GET /admin/settings/new: Form to create a new item""" |
|
103 | """GET /admin/settings/new: Form to create a new item""" | |
104 | # url('admin_new_setting') |
|
104 | # url('admin_new_setting') | |
105 |
|
105 | |||
106 | @HasPermissionAllDecorator('hg.admin') |
|
106 | @HasPermissionAllDecorator('hg.admin') | |
107 | def update(self, setting_id): |
|
107 | def update(self, setting_id): | |
108 | """PUT /admin/settings/setting_id: Update an existing item""" |
|
108 | """PUT /admin/settings/setting_id: Update an existing item""" | |
109 | # Forms posted to this method should contain a hidden field: |
|
109 | # Forms posted to this method should contain a hidden field: | |
110 | # <input type="hidden" name="_method" value="PUT" /> |
|
110 | # <input type="hidden" name="_method" value="PUT" /> | |
111 | # Or using helpers: |
|
111 | # Or using helpers: | |
112 | # h.form(url('admin_setting', setting_id=ID), |
|
112 | # h.form(url('admin_setting', setting_id=ID), | |
113 | # method='put') |
|
113 | # method='put') | |
114 | # url('admin_setting', setting_id=ID) |
|
114 | # url('admin_setting', setting_id=ID) | |
115 |
|
115 | |||
116 | if setting_id == 'mapping': |
|
116 | if setting_id == 'mapping': | |
117 | rm_obsolete = request.POST.get('destroy', False) |
|
117 | rm_obsolete = request.POST.get('destroy', False) | |
118 | log.debug('Rescanning directories with destroy=%s' % rm_obsolete) |
|
118 | log.debug('Rescanning directories with destroy=%s' % rm_obsolete) | |
119 | initial = ScmModel().repo_scan() |
|
119 | initial = ScmModel().repo_scan() | |
120 | log.debug('invalidating all repositories') |
|
120 | log.debug('invalidating all repositories') | |
121 | for repo_name in initial.keys(): |
|
121 | for repo_name in initial.keys(): | |
122 | invalidate_cache('get_repo_cached_%s' % repo_name) |
|
122 | invalidate_cache('get_repo_cached_%s' % repo_name) | |
123 |
|
123 | |||
124 | added, removed = repo2db_mapper(initial, rm_obsolete) |
|
124 | added, removed = repo2db_mapper(initial, rm_obsolete) | |
125 | _repr = lambda l: ', '.join(map(safe_unicode, l)) or '-' |
|
125 | _repr = lambda l: ', '.join(map(safe_unicode, l)) or '-' | |
126 | h.flash(_('Repositories successfully ' |
|
126 | h.flash(_('Repositories successfully ' | |
127 | 'rescanned added: %s ; removed: %s') % |
|
127 | 'rescanned added: %s ; removed: %s') % | |
128 | (_repr(added), _repr(removed)), |
|
128 | (_repr(added), _repr(removed)), | |
129 | category='success') |
|
129 | category='success') | |
130 |
|
130 | |||
131 | if setting_id == 'whoosh': |
|
131 | if setting_id == 'whoosh': | |
132 | repo_location = self._get_hg_ui_settings()['paths_root_path'] |
|
132 | repo_location = self._get_hg_ui_settings()['paths_root_path'] | |
133 | full_index = request.POST.get('full_index', False) |
|
133 | full_index = request.POST.get('full_index', False) | |
134 | run_task(tasks.whoosh_index, repo_location, full_index) |
|
134 | run_task(tasks.whoosh_index, repo_location, full_index) | |
135 | h.flash(_('Whoosh reindex task scheduled'), category='success') |
|
135 | h.flash(_('Whoosh reindex task scheduled'), category='success') | |
136 |
|
136 | |||
137 | if setting_id == 'global': |
|
137 | if setting_id == 'global': | |
138 |
|
138 | |||
139 | application_form = ApplicationSettingsForm()() |
|
139 | application_form = ApplicationSettingsForm()() | |
140 | try: |
|
140 | try: | |
141 | form_result = application_form.to_python(dict(request.POST)) |
|
141 | form_result = application_form.to_python(dict(request.POST)) | |
142 | except formencode.Invalid, errors: |
|
142 | except formencode.Invalid, errors: | |
143 | return htmlfill.render( |
|
143 | return htmlfill.render( | |
144 | render('admin/settings/settings.html'), |
|
144 | render('admin/settings/settings.html'), | |
145 | defaults=errors.value, |
|
145 | defaults=errors.value, | |
146 | errors=errors.error_dict or {}, |
|
146 | errors=errors.error_dict or {}, | |
147 | prefix_error=False, |
|
147 | prefix_error=False, | |
148 | encoding="UTF-8" |
|
148 | encoding="UTF-8" | |
149 | ) |
|
149 | ) | |
150 |
|
150 | |||
151 | try: |
|
151 | try: | |
152 | sett1 = RhodeCodeSetting.get_by_name_or_create('title') |
|
152 | sett1 = RhodeCodeSetting.get_by_name_or_create('title') | |
153 | sett1.app_settings_value = form_result['rhodecode_title'] |
|
153 | sett1.app_settings_value = form_result['rhodecode_title'] | |
154 | Session().add(sett1) |
|
154 | Session().add(sett1) | |
155 |
|
155 | |||
156 | sett2 = RhodeCodeSetting.get_by_name_or_create('realm') |
|
156 | sett2 = RhodeCodeSetting.get_by_name_or_create('realm') | |
157 | sett2.app_settings_value = form_result['rhodecode_realm'] |
|
157 | sett2.app_settings_value = form_result['rhodecode_realm'] | |
158 | Session().add(sett2) |
|
158 | Session().add(sett2) | |
159 |
|
159 | |||
160 | sett3 = RhodeCodeSetting.get_by_name_or_create('ga_code') |
|
160 | sett3 = RhodeCodeSetting.get_by_name_or_create('ga_code') | |
161 | sett3.app_settings_value = form_result['rhodecode_ga_code'] |
|
161 | sett3.app_settings_value = form_result['rhodecode_ga_code'] | |
162 | Session().add(sett3) |
|
162 | Session().add(sett3) | |
163 |
|
163 | |||
164 | Session().commit() |
|
164 | Session().commit() | |
165 | set_rhodecode_config(config) |
|
165 | set_rhodecode_config(config) | |
166 | h.flash(_('Updated application settings'), category='success') |
|
166 | h.flash(_('Updated application settings'), category='success') | |
167 |
|
167 | |||
168 | except Exception: |
|
168 | except Exception: | |
169 | log.error(traceback.format_exc()) |
|
169 | log.error(traceback.format_exc()) | |
170 | h.flash(_('error occurred during updating ' |
|
170 | h.flash(_('error occurred during updating ' | |
171 | 'application settings'), |
|
171 | 'application settings'), | |
172 | category='error') |
|
172 | category='error') | |
173 |
|
173 | |||
174 | if setting_id == 'visual': |
|
174 | if setting_id == 'visual': | |
175 |
|
175 | |||
176 | application_form = ApplicationVisualisationForm()() |
|
176 | application_form = ApplicationVisualisationForm()() | |
177 | try: |
|
177 | try: | |
178 | form_result = application_form.to_python(dict(request.POST)) |
|
178 | form_result = application_form.to_python(dict(request.POST)) | |
179 | except formencode.Invalid, errors: |
|
179 | except formencode.Invalid, errors: | |
180 | return htmlfill.render( |
|
180 | return htmlfill.render( | |
181 | render('admin/settings/settings.html'), |
|
181 | render('admin/settings/settings.html'), | |
182 | defaults=errors.value, |
|
182 | defaults=errors.value, | |
183 | errors=errors.error_dict or {}, |
|
183 | errors=errors.error_dict or {}, | |
184 | prefix_error=False, |
|
184 | prefix_error=False, | |
185 | encoding="UTF-8" |
|
185 | encoding="UTF-8" | |
186 | ) |
|
186 | ) | |
187 |
|
187 | |||
188 | try: |
|
188 | try: | |
189 | sett1 = RhodeCodeSetting.get_by_name_or_create('show_public_icon') |
|
189 | sett1 = RhodeCodeSetting.get_by_name_or_create('show_public_icon') | |
190 | sett1.app_settings_value = \ |
|
190 | sett1.app_settings_value = \ | |
191 | form_result['rhodecode_show_public_icon'] |
|
191 | form_result['rhodecode_show_public_icon'] | |
192 | Session().add(sett1) |
|
192 | Session().add(sett1) | |
193 |
|
193 | |||
194 | sett2 = RhodeCodeSetting.get_by_name_or_create('show_private_icon') |
|
194 | sett2 = RhodeCodeSetting.get_by_name_or_create('show_private_icon') | |
195 | sett2.app_settings_value = \ |
|
195 | sett2.app_settings_value = \ | |
196 | form_result['rhodecode_show_private_icon'] |
|
196 | form_result['rhodecode_show_private_icon'] | |
197 | Session().add(sett2) |
|
197 | Session().add(sett2) | |
198 |
|
198 | |||
199 | sett3 = RhodeCodeSetting.get_by_name_or_create('stylify_metatags') |
|
199 | sett3 = RhodeCodeSetting.get_by_name_or_create('stylify_metatags') | |
200 | sett3.app_settings_value = \ |
|
200 | sett3.app_settings_value = \ | |
201 | form_result['rhodecode_stylify_metatags'] |
|
201 | form_result['rhodecode_stylify_metatags'] | |
202 | Session().add(sett3) |
|
202 | Session().add(sett3) | |
203 |
|
203 | |||
204 | sett4 = RhodeCodeSetting.get_by_name_or_create('lightweight_dashboard') |
|
204 | sett4 = RhodeCodeSetting.get_by_name_or_create('lightweight_dashboard') | |
205 | sett4.app_settings_value = \ |
|
205 | sett4.app_settings_value = \ | |
206 | form_result['rhodecode_lightweight_dashboard'] |
|
206 | form_result['rhodecode_lightweight_dashboard'] | |
207 | Session().add(sett4) |
|
207 | Session().add(sett4) | |
208 |
|
208 | |||
209 | sett4 = RhodeCodeSetting.get_by_name_or_create('repository_fields') |
|
209 | sett4 = RhodeCodeSetting.get_by_name_or_create('repository_fields') | |
210 | sett4.app_settings_value = \ |
|
210 | sett4.app_settings_value = \ | |
211 | form_result['rhodecode_repository_fields'] |
|
211 | form_result['rhodecode_repository_fields'] | |
212 | Session().add(sett4) |
|
212 | Session().add(sett4) | |
213 |
|
213 | |||
214 | Session().commit() |
|
214 | Session().commit() | |
215 | set_rhodecode_config(config) |
|
215 | set_rhodecode_config(config) | |
216 | h.flash(_('Updated visualisation settings'), |
|
216 | h.flash(_('Updated visualisation settings'), | |
217 | category='success') |
|
217 | category='success') | |
218 |
|
218 | |||
219 | except Exception: |
|
219 | except Exception: | |
220 | log.error(traceback.format_exc()) |
|
220 | log.error(traceback.format_exc()) | |
221 | h.flash(_('error occurred during updating ' |
|
221 | h.flash(_('error occurred during updating ' | |
222 | 'visualisation settings'), |
|
222 | 'visualisation settings'), | |
223 | category='error') |
|
223 | category='error') | |
224 |
|
224 | |||
225 | if setting_id == 'vcs': |
|
225 | if setting_id == 'vcs': | |
226 | application_form = ApplicationUiSettingsForm()() |
|
226 | application_form = ApplicationUiSettingsForm()() | |
227 | try: |
|
227 | try: | |
228 | form_result = application_form.to_python(dict(request.POST)) |
|
228 | form_result = application_form.to_python(dict(request.POST)) | |
229 | except formencode.Invalid, errors: |
|
229 | except formencode.Invalid, errors: | |
230 | return htmlfill.render( |
|
230 | return htmlfill.render( | |
231 | render('admin/settings/settings.html'), |
|
231 | render('admin/settings/settings.html'), | |
232 | defaults=errors.value, |
|
232 | defaults=errors.value, | |
233 | errors=errors.error_dict or {}, |
|
233 | errors=errors.error_dict or {}, | |
234 | prefix_error=False, |
|
234 | prefix_error=False, | |
235 | encoding="UTF-8" |
|
235 | encoding="UTF-8" | |
236 | ) |
|
236 | ) | |
237 |
|
237 | |||
238 | try: |
|
238 | try: | |
239 | # fix namespaces for hooks and extensions |
|
239 | # fix namespaces for hooks and extensions | |
240 | _f = lambda s: s.replace('.', '_') |
|
240 | _f = lambda s: s.replace('.', '_') | |
241 |
|
241 | |||
242 | sett = RhodeCodeUi.get_by_key('push_ssl') |
|
242 | sett = RhodeCodeUi.get_by_key('push_ssl') | |
243 | sett.ui_value = form_result['web_push_ssl'] |
|
243 | sett.ui_value = form_result['web_push_ssl'] | |
244 | Session().add(sett) |
|
244 | Session().add(sett) | |
245 |
|
245 | |||
246 | sett = RhodeCodeUi.get_by_key('/') |
|
246 | sett = RhodeCodeUi.get_by_key('/') | |
247 | sett.ui_value = form_result['paths_root_path'] |
|
247 | sett.ui_value = form_result['paths_root_path'] | |
248 | Session().add(sett) |
|
248 | Session().add(sett) | |
249 |
|
249 | |||
250 | #HOOKS |
|
250 | #HOOKS | |
251 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_UPDATE) |
|
251 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_UPDATE) | |
252 | sett.ui_active = form_result[_f('hooks_%s' % |
|
252 | sett.ui_active = form_result[_f('hooks_%s' % | |
253 | RhodeCodeUi.HOOK_UPDATE)] |
|
253 | RhodeCodeUi.HOOK_UPDATE)] | |
254 | Session().add(sett) |
|
254 | Session().add(sett) | |
255 |
|
255 | |||
256 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_REPO_SIZE) |
|
256 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_REPO_SIZE) | |
257 | sett.ui_active = form_result[_f('hooks_%s' % |
|
257 | sett.ui_active = form_result[_f('hooks_%s' % | |
258 | RhodeCodeUi.HOOK_REPO_SIZE)] |
|
258 | RhodeCodeUi.HOOK_REPO_SIZE)] | |
259 | Session().add(sett) |
|
259 | Session().add(sett) | |
260 |
|
260 | |||
261 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_PUSH) |
|
261 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_PUSH) | |
262 | sett.ui_active = form_result[_f('hooks_%s' % |
|
262 | sett.ui_active = form_result[_f('hooks_%s' % | |
263 | RhodeCodeUi.HOOK_PUSH)] |
|
263 | RhodeCodeUi.HOOK_PUSH)] | |
264 | Session().add(sett) |
|
264 | Session().add(sett) | |
265 |
|
265 | |||
266 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_PULL) |
|
266 | sett = RhodeCodeUi.get_by_key(RhodeCodeUi.HOOK_PULL) | |
267 | sett.ui_active = form_result[_f('hooks_%s' % |
|
267 | sett.ui_active = form_result[_f('hooks_%s' % | |
268 | RhodeCodeUi.HOOK_PULL)] |
|
268 | RhodeCodeUi.HOOK_PULL)] | |
269 |
|
269 | |||
270 | Session().add(sett) |
|
270 | Session().add(sett) | |
271 |
|
271 | |||
272 | ## EXTENSIONS |
|
272 | ## EXTENSIONS | |
273 | sett = RhodeCodeUi.get_by_key('largefiles') |
|
273 | sett = RhodeCodeUi.get_by_key('largefiles') | |
274 | if not sett: |
|
274 | if not sett: | |
275 | #make one if it's not there ! |
|
275 | #make one if it's not there ! | |
276 | sett = RhodeCodeUi() |
|
276 | sett = RhodeCodeUi() | |
277 | sett.ui_key = 'largefiles' |
|
277 | sett.ui_key = 'largefiles' | |
278 | sett.ui_section = 'extensions' |
|
278 | sett.ui_section = 'extensions' | |
279 | sett.ui_active = form_result[_f('extensions_largefiles')] |
|
279 | sett.ui_active = form_result[_f('extensions_largefiles')] | |
280 | Session().add(sett) |
|
280 | Session().add(sett) | |
281 |
|
281 | |||
282 | sett = RhodeCodeUi.get_by_key('hgsubversion') |
|
282 | sett = RhodeCodeUi.get_by_key('hgsubversion') | |
283 | if not sett: |
|
283 | if not sett: | |
284 | #make one if it's not there ! |
|
284 | #make one if it's not there ! | |
285 | sett = RhodeCodeUi() |
|
285 | sett = RhodeCodeUi() | |
286 | sett.ui_key = 'hgsubversion' |
|
286 | sett.ui_key = 'hgsubversion' | |
287 | sett.ui_section = 'extensions' |
|
287 | sett.ui_section = 'extensions' | |
288 |
|
288 | |||
289 | sett.ui_active = form_result[_f('extensions_hgsubversion')] |
|
289 | sett.ui_active = form_result[_f('extensions_hgsubversion')] | |
290 | Session().add(sett) |
|
290 | Session().add(sett) | |
291 |
|
291 | |||
292 | # sett = RhodeCodeUi.get_by_key('hggit') |
|
292 | # sett = RhodeCodeUi.get_by_key('hggit') | |
293 | # if not sett: |
|
293 | # if not sett: | |
294 | # #make one if it's not there ! |
|
294 | # #make one if it's not there ! | |
295 | # sett = RhodeCodeUi() |
|
295 | # sett = RhodeCodeUi() | |
296 | # sett.ui_key = 'hggit' |
|
296 | # sett.ui_key = 'hggit' | |
297 | # sett.ui_section = 'extensions' |
|
297 | # sett.ui_section = 'extensions' | |
298 | # |
|
298 | # | |
299 | # sett.ui_active = form_result[_f('extensions_hggit')] |
|
299 | # sett.ui_active = form_result[_f('extensions_hggit')] | |
300 | # Session().add(sett) |
|
300 | # Session().add(sett) | |
301 |
|
301 | |||
302 | Session().commit() |
|
302 | Session().commit() | |
303 |
|
303 | |||
304 | h.flash(_('Updated VCS settings'), category='success') |
|
304 | h.flash(_('Updated VCS settings'), category='success') | |
305 |
|
305 | |||
306 | except Exception: |
|
306 | except Exception: | |
307 | log.error(traceback.format_exc()) |
|
307 | log.error(traceback.format_exc()) | |
308 | h.flash(_('error occurred during updating ' |
|
308 | h.flash(_('error occurred during updating ' | |
309 | 'application settings'), category='error') |
|
309 | 'application settings'), category='error') | |
310 |
|
310 | |||
311 | if setting_id == 'hooks': |
|
311 | if setting_id == 'hooks': | |
312 | ui_key = request.POST.get('new_hook_ui_key') |
|
312 | ui_key = request.POST.get('new_hook_ui_key') | |
313 | ui_value = request.POST.get('new_hook_ui_value') |
|
313 | ui_value = request.POST.get('new_hook_ui_value') | |
314 | try: |
|
314 | try: | |
315 |
|
315 | |||
316 | if ui_value and ui_key: |
|
316 | if ui_value and ui_key: | |
317 | RhodeCodeUi.create_or_update_hook(ui_key, ui_value) |
|
317 | RhodeCodeUi.create_or_update_hook(ui_key, ui_value) | |
318 | h.flash(_('Added new hook'), |
|
318 | h.flash(_('Added new hook'), | |
319 | category='success') |
|
319 | category='success') | |
320 |
|
320 | |||
321 | # check for edits |
|
321 | # check for edits | |
322 | update = False |
|
322 | update = False | |
323 | _d = request.POST.dict_of_lists() |
|
323 | _d = request.POST.dict_of_lists() | |
324 | for k, v in zip(_d.get('hook_ui_key', []), |
|
324 | for k, v in zip(_d.get('hook_ui_key', []), | |
325 | _d.get('hook_ui_value_new', [])): |
|
325 | _d.get('hook_ui_value_new', [])): | |
326 | RhodeCodeUi.create_or_update_hook(k, v) |
|
326 | RhodeCodeUi.create_or_update_hook(k, v) | |
327 | update = True |
|
327 | update = True | |
328 |
|
328 | |||
329 | if update: |
|
329 | if update: | |
330 | h.flash(_('Updated hooks'), category='success') |
|
330 | h.flash(_('Updated hooks'), category='success') | |
331 | Session().commit() |
|
331 | Session().commit() | |
332 | except Exception: |
|
332 | except Exception: | |
333 | log.error(traceback.format_exc()) |
|
333 | log.error(traceback.format_exc()) | |
334 | h.flash(_('error occurred during hook creation'), |
|
334 | h.flash(_('error occurred during hook creation'), | |
335 | category='error') |
|
335 | category='error') | |
336 |
|
336 | |||
337 | return redirect(url('admin_edit_setting', setting_id='hooks')) |
|
337 | return redirect(url('admin_edit_setting', setting_id='hooks')) | |
338 |
|
338 | |||
339 | if setting_id == 'email': |
|
339 | if setting_id == 'email': | |
340 | test_email = request.POST.get('test_email') |
|
340 | test_email = request.POST.get('test_email') | |
341 | test_email_subj = 'RhodeCode TestEmail' |
|
341 | test_email_subj = 'RhodeCode TestEmail' | |
342 | test_email_body = 'RhodeCode Email test' |
|
342 | test_email_body = 'RhodeCode Email test' | |
343 |
|
343 | |||
344 | test_email_html_body = EmailNotificationModel()\ |
|
344 | test_email_html_body = EmailNotificationModel()\ | |
345 | .get_email_tmpl(EmailNotificationModel.TYPE_DEFAULT, |
|
345 | .get_email_tmpl(EmailNotificationModel.TYPE_DEFAULT, | |
346 | body=test_email_body) |
|
346 | body=test_email_body) | |
347 |
|
347 | |||
348 | recipients = [test_email] if test_email else None |
|
348 | recipients = [test_email] if test_email else None | |
349 |
|
349 | |||
350 | run_task(tasks.send_email, recipients, test_email_subj, |
|
350 | run_task(tasks.send_email, recipients, test_email_subj, | |
351 | test_email_body, test_email_html_body) |
|
351 | test_email_body, test_email_html_body) | |
352 |
|
352 | |||
353 | h.flash(_('Email task created'), category='success') |
|
353 | h.flash(_('Email task created'), category='success') | |
354 | return redirect(url('admin_settings')) |
|
354 | return redirect(url('admin_settings')) | |
355 |
|
355 | |||
356 | @HasPermissionAllDecorator('hg.admin') |
|
356 | @HasPermissionAllDecorator('hg.admin') | |
357 | def delete(self, setting_id): |
|
357 | def delete(self, setting_id): | |
358 | """DELETE /admin/settings/setting_id: Delete an existing item""" |
|
358 | """DELETE /admin/settings/setting_id: Delete an existing item""" | |
359 | # Forms posted to this method should contain a hidden field: |
|
359 | # Forms posted to this method should contain a hidden field: | |
360 | # <input type="hidden" name="_method" value="DELETE" /> |
|
360 | # <input type="hidden" name="_method" value="DELETE" /> | |
361 | # Or using helpers: |
|
361 | # Or using helpers: | |
362 | # h.form(url('admin_setting', setting_id=ID), |
|
362 | # h.form(url('admin_setting', setting_id=ID), | |
363 | # method='delete') |
|
363 | # method='delete') | |
364 | # url('admin_setting', setting_id=ID) |
|
364 | # url('admin_setting', setting_id=ID) | |
365 | if setting_id == 'hooks': |
|
365 | if setting_id == 'hooks': | |
366 | hook_id = request.POST.get('hook_id') |
|
366 | hook_id = request.POST.get('hook_id') | |
367 | RhodeCodeUi.delete(hook_id) |
|
367 | RhodeCodeUi.delete(hook_id) | |
368 | Session().commit() |
|
368 | Session().commit() | |
369 |
|
369 | |||
370 | @HasPermissionAllDecorator('hg.admin') |
|
370 | @HasPermissionAllDecorator('hg.admin') | |
371 | def show(self, setting_id, format='html'): |
|
371 | def show(self, setting_id, format='html'): | |
372 | """ |
|
372 | """ | |
373 | GET /admin/settings/setting_id: Show a specific item""" |
|
373 | GET /admin/settings/setting_id: Show a specific item""" | |
374 | # url('admin_setting', setting_id=ID) |
|
374 | # url('admin_setting', setting_id=ID) | |
375 |
|
375 | |||
376 | @HasPermissionAllDecorator('hg.admin') |
|
376 | @HasPermissionAllDecorator('hg.admin') | |
377 | def edit(self, setting_id, format='html'): |
|
377 | def edit(self, setting_id, format='html'): | |
378 | """ |
|
378 | """ | |
379 | GET /admin/settings/setting_id/edit: Form to |
|
379 | GET /admin/settings/setting_id/edit: Form to | |
380 | edit an existing item""" |
|
380 | edit an existing item""" | |
381 | # url('admin_edit_setting', setting_id=ID) |
|
381 | # url('admin_edit_setting', setting_id=ID) | |
382 |
|
382 | |||
383 | c.hooks = RhodeCodeUi.get_builtin_hooks() |
|
383 | c.hooks = RhodeCodeUi.get_builtin_hooks() | |
384 | c.custom_hooks = RhodeCodeUi.get_custom_hooks() |
|
384 | c.custom_hooks = RhodeCodeUi.get_custom_hooks() | |
385 |
|
385 | |||
386 | return htmlfill.render( |
|
386 | return htmlfill.render( | |
387 | render('admin/settings/hooks.html'), |
|
387 | render('admin/settings/hooks.html'), | |
388 | defaults={}, |
|
388 | defaults={}, | |
389 | encoding="UTF-8", |
|
389 | encoding="UTF-8", | |
390 | force_defaults=False |
|
390 | force_defaults=False | |
391 | ) |
|
391 | ) | |
392 |
|
392 | |||
393 | def _load_my_repos_data(self): |
|
393 | def _load_my_repos_data(self): | |
394 | repos_list = Session().query(Repository)\ |
|
394 | repos_list = Session().query(Repository)\ | |
395 | .filter(Repository.user_id == |
|
395 | .filter(Repository.user_id == | |
396 | self.rhodecode_user.user_id)\ |
|
396 | self.rhodecode_user.user_id)\ | |
397 | .order_by(func.lower(Repository.repo_name)).all() |
|
397 | .order_by(func.lower(Repository.repo_name)).all() | |
398 |
|
398 | |||
399 | repos_data = RepoModel().get_repos_as_dict(repos_list=repos_list, |
|
399 | repos_data = RepoModel().get_repos_as_dict(repos_list=repos_list, | |
400 | admin=True) |
|
400 | admin=True) | |
401 | #json used to render the grid |
|
401 | #json used to render the grid | |
402 | return json.dumps(repos_data) |
|
402 | return json.dumps(repos_data) | |
403 |
|
403 | |||
404 | @NotAnonymous() |
|
404 | @NotAnonymous() | |
405 | def my_account(self): |
|
405 | def my_account(self): | |
406 | """ |
|
406 | """ | |
407 | GET /_admin/my_account Displays info about my account |
|
407 | GET /_admin/my_account Displays info about my account | |
408 | """ |
|
408 | """ | |
409 | # url('admin_settings_my_account') |
|
409 | # url('admin_settings_my_account') | |
410 |
|
410 | |||
411 | c.user = User.get(self.rhodecode_user.user_id) |
|
411 | c.user = User.get(self.rhodecode_user.user_id) | |
412 | c.ldap_dn = c.user.ldap_dn |
|
412 | c.ldap_dn = c.user.ldap_dn | |
413 |
|
413 | |||
414 | if c.user.username == 'default': |
|
414 | if c.user.username == 'default': | |
415 | h.flash(_("You can't edit this user since it's" |
|
415 | h.flash(_("You can't edit this user since it's" | |
416 | " crucial for entire application"), category='warning') |
|
416 | " crucial for entire application"), category='warning') | |
417 | return redirect(url('users')) |
|
417 | return redirect(url('users')) | |
418 |
|
418 | |||
419 | #json used to render the grid |
|
419 | #json used to render the grid | |
420 | c.data = self._load_my_repos_data() |
|
420 | c.data = self._load_my_repos_data() | |
421 |
|
421 | |||
422 | defaults = c.user.get_dict() |
|
422 | defaults = c.user.get_dict() | |
423 |
|
423 | |||
424 | c.form = htmlfill.render( |
|
424 | c.form = htmlfill.render( | |
425 | render('admin/users/user_edit_my_account_form.html'), |
|
425 | render('admin/users/user_edit_my_account_form.html'), | |
426 | defaults=defaults, |
|
426 | defaults=defaults, | |
427 | encoding="UTF-8", |
|
427 | encoding="UTF-8", | |
428 | force_defaults=False |
|
428 | force_defaults=False | |
429 | ) |
|
429 | ) | |
430 | return render('admin/users/user_edit_my_account.html') |
|
430 | return render('admin/users/user_edit_my_account.html') | |
431 |
|
431 | |||
432 | @NotAnonymous() |
|
432 | @NotAnonymous() | |
433 | def my_account_update(self): |
|
433 | def my_account_update(self): | |
434 | """PUT /_admin/my_account_update: Update an existing item""" |
|
434 | """PUT /_admin/my_account_update: Update an existing item""" | |
435 | # Forms posted to this method should contain a hidden field: |
|
435 | # Forms posted to this method should contain a hidden field: | |
436 | # <input type="hidden" name="_method" value="PUT" /> |
|
436 | # <input type="hidden" name="_method" value="PUT" /> | |
437 | # Or using helpers: |
|
437 | # Or using helpers: | |
438 | # h.form(url('admin_settings_my_account_update'), |
|
438 | # h.form(url('admin_settings_my_account_update'), | |
439 | # method='put') |
|
439 | # method='put') | |
440 | # url('admin_settings_my_account_update', id=ID) |
|
440 | # url('admin_settings_my_account_update', id=ID) | |
441 | uid = self.rhodecode_user.user_id |
|
441 | uid = self.rhodecode_user.user_id | |
442 | c.user = User.get(self.rhodecode_user.user_id) |
|
442 | c.user = User.get(self.rhodecode_user.user_id) | |
443 | c.ldap_dn = c.user.ldap_dn |
|
443 | c.ldap_dn = c.user.ldap_dn | |
444 | email = self.rhodecode_user.email |
|
444 | email = self.rhodecode_user.email | |
445 | _form = UserForm(edit=True, |
|
445 | _form = UserForm(edit=True, | |
446 | old_data={'user_id': uid, 'email': email})() |
|
446 | old_data={'user_id': uid, 'email': email})() | |
447 | form_result = {} |
|
447 | form_result = {} | |
448 | try: |
|
448 | try: | |
449 | form_result = _form.to_python(dict(request.POST)) |
|
449 | form_result = _form.to_python(dict(request.POST)) | |
450 | skip_attrs = ['admin', 'active'] # skip attr for my account |
|
450 | skip_attrs = ['admin', 'active'] # skip attr for my account | |
451 | if c.ldap_dn: |
|
451 | if c.ldap_dn: | |
452 | #forbid updating username for ldap accounts |
|
452 | #forbid updating username for ldap accounts | |
453 | skip_attrs.append('username') |
|
453 | skip_attrs.append('username') | |
454 | UserModel().update(uid, form_result, skip_attrs=skip_attrs) |
|
454 | UserModel().update(uid, form_result, skip_attrs=skip_attrs) | |
455 | h.flash(_('Your account was updated successfully'), |
|
455 | h.flash(_('Your account was updated successfully'), | |
456 | category='success') |
|
456 | category='success') | |
457 | Session().commit() |
|
457 | Session().commit() | |
458 | except formencode.Invalid, errors: |
|
458 | except formencode.Invalid, errors: | |
459 | #json used to render the grid |
|
459 | #json used to render the grid | |
460 | c.data = self._load_my_repos_data() |
|
460 | c.data = self._load_my_repos_data() | |
461 | c.form = htmlfill.render( |
|
461 | c.form = htmlfill.render( | |
462 | render('admin/users/user_edit_my_account_form.html'), |
|
462 | render('admin/users/user_edit_my_account_form.html'), | |
463 | defaults=errors.value, |
|
463 | defaults=errors.value, | |
464 | errors=errors.error_dict or {}, |
|
464 | errors=errors.error_dict or {}, | |
465 | prefix_error=False, |
|
465 | prefix_error=False, | |
466 | encoding="UTF-8") |
|
466 | encoding="UTF-8") | |
467 | return render('admin/users/user_edit_my_account.html') |
|
467 | return render('admin/users/user_edit_my_account.html') | |
468 | except Exception: |
|
468 | except Exception: | |
469 | log.error(traceback.format_exc()) |
|
469 | log.error(traceback.format_exc()) | |
470 | h.flash(_('error occurred during update of user %s') \ |
|
470 | h.flash(_('error occurred during update of user %s') \ | |
471 | % form_result.get('username'), category='error') |
|
471 | % form_result.get('username'), category='error') | |
472 |
|
472 | |||
473 | return redirect(url('my_account')) |
|
473 | return redirect(url('my_account')) | |
474 |
|
474 | |||
475 | @NotAnonymous() |
|
475 | @NotAnonymous() | |
476 | def my_account_my_pullrequests(self): |
|
476 | def my_account_my_pullrequests(self): | |
477 | c.my_pull_requests = PullRequest.query()\ |
|
477 | c.my_pull_requests = PullRequest.query()\ | |
478 | .filter(PullRequest.user_id == |
|
478 | .filter(PullRequest.user_id == | |
479 | self.rhodecode_user.user_id)\ |
|
479 | self.rhodecode_user.user_id)\ | |
|
480 | .order_by(PullRequest.created_on.desc())\ | |||
480 | .all() |
|
481 | .all() | |
481 | c.participate_in_pull_requests = \ |
|
482 | ||
|
483 | c.participate_in_pull_requests = sorted( | |||
482 | [x.pull_request for x in PullRequestReviewers.query()\ |
|
484 | [x.pull_request for x in PullRequestReviewers.query()\ | |
483 | .filter(PullRequestReviewers.user_id == |
|
485 | .filter(PullRequestReviewers.user_id == | |
484 | self.rhodecode_user.user_id)\ |
|
486 | self.rhodecode_user.user_id)\ | |
485 | .all()] |
|
487 | .all()], | |
|
488 | key=lambda o: o.created_on, reverse=True) | |||
|
489 | ||||
486 | return render('admin/users/user_edit_my_account_pullrequests.html') |
|
490 | return render('admin/users/user_edit_my_account_pullrequests.html') | |
487 |
|
491 | |||
488 | @NotAnonymous() |
|
492 | @NotAnonymous() | |
489 | def create_repository(self): |
|
493 | def create_repository(self): | |
490 | """GET /_admin/create_repository: Form to create a new item""" |
|
494 | """GET /_admin/create_repository: Form to create a new item""" | |
491 | new_repo = request.GET.get('repo', '') |
|
495 | new_repo = request.GET.get('repo', '') | |
492 | parent_group = request.GET.get('parent_group') |
|
496 | parent_group = request.GET.get('parent_group') | |
493 | if not HasPermissionAny('hg.admin', 'hg.create.repository')(): |
|
497 | if not HasPermissionAny('hg.admin', 'hg.create.repository')(): | |
494 | #you're not super admin nor have global create permissions, |
|
498 | #you're not super admin nor have global create permissions, | |
495 | #but maybe you have at least write permission to a parent group ? |
|
499 | #but maybe you have at least write permission to a parent group ? | |
496 | _gr = RepoGroup.get(parent_group) |
|
500 | _gr = RepoGroup.get(parent_group) | |
497 | gr_name = _gr.group_name if _gr else None |
|
501 | gr_name = _gr.group_name if _gr else None | |
498 | if not HasReposGroupPermissionAny('group.admin', 'group.write')(group_name=gr_name): |
|
502 | if not HasReposGroupPermissionAny('group.admin', 'group.write')(group_name=gr_name): | |
499 | raise HTTPForbidden |
|
503 | raise HTTPForbidden | |
500 |
|
504 | |||
501 | acl_groups = GroupList(RepoGroup.query().all(), |
|
505 | acl_groups = GroupList(RepoGroup.query().all(), | |
502 | perm_set=['group.write', 'group.admin']) |
|
506 | perm_set=['group.write', 'group.admin']) | |
503 | c.repo_groups = RepoGroup.groups_choices(groups=acl_groups) |
|
507 | c.repo_groups = RepoGroup.groups_choices(groups=acl_groups) | |
504 | c.repo_groups_choices = map(lambda k: unicode(k[0]), c.repo_groups) |
|
508 | c.repo_groups_choices = map(lambda k: unicode(k[0]), c.repo_groups) | |
505 | choices, c.landing_revs = ScmModel().get_repo_landing_revs() |
|
509 | choices, c.landing_revs = ScmModel().get_repo_landing_revs() | |
506 |
|
510 | |||
507 | c.new_repo = repo_name_slug(new_repo) |
|
511 | c.new_repo = repo_name_slug(new_repo) | |
508 |
|
512 | |||
509 | ## apply the defaults from defaults page |
|
513 | ## apply the defaults from defaults page | |
510 | defaults = RhodeCodeSetting.get_default_repo_settings(strip_prefix=True) |
|
514 | defaults = RhodeCodeSetting.get_default_repo_settings(strip_prefix=True) | |
511 | if parent_group: |
|
515 | if parent_group: | |
512 | defaults.update({'repo_group': parent_group}) |
|
516 | defaults.update({'repo_group': parent_group}) | |
513 |
|
517 | |||
514 | return htmlfill.render( |
|
518 | return htmlfill.render( | |
515 | render('admin/repos/repo_add.html'), |
|
519 | render('admin/repos/repo_add.html'), | |
516 | defaults=defaults, |
|
520 | defaults=defaults, | |
517 | errors={}, |
|
521 | errors={}, | |
518 | prefix_error=False, |
|
522 | prefix_error=False, | |
519 | encoding="UTF-8" |
|
523 | encoding="UTF-8" | |
520 | ) |
|
524 | ) | |
521 |
|
525 | |||
522 | def _get_hg_ui_settings(self): |
|
526 | def _get_hg_ui_settings(self): | |
523 | ret = RhodeCodeUi.query().all() |
|
527 | ret = RhodeCodeUi.query().all() | |
524 |
|
528 | |||
525 | if not ret: |
|
529 | if not ret: | |
526 | raise Exception('Could not get application ui settings !') |
|
530 | raise Exception('Could not get application ui settings !') | |
527 | settings = {} |
|
531 | settings = {} | |
528 | for each in ret: |
|
532 | for each in ret: | |
529 | k = each.ui_key |
|
533 | k = each.ui_key | |
530 | v = each.ui_value |
|
534 | v = each.ui_value | |
531 | if k == '/': |
|
535 | if k == '/': | |
532 | k = 'root_path' |
|
536 | k = 'root_path' | |
533 |
|
537 | |||
534 | if k == 'push_ssl': |
|
538 | if k == 'push_ssl': | |
535 | v = str2bool(v) |
|
539 | v = str2bool(v) | |
536 |
|
540 | |||
537 | if k.find('.') != -1: |
|
541 | if k.find('.') != -1: | |
538 | k = k.replace('.', '_') |
|
542 | k = k.replace('.', '_') | |
539 |
|
543 | |||
540 | if each.ui_section in ['hooks', 'extensions']: |
|
544 | if each.ui_section in ['hooks', 'extensions']: | |
541 | v = each.ui_active |
|
545 | v = each.ui_active | |
542 |
|
546 | |||
543 | settings[each.ui_section + '_' + k] = v |
|
547 | settings[each.ui_section + '_' + k] = v | |
544 | return settings |
|
548 | return settings |
@@ -1,257 +1,258 b'' | |||||
1 | # -*- coding: utf-8 -*- |
|
1 | # -*- coding: utf-8 -*- | |
2 | """ |
|
2 | """ | |
3 | rhodecode.model.pull_request |
|
3 | rhodecode.model.pull_request | |
4 | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ |
|
4 | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | |
5 |
|
5 | |||
6 | pull request model for RhodeCode |
|
6 | pull request model for RhodeCode | |
7 |
|
7 | |||
8 | :created_on: Jun 6, 2012 |
|
8 | :created_on: Jun 6, 2012 | |
9 | :author: marcink |
|
9 | :author: marcink | |
10 | :copyright: (C) 2012-2012 Marcin Kuzminski <marcin@python-works.com> |
|
10 | :copyright: (C) 2012-2012 Marcin Kuzminski <marcin@python-works.com> | |
11 | :license: GPLv3, see COPYING for more details. |
|
11 | :license: GPLv3, see COPYING for more details. | |
12 | """ |
|
12 | """ | |
13 | # This program is free software: you can redistribute it and/or modify |
|
13 | # This program is free software: you can redistribute it and/or modify | |
14 | # it under the terms of the GNU General Public License as published by |
|
14 | # it under the terms of the GNU General Public License as published by | |
15 | # the Free Software Foundation, either version 3 of the License, or |
|
15 | # the Free Software Foundation, either version 3 of the License, or | |
16 | # (at your option) any later version. |
|
16 | # (at your option) any later version. | |
17 | # |
|
17 | # | |
18 | # This program is distributed in the hope that it will be useful, |
|
18 | # This program is distributed in the hope that it will be useful, | |
19 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
19 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | |
20 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
20 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
21 | # GNU General Public License for more details. |
|
21 | # GNU General Public License for more details. | |
22 | # |
|
22 | # | |
23 | # You should have received a copy of the GNU General Public License |
|
23 | # You should have received a copy of the GNU General Public License | |
24 | # along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
24 | # along with this program. If not, see <http://www.gnu.org/licenses/>. | |
25 |
|
25 | |||
26 | import logging |
|
26 | import logging | |
27 | import datetime |
|
27 | import datetime | |
28 | import re |
|
28 | import re | |
29 |
|
29 | |||
30 | from pylons.i18n.translation import _ |
|
30 | from pylons.i18n.translation import _ | |
31 |
|
31 | |||
32 | from rhodecode.model.meta import Session |
|
32 | from rhodecode.model.meta import Session | |
33 | from rhodecode.lib import helpers as h, unionrepo |
|
33 | from rhodecode.lib import helpers as h, unionrepo | |
34 | from rhodecode.model import BaseModel |
|
34 | from rhodecode.model import BaseModel | |
35 | from rhodecode.model.db import PullRequest, PullRequestReviewers, Notification,\ |
|
35 | from rhodecode.model.db import PullRequest, PullRequestReviewers, Notification,\ | |
36 | ChangesetStatus |
|
36 | ChangesetStatus | |
37 | from rhodecode.model.notification import NotificationModel |
|
37 | from rhodecode.model.notification import NotificationModel | |
38 | from rhodecode.lib.utils2 import safe_unicode |
|
38 | from rhodecode.lib.utils2 import safe_unicode | |
39 |
|
39 | |||
40 | from rhodecode.lib.vcs.utils.hgcompat import scmutil |
|
40 | from rhodecode.lib.vcs.utils.hgcompat import scmutil | |
41 | from rhodecode.lib.vcs.utils import safe_str |
|
41 | from rhodecode.lib.vcs.utils import safe_str | |
42 | from rhodecode.lib.vcs.backends.base import EmptyChangeset |
|
42 | from rhodecode.lib.vcs.backends.base import EmptyChangeset | |
43 |
|
43 | |||
44 | log = logging.getLogger(__name__) |
|
44 | log = logging.getLogger(__name__) | |
45 |
|
45 | |||
46 |
|
46 | |||
47 | class PullRequestModel(BaseModel): |
|
47 | class PullRequestModel(BaseModel): | |
48 |
|
48 | |||
49 | cls = PullRequest |
|
49 | cls = PullRequest | |
50 |
|
50 | |||
51 | def __get_pull_request(self, pull_request): |
|
51 | def __get_pull_request(self, pull_request): | |
52 | return self._get_instance(PullRequest, pull_request) |
|
52 | return self._get_instance(PullRequest, pull_request) | |
53 |
|
53 | |||
54 | def get_all(self, repo): |
|
54 | def get_all(self, repo): | |
55 | repo = self._get_repo(repo) |
|
55 | repo = self._get_repo(repo) | |
56 | return PullRequest.query()\ |
|
56 | return PullRequest.query()\ | |
57 | .filter(PullRequest.other_repo == repo)\ |
|
57 | .filter(PullRequest.other_repo == repo)\ | |
58 | .order_by(PullRequest.created_on)\ |
|
58 | .order_by(PullRequest.created_on.desc())\ | |
59 | .all() |
|
59 | .all() | |
60 |
|
60 | |||
61 | def create(self, created_by, org_repo, org_ref, other_repo, other_ref, |
|
61 | def create(self, created_by, org_repo, org_ref, other_repo, other_ref, | |
62 | revisions, reviewers, title, description=None): |
|
62 | revisions, reviewers, title, description=None): | |
63 | from rhodecode.model.changeset_status import ChangesetStatusModel |
|
63 | from rhodecode.model.changeset_status import ChangesetStatusModel | |
64 |
|
64 | |||
65 | created_by_user = self._get_user(created_by) |
|
65 | created_by_user = self._get_user(created_by) | |
66 | org_repo = self._get_repo(org_repo) |
|
66 | org_repo = self._get_repo(org_repo) | |
67 | other_repo = self._get_repo(other_repo) |
|
67 | other_repo = self._get_repo(other_repo) | |
68 |
|
68 | |||
69 | new = PullRequest() |
|
69 | new = PullRequest() | |
70 | new.org_repo = org_repo |
|
70 | new.org_repo = org_repo | |
71 | new.org_ref = org_ref |
|
71 | new.org_ref = org_ref | |
72 | new.other_repo = other_repo |
|
72 | new.other_repo = other_repo | |
73 | new.other_ref = other_ref |
|
73 | new.other_ref = other_ref | |
74 | new.revisions = revisions |
|
74 | new.revisions = revisions | |
75 | new.title = title |
|
75 | new.title = title | |
76 | new.description = description |
|
76 | new.description = description | |
77 | new.author = created_by_user |
|
77 | new.author = created_by_user | |
78 | self.sa.add(new) |
|
78 | self.sa.add(new) | |
79 | Session().flush() |
|
79 | Session().flush() | |
80 | #members |
|
80 | #members | |
81 | for member in reviewers: |
|
81 | for member in set(reviewers): | |
82 | _usr = self._get_user(member) |
|
82 | _usr = self._get_user(member) | |
83 | reviewer = PullRequestReviewers(_usr, new) |
|
83 | reviewer = PullRequestReviewers(_usr, new) | |
84 | self.sa.add(reviewer) |
|
84 | self.sa.add(reviewer) | |
85 |
|
85 | |||
86 | #reset state to under-review |
|
86 | #reset state to under-review | |
87 | ChangesetStatusModel().set_status( |
|
87 | ChangesetStatusModel().set_status( | |
88 | repo=org_repo, |
|
88 | repo=org_repo, | |
89 | status=ChangesetStatus.STATUS_UNDER_REVIEW, |
|
89 | status=ChangesetStatus.STATUS_UNDER_REVIEW, | |
90 | user=created_by_user, |
|
90 | user=created_by_user, | |
91 | pull_request=new |
|
91 | pull_request=new | |
92 | ) |
|
92 | ) | |
93 |
|
93 | |||
94 | #notification to reviewers |
|
94 | #notification to reviewers | |
95 | notif = NotificationModel() |
|
95 | notif = NotificationModel() | |
96 |
|
96 | |||
97 | pr_url = h.url('pullrequest_show', repo_name=other_repo.repo_name, |
|
97 | pr_url = h.url('pullrequest_show', repo_name=other_repo.repo_name, | |
98 | pull_request_id=new.pull_request_id, |
|
98 | pull_request_id=new.pull_request_id, | |
99 | qualified=True, |
|
99 | qualified=True, | |
100 | ) |
|
100 | ) | |
101 | subject = safe_unicode( |
|
101 | subject = safe_unicode( | |
102 | h.link_to( |
|
102 | h.link_to( | |
103 | _('%(user)s wants you to review pull request #%(pr_id)s: %(pr_title)s') % \ |
|
103 | _('%(user)s wants you to review pull request #%(pr_id)s: %(pr_title)s') % \ | |
104 | {'user': created_by_user.username, |
|
104 | {'user': created_by_user.username, | |
105 | 'pr_title': new.title, |
|
105 | 'pr_title': new.title, | |
106 | 'pr_id': new.pull_request_id}, |
|
106 | 'pr_id': new.pull_request_id}, | |
107 | pr_url |
|
107 | pr_url | |
108 | ) |
|
108 | ) | |
109 | ) |
|
109 | ) | |
110 | body = description |
|
110 | body = description | |
111 | kwargs = { |
|
111 | kwargs = { | |
112 | 'pr_title': title, |
|
112 | 'pr_title': title, | |
113 | 'pr_user_created': h.person(created_by_user.email), |
|
113 | 'pr_user_created': h.person(created_by_user.email), | |
114 | 'pr_repo_url': h.url('summary_home', repo_name=other_repo.repo_name, |
|
114 | 'pr_repo_url': h.url('summary_home', repo_name=other_repo.repo_name, | |
115 | qualified=True,), |
|
115 | qualified=True,), | |
116 | 'pr_url': pr_url, |
|
116 | 'pr_url': pr_url, | |
117 | 'pr_revisions': revisions |
|
117 | 'pr_revisions': revisions | |
118 | } |
|
118 | } | |
|
119 | ||||
119 | notif.create(created_by=created_by_user, subject=subject, body=body, |
|
120 | notif.create(created_by=created_by_user, subject=subject, body=body, | |
120 | recipients=reviewers, |
|
121 | recipients=reviewers, | |
121 | type_=Notification.TYPE_PULL_REQUEST, email_kwargs=kwargs) |
|
122 | type_=Notification.TYPE_PULL_REQUEST, email_kwargs=kwargs) | |
122 | return new |
|
123 | return new | |
123 |
|
124 | |||
124 | def update_reviewers(self, pull_request, reviewers_ids): |
|
125 | def update_reviewers(self, pull_request, reviewers_ids): | |
125 | reviewers_ids = set(reviewers_ids) |
|
126 | reviewers_ids = set(reviewers_ids) | |
126 | pull_request = self.__get_pull_request(pull_request) |
|
127 | pull_request = self.__get_pull_request(pull_request) | |
127 | current_reviewers = PullRequestReviewers.query()\ |
|
128 | current_reviewers = PullRequestReviewers.query()\ | |
128 | .filter(PullRequestReviewers.pull_request== |
|
129 | .filter(PullRequestReviewers.pull_request== | |
129 | pull_request)\ |
|
130 | pull_request)\ | |
130 | .all() |
|
131 | .all() | |
131 | current_reviewers_ids = set([x.user.user_id for x in current_reviewers]) |
|
132 | current_reviewers_ids = set([x.user.user_id for x in current_reviewers]) | |
132 |
|
133 | |||
133 | to_add = reviewers_ids.difference(current_reviewers_ids) |
|
134 | to_add = reviewers_ids.difference(current_reviewers_ids) | |
134 | to_remove = current_reviewers_ids.difference(reviewers_ids) |
|
135 | to_remove = current_reviewers_ids.difference(reviewers_ids) | |
135 |
|
136 | |||
136 | log.debug("Adding %s reviewers" % to_add) |
|
137 | log.debug("Adding %s reviewers" % to_add) | |
137 | log.debug("Removing %s reviewers" % to_remove) |
|
138 | log.debug("Removing %s reviewers" % to_remove) | |
138 |
|
139 | |||
139 | for uid in to_add: |
|
140 | for uid in to_add: | |
140 | _usr = self._get_user(uid) |
|
141 | _usr = self._get_user(uid) | |
141 | reviewer = PullRequestReviewers(_usr, pull_request) |
|
142 | reviewer = PullRequestReviewers(_usr, pull_request) | |
142 | self.sa.add(reviewer) |
|
143 | self.sa.add(reviewer) | |
143 |
|
144 | |||
144 | for uid in to_remove: |
|
145 | for uid in to_remove: | |
145 | reviewer = PullRequestReviewers.query()\ |
|
146 | reviewer = PullRequestReviewers.query()\ | |
146 | .filter(PullRequestReviewers.user_id==uid, |
|
147 | .filter(PullRequestReviewers.user_id==uid, | |
147 | PullRequestReviewers.pull_request==pull_request)\ |
|
148 | PullRequestReviewers.pull_request==pull_request)\ | |
148 | .scalar() |
|
149 | .scalar() | |
149 | if reviewer: |
|
150 | if reviewer: | |
150 | self.sa.delete(reviewer) |
|
151 | self.sa.delete(reviewer) | |
151 |
|
152 | |||
152 | def delete(self, pull_request): |
|
153 | def delete(self, pull_request): | |
153 | pull_request = self.__get_pull_request(pull_request) |
|
154 | pull_request = self.__get_pull_request(pull_request) | |
154 | Session().delete(pull_request) |
|
155 | Session().delete(pull_request) | |
155 |
|
156 | |||
156 | def close_pull_request(self, pull_request): |
|
157 | def close_pull_request(self, pull_request): | |
157 | pull_request = self.__get_pull_request(pull_request) |
|
158 | pull_request = self.__get_pull_request(pull_request) | |
158 | pull_request.status = PullRequest.STATUS_CLOSED |
|
159 | pull_request.status = PullRequest.STATUS_CLOSED | |
159 | pull_request.updated_on = datetime.datetime.now() |
|
160 | pull_request.updated_on = datetime.datetime.now() | |
160 | self.sa.add(pull_request) |
|
161 | self.sa.add(pull_request) | |
161 |
|
162 | |||
162 | def _get_changesets(self, alias, org_repo, org_ref, other_repo, other_ref): |
|
163 | def _get_changesets(self, alias, org_repo, org_ref, other_repo, other_ref): | |
163 | """ |
|
164 | """ | |
164 | Returns a list of changesets that can be merged from org_repo@org_ref |
|
165 | Returns a list of changesets that can be merged from org_repo@org_ref | |
165 | to other_repo@other_ref ... and the ancestor that would be used for merge |
|
166 | to other_repo@other_ref ... and the ancestor that would be used for merge | |
166 |
|
167 | |||
167 | :param org_repo: |
|
168 | :param org_repo: | |
168 | :param org_ref: |
|
169 | :param org_ref: | |
169 | :param other_repo: |
|
170 | :param other_repo: | |
170 | :param other_ref: |
|
171 | :param other_ref: | |
171 | :param tmp: |
|
172 | :param tmp: | |
172 | """ |
|
173 | """ | |
173 |
|
174 | |||
174 | ancestor = None |
|
175 | ancestor = None | |
175 |
|
176 | |||
176 | if alias == 'hg': |
|
177 | if alias == 'hg': | |
177 | # lookup up the exact node id |
|
178 | # lookup up the exact node id | |
178 | _revset_predicates = { |
|
179 | _revset_predicates = { | |
179 | 'branch': 'branch', |
|
180 | 'branch': 'branch', | |
180 | 'book': 'bookmark', |
|
181 | 'book': 'bookmark', | |
181 | 'tag': 'tag', |
|
182 | 'tag': 'tag', | |
182 | 'rev': 'id', |
|
183 | 'rev': 'id', | |
183 | } |
|
184 | } | |
184 |
|
185 | |||
185 | org_rev_spec = "%s('%s')" % (_revset_predicates[org_ref[0]], |
|
186 | org_rev_spec = "%s('%s')" % (_revset_predicates[org_ref[0]], | |
186 | safe_str(org_ref[1])) |
|
187 | safe_str(org_ref[1])) | |
187 | if org_ref[1] == EmptyChangeset().raw_id: |
|
188 | if org_ref[1] == EmptyChangeset().raw_id: | |
188 | org_rev = org_ref[1] |
|
189 | org_rev = org_ref[1] | |
189 | else: |
|
190 | else: | |
190 | org_rev = org_repo._repo[scmutil.revrange(org_repo._repo, |
|
191 | org_rev = org_repo._repo[scmutil.revrange(org_repo._repo, | |
191 | [org_rev_spec])[-1]] |
|
192 | [org_rev_spec])[-1]] | |
192 | other_rev_spec = "%s('%s')" % (_revset_predicates[other_ref[0]], |
|
193 | other_rev_spec = "%s('%s')" % (_revset_predicates[other_ref[0]], | |
193 | safe_str(other_ref[1])) |
|
194 | safe_str(other_ref[1])) | |
194 | if other_ref[1] == EmptyChangeset().raw_id: |
|
195 | if other_ref[1] == EmptyChangeset().raw_id: | |
195 | other_rev = other_ref[1] |
|
196 | other_rev = other_ref[1] | |
196 | else: |
|
197 | else: | |
197 | other_rev = other_repo._repo[scmutil.revrange(other_repo._repo, |
|
198 | other_rev = other_repo._repo[scmutil.revrange(other_repo._repo, | |
198 | [other_rev_spec])[-1]] |
|
199 | [other_rev_spec])[-1]] | |
199 |
|
200 | |||
200 | #case two independent repos |
|
201 | #case two independent repos | |
201 | if org_repo != other_repo: |
|
202 | if org_repo != other_repo: | |
202 | hgrepo = unionrepo.unionrepository(other_repo.baseui, |
|
203 | hgrepo = unionrepo.unionrepository(other_repo.baseui, | |
203 | other_repo.path, |
|
204 | other_repo.path, | |
204 | org_repo.path) |
|
205 | org_repo.path) | |
205 | # all the changesets we are looking for will be in other_repo, |
|
206 | # all the changesets we are looking for will be in other_repo, | |
206 | # so rev numbers from hgrepo can be used in other_repo |
|
207 | # so rev numbers from hgrepo can be used in other_repo | |
207 |
|
208 | |||
208 | #no remote compare do it on the same repository |
|
209 | #no remote compare do it on the same repository | |
209 | else: |
|
210 | else: | |
210 | hgrepo = other_repo._repo |
|
211 | hgrepo = other_repo._repo | |
211 |
|
212 | |||
212 | revs = ["ancestors(id('%s')) and not ancestors(id('%s'))" % |
|
213 | revs = ["ancestors(id('%s')) and not ancestors(id('%s'))" % | |
213 | (other_rev, org_rev)] |
|
214 | (other_rev, org_rev)] | |
214 | changesets = [other_repo.get_changeset(cs) |
|
215 | changesets = [other_repo.get_changeset(cs) | |
215 | for cs in scmutil.revrange(hgrepo, revs)] |
|
216 | for cs in scmutil.revrange(hgrepo, revs)] | |
216 |
|
217 | |||
217 | if org_repo != other_repo: |
|
218 | if org_repo != other_repo: | |
218 | ancestors = scmutil.revrange(hgrepo, |
|
219 | ancestors = scmutil.revrange(hgrepo, | |
219 | ["ancestor(id('%s'), id('%s'))" % (org_rev, other_rev)]) |
|
220 | ["ancestor(id('%s'), id('%s'))" % (org_rev, other_rev)]) | |
220 | if len(ancestors) == 1: |
|
221 | if len(ancestors) == 1: | |
221 | ancestor = hgrepo[ancestors[0]].hex() |
|
222 | ancestor = hgrepo[ancestors[0]].hex() | |
222 |
|
223 | |||
223 | elif alias == 'git': |
|
224 | elif alias == 'git': | |
224 | assert org_repo == other_repo, (org_repo, other_repo) # no git support for different repos |
|
225 | assert org_repo == other_repo, (org_repo, other_repo) # no git support for different repos | |
225 | so, se = org_repo.run_git_command( |
|
226 | so, se = org_repo.run_git_command( | |
226 | 'log --reverse --pretty="format: %%H" -s -p %s..%s' % (org_ref[1], |
|
227 | 'log --reverse --pretty="format: %%H" -s -p %s..%s' % (org_ref[1], | |
227 | other_ref[1]) |
|
228 | other_ref[1]) | |
228 | ) |
|
229 | ) | |
229 | changesets = [org_repo.get_changeset(cs) |
|
230 | changesets = [org_repo.get_changeset(cs) | |
230 | for cs in re.findall(r'[0-9a-fA-F]{40}', so)] |
|
231 | for cs in re.findall(r'[0-9a-fA-F]{40}', so)] | |
231 |
|
232 | |||
232 | return changesets, ancestor |
|
233 | return changesets, ancestor | |
233 |
|
234 | |||
234 | def get_compare_data(self, org_repo, org_ref, other_repo, other_ref): |
|
235 | def get_compare_data(self, org_repo, org_ref, other_repo, other_ref): | |
235 | """ |
|
236 | """ | |
236 | Returns incoming changesets for mercurial repositories |
|
237 | Returns incoming changesets for mercurial repositories | |
237 |
|
238 | |||
238 | :param org_repo: |
|
239 | :param org_repo: | |
239 | :param org_ref: |
|
240 | :param org_ref: | |
240 | :param other_repo: |
|
241 | :param other_repo: | |
241 | :param other_ref: |
|
242 | :param other_ref: | |
242 | """ |
|
243 | """ | |
243 |
|
244 | |||
244 | if len(org_ref) != 2 or not isinstance(org_ref, (list, tuple)): |
|
245 | if len(org_ref) != 2 or not isinstance(org_ref, (list, tuple)): | |
245 | raise Exception('org_ref must be a two element list/tuple') |
|
246 | raise Exception('org_ref must be a two element list/tuple') | |
246 |
|
247 | |||
247 | if len(other_ref) != 2 or not isinstance(org_ref, (list, tuple)): |
|
248 | if len(other_ref) != 2 or not isinstance(org_ref, (list, tuple)): | |
248 | raise Exception('other_ref must be a two element list/tuple') |
|
249 | raise Exception('other_ref must be a two element list/tuple') | |
249 |
|
250 | |||
250 | org_repo_scm = org_repo.scm_instance |
|
251 | org_repo_scm = org_repo.scm_instance | |
251 | other_repo_scm = other_repo.scm_instance |
|
252 | other_repo_scm = other_repo.scm_instance | |
252 |
|
253 | |||
253 | alias = org_repo.scm_instance.alias |
|
254 | alias = org_repo.scm_instance.alias | |
254 | cs_ranges, ancestor = self._get_changesets(alias, |
|
255 | cs_ranges, ancestor = self._get_changesets(alias, | |
255 | org_repo_scm, org_ref, |
|
256 | org_repo_scm, org_ref, | |
256 | other_repo_scm, other_ref) |
|
257 | other_repo_scm, other_ref) | |
257 | return cs_ranges, ancestor |
|
258 | return cs_ranges, ancestor |
@@ -1,41 +1,49 b'' | |||||
1 |
|
1 | |||
2 | <div class="pullrequests_section_head">${_('Opened by me')}</div> |
|
2 | <div class="pullrequests_section_head">${_('Opened by me')}</div> | |
3 | <ul> |
|
3 | <ul> | |
4 | %if c.my_pull_requests: |
|
4 | %if c.my_pull_requests: | |
5 | %for pull_request in c.my_pull_requests: |
|
5 | %for pull_request in c.my_pull_requests: | |
6 | <li> |
|
6 | <li> | |
7 | <div style="height: 12px"> |
|
7 | <div style="height: 12px"> | |
8 | <div style="float:left"> |
|
8 | <div style="float:left"> | |
|
9 | %if pull_request.is_closed(): | |||
|
10 | <img src="${h.url('/images/icons/lock_go.png')}" title="${_('Closed')}"/> | |||
|
11 | %endif | |||
|
12 | <img src="${h.url('/images/icons/flag_status_%s.png' % str(pull_request.last_review_status))}" /> | |||
9 | <a href="${h.url('pullrequest_show',repo_name=pull_request.other_repo.repo_name,pull_request_id=pull_request.pull_request_id)}"> |
|
13 | <a href="${h.url('pullrequest_show',repo_name=pull_request.other_repo.repo_name,pull_request_id=pull_request.pull_request_id)}"> | |
10 | ${_('Pull request #%s opened on %s') % (pull_request.pull_request_id, h.fmt_date(pull_request.created_on))} |
|
14 | ${_('Pull request #%s opened on %s') % (pull_request.pull_request_id, h.fmt_date(pull_request.created_on))} | |
11 | </a> |
|
15 | </a> | |
12 | </div> |
|
16 | </div> | |
13 | <div style="float:left;margin-top: -5px"> |
|
17 | <div style="float:left;margin-top: -5px"> | |
14 | ${h.form(url('pullrequest_delete', repo_name=pull_request.other_repo.repo_name, pull_request_id=pull_request.pull_request_id),method='delete')} |
|
18 | ${h.form(url('pullrequest_delete', repo_name=pull_request.other_repo.repo_name, pull_request_id=pull_request.pull_request_id),method='delete')} | |
15 | ${h.submit('remove_%s' % pull_request.pull_request_id,'',class_="delete_icon action_button",onclick="return confirm('"+_('Confirm to delete this pull request')+"');")} |
|
19 | ${h.submit('remove_%s' % pull_request.pull_request_id,'',class_="delete_icon action_button",onclick="return confirm('"+_('Confirm to delete this pull request')+"');")} | |
16 | ${h.end_form()} |
|
20 | ${h.end_form()} | |
17 | </div> |
|
21 | </div> | |
18 | </div> |
|
22 | </div> | |
19 | </li> |
|
23 | </li> | |
20 | %endfor |
|
24 | %endfor | |
21 | %else: |
|
25 | %else: | |
22 | <li><span class="empty_data">${_('Nothing here yet')}</span></li> |
|
26 | <li><span class="empty_data">${_('Nothing here yet')}</span></li> | |
23 | %endif |
|
27 | %endif | |
24 | </ul> |
|
28 | </ul> | |
25 |
|
29 | |||
26 | <div class="pullrequests_section_head" style="clear:both">${_('I participate in')}</div> |
|
30 | <div class="pullrequests_section_head" style="clear:both">${_('I participate in')}</div> | |
27 | <ul> |
|
31 | <ul> | |
28 | %if c.participate_in_pull_requests: |
|
32 | %if c.participate_in_pull_requests: | |
29 | %for pull_request in c.participate_in_pull_requests: |
|
33 | %for pull_request in c.participate_in_pull_requests: | |
30 | <li> |
|
34 | <li> | |
31 | <div style="height: 12px"> |
|
35 | <div style="height: 12px"> | |
|
36 | %if pull_request.is_closed(): | |||
|
37 | <img src="${h.url('/images/icons/lock_go.png')}" title="${_('Closed')}"/> | |||
|
38 | %endif | |||
|
39 | <img src="${h.url('/images/icons/flag_status_%s.png' % str(pull_request.last_review_status))}" /> | |||
32 | <a href="${h.url('pullrequest_show',repo_name=pull_request.other_repo.repo_name,pull_request_id=pull_request.pull_request_id)}"> |
|
40 | <a href="${h.url('pullrequest_show',repo_name=pull_request.other_repo.repo_name,pull_request_id=pull_request.pull_request_id)}"> | |
33 | ${_('Pull request #%s opened by %s on %s') % (pull_request.pull_request_id, pull_request.author.full_name, h.fmt_date(pull_request.created_on))} |
|
41 | ${_('Pull request #%s opened by %s on %s') % (pull_request.pull_request_id, pull_request.author.full_name, h.fmt_date(pull_request.created_on))} | |
34 | </a> |
|
42 | </a> | |
35 | </div> |
|
43 | </div> | |
36 | </li> |
|
44 | </li> | |
37 | %endfor |
|
45 | %endfor | |
38 | %else: |
|
46 | %else: | |
39 | <li><span class="empty_data">${_('Nothing here yet')}</span></li> |
|
47 | <li><span class="empty_data">${_('Nothing here yet')}</span></li> | |
40 | %endif |
|
48 | %endif | |
41 | </ul> |
|
49 | </ul> |
General Comments 0
You need to be logged in to leave comments.
Login now