changeset.py
298 lines
| 11.9 KiB
| text/x-python
|
PythonLexer
r812 | # -*- coding: utf-8 -*- | |||
""" | ||||
rhodecode.controllers.changeset | ||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||
r1203 | changeset controller for pylons showoing changes beetween | |||
r977 | revisions | |||
r1203 | ||||
r812 | :created_on: Apr 25, 2010 | |||
:author: marcink | ||||
r1203 | :copyright: (C) 2009-2011 Marcin Kuzminski <marcin@python-works.com> | |||
r812 | :license: GPLv3, see COPYING for more details. | |||
""" | ||||
r1206 | # This program is free software: you can redistribute it and/or modify | |||
# it under the terms of the GNU General Public License as published by | ||||
# the Free Software Foundation, either version 3 of the License, or | ||||
# (at your option) any later version. | ||||
r1203 | # | |||
r547 | # This program is distributed in the hope that it will be useful, | |||
# but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||
# GNU General Public License for more details. | ||||
r1203 | # | |||
r547 | # You should have received a copy of the GNU General Public License | |||
r1206 | # along with this program. If not, see <http://www.gnu.org/licenses/>. | |||
r812 | import logging | |||
import traceback | ||||
r547 | from pylons import tmpl_context as c, url, request, response | |||
from pylons.i18n.translation import _ | ||||
from pylons.controllers.util import redirect | ||||
r1670 | from pylons.decorators import jsonify | |||
r812 | ||||
import rhodecode.lib.helpers as h | ||||
r1670 | from rhodecode.lib.auth import LoginRequired, HasRepoPermissionAnyDecorator, \ | |||
NotAnonymous | ||||
r1045 | from rhodecode.lib.base import BaseRepoController, render | |||
r649 | from rhodecode.lib.utils import EmptyChangeset | |||
r1514 | from rhodecode.lib.compat import OrderedDict | |||
r1670 | from rhodecode.model.db import ChangesetComment | |||
from rhodecode.model.comment import ChangesetCommentsModel | ||||
r812 | ||||
r978 | from vcs.exceptions import RepositoryError, ChangesetError, \ | |||
r1670 | ChangesetDoesNotExistError | |||
r547 | from vcs.nodes import FileNode | |||
from vcs.utils import diffs as differ | ||||
r1674 | from webob.exc import HTTPForbidden | |||
r547 | ||||
log = logging.getLogger(__name__) | ||||
r1212 | ||||
r1045 | class ChangesetController(BaseRepoController): | |||
r636 | ||||
r547 | @LoginRequired() | |||
@HasRepoPermissionAnyDecorator('repository.read', 'repository.write', | ||||
r636 | 'repository.admin') | |||
r547 | def __before__(self): | |||
super(ChangesetController, self).__before__() | ||||
r1130 | c.affected_files_cut_off = 60 | |||
r636 | ||||
r547 | def index(self, revision): | |||
r636 | ||||
r547 | def wrap_to_table(str): | |||
r636 | ||||
r547 | return '''<table class="code-difftable"> | |||
<tr class="line"> | ||||
<td class="lineno new"></td> | ||||
<td class="code"><pre>%s</pre></td> | ||||
</tr> | ||||
</table>''' % str | ||||
r636 | ||||
r977 | #get ranges of revisions if preset | |||
rev_range = revision.split('...')[:2] | ||||
r1670 | ||||
r547 | try: | |||
r977 | if len(rev_range) == 2: | |||
rev_start = rev_range[0] | ||||
rev_end = rev_range[1] | ||||
r1107 | rev_ranges = c.rhodecode_repo.get_changesets(start=rev_start, | |||
end=rev_end) | ||||
r977 | else: | |||
r1108 | rev_ranges = [c.rhodecode_repo.get_changeset(revision)] | |||
r983 | ||||
c.cs_ranges = list(rev_ranges) | ||||
r1656 | if not c.cs_ranges: | |||
raise RepositoryError('Changeset range returned empty result') | ||||
r983 | ||||
r978 | except (RepositoryError, ChangesetDoesNotExistError, Exception), e: | |||
r547 | log.error(traceback.format_exc()) | |||
r644 | h.flash(str(e), category='warning') | |||
r636 | return redirect(url('home')) | |||
r977 | ||||
c.changes = OrderedDict() | ||||
c.sum_added = 0 | ||||
c.sum_removed = 0 | ||||
r1257 | c.lines_added = 0 | |||
c.lines_deleted = 0 | ||||
r1280 | c.cut_off = False # defines if cut off limit is reached | |||
r636 | ||||
r1670 | c.comments = [] | |||
r1677 | c.inline_comments = [] | |||
c.inline_cnt = 0 | ||||
r1274 | # Iterate over ranges (default changeset view is always one changeset) | |||
r977 | for changeset in c.cs_ranges: | |||
r1675 | c.comments.extend(ChangesetCommentsModel()\ | |||
.get_comments(c.rhodecode_db_repo.repo_id, | ||||
changeset.raw_id)) | ||||
r1677 | inlines = ChangesetCommentsModel()\ | |||
.get_inline_comments(c.rhodecode_db_repo.repo_id, | ||||
changeset.raw_id) | ||||
c.inline_comments.extend(inlines) | ||||
r977 | c.changes[changeset.raw_id] = [] | |||
try: | ||||
changeset_parent = changeset.parents[0] | ||||
except IndexError: | ||||
changeset_parent = None | ||||
#================================================================== | ||||
r547 | # ADDED FILES | |||
r977 | #================================================================== | |||
for node in changeset.added: | ||||
r1274 | ||||
r566 | filenode_old = FileNode(node.path, '', EmptyChangeset()) | |||
r547 | if filenode_old.is_binary or node.is_binary: | |||
diff = wrap_to_table(_('binary file')) | ||||
r1259 | st = (0, 0) | |||
r547 | else: | |||
r1274 | # in this case node.size is good parameter since those are | |||
# added nodes and their size defines how many changes were | ||||
# made | ||||
r547 | c.sum_added += node.size | |||
r812 | if c.sum_added < self.cut_off_limit: | |||
r1125 | f_gitdiff = differ.get_gitdiff(filenode_old, node) | |||
r1257 | d = differ.DiffProcessor(f_gitdiff, format='gitdiff') | |||
r1274 | ||||
st = d.stat() | ||||
r1257 | diff = d.as_html() | |||
r1274 | ||||
r547 | else: | |||
r1212 | diff = wrap_to_table(_('Changeset is to big and ' | |||
'was cut off, see raw ' | ||||
'changeset instead')) | ||||
r1130 | c.cut_off = True | |||
break | ||||
r636 | ||||
r547 | cs1 = None | |||
r636 | cs2 = node.last_changeset.raw_id | |||
r1257 | c.lines_added += st[0] | |||
c.lines_deleted += st[1] | ||||
c.changes[changeset.raw_id].append(('added', node, diff, | ||||
cs1, cs2, st)) | ||||
r636 | ||||
r977 | #================================================================== | |||
r547 | # CHANGED FILES | |||
r977 | #================================================================== | |||
r1130 | if not c.cut_off: | |||
for node in changeset.changed: | ||||
try: | ||||
filenode_old = changeset_parent.get_node(node.path) | ||||
except ChangesetError: | ||||
r1274 | log.warning('Unable to fetch parent node for diff') | |||
r1212 | filenode_old = FileNode(node.path, '', | |||
EmptyChangeset()) | ||||
r636 | ||||
r1130 | if filenode_old.is_binary or node.is_binary: | |||
diff = wrap_to_table(_('binary file')) | ||||
r1259 | st = (0, 0) | |||
r1130 | else: | |||
r636 | ||||
r1130 | if c.sum_removed < self.cut_off_limit: | |||
f_gitdiff = differ.get_gitdiff(filenode_old, node) | ||||
r1257 | d = differ.DiffProcessor(f_gitdiff, | |||
format='gitdiff') | ||||
r1259 | st = d.stat() | |||
r1274 | if (st[0] + st[1]) * 256 > self.cut_off_limit: | |||
diff = wrap_to_table(_('Diff is to big ' | ||||
'and was cut off, see ' | ||||
'raw diff instead')) | ||||
else: | ||||
diff = d.as_html() | ||||
r1130 | if diff: | |||
c.sum_removed += len(diff) | ||||
else: | ||||
r1212 | diff = wrap_to_table(_('Changeset is to big and ' | |||
'was cut off, see raw ' | ||||
'changeset instead')) | ||||
r1130 | c.cut_off = True | |||
break | ||||
r636 | ||||
r1130 | cs1 = filenode_old.last_changeset.raw_id | |||
cs2 = node.last_changeset.raw_id | ||||
r1257 | c.lines_added += st[0] | |||
c.lines_deleted += st[1] | ||||
c.changes[changeset.raw_id].append(('changed', node, diff, | ||||
cs1, cs2, st)) | ||||
r636 | ||||
r977 | #================================================================== | |||
r1203 | # REMOVED FILES | |||
r977 | #================================================================== | |||
r1130 | if not c.cut_off: | |||
for node in changeset.removed: | ||||
r1212 | c.changes[changeset.raw_id].append(('removed', node, None, | |||
r1259 | None, None, (0, 0))) | |||
r636 | ||||
r1677 | # count inline comments | |||
for path, lines in c.inline_comments: | ||||
for comments in lines.values(): | ||||
c.inline_cnt += len(comments) | ||||
r977 | if len(c.cs_ranges) == 1: | |||
c.changeset = c.cs_ranges[0] | ||||
c.changes = c.changes[c.changeset.raw_id] | ||||
return render('changeset/changeset.html') | ||||
else: | ||||
return render('changeset/changeset_range.html') | ||||
r547 | ||||
def raw_changeset(self, revision): | ||||
r636 | ||||
r547 | method = request.GET.get('diff', 'show') | |||
try: | ||||
r1045 | c.scm_type = c.rhodecode_repo.alias | |||
c.changeset = c.rhodecode_repo.get_changeset(revision) | ||||
r547 | except RepositoryError: | |||
log.error(traceback.format_exc()) | ||||
r636 | return redirect(url('home')) | |||
r547 | else: | |||
try: | ||||
r977 | c.changeset_parent = c.changeset.parents[0] | |||
r547 | except IndexError: | |||
r977 | c.changeset_parent = None | |||
r547 | c.changes = [] | |||
r636 | ||||
r547 | for node in c.changeset.added: | |||
filenode_old = FileNode(node.path, '') | ||||
if filenode_old.is_binary or node.is_binary: | ||||
r812 | diff = _('binary file') + '\n' | |||
r636 | else: | |||
r1125 | f_gitdiff = differ.get_gitdiff(filenode_old, node) | |||
r1212 | diff = differ.DiffProcessor(f_gitdiff, | |||
format='gitdiff').raw_diff() | ||||
r547 | ||||
cs1 = None | ||||
r636 | cs2 = node.last_changeset.raw_id | |||
r547 | c.changes.append(('added', node, diff, cs1, cs2)) | |||
r636 | ||||
r547 | for node in c.changeset.changed: | |||
r977 | filenode_old = c.changeset_parent.get_node(node.path) | |||
r547 | if filenode_old.is_binary or node.is_binary: | |||
diff = _('binary file') | ||||
r636 | else: | |||
r1125 | f_gitdiff = differ.get_gitdiff(filenode_old, node) | |||
r1212 | diff = differ.DiffProcessor(f_gitdiff, | |||
format='gitdiff').raw_diff() | ||||
r547 | ||||
r636 | cs1 = filenode_old.last_changeset.raw_id | |||
cs2 = node.last_changeset.raw_id | ||||
c.changes.append(('changed', node, diff, cs1, cs2)) | ||||
r547 | response.content_type = 'text/plain' | |||
r812 | ||||
r547 | if method == 'download': | |||
r1212 | response.content_disposition = 'attachment; filename=%s.patch' \ | |||
% revision | ||||
r812 | ||||
r1402 | c.parent_tmpl = ''.join(['# Parent %s\n' % x.raw_id for x in | |||
c.changeset.parents]) | ||||
r636 | ||||
r547 | c.diffs = '' | |||
for x in c.changes: | ||||
c.diffs += x[2] | ||||
r812 | ||||
r547 | return render('changeset/raw_changeset.html') | |||
r1670 | ||||
def comment(self, repo_name, revision): | ||||
ccmodel = ChangesetCommentsModel() | ||||
ccmodel.create(text=request.POST.get('text'), | ||||
r1676 | repo_id=c.rhodecode_db_repo.repo_id, | |||
user_id=c.rhodecode_user.user_id, | ||||
revision=revision, f_path=request.POST.get('f_path'), | ||||
line_no=request.POST.get('line')) | ||||
r1670 | ||||
return redirect(h.url('changeset_home', repo_name=repo_name, | ||||
revision=revision)) | ||||
@jsonify | ||||
def delete_comment(self, comment_id): | ||||
r1674 | co = ChangesetComment.get(comment_id) | |||
if (h.HasPermissionAny('hg.admin', 'repository.admin')() or | ||||
co.author.user_id == c.rhodecode_user.user_id): | ||||
ccmodel = ChangesetCommentsModel() | ||||
ccmodel.delete(comment_id=comment_id) | ||||
return True | ||||
else: | ||||
raise HTTPForbidden() | ||||