Show More
@@ -325,6 +325,21 b' class RepoFilesView(RepoAppView):' | |||
|
325 | 325 | |
|
326 | 326 | return lf_enabled |
|
327 | 327 | |
|
328 | def _get_archive_name(self, db_repo_name, commit_sha, ext, subrepos=False, path_sha=''): | |
|
329 | # original backward compat name of archive | |
|
330 | clean_name = safe_str(db_repo_name.replace('/', '_')) | |
|
331 | ||
|
332 | # e.g vcsserver.zip | |
|
333 | # e.g vcsserver-abcdefgh.zip | |
|
334 | # e.g vcsserver-abcdefgh-defghijk.zip | |
|
335 | archive_name = '{}{}{}{}{}'.format( | |
|
336 | clean_name, | |
|
337 | '-sub' if subrepos else '', | |
|
338 | commit_sha, | |
|
339 | '-{}'.format(path_sha) if path_sha else '', | |
|
340 | ext) | |
|
341 | return archive_name | |
|
342 | ||
|
328 | 343 | @LoginRequired() |
|
329 | 344 | @HasRepoPermissionAnyDecorator( |
|
330 | 345 | 'repository.read', 'repository.write', 'repository.admin') |
@@ -339,6 +354,7 b' class RepoFilesView(RepoAppView):' | |||
|
339 | 354 | default_at_path = '/' |
|
340 | 355 | fname = self.request.matchdict['fname'] |
|
341 | 356 | subrepos = self.request.GET.get('subrepos') == 'true' |
|
357 | with_hash = str2bool(self.request.GET.get('with_hash', '1')) | |
|
342 | 358 | at_path = self.request.GET.get('at_path') or default_at_path |
|
343 | 359 | |
|
344 | 360 | if not self.db_repo.enable_downloads: |
@@ -364,26 +380,26 b' class RepoFilesView(RepoAppView):' | |||
|
364 | 380 | except Exception: |
|
365 | 381 | return Response(_('No node at path {} for this repository').format(at_path)) |
|
366 | 382 | |
|
367 | path_sha = sha1(at_path)[:8] | |
|
368 | ||
|
369 | # original backward compat name of archive | |
|
370 | clean_name = safe_str(self.db_repo_name.replace('/', '_')) | |
|
371 | short_sha = safe_str(commit.short_id) | |
|
383 | # path sha is part of subdir | |
|
384 | path_sha = '' | |
|
385 | if at_path != default_at_path: | |
|
386 | path_sha = sha1(at_path)[:8] | |
|
387 | short_sha = '-{}'.format(safe_str(commit.short_id)) | |
|
388 | # used for cache etc | |
|
389 | archive_name = self._get_archive_name( | |
|
390 | self.db_repo_name, commit_sha=short_sha, ext=ext, subrepos=subrepos, | |
|
391 | path_sha=path_sha) | |
|
372 | 392 | |
|
373 | if at_path == default_at_path: | |
|
374 | archive_name = '{}-{}{}{}'.format( | |
|
375 | clean_name, | |
|
376 | '-sub' if subrepos else '', | |
|
377 | short_sha, | |
|
378 | ext) | |
|
379 | # custom path and new name | |
|
380 | else: | |
|
381 | archive_name = '{}-{}{}-{}{}'.format( | |
|
382 | clean_name, | |
|
383 | '-sub' if subrepos else '', | |
|
384 | short_sha, | |
|
385 | path_sha, | |
|
386 | ext) | |
|
393 | if not with_hash: | |
|
394 | short_sha = '' | |
|
395 | path_sha = '' | |
|
396 | ||
|
397 | # what end client gets served | |
|
398 | response_archive_name = self._get_archive_name( | |
|
399 | self.db_repo_name, commit_sha=short_sha, ext=ext, subrepos=subrepos, | |
|
400 | path_sha=path_sha) | |
|
401 | # remove extension from our archive directory name | |
|
402 | archive_dir_name = response_archive_name[:-len(ext)] | |
|
387 | 403 | |
|
388 | 404 | use_cached_archive = False |
|
389 | 405 | archive_cache_enabled = CONFIG.get( |
@@ -403,13 +419,15 b' class RepoFilesView(RepoAppView):' | |||
|
403 | 419 | else: |
|
404 | 420 | log.debug('Archive %s is not yet cached', archive_name) |
|
405 | 421 | |
|
422 | # generate new archive, as previous was not found in the cache | |
|
406 | 423 | if not use_cached_archive: |
|
407 | # generate new archive | |
|
424 | ||
|
408 | 425 | fd, archive = tempfile.mkstemp() |
|
409 | 426 | log.debug('Creating new temp archive in %s', archive) |
|
410 | 427 | try: |
|
411 |
commit.archive_repo(archive, |
|
|
412 |
|
|
|
428 | commit.archive_repo(archive, archive_dir_name=archive_dir_name, | |
|
429 | kind=fileformat, subrepos=subrepos, | |
|
430 | archive_at_path=at_path, with_hash=with_hash) | |
|
413 | 431 | except ImproperArchiveTypeError: |
|
414 | 432 | return _('Unknown archive type') |
|
415 | 433 | if archive_cache_enabled: |
@@ -445,8 +463,7 b' class RepoFilesView(RepoAppView):' | |||
|
445 | 463 | yield data |
|
446 | 464 | |
|
447 | 465 | response = Response(app_iter=get_chunked_archive(archive)) |
|
448 | response.content_disposition = str( | |
|
449 | 'attachment; filename=%s' % archive_name) | |
|
466 | response.content_disposition = str('attachment; filename=%s' % response_archive_name) | |
|
450 | 467 | response.content_type = str(content_type) |
|
451 | 468 | |
|
452 | 469 | return response |
@@ -1192,13 +1192,14 b' class BaseCommit(object):' | |||
|
1192 | 1192 | return None |
|
1193 | 1193 | |
|
1194 | 1194 | def archive_repo(self, archive_dest_path, kind='tgz', subrepos=None, |
|
1195 |
|
|
|
1195 | archive_dir_name=None, write_metadata=False, mtime=None, | |
|
1196 | archive_at_path='/', with_hash=True): | |
|
1196 | 1197 | """ |
|
1197 | 1198 | Creates an archive containing the contents of the repository. |
|
1198 | 1199 | |
|
1199 | 1200 | :param archive_dest_path: path to the file which to create the archive. |
|
1200 | 1201 | :param kind: one of following: ``"tbz2"``, ``"tgz"``, ``"zip"``. |
|
1201 |
:param |
|
|
1202 | :param archive_dir_name: name of root directory in archive. | |
|
1202 | 1203 | Default is repository name and commit's short_id joined with dash: |
|
1203 | 1204 | ``"{repo_name}-{short_id}"``. |
|
1204 | 1205 | :param write_metadata: write a metadata file into archive. |
@@ -1214,7 +1215,7 b' class BaseCommit(object):' | |||
|
1214 | 1215 | 'Archive kind (%s) not supported use one of %s' % |
|
1215 | 1216 | (kind, allowed_kinds)) |
|
1216 | 1217 | |
|
1217 |
|
|
|
1218 | archive_dir_name = self._validate_archive_prefix(archive_dir_name) | |
|
1218 | 1219 | |
|
1219 | 1220 | mtime = mtime is not None or time.mktime(self.date.timetuple()) |
|
1220 | 1221 | |
@@ -1222,7 +1223,7 b' class BaseCommit(object):' | |||
|
1222 | 1223 | cur_rev = self.repository.get_commit(commit_id=self.raw_id) |
|
1223 | 1224 | for _r, _d, files in cur_rev.walk(archive_at_path): |
|
1224 | 1225 | for f in files: |
|
1225 |
f_path = os.path.join( |
|
|
1226 | f_path = os.path.join(archive_dir_name, f.path) | |
|
1226 | 1227 | file_info.append( |
|
1227 | 1228 | (f_path, f.mode, f.is_link(), f.raw_bytes)) |
|
1228 | 1229 | |
@@ -1239,18 +1240,18 b' class BaseCommit(object):' | |||
|
1239 | 1240 | |
|
1240 | 1241 | connection.Hg.archive_repo(archive_dest_path, mtime, file_info, kind) |
|
1241 | 1242 | |
|
1242 |
def _validate_archive_prefix(self, |
|
|
1243 |
if |
|
|
1244 |
|
|
|
1243 | def _validate_archive_prefix(self, archive_dir_name): | |
|
1244 | if archive_dir_name is None: | |
|
1245 | archive_dir_name = self._ARCHIVE_PREFIX_TEMPLATE.format( | |
|
1245 | 1246 | repo_name=safe_str(self.repository.name), |
|
1246 | 1247 | short_id=self.short_id) |
|
1247 |
elif not isinstance( |
|
|
1248 |
raise ValueError("prefix not a bytes object: %s" % repr( |
|
|
1249 |
elif |
|
|
1248 | elif not isinstance(archive_dir_name, str): | |
|
1249 | raise ValueError("prefix not a bytes object: %s" % repr(archive_dir_name)) | |
|
1250 | elif archive_dir_name.startswith('/'): | |
|
1250 | 1251 | raise VCSError("Prefix cannot start with leading slash") |
|
1251 |
elif |
|
|
1252 | elif archive_dir_name.strip() == '': | |
|
1252 | 1253 | raise VCSError("Prefix cannot be empty") |
|
1253 |
return |
|
|
1254 | return archive_dir_name | |
|
1254 | 1255 | |
|
1255 | 1256 | @LazyProperty |
|
1256 | 1257 | def root(self): |
@@ -51,7 +51,7 b'' | |||
|
51 | 51 | ${_('Download full tree ZIP')} |
|
52 | 52 | </a> |
|
53 | 53 | % else: |
|
54 | <a href="${h.route_path('repo_archivefile',repo_name=c.repo_name, fname='{}.zip'.format(c.commit.raw_id), _query={'at_path':c.f_path})}"> | |
|
54 | <a href="${h.route_path('repo_archivefile',repo_name=c.repo_name, fname='{}.zip'.format(c.commit.raw_id), _query={'at_path':c.f_path, 'with_hash': '1'})}"> | |
|
55 | 55 | ${_('Download this tree ZIP')} |
|
56 | 56 | </a> |
|
57 | 57 | % endif |
@@ -187,7 +187,7 b'' | |||
|
187 | 187 | <div class="enabled pull-left" style="margin-right: 10px"> |
|
188 | 188 | |
|
189 | 189 | <div class="btn-group btn-group-actions"> |
|
190 |
<a class="archive_link btn btn-small" data-ext=".zip" href="${h.route_path('repo_archivefile',repo_name=c.rhodecode_db_repo.repo_name, fname=c.rhodecode_db_repo.landing_ref_name+'.zip' |
|
|
190 | <a class="archive_link btn btn-small" data-ext=".zip" href="${h.route_path('repo_archivefile',repo_name=c.rhodecode_db_repo.repo_name, fname=c.rhodecode_db_repo.landing_ref_name+'.zip', _query={'with_hash': '1'})}"> | |
|
191 | 191 | <i class="icon-download"></i> |
|
192 | 192 | ${c.rhodecode_db_repo.landing_ref_name}.zip |
|
193 | 193 | ## replaced by some JS on select |
@@ -202,8 +202,7 b'' | |||
|
202 | 202 | % for a_type, content_type, extension in h.ARCHIVE_SPECS: |
|
203 | 203 | % if extension not in ['.zip']: |
|
204 | 204 | <li> |
|
205 | ||
|
206 | <a class="archive_link" data-ext="${extension}" href="${h.route_path('repo_archivefile',repo_name=c.rhodecode_db_repo.repo_name, fname=c.rhodecode_db_repo.landing_ref_name+extension)}"> | |
|
205 | <a class="archive_link" data-ext="${extension}" href="${h.route_path('repo_archivefile',repo_name=c.rhodecode_db_repo.repo_name, fname=c.rhodecode_db_repo.landing_ref_name+extension, _query={'with_hash': '1'})}"> | |
|
207 | 206 | <i class="icon-download"></i> |
|
208 | 207 | ${c.rhodecode_db_repo.landing_ref_name+extension} |
|
209 | 208 | </a> |
General Comments 0
You need to be logged in to leave comments.
Login now