changeset.py
407 lines
| 15.4 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 | ||||
r1776 | from collections import defaultdict | |||
from webob.exc import HTTPForbidden | ||||
r812 | ||||
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 | ||||
r1776 | from vcs.exceptions import RepositoryError, ChangesetError, \ | |||
ChangesetDoesNotExistError | ||||
from vcs.nodes import FileNode | ||||
r812 | import rhodecode.lib.helpers as h | |||
r1712 | from rhodecode.lib.auth import LoginRequired, HasRepoPermissionAnyDecorator | |||
r1045 | from rhodecode.lib.base import BaseRepoController, render | |||
r649 | from rhodecode.lib.utils import EmptyChangeset | |||
r1514 | from rhodecode.lib.compat import OrderedDict | |||
r1753 | from rhodecode.lib import diffs | |||
r1670 | from rhodecode.model.db import ChangesetComment | |||
from rhodecode.model.comment import ChangesetCommentsModel | ||||
r1713 | from rhodecode.model.meta import Session | |||
r547 | ||||
log = logging.getLogger(__name__) | ||||
r1212 | ||||
r1776 | def anchor_url(revision,path): | |||
fid = h.FID(revision, path) | ||||
return h.url.current(anchor=fid,**request.GET) | ||||
def get_ignore_ws(fid, GET): | ||||
ig_ws_global = request.GET.get('ignorews') | ||||
ig_ws = filter(lambda k:k.startswith('WS'),GET.getall(fid)) | ||||
if ig_ws: | ||||
try: | ||||
return int(ig_ws[0].split(':')[-1]) | ||||
except: | ||||
pass | ||||
return ig_ws_global | ||||
def _ignorews_url(fileid=None): | ||||
params = defaultdict(list) | ||||
lbl = _('show white space') | ||||
ig_ws = get_ignore_ws(fileid, request.GET) | ||||
ln_ctx = get_line_ctx(fileid, request.GET) | ||||
# global option | ||||
if fileid is None: | ||||
if ig_ws is None: | ||||
params['ignorews'] += [1] | ||||
lbl = _('ignore white space') | ||||
ctx_key = 'context' | ||||
ctx_val = ln_ctx | ||||
# per file options | ||||
else: | ||||
if ig_ws is None: | ||||
params[fileid] += ['WS:1'] | ||||
lbl = _('ignore white space') | ||||
ctx_key = fileid | ||||
ctx_val = 'C:%s' % ln_ctx | ||||
# if we have passed in ln_ctx pass it along to our params | ||||
if ln_ctx: | ||||
params[ctx_key] += [ctx_val] | ||||
params['anchor'] = fileid | ||||
return h.link_to(lbl, h.url.current(**params)) | ||||
def get_line_ctx(fid, GET): | ||||
ln_ctx_global = request.GET.get('context') | ||||
ln_ctx = filter(lambda k:k.startswith('C'),GET.getall(fid)) | ||||
if ln_ctx: | ||||
retval = ln_ctx[0].split(':')[-1] | ||||
else: | ||||
retval = ln_ctx_global | ||||
try: | ||||
return int(retval) | ||||
except: | ||||
return | ||||
def _context_url(fileid=None): | ||||
""" | ||||
Generates url for context lines | ||||
:param fileid: | ||||
""" | ||||
ig_ws = get_ignore_ws(fileid, request.GET) | ||||
ln_ctx = (get_line_ctx(fileid, request.GET) or 3) * 2 | ||||
params = defaultdict(list) | ||||
# global option | ||||
if fileid is None: | ||||
if ln_ctx > 0: | ||||
params['context'] += [ln_ctx] | ||||
if ig_ws: | ||||
ig_ws_key = 'ignorews' | ||||
ig_ws_val = 1 | ||||
# per file option | ||||
else: | ||||
params[fileid] += ['C:%s' % ln_ctx] | ||||
ig_ws_key = fileid | ||||
ig_ws_val = 'WS:%s' % 1 | ||||
if ig_ws: | ||||
params[ig_ws_key] += [ig_ws_val] | ||||
lbl = _('%s line context') % ln_ctx | ||||
params['anchor'] = fileid | ||||
return h.link_to(lbl, h.url.current(**params)) | ||||
def wrap_to_table(str_): | ||||
return '''<table class="code-difftable"> | ||||
<tr class="line"> | ||||
<td class="lineno new"></td> | ||||
<td class="code"><pre>%s</pre></td> | ||||
</tr> | ||||
</table>''' % str_ | ||||
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 | ||||
r1776 | c.anchor_url = anchor_url | |||
c.ignorews_url = _ignorews_url | ||||
c.context_url = _context_url | ||||
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 | |||
r1776 | fid = h.FID(revision, node.path) | |||
line_context_lcl = get_line_ctx(fid, request.GET) | ||||
ignore_whitespace_lcl = get_ignore_ws(fid, request.GET) | ||||
r812 | if c.sum_added < self.cut_off_limit: | |||
r1753 | f_gitdiff = diffs.get_gitdiff(filenode_old, node, | |||
r1776 | ignore_whitespace=ignore_whitespace_lcl, | |||
context=line_context_lcl) | ||||
r1753 | d = diffs.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: | |||
r1776 | fid = h.FID(revision, node.path) | |||
line_context_lcl = get_line_ctx(fid, request.GET) | ||||
ignore_whitespace_lcl = get_ignore_ws(fid, request.GET,) | ||||
r1753 | f_gitdiff = diffs.get_gitdiff(filenode_old, node, | |||
r1776 | ignore_whitespace=ignore_whitespace_lcl, | |||
context=line_context_lcl) | ||||
r1753 | d = diffs.DiffProcessor(f_gitdiff, | |||
r1257 | 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') | |||
r1752 | ignore_whitespace = request.GET.get('ignorews') == '1' | |||
r1768 | line_context = request.GET.get('context', 3) | |||
r547 | 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: | |||
r1753 | f_gitdiff = diffs.get_gitdiff(filenode_old, node, | |||
r1768 | ignore_whitespace=ignore_whitespace, | |||
context=line_context) | ||||
r1753 | diff = diffs.DiffProcessor(f_gitdiff, | |||
r1212 | 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: | |||
r1753 | f_gitdiff = diffs.get_gitdiff(filenode_old, node, | |||
r1768 | ignore_whitespace=ignore_whitespace, | |||
context=line_context) | ||||
r1753 | diff = diffs.DiffProcessor(f_gitdiff, | |||
r1212 | 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): | ||||
r1712 | ChangesetCommentsModel().create(text=request.POST.get('text'), | |||
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')) | ||||
r1749 | Session.commit() | |||
r1670 | return redirect(h.url('changeset_home', repo_name=repo_name, | |||
revision=revision)) | ||||
@jsonify | ||||
r1716 | def delete_comment(self, repo_name, comment_id): | |||
r1674 | co = ChangesetComment.get(comment_id) | |||
r1712 | owner = lambda : co.author.user_id == c.rhodecode_user.user_id | |||
if h.HasPermissionAny('hg.admin', 'repository.admin')() or owner: | ||||
r1713 | ChangesetCommentsModel().delete(comment=co) | |||
r1749 | Session.commit() | |||
r1674 | return True | |||
else: | ||||
raise HTTPForbidden() | ||||