##// END OF EJS Templates
fixed binary file issues
marcink -
r1043:6ec53c16 beta
parent child Browse files
Show More
@@ -1,267 +1,271 b''
1 1 # -*- coding: utf-8 -*-
2 2 """
3 3 rhodecode.controllers.files
4 4 ~~~~~~~~~~~~~~~~~~~~~~~~~~~
5 5
6 6 Files controller for RhodeCode
7 7
8 8 :created_on: Apr 21, 2010
9 9 :author: marcink
10 10 :copyright: (C) 2009-2011 Marcin Kuzminski <marcin@python-works.com>
11 11 :license: GPLv3, see COPYING for more details.
12 12 """
13 13 # This program is free software; you can redistribute it and/or
14 14 # modify it under the terms of the GNU General Public License
15 15 # as published by the Free Software Foundation; version 2
16 16 # of the License or (at your opinion) any later version of the license.
17 17 #
18 18 # This program is distributed in the hope that it will be useful,
19 19 # but WITHOUT ANY WARRANTY; without even the implied warranty of
20 20 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21 21 # GNU General Public License for more details.
22 22 #
23 23 # You should have received a copy of the GNU General Public License
24 24 # along with this program; if not, write to the Free Software
25 25 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
26 26 # MA 02110-1301, USA.
27 27 import tempfile
28 28 import logging
29 29 import rhodecode.lib.helpers as h
30 30
31 31 from mercurial import archival
32 32
33 33 from pylons import request, response, session, tmpl_context as c, url
34 34 from pylons.i18n.translation import _
35 35 from pylons.controllers.util import redirect
36 36
37 37 from rhodecode.lib.auth import LoginRequired, HasRepoPermissionAnyDecorator
38 38 from rhodecode.lib.base import BaseController, render
39 39 from rhodecode.lib.utils import EmptyChangeset
40 40 from rhodecode.model.scm import ScmModel
41 41
42 42 from vcs.backends import ARCHIVE_SPECS
43 43 from vcs.exceptions import RepositoryError, ChangesetError, \
44 44 ChangesetDoesNotExistError, EmptyRepositoryError, ImproperArchiveTypeError
45 45 from vcs.nodes import FileNode
46 46 from vcs.utils import diffs as differ
47 47
48 48 log = logging.getLogger(__name__)
49 49
50 50 class FilesController(BaseController):
51 51
52 52 @LoginRequired()
53 53 @HasRepoPermissionAnyDecorator('repository.read', 'repository.write',
54 54 'repository.admin')
55 55 def __before__(self):
56 56 super(FilesController, self).__before__()
57 57 c.cut_off_limit = self.cut_off_limit
58 58
59 59 def index(self, repo_name, revision, f_path):
60 60 c.repo, dbrepo = ScmModel().get(c.repo_name, retval='repo')
61 61
62 62
63 63 try:
64 64 #reditect to given revision from form
65 65 post_revision = request.POST.get('at_rev', None)
66 66 if post_revision:
67 67 post_revision = c.repo.get_changeset(post_revision).raw_id
68 68 redirect(url('files_home', repo_name=c.repo_name,
69 69 revision=post_revision, f_path=f_path))
70 70
71 71 c.branch = request.GET.get('branch', None)
72 72
73 73 c.f_path = f_path
74 74
75 75 c.changeset = c.repo.get_changeset(revision)
76 76 cur_rev = c.changeset.revision
77 77
78 78 #prev link
79 79 try:
80 80 prev_rev = c.repo.get_changeset(cur_rev).prev(c.branch).raw_id
81 81 c.url_prev = url('files_home', repo_name=c.repo_name,
82 82 revision=prev_rev, f_path=f_path)
83 83 if c.branch:
84 84 c.url_prev += '?branch=%s' % c.branch
85 85 except ChangesetDoesNotExistError:
86 86 c.url_prev = '#'
87 87
88 88 #next link
89 89 try:
90 90 next_rev = c.repo.get_changeset(cur_rev).next(c.branch).raw_id
91 91 c.url_next = url('files_home', repo_name=c.repo_name,
92 92 revision=next_rev, f_path=f_path)
93 93 if c.branch:
94 94 c.url_next += '?branch=%s' % c.branch
95 95 except ChangesetDoesNotExistError:
96 96 c.url_next = '#'
97 97
98 98 #files
99 99 try:
100 100 c.files_list = c.changeset.get_node(f_path)
101 101 c.file_history = self._get_history(c.repo, c.files_list, f_path)
102 102 except RepositoryError, e:
103 103 h.flash(str(e), category='warning')
104 104 redirect(h.url('files_home', repo_name=repo_name, revision=revision))
105 105
106 106 except EmptyRepositoryError, e:
107 107 h.flash(_('There are no files yet'), category='warning')
108 108 redirect(h.url('summary_home', repo_name=repo_name))
109 109
110 110 except RepositoryError, e:
111 111 h.flash(str(e), category='warning')
112 112 redirect(h.url('files_home', repo_name=repo_name, revision='tip'))
113 113
114 114
115 115
116 116 return render('files/files.html')
117 117
118 118 def rawfile(self, repo_name, revision, f_path):
119 119 c.repo, dbrepo = ScmModel().get(c.repo_name, retval='repo')
120 120 file_node = c.repo.get_changeset(revision).get_node(f_path)
121 121 response.content_type = file_node.mimetype
122 122 response.content_disposition = 'attachment; filename=%s' \
123 123 % f_path.split('/')[-1]
124 124 return file_node.content
125 125
126 126 def raw(self, repo_name, revision, f_path):
127 127 c.repo, dbrepo = ScmModel().get(c.repo_name, retval='repo')
128 128 file_node = c.repo.get_changeset(revision).get_node(f_path)
129 129 response.content_type = 'text/plain'
130 130
131 131 return file_node.content
132 132
133 133 def annotate(self, repo_name, revision, f_path):
134 134 c.repo, dbrepo = ScmModel().get(c.repo_name, retval='repo')
135 135
136 136 try:
137 137 c.cs = c.repo.get_changeset(revision)
138 138 c.file = c.cs.get_node(f_path)
139 139 except RepositoryError, e:
140 140 h.flash(str(e), category='warning')
141 141 redirect(h.url('files_home', repo_name=repo_name, revision=revision))
142 142
143 143 c.file_history = self._get_history(c.repo, c.file, f_path)
144 144
145 145 c.f_path = f_path
146 146
147 147 return render('files/files_annotate.html')
148 148
149 149 def archivefile(self, repo_name, fname):
150 150
151 151 fileformat = None
152 152 revision = None
153 153 ext = None
154 154
155 155 for a_type, ext_data in ARCHIVE_SPECS.items():
156 156 archive_spec = fname.split(ext_data[1])
157 157 if len(archive_spec) == 2 and archive_spec[1] == '':
158 158 fileformat = a_type or ext_data[1]
159 159 revision = archive_spec[0]
160 160 ext = ext_data[1]
161 161
162 162 try:
163 163 repo, dbrepo = ScmModel().get(repo_name)
164 164
165 165 if dbrepo.enable_downloads is False:
166 166 return _('downloads disabled')
167 167
168 168 cs = repo.get_changeset(revision)
169 169 content_type = ARCHIVE_SPECS[fileformat][0]
170 170 except ChangesetDoesNotExistError:
171 171 return _('Unknown revision %s') % revision
172 172 except EmptyRepositoryError:
173 173 return _('Empty repository')
174 174 except (ImproperArchiveTypeError, KeyError):
175 175 return _('Unknown archive type')
176 176
177 177 response.content_type = content_type
178 178 response.content_disposition = 'attachment; filename=%s-%s%s' \
179 179 % (repo_name, revision, ext)
180 180
181 181 return cs.get_chunked_archive(kind=fileformat)
182 182
183 183
184 184 def diff(self, repo_name, f_path):
185 185 diff1 = request.GET.get('diff1')
186 186 diff2 = request.GET.get('diff2')
187 187 c.action = request.GET.get('diff')
188 188 c.no_changes = diff1 == diff2
189 189 c.f_path = f_path
190 190 c.repo, dbrepo = ScmModel().get(c.repo_name, retval='repo')
191 191
192 192 try:
193 193 if diff1 not in ['', None, 'None', '0' * 12, '0' * 40]:
194 194 c.changeset_1 = c.repo.get_changeset(diff1)
195 195 node1 = c.changeset_1.get_node(f_path)
196 196 else:
197 197 c.changeset_1 = EmptyChangeset()
198 198 node1 = FileNode('.', '', changeset=c.changeset_1)
199 199
200 200 if diff2 not in ['', None, 'None', '0' * 12, '0' * 40]:
201 201 c.changeset_2 = c.repo.get_changeset(diff2)
202 202 node2 = c.changeset_2.get_node(f_path)
203 203 else:
204 204 c.changeset_2 = EmptyChangeset()
205 205 node2 = FileNode('.', '', changeset=c.changeset_2)
206 206 except RepositoryError:
207 207 return redirect(url('files_home',
208 208 repo_name=c.repo_name, f_path=f_path))
209 209
210 210 f_udiff = differ.get_udiff(node1, node2)
211 211 diff = differ.DiffProcessor(f_udiff)
212 212
213 213 if c.action == 'download':
214 214 diff_name = '%s_vs_%s.diff' % (diff1, diff2)
215 215 response.content_type = 'text/plain'
216 216 response.content_disposition = 'attachment; filename=%s' \
217 217 % diff_name
218 218 return diff.raw_diff()
219 219
220 220 elif c.action == 'raw':
221 221 response.content_type = 'text/plain'
222 222 return diff.raw_diff()
223 223
224 224 elif c.action == 'diff':
225 225 if node1.size > self.cut_off_limit or node2.size > self.cut_off_limit:
226 226 c.cur_diff = _('Diff is to big to display')
227 elif node1.is_binary or node2.is_binary:
228 c.cur_diff = _('Binary file')
227 229 else:
228 230 c.cur_diff = diff.as_html()
229 231 else:
230 232 #default option
231 233 if node1.size > self.cut_off_limit or node2.size > self.cut_off_limit:
232 234 c.cur_diff = _('Diff is to big to display')
235 elif node1.is_binary or node2.is_binary:
236 c.cur_diff = _('Binary file')
233 237 else:
234 238 c.cur_diff = diff.as_html()
235 239
236 240 if not c.cur_diff: c.no_changes = True
237 241 return render('files/file_diff.html')
238 242
239 243 def _get_history(self, repo, node, f_path):
240 244 from vcs.nodes import NodeKind
241 245 if not node.kind is NodeKind.FILE:
242 246 return []
243 247 changesets = node.history
244 248 hist_l = []
245 249
246 250 changesets_group = ([], _("Changesets"))
247 251 branches_group = ([], _("Branches"))
248 252 tags_group = ([], _("Tags"))
249 253
250 254 for chs in changesets:
251 255 n_desc = 'r%s:%s' % (chs.revision, chs.short_id)
252 256 changesets_group[0].append((chs.raw_id, n_desc,))
253 257
254 258 hist_l.append(changesets_group)
255 259
256 260 for name, chs in c.repository_branches.items():
257 261 #chs = chs.split(':')[-1]
258 262 branches_group[0].append((chs, name),)
259 263 hist_l.append(branches_group)
260 264
261 265 for name, chs in c.repository_tags.items():
262 266 #chs = chs.split(':')[-1]
263 267 tags_group[0].append((chs, name),)
264 268 hist_l.append(tags_group)
265 269
266 270 return hist_l
267 271
@@ -1,87 +1,91 b''
1 1 <%inherit file="/base/base.html"/>
2 2
3 3 <%def name="title()">
4 4 ${c.repo_name} ${_('File annotate')} - ${c.rhodecode_name}
5 5 </%def>
6 6
7 7 <%def name="breadcrumbs_links()">
8 8 ${h.link_to(u'Home',h.url('/'))}
9 9 &raquo;
10 10 ${h.link_to(c.repo_name,h.url('summary_home',repo_name=c.repo_name))}
11 11 &raquo;
12 12 ${_('annotate')} @ R${c.cs.revision}:${h.short_id(c.cs.raw_id)}
13 13 </%def>
14 14
15 15 <%def name="page_nav()">
16 16 ${self.menu('files')}
17 17 </%def>
18 18 <%def name="main()">
19 19 <div class="box">
20 20 <!-- box / title -->
21 21 <div class="title">
22 22 ${self.breadcrumbs()}
23 23 <ul class="links">
24 24 <li>
25 25 <span style="text-transform: uppercase;"><a href="#">${_('branch')}: ${c.cs.branch}</a></span>
26 26 </li>
27 27 </ul>
28 28 </div>
29 29 <div class="table">
30 30 <div id="files_data">
31 31 <h3 class="files_location">${_('Location')}: ${h.files_breadcrumbs(c.repo_name,c.cs.revision,c.file.path)}</h3>
32 32 <dl class="overview">
33 33 <dt>${_('Revision')}</dt>
34 34 <dd>${h.link_to("r%s:%s" % (c.file.last_changeset.revision,h.short_id(c.file.last_changeset.raw_id)),
35 35 h.url('changeset_home',repo_name=c.repo_name,revision=c.file.last_changeset.raw_id))} </dd>
36 36 <dt>${_('Size')}</dt>
37 37 <dd>${h.format_byte_size(c.file.size,binary=True)}</dd>
38 38 <dt>${_('Mimetype')}</dt>
39 39 <dd>${c.file.mimetype}</dd>
40 40 <dt>${_('Options')}</dt>
41 41 <dd>${h.link_to(_('show source'),
42 42 h.url('files_home',repo_name=c.repo_name,revision=c.cs.raw_id,f_path=c.f_path))}
43 43 / ${h.link_to(_('show as raw'),
44 44 h.url('files_raw_home',repo_name=c.repo_name,revision=c.cs.raw_id,f_path=c.f_path))}
45 45 / ${h.link_to(_('download as raw'),
46 46 h.url('files_rawfile_home',repo_name=c.repo_name,revision=c.cs.raw_id,f_path=c.f_path))}
47 47 </dd>
48 48 <dt>${_('History')}</dt>
49 49 <dd>
50 50 <div>
51 51 ${h.form(h.url('files_diff_home',repo_name=c.repo_name,f_path=c.f_path),method='get')}
52 52 ${h.hidden('diff2',c.file.last_changeset.raw_id)}
53 53 ${h.select('diff1',c.file.last_changeset.raw_id,c.file_history)}
54 54 ${h.submit('diff','diff to revision',class_="ui-button")}
55 55 ${h.submit('show_rev','show at revision',class_="ui-button")}
56 56 ${h.end_form()}
57 57 </div>
58 58 </dd>
59 59 </dl>
60 60 <div id="body" class="codeblock">
61 61 <div class="code-header">
62 62 <div class="revision">${c.file.name}@r${c.file.last_changeset.revision}:${h.short_id(c.file.last_changeset.raw_id)}</div>
63 63 <div class="commit">"${c.file.message}"</div>
64 64 </div>
65 65 <div class="code-body">
66 %if c.file.is_binary:
67 ${_('Binary file')}
68 %else:
66 69 % if c.file.size < c.cut_off_limit:
67 70 ${h.pygmentize_annotation(c.file,linenos=True,anchorlinenos=True,lineanchors='S',cssclass="code-highlight")}
68 71 %else:
69 72 ${_('File is to big to display')} ${h.link_to(_('show as raw'),
70 73 h.url('files_raw_home',repo_name=c.repo_name,revision=c.cs.revision,f_path=c.f_path))}
71 %endif
74 %endif
75 <script type="text/javascript">
76 YAHOO.util.Event.onDOMReady(function(){
77 YAHOO.util.Event.addListener('show_rev','click',function(e){
78 YAHOO.util.Event.preventDefault(e);
79 var cs = YAHOO.util.Dom.get('diff1').value;
80 var url = "${h.url('files_annotate_home',repo_name=c.repo_name,revision='__CS__',f_path=c.f_path)}".replace('__CS__',cs);
81 window.location = url;
82 });
83 });
84 </script>
85 %endif
72 86 </div>
73 87 </div>
74 <script type="text/javascript">
75 YAHOO.util.Event.onDOMReady(function(){
76 YAHOO.util.Event.addListener('show_rev','click',function(e){
77 YAHOO.util.Event.preventDefault(e);
78 var cs = YAHOO.util.Dom.get('diff1').value;
79 var url = "${h.url('files_annotate_home',repo_name=c.repo_name,revision='__CS__',f_path=c.f_path)}".replace('__CS__',cs);
80 window.location = url;
81 });
82 });
83 </script>
84 88 </div>
85 89 </div>
86 90 </div>
87 91 </%def> No newline at end of file
@@ -1,87 +1,91 b''
1 1 <dl>
2 2 <dt>${_('Revision')}</dt>
3 3 <dd>
4 4 ${h.link_to("r%s:%s" % (c.files_list.last_changeset.revision,h.short_id(c.files_list.last_changeset.raw_id)),
5 5 h.url('changeset_home',repo_name=c.repo_name,revision=c.files_list.last_changeset.raw_id))}
6 6 </dd>
7 7 <dt>${_('Size')}</dt>
8 8 <dd>${h.format_byte_size(c.files_list.size,binary=True)}</dd>
9 9 <dt>${_('Mimetype')}</dt>
10 10 <dd>${c.files_list.mimetype}</dd>
11 11 <dt>${_('Options')}</dt>
12 12 <dd>${h.link_to(_('show annotation'),
13 13 h.url('files_annotate_home',repo_name=c.repo_name,revision=c.changeset.raw_id,f_path=c.f_path))}
14 14 / ${h.link_to(_('show as raw'),
15 15 h.url('files_raw_home',repo_name=c.repo_name,revision=c.changeset.raw_id,f_path=c.f_path))}
16 16 / ${h.link_to(_('download as raw'),
17 17 h.url('files_rawfile_home',repo_name=c.repo_name,revision=c.changeset.raw_id,f_path=c.f_path))}
18 18 </dd>
19 19 <dt>${_('History')}</dt>
20 20 <dd>
21 21 <div>
22 22 ${h.form(h.url('files_diff_home',repo_name=c.repo_name,f_path=c.f_path),method='get')}
23 23 ${h.hidden('diff2',c.files_list.last_changeset.raw_id)}
24 24 ${h.select('diff1',c.files_list.last_changeset.raw_id,c.file_history)}
25 25 ${h.submit('diff','diff to revision',class_="ui-button")}
26 26 ${h.submit('show_rev','show at revision',class_="ui-button")}
27 27 ${h.end_form()}
28 28 </div>
29 29 </dd>
30 30 </dl>
31 31
32 32
33 33 <div id="body" class="codeblock">
34 34 <div class="code-header">
35 35 <div class="revision">${c.files_list.name}@r${c.files_list.last_changeset.revision}:${h.short_id(c.files_list.last_changeset.raw_id)}</div>
36 36 <div class="commit">"${c.files_list.last_changeset.message}"</div>
37 37 </div>
38 38 <div class="code-body">
39 %if c.files_list.is_binary:
40 ${_('Binary file')}
41 %else:
39 42 % if c.files_list.size < c.cut_off_limit:
40 43 ${h.pygmentize(c.files_list,linenos=True,anchorlinenos=True,lineanchors='L',cssclass="code-highlight")}
41 44 %else:
42 45 ${_('File is to big to display')} ${h.link_to(_('show as raw'),
43 46 h.url('files_raw_home',repo_name=c.repo_name,revision=c.changeset.raw_id,f_path=c.f_path))}
44 47 %endif
45 48
46 49 <script type="text/javascript">
47 50 function highlight_lines(lines){
48 51 for(pos in lines){
49 52 YUD.setStyle('L'+lines[pos],'background-color','#FFFFBE');
50 53 }
51 54 }
52 55 page_highlights = location.href.substring(location.href.indexOf('#')+1).split('L');
53 56 if (page_highlights.length == 2){
54 57 highlight_ranges = page_highlights[1].split(",");
55 58
56 59 var h_lines = [];
57 60 for (pos in highlight_ranges){
58 61 var _range = highlight_ranges[pos].split('-');
59 62 if(_range.length == 2){
60 63 var start = parseInt(_range[0]);
61 64 var end = parseInt(_range[1]);
62 65 if (start < end){
63 66 for(var i=start;i<=end;i++){
64 67 h_lines.push(i);
65 68 }
66 69 }
67 70 }
68 71 else{
69 72 h_lines.push(parseInt(highlight_ranges[pos]));
70 73 }
71 74 }
72 75 highlight_lines(h_lines);
73 76 }
74 </script>
77 </script>
78 %endif
75 79 </div>
76 80 </div>
77 81
78 82 <script type="text/javascript">
79 83 YAHOO.util.Event.onDOMReady(function(){
80 84 YAHOO.util.Event.addListener('show_rev','click',function(e){
81 85 YAHOO.util.Event.preventDefault(e);
82 86 var cs = YAHOO.util.Dom.get('diff1').value;
83 87 var url = "${h.url('files_home',repo_name=c.repo_name,revision='__CS__',f_path=c.f_path)}".replace('__CS__',cs);
84 88 window.location = url;
85 89 });
86 90 });
87 91 </script> No newline at end of file
General Comments 0
You need to be logged in to leave comments. Login now