##// END OF EJS Templates
pull requests: give slightly more useful error messages on failing form validation
Mads Kiilerich -
r4080:b622e684 default
parent child Browse files
Show More
@@ -1,552 +1,552 b''
1 # -*- coding: utf-8 -*-
1 # -*- coding: utf-8 -*-
2 """
2 """
3 rhodecode.controllers.pullrequests
3 rhodecode.controllers.pullrequests
4 ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
4 ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
5
5
6 pull requests controller for rhodecode for initializing pull requests
6 pull requests controller for rhodecode for initializing pull requests
7
7
8 :created_on: May 7, 2012
8 :created_on: May 7, 2012
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 import logging
25 import logging
26 import traceback
26 import traceback
27 import formencode
27 import formencode
28
28
29 from webob.exc import HTTPNotFound, HTTPForbidden
29 from webob.exc import HTTPNotFound, HTTPForbidden
30 from collections import defaultdict
30 from collections import defaultdict
31 from itertools import groupby
31 from itertools import groupby
32
32
33 from pylons import request, tmpl_context as c, url
33 from pylons import request, tmpl_context as c, url
34 from pylons.controllers.util import redirect
34 from pylons.controllers.util import redirect
35 from pylons.i18n.translation import _
35 from pylons.i18n.translation import _
36
36
37 from rhodecode.lib.compat import json
37 from rhodecode.lib.compat import json
38 from rhodecode.lib.base import BaseRepoController, render
38 from rhodecode.lib.base import BaseRepoController, render
39 from rhodecode.lib.auth import LoginRequired, HasRepoPermissionAnyDecorator,\
39 from rhodecode.lib.auth import LoginRequired, HasRepoPermissionAnyDecorator,\
40 NotAnonymous
40 NotAnonymous
41 from rhodecode.lib.helpers import Page
41 from rhodecode.lib.helpers import Page
42 from rhodecode.lib import helpers as h
42 from rhodecode.lib import helpers as h
43 from rhodecode.lib import diffs
43 from rhodecode.lib import diffs
44 from rhodecode.lib.utils import action_logger, jsonify
44 from rhodecode.lib.utils import action_logger, jsonify
45 from rhodecode.lib.vcs.utils import safe_str
45 from rhodecode.lib.vcs.utils import safe_str
46 from rhodecode.lib.vcs.exceptions import EmptyRepositoryError
46 from rhodecode.lib.vcs.exceptions import EmptyRepositoryError
47 from rhodecode.lib.diffs import LimitedDiffContainer
47 from rhodecode.lib.diffs import LimitedDiffContainer
48 from rhodecode.model.db import PullRequest, ChangesetStatus, ChangesetComment
48 from rhodecode.model.db import PullRequest, ChangesetStatus, ChangesetComment
49 from rhodecode.model.pull_request import PullRequestModel
49 from rhodecode.model.pull_request import PullRequestModel
50 from rhodecode.model.meta import Session
50 from rhodecode.model.meta import Session
51 from rhodecode.model.repo import RepoModel
51 from rhodecode.model.repo import RepoModel
52 from rhodecode.model.comment import ChangesetCommentsModel
52 from rhodecode.model.comment import ChangesetCommentsModel
53 from rhodecode.model.changeset_status import ChangesetStatusModel
53 from rhodecode.model.changeset_status import ChangesetStatusModel
54 from rhodecode.model.forms import PullRequestForm
54 from rhodecode.model.forms import PullRequestForm
55 from rhodecode.lib.utils2 import safe_int
55 from rhodecode.lib.utils2 import safe_int
56
56
57 log = logging.getLogger(__name__)
57 log = logging.getLogger(__name__)
58
58
59
59
60 class PullrequestsController(BaseRepoController):
60 class PullrequestsController(BaseRepoController):
61
61
62 def __before__(self):
62 def __before__(self):
63 super(PullrequestsController, self).__before__()
63 super(PullrequestsController, self).__before__()
64 repo_model = RepoModel()
64 repo_model = RepoModel()
65 c.users_array = repo_model.get_users_js()
65 c.users_array = repo_model.get_users_js()
66 c.users_groups_array = repo_model.get_users_groups_js()
66 c.users_groups_array = repo_model.get_users_groups_js()
67
67
68 def _get_repo_refs(self, repo, rev=None, branch=None, branch_rev=None):
68 def _get_repo_refs(self, repo, rev=None, branch=None, branch_rev=None):
69 """return a structure with repo's interesting changesets, suitable for
69 """return a structure with repo's interesting changesets, suitable for
70 the selectors in pullrequest.html
70 the selectors in pullrequest.html
71
71
72 rev: a revision that must be in the list somehow and selected by default
72 rev: a revision that must be in the list somehow and selected by default
73 branch: a branch that must be in the list and selected by default - even if closed
73 branch: a branch that must be in the list and selected by default - even if closed
74 branch_rev: a revision of which peers should be preferred and available."""
74 branch_rev: a revision of which peers should be preferred and available."""
75 # list named branches that has been merged to this named branch - it should probably merge back
75 # list named branches that has been merged to this named branch - it should probably merge back
76 peers = []
76 peers = []
77
77
78 if rev:
78 if rev:
79 rev = safe_str(rev)
79 rev = safe_str(rev)
80
80
81 if branch:
81 if branch:
82 branch = safe_str(branch)
82 branch = safe_str(branch)
83
83
84 if branch_rev:
84 if branch_rev:
85 branch_rev = safe_str(branch_rev)
85 branch_rev = safe_str(branch_rev)
86 # not restricting to merge() would also get branch point and be better
86 # not restricting to merge() would also get branch point and be better
87 # (especially because it would get the branch point) ... but is currently too expensive
87 # (especially because it would get the branch point) ... but is currently too expensive
88 otherbranches = {}
88 otherbranches = {}
89 for i in repo._repo.revs(
89 for i in repo._repo.revs(
90 "sort(parents(branch(id(%s)) and merge()) - branch(id(%s)))",
90 "sort(parents(branch(id(%s)) and merge()) - branch(id(%s)))",
91 branch_rev, branch_rev):
91 branch_rev, branch_rev):
92 cs = repo.get_changeset(i)
92 cs = repo.get_changeset(i)
93 otherbranches[cs.branch] = cs.raw_id
93 otherbranches[cs.branch] = cs.raw_id
94 for abranch, node in otherbranches.iteritems():
94 for abranch, node in otherbranches.iteritems():
95 selected = 'branch:%s:%s' % (abranch, node)
95 selected = 'branch:%s:%s' % (abranch, node)
96 peers.append((selected, abranch))
96 peers.append((selected, abranch))
97
97
98 selected = None
98 selected = None
99
99
100 branches = []
100 branches = []
101 for abranch, branchrev in repo.branches.iteritems():
101 for abranch, branchrev in repo.branches.iteritems():
102 n = 'branch:%s:%s' % (abranch, branchrev)
102 n = 'branch:%s:%s' % (abranch, branchrev)
103 branches.append((n, abranch))
103 branches.append((n, abranch))
104 if rev == branchrev:
104 if rev == branchrev:
105 selected = n
105 selected = n
106 if branch == abranch:
106 if branch == abranch:
107 selected = n
107 selected = n
108 branch = None
108 branch = None
109 if branch: # branch not in list - it is probably closed
109 if branch: # branch not in list - it is probably closed
110 revs = repo._repo.revs('max(branch(%s))', branch)
110 revs = repo._repo.revs('max(branch(%s))', branch)
111 if revs:
111 if revs:
112 cs = repo.get_changeset(revs[0])
112 cs = repo.get_changeset(revs[0])
113 selected = 'branch:%s:%s' % (branch, cs.raw_id)
113 selected = 'branch:%s:%s' % (branch, cs.raw_id)
114 branches.append((selected, branch))
114 branches.append((selected, branch))
115
115
116 bookmarks = []
116 bookmarks = []
117 for bookmark, bookmarkrev in repo.bookmarks.iteritems():
117 for bookmark, bookmarkrev in repo.bookmarks.iteritems():
118 n = 'book:%s:%s' % (bookmark, bookmarkrev)
118 n = 'book:%s:%s' % (bookmark, bookmarkrev)
119 bookmarks.append((n, bookmark))
119 bookmarks.append((n, bookmark))
120 if rev == bookmarkrev:
120 if rev == bookmarkrev:
121 selected = n
121 selected = n
122
122
123 tags = []
123 tags = []
124 for tag, tagrev in repo.tags.iteritems():
124 for tag, tagrev in repo.tags.iteritems():
125 n = 'tag:%s:%s' % (tag, tagrev)
125 n = 'tag:%s:%s' % (tag, tagrev)
126 tags.append((n, tag))
126 tags.append((n, tag))
127 if rev == tagrev and tag != 'tip': # tip is not a real tag - and its branch is better
127 if rev == tagrev and tag != 'tip': # tip is not a real tag - and its branch is better
128 selected = n
128 selected = n
129
129
130 # prio 1: rev was selected as existing entry above
130 # prio 1: rev was selected as existing entry above
131
131
132 # prio 2: create special entry for rev; rev _must_ be used
132 # prio 2: create special entry for rev; rev _must_ be used
133 specials = []
133 specials = []
134 if rev and selected is None:
134 if rev and selected is None:
135 selected = 'rev:%s:%s' % (rev, rev)
135 selected = 'rev:%s:%s' % (rev, rev)
136 specials = [(selected, '%s: %s' % (_("Changeset"), rev[:12]))]
136 specials = [(selected, '%s: %s' % (_("Changeset"), rev[:12]))]
137
137
138 # prio 3: most recent peer branch
138 # prio 3: most recent peer branch
139 if peers and not selected:
139 if peers and not selected:
140 selected = peers[0][0][0]
140 selected = peers[0][0][0]
141
141
142 # prio 4: tip revision
142 # prio 4: tip revision
143 if not selected:
143 if not selected:
144 selected = 'tag:tip:%s' % repo.tags['tip']
144 selected = 'tag:tip:%s' % repo.tags['tip']
145
145
146 groups = [(specials, _("Special")),
146 groups = [(specials, _("Special")),
147 (peers, _("Peer branches")),
147 (peers, _("Peer branches")),
148 (bookmarks, _("Bookmarks")),
148 (bookmarks, _("Bookmarks")),
149 (branches, _("Branches")),
149 (branches, _("Branches")),
150 (tags, _("Tags")),
150 (tags, _("Tags")),
151 ]
151 ]
152 return [g for g in groups if g[0]], selected
152 return [g for g in groups if g[0]], selected
153
153
154 def _get_is_allowed_change_status(self, pull_request):
154 def _get_is_allowed_change_status(self, pull_request):
155 owner = self.rhodecode_user.user_id == pull_request.user_id
155 owner = self.rhodecode_user.user_id == pull_request.user_id
156 reviewer = self.rhodecode_user.user_id in [x.user_id for x in
156 reviewer = self.rhodecode_user.user_id in [x.user_id for x in
157 pull_request.reviewers]
157 pull_request.reviewers]
158 return (self.rhodecode_user.admin or owner or reviewer)
158 return (self.rhodecode_user.admin or owner or reviewer)
159
159
160 def _load_compare_data(self, pull_request, enable_comments=True):
160 def _load_compare_data(self, pull_request, enable_comments=True):
161 """
161 """
162 Load context data needed for generating compare diff
162 Load context data needed for generating compare diff
163
163
164 :param pull_request:
164 :param pull_request:
165 """
165 """
166 org_repo = pull_request.org_repo
166 org_repo = pull_request.org_repo
167 (org_ref_type,
167 (org_ref_type,
168 org_ref_name,
168 org_ref_name,
169 org_ref_rev) = pull_request.org_ref.split(':')
169 org_ref_rev) = pull_request.org_ref.split(':')
170
170
171 other_repo = org_repo
171 other_repo = org_repo
172 (other_ref_type,
172 (other_ref_type,
173 other_ref_name,
173 other_ref_name,
174 other_ref_rev) = pull_request.other_ref.split(':')
174 other_ref_rev) = pull_request.other_ref.split(':')
175
175
176 # despite opening revisions for bookmarks/branches/tags, we always
176 # despite opening revisions for bookmarks/branches/tags, we always
177 # convert this to rev to prevent changes after bookmark or branch change
177 # convert this to rev to prevent changes after bookmark or branch change
178 org_ref = ('rev', org_ref_rev)
178 org_ref = ('rev', org_ref_rev)
179 other_ref = ('rev', other_ref_rev)
179 other_ref = ('rev', other_ref_rev)
180
180
181 c.org_repo = org_repo
181 c.org_repo = org_repo
182 c.other_repo = other_repo
182 c.other_repo = other_repo
183
183
184 c.fulldiff = fulldiff = request.GET.get('fulldiff')
184 c.fulldiff = fulldiff = request.GET.get('fulldiff')
185
185
186 c.cs_ranges = [org_repo.get_changeset(x) for x in pull_request.revisions]
186 c.cs_ranges = [org_repo.get_changeset(x) for x in pull_request.revisions]
187
187
188 c.statuses = org_repo.statuses([x.raw_id for x in c.cs_ranges])
188 c.statuses = org_repo.statuses([x.raw_id for x in c.cs_ranges])
189
189
190 c.org_ref = org_ref[1]
190 c.org_ref = org_ref[1]
191 c.org_ref_type = org_ref[0]
191 c.org_ref_type = org_ref[0]
192 c.other_ref = other_ref[1]
192 c.other_ref = other_ref[1]
193 c.other_ref_type = other_ref[0]
193 c.other_ref_type = other_ref[0]
194
194
195 diff_limit = self.cut_off_limit if not fulldiff else None
195 diff_limit = self.cut_off_limit if not fulldiff else None
196
196
197 # we swap org/other ref since we run a simple diff on one repo
197 # we swap org/other ref since we run a simple diff on one repo
198 log.debug('running diff between %s and %s in %s'
198 log.debug('running diff between %s and %s in %s'
199 % (other_ref, org_ref, org_repo.scm_instance.path))
199 % (other_ref, org_ref, org_repo.scm_instance.path))
200 txtdiff = org_repo.scm_instance.get_diff(rev1=safe_str(other_ref[1]), rev2=safe_str(org_ref[1]))
200 txtdiff = org_repo.scm_instance.get_diff(rev1=safe_str(other_ref[1]), rev2=safe_str(org_ref[1]))
201
201
202 diff_processor = diffs.DiffProcessor(txtdiff or '', format='gitdiff',
202 diff_processor = diffs.DiffProcessor(txtdiff or '', format='gitdiff',
203 diff_limit=diff_limit)
203 diff_limit=diff_limit)
204 _parsed = diff_processor.prepare()
204 _parsed = diff_processor.prepare()
205
205
206 c.limited_diff = False
206 c.limited_diff = False
207 if isinstance(_parsed, LimitedDiffContainer):
207 if isinstance(_parsed, LimitedDiffContainer):
208 c.limited_diff = True
208 c.limited_diff = True
209
209
210 c.files = []
210 c.files = []
211 c.changes = {}
211 c.changes = {}
212 c.lines_added = 0
212 c.lines_added = 0
213 c.lines_deleted = 0
213 c.lines_deleted = 0
214
214
215 for f in _parsed:
215 for f in _parsed:
216 st = f['stats']
216 st = f['stats']
217 c.lines_added += st['added']
217 c.lines_added += st['added']
218 c.lines_deleted += st['deleted']
218 c.lines_deleted += st['deleted']
219 fid = h.FID('', f['filename'])
219 fid = h.FID('', f['filename'])
220 c.files.append([fid, f['operation'], f['filename'], f['stats']])
220 c.files.append([fid, f['operation'], f['filename'], f['stats']])
221 htmldiff = diff_processor.as_html(enable_comments=enable_comments,
221 htmldiff = diff_processor.as_html(enable_comments=enable_comments,
222 parsed_lines=[f])
222 parsed_lines=[f])
223 c.changes[fid] = [f['operation'], f['filename'], htmldiff]
223 c.changes[fid] = [f['operation'], f['filename'], htmldiff]
224
224
225 @LoginRequired()
225 @LoginRequired()
226 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
226 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
227 'repository.admin')
227 'repository.admin')
228 def show_all(self, repo_name):
228 def show_all(self, repo_name):
229 c.from_ = request.GET.get('from_') or ''
229 c.from_ = request.GET.get('from_') or ''
230 c.closed = request.GET.get('closed') or ''
230 c.closed = request.GET.get('closed') or ''
231 c.pull_requests = PullRequestModel().get_all(repo_name, from_=c.from_, closed=c.closed)
231 c.pull_requests = PullRequestModel().get_all(repo_name, from_=c.from_, closed=c.closed)
232 c.repo_name = repo_name
232 c.repo_name = repo_name
233 p = safe_int(request.GET.get('page', 1), 1)
233 p = safe_int(request.GET.get('page', 1), 1)
234
234
235 c.pullrequests_pager = Page(c.pull_requests, page=p, items_per_page=10)
235 c.pullrequests_pager = Page(c.pull_requests, page=p, items_per_page=10)
236
236
237 c.pullrequest_data = render('/pullrequests/pullrequest_data.html')
237 c.pullrequest_data = render('/pullrequests/pullrequest_data.html')
238
238
239 if request.environ.get('HTTP_X_PARTIAL_XHR'):
239 if request.environ.get('HTTP_X_PARTIAL_XHR'):
240 return c.pullrequest_data
240 return c.pullrequest_data
241
241
242 return render('/pullrequests/pullrequest_show_all.html')
242 return render('/pullrequests/pullrequest_show_all.html')
243
243
244 @LoginRequired()
244 @LoginRequired()
245 @NotAnonymous()
245 @NotAnonymous()
246 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
246 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
247 'repository.admin')
247 'repository.admin')
248 def index(self):
248 def index(self):
249 org_repo = c.rhodecode_db_repo
249 org_repo = c.rhodecode_db_repo
250
250
251 if org_repo.scm_instance.alias != 'hg':
251 if org_repo.scm_instance.alias != 'hg':
252 log.error('Review not available for GIT REPOS')
252 log.error('Review not available for GIT REPOS')
253 raise HTTPNotFound
253 raise HTTPNotFound
254
254
255 try:
255 try:
256 org_repo.scm_instance.get_changeset()
256 org_repo.scm_instance.get_changeset()
257 except EmptyRepositoryError, e:
257 except EmptyRepositoryError, e:
258 h.flash(h.literal(_('There are no changesets yet')),
258 h.flash(h.literal(_('There are no changesets yet')),
259 category='warning')
259 category='warning')
260 redirect(url('summary_home', repo_name=org_repo.repo_name))
260 redirect(url('summary_home', repo_name=org_repo.repo_name))
261
261
262 org_rev = request.GET.get('rev_end')
262 org_rev = request.GET.get('rev_end')
263 # rev_start is not directly useful - its parent could however be used
263 # rev_start is not directly useful - its parent could however be used
264 # as default for other and thus give a simple compare view
264 # as default for other and thus give a simple compare view
265 #other_rev = request.POST.get('rev_start')
265 #other_rev = request.POST.get('rev_start')
266 branch = request.GET.get('branch')
266 branch = request.GET.get('branch')
267
267
268 c.org_repos = []
268 c.org_repos = []
269 c.org_repos.append((org_repo.repo_name, org_repo.repo_name))
269 c.org_repos.append((org_repo.repo_name, org_repo.repo_name))
270 c.default_org_repo = org_repo.repo_name
270 c.default_org_repo = org_repo.repo_name
271 c.org_refs, c.default_org_ref = self._get_repo_refs(org_repo.scm_instance, rev=org_rev, branch=branch)
271 c.org_refs, c.default_org_ref = self._get_repo_refs(org_repo.scm_instance, rev=org_rev, branch=branch)
272
272
273 c.other_repos = []
273 c.other_repos = []
274 other_repos_info = {}
274 other_repos_info = {}
275
275
276 def add_other_repo(repo, branch_rev=None):
276 def add_other_repo(repo, branch_rev=None):
277 if repo.repo_name in other_repos_info: # shouldn't happen
277 if repo.repo_name in other_repos_info: # shouldn't happen
278 return
278 return
279 c.other_repos.append((repo.repo_name, repo.repo_name))
279 c.other_repos.append((repo.repo_name, repo.repo_name))
280 other_refs, selected_other_ref = self._get_repo_refs(repo.scm_instance, branch_rev=branch_rev)
280 other_refs, selected_other_ref = self._get_repo_refs(repo.scm_instance, branch_rev=branch_rev)
281 other_repos_info[repo.repo_name] = {
281 other_repos_info[repo.repo_name] = {
282 'user': dict(user_id=repo.user.user_id,
282 'user': dict(user_id=repo.user.user_id,
283 username=repo.user.username,
283 username=repo.user.username,
284 firstname=repo.user.firstname,
284 firstname=repo.user.firstname,
285 lastname=repo.user.lastname,
285 lastname=repo.user.lastname,
286 gravatar_link=h.gravatar_url(repo.user.email, 14)),
286 gravatar_link=h.gravatar_url(repo.user.email, 14)),
287 'description': repo.description.split('\n', 1)[0],
287 'description': repo.description.split('\n', 1)[0],
288 'revs': h.select('other_ref', selected_other_ref, other_refs, class_='refs')
288 'revs': h.select('other_ref', selected_other_ref, other_refs, class_='refs')
289 }
289 }
290
290
291 # add org repo to other so we can open pull request against peer branches on itself
291 # add org repo to other so we can open pull request against peer branches on itself
292 add_other_repo(org_repo, branch_rev=org_rev)
292 add_other_repo(org_repo, branch_rev=org_rev)
293 c.default_other_repo = org_repo.repo_name
293 c.default_other_repo = org_repo.repo_name
294
294
295 # gather forks and add to this list ... even though it is rare to
295 # gather forks and add to this list ... even though it is rare to
296 # request forks to pull from their parent
296 # request forks to pull from their parent
297 for fork in org_repo.forks:
297 for fork in org_repo.forks:
298 add_other_repo(fork)
298 add_other_repo(fork)
299
299
300 # add parents of this fork also, but only if it's not empty
300 # add parents of this fork also, but only if it's not empty
301 if org_repo.parent and org_repo.parent.scm_instance.revisions:
301 if org_repo.parent and org_repo.parent.scm_instance.revisions:
302 add_other_repo(org_repo.parent)
302 add_other_repo(org_repo.parent)
303 c.default_other_repo = org_repo.parent.repo_name
303 c.default_other_repo = org_repo.parent.repo_name
304
304
305 c.default_other_repo_info = other_repos_info[c.default_other_repo]
305 c.default_other_repo_info = other_repos_info[c.default_other_repo]
306 c.other_repos_info = json.dumps(other_repos_info)
306 c.other_repos_info = json.dumps(other_repos_info)
307
307
308 return render('/pullrequests/pullrequest.html')
308 return render('/pullrequests/pullrequest.html')
309
309
310 @LoginRequired()
310 @LoginRequired()
311 @NotAnonymous()
311 @NotAnonymous()
312 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
312 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
313 'repository.admin')
313 'repository.admin')
314 def create(self, repo_name):
314 def create(self, repo_name):
315 repo = RepoModel()._get_repo(repo_name)
315 repo = RepoModel()._get_repo(repo_name)
316 try:
316 try:
317 _form = PullRequestForm(repo.repo_id)().to_python(request.POST)
317 _form = PullRequestForm(repo.repo_id)().to_python(request.POST)
318 except formencode.Invalid, errors:
318 except formencode.Invalid, errors:
319 log.error(traceback.format_exc())
319 log.error(traceback.format_exc())
320 if errors.error_dict.get('revisions'):
320 if errors.error_dict.get('revisions'):
321 msg = 'Revisions: %s' % errors.error_dict['revisions']
321 msg = 'Revisions: %s' % errors.error_dict['revisions']
322 elif errors.error_dict.get('pullrequest_title'):
322 elif errors.error_dict.get('pullrequest_title'):
323 msg = _('Pull request requires a title with min. 3 chars')
323 msg = _('Pull request requires a title with min. 3 chars')
324 else:
324 else:
325 msg = _('Error creating pull request')
325 msg = _('Error creating pull request: %s') % errors.msg
326
326
327 h.flash(msg, 'error')
327 h.flash(msg, 'error')
328 return redirect(url('pullrequest_home', repo_name=repo_name))
328 return redirect(url('pullrequest_home', repo_name=repo_name)) ## would rather just go back to form ...
329
329
330 org_repo = _form['org_repo']
330 org_repo = _form['org_repo']
331 org_ref = _form['org_ref'] # will end with merge_rev but have symbolic name
331 org_ref = _form['org_ref'] # will end with merge_rev but have symbolic name
332 other_repo = _form['other_repo']
332 other_repo = _form['other_repo']
333 other_ref = 'rev:ancestor:%s' % _form['ancestor_rev'] # could be calculated from other_ref ...
333 other_ref = 'rev:ancestor:%s' % _form['ancestor_rev'] # could be calculated from other_ref ...
334 revisions = [x for x in reversed(_form['revisions'])]
334 revisions = [x for x in reversed(_form['revisions'])]
335 reviewers = _form['review_members']
335 reviewers = _form['review_members']
336
336
337 title = _form['pullrequest_title']
337 title = _form['pullrequest_title']
338 if not title:
338 if not title:
339 title = '%s#%s to %s' % (org_repo, org_ref.split(':', 2)[1], other_repo)
339 title = '%s#%s to %s' % (org_repo, org_ref.split(':', 2)[1], other_repo)
340 description = _form['pullrequest_desc']
340 description = _form['pullrequest_desc']
341 try:
341 try:
342 pull_request = PullRequestModel().create(
342 pull_request = PullRequestModel().create(
343 self.rhodecode_user.user_id, org_repo, org_ref, other_repo,
343 self.rhodecode_user.user_id, org_repo, org_ref, other_repo,
344 other_ref, revisions, reviewers, title, description
344 other_ref, revisions, reviewers, title, description
345 )
345 )
346 Session().commit()
346 Session().commit()
347 h.flash(_('Successfully opened new pull request'),
347 h.flash(_('Successfully opened new pull request'),
348 category='success')
348 category='success')
349 except Exception:
349 except Exception:
350 h.flash(_('Error occurred during sending pull request'),
350 h.flash(_('Error occurred during sending pull request'),
351 category='error')
351 category='error')
352 log.error(traceback.format_exc())
352 log.error(traceback.format_exc())
353 return redirect(url('pullrequest_home', repo_name=repo_name))
353 return redirect(url('pullrequest_home', repo_name=repo_name))
354
354
355 return redirect(url('pullrequest_show', repo_name=other_repo,
355 return redirect(url('pullrequest_show', repo_name=other_repo,
356 pull_request_id=pull_request.pull_request_id))
356 pull_request_id=pull_request.pull_request_id))
357
357
358 @LoginRequired()
358 @LoginRequired()
359 @NotAnonymous()
359 @NotAnonymous()
360 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
360 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
361 'repository.admin')
361 'repository.admin')
362 @jsonify
362 @jsonify
363 def update(self, repo_name, pull_request_id):
363 def update(self, repo_name, pull_request_id):
364 pull_request = PullRequest.get_or_404(pull_request_id)
364 pull_request = PullRequest.get_or_404(pull_request_id)
365 if pull_request.is_closed():
365 if pull_request.is_closed():
366 raise HTTPForbidden()
366 raise HTTPForbidden()
367 #only owner or admin can update it
367 #only owner or admin can update it
368 owner = pull_request.author.user_id == c.rhodecode_user.user_id
368 owner = pull_request.author.user_id == c.rhodecode_user.user_id
369 if h.HasPermissionAny('hg.admin', 'repository.admin')() or owner:
369 if h.HasPermissionAny('hg.admin', 'repository.admin')() or owner:
370 reviewers_ids = map(int, filter(lambda v: v not in [None, ''],
370 reviewers_ids = map(int, filter(lambda v: v not in [None, ''],
371 request.POST.get('reviewers_ids', '').split(',')))
371 request.POST.get('reviewers_ids', '').split(',')))
372
372
373 PullRequestModel().update_reviewers(pull_request_id, reviewers_ids)
373 PullRequestModel().update_reviewers(pull_request_id, reviewers_ids)
374 Session().commit()
374 Session().commit()
375 return True
375 return True
376 raise HTTPForbidden()
376 raise HTTPForbidden()
377
377
378 @LoginRequired()
378 @LoginRequired()
379 @NotAnonymous()
379 @NotAnonymous()
380 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
380 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
381 'repository.admin')
381 'repository.admin')
382 @jsonify
382 @jsonify
383 def delete(self, repo_name, pull_request_id):
383 def delete(self, repo_name, pull_request_id):
384 pull_request = PullRequest.get_or_404(pull_request_id)
384 pull_request = PullRequest.get_or_404(pull_request_id)
385 #only owner can delete it !
385 #only owner can delete it !
386 if pull_request.author.user_id == c.rhodecode_user.user_id:
386 if pull_request.author.user_id == c.rhodecode_user.user_id:
387 PullRequestModel().delete(pull_request)
387 PullRequestModel().delete(pull_request)
388 Session().commit()
388 Session().commit()
389 h.flash(_('Successfully deleted pull request'),
389 h.flash(_('Successfully deleted pull request'),
390 category='success')
390 category='success')
391 return redirect(url('admin_settings_my_account', anchor='pullrequests'))
391 return redirect(url('admin_settings_my_account', anchor='pullrequests'))
392 raise HTTPForbidden()
392 raise HTTPForbidden()
393
393
394 @LoginRequired()
394 @LoginRequired()
395 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
395 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
396 'repository.admin')
396 'repository.admin')
397 def show(self, repo_name, pull_request_id):
397 def show(self, repo_name, pull_request_id):
398 repo_model = RepoModel()
398 repo_model = RepoModel()
399 c.users_array = repo_model.get_users_js()
399 c.users_array = repo_model.get_users_js()
400 c.users_groups_array = repo_model.get_users_groups_js()
400 c.users_groups_array = repo_model.get_users_groups_js()
401 c.pull_request = PullRequest.get_or_404(pull_request_id)
401 c.pull_request = PullRequest.get_or_404(pull_request_id)
402 c.allowed_to_change_status = self._get_is_allowed_change_status(c.pull_request)
402 c.allowed_to_change_status = self._get_is_allowed_change_status(c.pull_request)
403 cc_model = ChangesetCommentsModel()
403 cc_model = ChangesetCommentsModel()
404 cs_model = ChangesetStatusModel()
404 cs_model = ChangesetStatusModel()
405 _cs_statuses = cs_model.get_statuses(c.pull_request.org_repo,
405 _cs_statuses = cs_model.get_statuses(c.pull_request.org_repo,
406 pull_request=c.pull_request,
406 pull_request=c.pull_request,
407 with_revisions=True)
407 with_revisions=True)
408
408
409 cs_statuses = defaultdict(list)
409 cs_statuses = defaultdict(list)
410 for st in _cs_statuses:
410 for st in _cs_statuses:
411 cs_statuses[st.author.username] += [st]
411 cs_statuses[st.author.username] += [st]
412
412
413 c.pull_request_reviewers = []
413 c.pull_request_reviewers = []
414 c.pull_request_pending_reviewers = []
414 c.pull_request_pending_reviewers = []
415 for o in c.pull_request.reviewers:
415 for o in c.pull_request.reviewers:
416 st = cs_statuses.get(o.user.username, None)
416 st = cs_statuses.get(o.user.username, None)
417 if st:
417 if st:
418 sorter = lambda k: k.version
418 sorter = lambda k: k.version
419 st = [(x, list(y)[0])
419 st = [(x, list(y)[0])
420 for x, y in (groupby(sorted(st, key=sorter), sorter))]
420 for x, y in (groupby(sorted(st, key=sorter), sorter))]
421 else:
421 else:
422 c.pull_request_pending_reviewers.append(o.user)
422 c.pull_request_pending_reviewers.append(o.user)
423 c.pull_request_reviewers.append([o.user, st])
423 c.pull_request_reviewers.append([o.user, st])
424
424
425 # pull_requests repo_name we opened it against
425 # pull_requests repo_name we opened it against
426 # ie. other_repo must match
426 # ie. other_repo must match
427 if repo_name != c.pull_request.other_repo.repo_name:
427 if repo_name != c.pull_request.other_repo.repo_name:
428 raise HTTPNotFound
428 raise HTTPNotFound
429
429
430 # load compare data into template context
430 # load compare data into template context
431 enable_comments = not c.pull_request.is_closed()
431 enable_comments = not c.pull_request.is_closed()
432 self._load_compare_data(c.pull_request, enable_comments=enable_comments)
432 self._load_compare_data(c.pull_request, enable_comments=enable_comments)
433
433
434 # inline comments
434 # inline comments
435 c.inline_cnt = 0
435 c.inline_cnt = 0
436 c.inline_comments = cc_model.get_inline_comments(
436 c.inline_comments = cc_model.get_inline_comments(
437 c.rhodecode_db_repo.repo_id,
437 c.rhodecode_db_repo.repo_id,
438 pull_request=pull_request_id)
438 pull_request=pull_request_id)
439 # count inline comments
439 # count inline comments
440 for __, lines in c.inline_comments:
440 for __, lines in c.inline_comments:
441 for comments in lines.values():
441 for comments in lines.values():
442 c.inline_cnt += len(comments)
442 c.inline_cnt += len(comments)
443 # comments
443 # comments
444 c.comments = cc_model.get_comments(c.rhodecode_db_repo.repo_id,
444 c.comments = cc_model.get_comments(c.rhodecode_db_repo.repo_id,
445 pull_request=pull_request_id)
445 pull_request=pull_request_id)
446
446
447 # (badly named) pull-request status calculation based on reviewer votes
447 # (badly named) pull-request status calculation based on reviewer votes
448 c.current_changeset_status = cs_model.calculate_status(
448 c.current_changeset_status = cs_model.calculate_status(
449 c.pull_request_reviewers,
449 c.pull_request_reviewers,
450 )
450 )
451 c.changeset_statuses = ChangesetStatus.STATUSES
451 c.changeset_statuses = ChangesetStatus.STATUSES
452
452
453 c.as_form = False
453 c.as_form = False
454 c.ancestor = None # there is one - but right here we don't know which
454 c.ancestor = None # there is one - but right here we don't know which
455 return render('/pullrequests/pullrequest_show.html')
455 return render('/pullrequests/pullrequest_show.html')
456
456
457 @LoginRequired()
457 @LoginRequired()
458 @NotAnonymous()
458 @NotAnonymous()
459 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
459 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
460 'repository.admin')
460 'repository.admin')
461 @jsonify
461 @jsonify
462 def comment(self, repo_name, pull_request_id):
462 def comment(self, repo_name, pull_request_id):
463 pull_request = PullRequest.get_or_404(pull_request_id)
463 pull_request = PullRequest.get_or_404(pull_request_id)
464 if pull_request.is_closed():
464 if pull_request.is_closed():
465 raise HTTPForbidden()
465 raise HTTPForbidden()
466
466
467 status = request.POST.get('changeset_status')
467 status = request.POST.get('changeset_status')
468 change_status = request.POST.get('change_changeset_status')
468 change_status = request.POST.get('change_changeset_status')
469 text = request.POST.get('text')
469 text = request.POST.get('text')
470 close_pr = request.POST.get('save_close')
470 close_pr = request.POST.get('save_close')
471
471
472 allowed_to_change_status = self._get_is_allowed_change_status(pull_request)
472 allowed_to_change_status = self._get_is_allowed_change_status(pull_request)
473 if status and change_status and allowed_to_change_status:
473 if status and change_status and allowed_to_change_status:
474 _def = (_('Status change -> %s')
474 _def = (_('Status change -> %s')
475 % ChangesetStatus.get_status_lbl(status))
475 % ChangesetStatus.get_status_lbl(status))
476 if close_pr:
476 if close_pr:
477 _def = _('Closing with') + ' ' + _def
477 _def = _('Closing with') + ' ' + _def
478 text = text or _def
478 text = text or _def
479 comm = ChangesetCommentsModel().create(
479 comm = ChangesetCommentsModel().create(
480 text=text,
480 text=text,
481 repo=c.rhodecode_db_repo.repo_id,
481 repo=c.rhodecode_db_repo.repo_id,
482 user=c.rhodecode_user.user_id,
482 user=c.rhodecode_user.user_id,
483 pull_request=pull_request_id,
483 pull_request=pull_request_id,
484 f_path=request.POST.get('f_path'),
484 f_path=request.POST.get('f_path'),
485 line_no=request.POST.get('line'),
485 line_no=request.POST.get('line'),
486 status_change=(ChangesetStatus.get_status_lbl(status)
486 status_change=(ChangesetStatus.get_status_lbl(status)
487 if status and change_status
487 if status and change_status
488 and allowed_to_change_status else None),
488 and allowed_to_change_status else None),
489 closing_pr=close_pr
489 closing_pr=close_pr
490 )
490 )
491
491
492 action_logger(self.rhodecode_user,
492 action_logger(self.rhodecode_user,
493 'user_commented_pull_request:%s' % pull_request_id,
493 'user_commented_pull_request:%s' % pull_request_id,
494 c.rhodecode_db_repo, self.ip_addr, self.sa)
494 c.rhodecode_db_repo, self.ip_addr, self.sa)
495
495
496 if allowed_to_change_status:
496 if allowed_to_change_status:
497 # get status if set !
497 # get status if set !
498 if status and change_status:
498 if status and change_status:
499 ChangesetStatusModel().set_status(
499 ChangesetStatusModel().set_status(
500 c.rhodecode_db_repo.repo_id,
500 c.rhodecode_db_repo.repo_id,
501 status,
501 status,
502 c.rhodecode_user.user_id,
502 c.rhodecode_user.user_id,
503 comm,
503 comm,
504 pull_request=pull_request_id
504 pull_request=pull_request_id
505 )
505 )
506
506
507 if close_pr:
507 if close_pr:
508 if status in ['rejected', 'approved']:
508 if status in ['rejected', 'approved']:
509 PullRequestModel().close_pull_request(pull_request_id)
509 PullRequestModel().close_pull_request(pull_request_id)
510 action_logger(self.rhodecode_user,
510 action_logger(self.rhodecode_user,
511 'user_closed_pull_request:%s' % pull_request_id,
511 'user_closed_pull_request:%s' % pull_request_id,
512 c.rhodecode_db_repo, self.ip_addr, self.sa)
512 c.rhodecode_db_repo, self.ip_addr, self.sa)
513 else:
513 else:
514 h.flash(_('Closing pull request on other statuses than '
514 h.flash(_('Closing pull request on other statuses than '
515 'rejected or approved forbidden'),
515 'rejected or approved forbidden'),
516 category='warning')
516 category='warning')
517
517
518 Session().commit()
518 Session().commit()
519
519
520 if not request.environ.get('HTTP_X_PARTIAL_XHR'):
520 if not request.environ.get('HTTP_X_PARTIAL_XHR'):
521 return redirect(h.url('pullrequest_show', repo_name=repo_name,
521 return redirect(h.url('pullrequest_show', repo_name=repo_name,
522 pull_request_id=pull_request_id))
522 pull_request_id=pull_request_id))
523
523
524 data = {
524 data = {
525 'target_id': h.safeid(h.safe_unicode(request.POST.get('f_path'))),
525 'target_id': h.safeid(h.safe_unicode(request.POST.get('f_path'))),
526 }
526 }
527 if comm:
527 if comm:
528 c.co = comm
528 c.co = comm
529 data.update(comm.get_dict())
529 data.update(comm.get_dict())
530 data.update({'rendered_text':
530 data.update({'rendered_text':
531 render('changeset/changeset_comment_block.html')})
531 render('changeset/changeset_comment_block.html')})
532
532
533 return data
533 return data
534
534
535 @LoginRequired()
535 @LoginRequired()
536 @NotAnonymous()
536 @NotAnonymous()
537 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
537 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
538 'repository.admin')
538 'repository.admin')
539 @jsonify
539 @jsonify
540 def delete_comment(self, repo_name, comment_id):
540 def delete_comment(self, repo_name, comment_id):
541 co = ChangesetComment.get(comment_id)
541 co = ChangesetComment.get(comment_id)
542 if co.pull_request.is_closed():
542 if co.pull_request.is_closed():
543 #don't allow deleting comments on closed pull request
543 #don't allow deleting comments on closed pull request
544 raise HTTPForbidden()
544 raise HTTPForbidden()
545
545
546 owner = co.author.user_id == c.rhodecode_user.user_id
546 owner = co.author.user_id == c.rhodecode_user.user_id
547 if h.HasPermissionAny('hg.admin', 'repository.admin')() or owner:
547 if h.HasPermissionAny('hg.admin', 'repository.admin')() or owner:
548 ChangesetCommentsModel().delete(comment=co)
548 ChangesetCommentsModel().delete(comment=co)
549 Session().commit()
549 Session().commit()
550 return True
550 return True
551 else:
551 else:
552 raise HTTPForbidden()
552 raise HTTPForbidden()
General Comments 0
You need to be logged in to leave comments. Login now