diff --git a/trac/util/__init__.py b/trac/util/__init__.py
|
a
|
b
|
|
| 870 | 870 | if max is not None and value > max: |
| 871 | 871 | value = max |
| 872 | 872 | return value |
| | 873 | |
| | 874 | def pathjoin(*args): |
| | 875 | """Strip `/` from the arguments and join them with a single `/`.""" |
| | 876 | return '/'.join(filter(None, (each.strip('/') for each in args if each))) |
diff --git a/trac/versioncontrol/templates/browser.html b/trac/versioncontrol/templates/browser.html
|
a
|
b
|
|
| 111 | 111 | <tr py:if="file"> |
| 112 | 112 | <td class="message searchable" py:choose=""> |
| 113 | 113 | <py:when test="wiki_format_messages" xml:space="preserve"> |
| 114 | | ${wiki_to_html(context('changeset', file.changeset.rev, parent=repos_resource), |
| | 114 | ${wiki_to_html(context('changeset', file.changeset.rev, parent=repos.resource), |
| 115 | 115 | file.changeset.message, escape_newlines=True)} |
| 116 | 116 | </py:when> |
| 117 | 117 | <py:otherwise>${file.changeset.message}</py:otherwise> |
| … |
… |
|
| 159 | 159 | <div id="anydiff"> |
| 160 | 160 | <form action="${href.diff()}" method="get"> |
| 161 | 161 | <div class="buttons"> |
| 162 | | <input type="hidden" name="new_path" value="$reponame/$path" /> |
| 163 | | <input type="hidden" name="old_path" value="$reponame/$path" /> |
| | 162 | <input type="hidden" name="new_path" value="${'/' + pathjoin(reponame, path)}" /> |
| | 163 | <input type="hidden" name="old_path" value="${'/' + pathjoin(reponame, path)}" /> |
| 164 | 164 | <input type="hidden" name="new_rev" value="$stickyrev" /> |
| 165 | 165 | <input type="hidden" name="old_rev" value="$stickyrev" /> |
| 166 | 166 | <input type="submit" value="${_('View changes...')}" title="${_('Select paths and revs for Diff')}" /> |
diff --git a/trac/versioncontrol/templates/changeset.html b/trac/versioncontrol/templates/changeset.html
|
a
|
b
|
|
| 51 | 51 | id="prefs" action=""> |
| 52 | 52 | <div> |
| 53 | 53 | <py:if test="not changeset"> |
| 54 | | <input type="hidden" name="old_path" value="$reponame/$old_path" /> |
| 55 | | <input type="hidden" name="new_path" value="$reponame/$new_path" /> |
| | 54 | <input type="hidden" name="old_path" value="${'/' + pathjoin(reponame, old_path)}" /> |
| | 55 | <input type="hidden" name="new_path" value="${'/' + pathjoin(reponame, new_path)}" /> |
| 56 | 56 | <input type="hidden" name="old" value="$old_rev" /> |
| 57 | 57 | <input type="hidden" name="new" value="$new_rev" /> |
| 58 | 58 | </py:if> |
diff --git a/trac/versioncontrol/templates/dir_entries.html b/trac/versioncontrol/templates/dir_entries.html
|
a
|
b
|
|
| 7 | 7 | </py:if> |
| 8 | 8 | <py:for each="idx, entry in enumerate(dir.entries)"> |
| 9 | 9 | <py:with vars="change = dir.changes[entry.rev]; |
| 10 | | chgset_context = change and context('changeset', change.rev, parent=repos_resource); |
| | 10 | chgset_context = change and context('changeset', change.rev, parent=repos.resource); |
| 11 | 11 | chgset_view = change and change.can_view(perm)"> |
| 12 | 12 | <tr class="${idx % 2 and 'even' or 'odd'}"> |
| 13 | 13 | <td class="name"> |
diff --git a/trac/versioncontrol/templates/repository_index.html b/trac/versioncontrol/templates/repository_index.html
|
a
|
b
|
|
| 5 | 5 | <table class="listing dirlist" id="${repoindex or None}"> |
| 6 | 6 | <xi:include href="dirlist_thead.html" /> |
| 7 | 7 | <tbody> |
| 8 | | <py:for each="idx, (reponame, repoinfo, change, err) in enumerate(repo.repositories)" |
| 9 | | py:with="chgset_context = change and context('changeset', change.rev, parent=Resource('repository', reponame)); |
| | 8 | <py:for each="idx, (reponame, repoinfo, repos, change, err) in enumerate(repo.repositories)" |
| | 9 | py:with="chgset_context = change and context('changeset', change.rev, parent=repos.resource); |
| 10 | 10 | chgset_view = change and change.can_view(perm)"> |
| 11 | 11 | <tr class="${idx % 2 and 'even' or 'odd'}"> |
| 12 | 12 | <td class="name"> |
| 13 | 13 | <em py:strip="not err"> |
| 14 | 14 | <b py:strip="repoinfo.alias != ''"> |
| 15 | 15 | <a class="dir" title="View Root Directory" |
| 16 | | href="${href.browser(reponame, order=(order != 'name' and order or None), desc=desc)}">$reponame</a> |
| | 16 | href="${href.browser(repos.reponame, order=(order != 'name' and order or None), desc=desc)}">$reponame</a> |
| 17 | 17 | </b> |
| 18 | 18 | </em> |
| 19 | 19 | </td> |
| 20 | 20 | <td class="size" /> |
| 21 | 21 | <td class="rev"> |
| 22 | 22 | <py:if test="not err"> |
| 23 | | <a title="View Revision Log" href="${href.log(reponame)}">$change.rev</a> |
| 24 | | <a title="View Changeset" class="chgset" href="${href.changeset(change.rev, reponame)}"> </a> |
| | 23 | <a title="View Revision Log" href="${href.log(repos.reponame)}">$change.rev</a> |
| | 24 | <a title="View Changeset" class="chgset" href="${href.changeset(change.rev, repos.reponame)}"> </a> |
| 25 | 25 | </py:if> |
| 26 | 26 | </td> |
| 27 | 27 | <td class="age" style="${chgset_view and change and repo.timerange and 'border-color: rgb(%s,%s,%s)' % |
diff --git a/trac/versioncontrol/templates/revisionlog.html b/trac/versioncontrol/templates/revisionlog.html
|
a
|
b
|
|
| 110 | 110 | <py:for each="idx, item in enumerate(items)"> |
| 111 | 111 | <py:with vars="change = changes[item.rev]; |
| 112 | 112 | is_separator = item.change is None; |
| 113 | | chgset_context = context('changeset', change.rev, parent=context.resource.parent); |
| | 113 | chgset_context = context('changeset', change.rev, parent=repos.resource); |
| 114 | 114 | odd_even = idx % 2 and 'odd' or 'even'"> |
| 115 | 115 | <!--! highlight copy or rename operations --> |
| 116 | 116 | <tr py:if="not is_separator and item.get('copyfrom_path')" class="$odd_even"> |
diff --git a/trac/versioncontrol/templates/revisionlog.rss b/trac/versioncontrol/templates/revisionlog.rss
|
a
|
b
|
|
| 17 | 17 | |
| 18 | 18 | <item py:for="item in items" |
| 19 | 19 | py:with="change = changes[item.rev]; |
| 20 | | item_context = context('changeset', change.rev, parent=context.resource.parent)"> |
| | 20 | item_context = context('changeset', change.rev, parent=repos.resource)"> |
| 21 | 21 | ${author_or_creator(change.author, email_map)} |
| 22 | 22 | <pubDate>${http_date(change.date)}</pubDate> |
| 23 | 23 | <title>Revision $item.rev: ${shorten_line(change.message)}</title> |
diff --git a/trac/versioncontrol/templates/revisionlog.txt b/trac/versioncontrol/templates/revisionlog.txt
|
a
|
b
|
|
| 1 | 1 | # |
| 2 | | # ChangeLog for $path${reponame and _(" in $(reponame)s", reponame=reponame) or ''} |
| | 2 | # ChangeLog for $path${reponame and _(" in $(reponame)s", reponame=reponame) or None} |
| 3 | 3 | # |
| 4 | 4 | # Generated by Trac $trac.version |
| 5 | 5 | # ${format_datetime()} |
diff --git a/trac/versioncontrol/web_ui/browser.py b/trac/versioncontrol/web_ui/browser.py
|
a
|
b
|
|
| 26 | 26 | from trac.mimeview.api import Mimeview, is_binary, \ |
| 27 | 27 | IHTMLPreviewAnnotator, Context |
| 28 | 28 | from trac.perm import IPermissionRequestor |
| 29 | | from trac.resource import Resource, ResourceNotFound |
| | 29 | from trac.resource import ResourceNotFound |
| 30 | 30 | from trac.util import embedded_numbers |
| 31 | 31 | from trac.util.compat import any |
| 32 | 32 | from trac.util.datefmt import http_date, utc |
| … |
… |
|
| 291 | 291 | |
| 292 | 292 | def get_navigation_items(self, req): |
| 293 | 293 | rm = RepositoryManager(self.env) |
| 294 | | all_repos = rm.get_all_repositories() |
| 295 | | if any(info.get('hidden') not in _TRUE_VALUES |
| 296 | | and rm.get_repository(reponame).can_view(req.perm) |
| 297 | | for reponame, info in all_repos.iteritems()): |
| | 294 | if any(repos and repos.params.get('hidden') not in _TRUE_VALUES |
| | 295 | and repos.can_view(req.perm) |
| | 296 | for repos in rm.get_real_repositories()): |
| 298 | 297 | yield ('mainnav', 'browser', |
| 299 | 298 | tag.a(_('Browse Source'), href=req.href.browser())) |
| 300 | 299 | |
| … |
… |
|
| 347 | 346 | raise ResourceNotFound(_("No repository '%(repo)s' found", |
| 348 | 347 | repo=reponame)) |
| 349 | 348 | |
| | 349 | if reponame != repos.reponame: # Redirect alias |
| | 350 | qs = req.query_string |
| | 351 | req.redirect(req.href.browser(repos.reponame or None, path) |
| | 352 | + (qs and '?' + qs or '')) |
| | 353 | |
| 350 | 354 | # Find node for the requested path/rev |
| 351 | 355 | context = Context.from_request(req) |
| 352 | 356 | node = None |
| … |
… |
|
| 362 | 366 | raise ResourceNotFound(e.message, |
| 363 | 367 | _('Invalid changeset number')) |
| 364 | 368 | |
| 365 | | repos_resource = Resource('repository', reponame) |
| 366 | | context = context(repos_resource.child('source', path, |
| | 369 | context = context(repos.resource.child('source', path, |
| 367 | 370 | version=node.created_rev)) |
| 368 | 371 | |
| 369 | 372 | # Prepare template data |
| 370 | | path_links = get_path_links(req.href, reponame, path, rev, order, desc) |
| | 373 | path_links = get_path_links(req.href, repos.reponame, path, rev, |
| | 374 | order, desc) |
| 371 | 375 | |
| 372 | 376 | repo_data = dir_data = file_data = None |
| 373 | 377 | if all_repositories: |
| … |
… |
|
| 375 | 379 | context, all_repositories, order, desc) |
| 376 | 380 | if node: |
| 377 | 381 | if node.isdir: |
| 378 | | dir_data = self._render_dir( |
| 379 | | req, reponame, repos, node, rev, order, desc) |
| | 382 | dir_data = self._render_dir(req, repos, node, rev, order, desc) |
| 380 | 383 | elif node.isfile: |
| 381 | | file_data = self._render_file( |
| 382 | | req, context, reponame, repos, node, rev) |
| | 384 | file_data = self._render_file(req, context, repos, node, rev) |
| 383 | 385 | |
| 384 | 386 | quickjump_data = properties_data = None |
| 385 | 387 | if node and not xhr: |
| … |
… |
|
| 387 | 389 | 'browser', context, node.get_properties()) |
| 388 | 390 | quickjump_data = list(repos.get_quickjump_entries(rev)) |
| 389 | 391 | |
| | 392 | reponame = repos.reponame or None |
| 390 | 393 | data = { |
| 391 | | 'context': context, 'reponame': reponame or None, |
| 392 | | 'repos_resource': repos and repos_resource, |
| | 394 | 'context': context, 'reponame': reponame, |
| | 395 | 'repos': repos, |
| 393 | 396 | 'path': path, 'rev': node and node.rev, 'stickyrev': rev, |
| 394 | 397 | 'created_path': node and node.created_path, |
| 395 | 398 | 'created_rev': node and node.created_rev, |
| … |
… |
|
| 415 | 418 | prev_rev = repos.previous_rev(rev=node.rev, |
| 416 | 419 | path=node.created_path) |
| 417 | 420 | if prev_rev: |
| 418 | | href = req.href.browser(reponame or None, |
| | 421 | href = req.href.browser(reponame, |
| 419 | 422 | node.created_path, rev=prev_rev) |
| 420 | 423 | add_link(req, 'prev', href, |
| 421 | 424 | _('Revision %(num)s', num=prev_rev)) |
| 422 | 425 | if rev is not None: |
| 423 | | add_link(req, 'up', req.href.browser(reponame or None, |
| | 426 | add_link(req, 'up', req.href.browser(reponame, |
| 424 | 427 | node.created_path)) |
| 425 | 428 | next_rev = repos.next_rev(rev=node.rev, |
| 426 | 429 | path=node.created_path) |
| 427 | 430 | if next_rev: |
| 428 | | href = req.href.browser(reponame or None, |
| 429 | | node.created_path, rev=next_rev) |
| | 431 | href = req.href.browser(reponame, node.created_path, |
| | 432 | rev=next_rev) |
| 430 | 433 | add_link(req, 'next', href, |
| 431 | 434 | _('Revision %(num)s', num=next_rev)) |
| 432 | 435 | prevnext_nav(req, _('Previous Revision'), _('Next Revision'), |
| … |
… |
|
| 436 | 439 | add_link(req, 'up', path_links[-2]['href'], |
| 437 | 440 | _('Parent directory')) |
| 438 | 441 | add_ctxtnav(req, tag.a(_('Last Change'), |
| 439 | | href=req.href.changeset(node.rev, reponame or None, |
| | 442 | href=req.href.changeset(node.rev, reponame, |
| 440 | 443 | node.created_path))) |
| 441 | 444 | if node.isfile: |
| 442 | 445 | if data['file']['annotate']: |
| 443 | 446 | add_ctxtnav(req, _('Normal'), |
| 444 | 447 | title=_('View file without annotations'), |
| 445 | | href=req.href.browser(reponame or None, |
| | 448 | href=req.href.browser(reponame, |
| 446 | 449 | node.created_path, |
| 447 | 450 | rev=node.rev)) |
| 448 | 451 | else: |
| … |
… |
|
| 450 | 453 | title=_('Annotate each line with the last ' |
| 451 | 454 | 'changed revision ' |
| 452 | 455 | '(this can be time consuming...)'), |
| 453 | | href=req.href.browser(reponame or None, |
| | 456 | href=req.href.browser(reponame, |
| 454 | 457 | node.created_path, |
| 455 | 458 | rev=node.rev, |
| 456 | 459 | annotate='blame')) |
| 457 | 460 | add_ctxtnav(req, _('Revision Log'), |
| 458 | | href=req.href.log(reponame or None, path, rev=rev)) |
| | 461 | href=req.href.log(reponame, path, rev=rev)) |
| 459 | 462 | path_url = repos.get_path_url(path, rev) |
| 460 | 463 | if path_url: |
| 461 | 464 | if path_url.startswith('//'): |
| … |
… |
|
| 489 | 492 | timerange = TimeRange(youngest.date) |
| 490 | 493 | else: |
| 491 | 494 | timerange.insert(youngest.date) |
| 492 | | entry = (reponame, repoinfo, youngest, None) |
| | 495 | entry = (reponame, repoinfo, repos, youngest, None) |
| 493 | 496 | else: |
| 494 | | entry = (reponame, repoinfo, None, "XXX") |
| | 497 | entry = (reponame, repoinfo, None, None, "XXX") |
| 495 | 498 | except TracError, err: |
| 496 | | entry = (reponame, repoinfo, None, exception_to_unicode(err)) |
| | 499 | entry = (reponame, repoinfo, None, None, |
| | 500 | exception_to_unicode(err)) |
| 497 | 501 | repositories.append(entry) |
| 498 | 502 | |
| 499 | 503 | # Ordering of repositories |
| 500 | 504 | if order == 'date': |
| 501 | | def repo_order((reponame, repoinfo, repos, youngest)): |
| | 505 | def repo_order((reponame, repoinfo, repos, youngest, err)): |
| 502 | 506 | return youngest and youngest.date |
| 503 | 507 | else: |
| 504 | | def repo_order((reponame, repoinfo, repos, youngest)): |
| | 508 | def repo_order((reponame, repoinfo, repos, youngest, err)): |
| 505 | 509 | return embedded_numbers(reponame.lower()) |
| 506 | 510 | |
| 507 | 511 | repositories = sorted(repositories, key=repo_order, reverse=desc) |
| … |
… |
|
| 509 | 513 | return {'repositories' : repositories, |
| 510 | 514 | 'timerange': timerange, 'colorize_age': custom_colorizer} |
| 511 | 515 | |
| 512 | | def _render_dir(self, req, reponame, repos, node, rev, order, desc): |
| | 516 | def _render_dir(self, req, repos, node, rev, order, desc): |
| 513 | 517 | req.perm(node.resource).require('BROWSER_VIEW') |
| 514 | 518 | |
| 515 | 519 | # Entries metadata |
| … |
… |
|
| 564 | 568 | if node.path and patterns and \ |
| 565 | 569 | filter(None, [fnmatchcase(node.path, p) for p in patterns]): |
| 566 | 570 | zip_href = req.href.changeset(rev or repos.youngest_rev, |
| 567 | | reponame, node.path, old=rev, |
| 568 | | old_path=reponame or '/', |
| | 571 | repos.reponame or None, node.path, |
| | 572 | old=rev, |
| | 573 | old_path=repos.reponame or '/', |
| 569 | 574 | format='zip') |
| 570 | 575 | add_link(req, 'alternate', zip_href, _('Zip Archive'), |
| 571 | 576 | 'application/zip', 'zip') |
| … |
… |
|
| 578 | 583 | timerange.to_seconds(timerange.oldest)), |
| 579 | 584 | } |
| 580 | 585 | |
| 581 | | def _render_file(self, req, context, reponame, repos, node, rev=None): |
| | 586 | def _render_file(self, req, context, repos, node, rev=None): |
| 582 | 587 | req.perm(node.resource).require('FILE_VIEW') |
| 583 | 588 | |
| 584 | 589 | mimeview = Mimeview(self.env) |
| … |
… |
|
| 618 | 623 | |
| 619 | 624 | # add ''Plain Text'' alternate link if needed |
| 620 | 625 | if not is_binary(chunk) and mime_type != 'text/plain': |
| 621 | | plain_href = req.href.browser(reponame or None, node.path, |
| 622 | | rev=rev, format='txt') |
| | 626 | plain_href = req.href.browser(repos.reponame or None, |
| | 627 | node.path, rev=rev, format='txt') |
| 623 | 628 | add_link(req, 'alternate', plain_href, _('Plain Text'), |
| 624 | 629 | 'text/plain') |
| 625 | 630 | |
| 626 | 631 | # add ''Original Format'' alternate link (always) |
| 627 | 632 | raw_href = req.href.export(rev or repos.youngest_rev, |
| 628 | | reponame, node.path) |
| | 633 | repos.reponame or None, node.path) |
| 629 | 634 | add_link(req, 'alternate', raw_href, _('Original Format'), |
| 630 | 635 | mime_type) |
| 631 | 636 | |
| … |
… |
|
| 784 | 789 | if fnmatchcase(rdata[0], glob)) |
| 785 | 790 | |
| 786 | 791 | if format == 'table': |
| 787 | | data = self._render_repository_index( |
| 788 | | formatter.context, all_repos, order, desc) |
| | 792 | repo = self._render_repository_index(formatter.context, all_repos, |
| | 793 | order, desc) |
| 789 | 794 | |
| 790 | 795 | add_stylesheet(formatter.req, 'common/css/browser.css') |
| | 796 | data = {'repo': repo, 'desc': desc and 1 or None, |
| | 797 | 'reponame': None, 'path': '/', 'stickyrev': None} |
| 791 | 798 | from trac.web.chrome import Chrome |
| 792 | 799 | return Chrome(self.env).render_template( |
| 793 | | formatter.req, 'repository_index.html', |
| 794 | | {'repo': data}, None, fragment=True) |
| | 800 | formatter.req, 'repository_index.html', data, None, |
| | 801 | fragment=True) |
| 795 | 802 | |
| 796 | | def repolink(reponame): |
| | 803 | def repolink(reponame, repos): |
| 797 | 804 | label = reponame or _('(default)') |
| 798 | 805 | return Markup(tag.a(label, |
| 799 | 806 | title=_('View repository %(repo)s', repo=label), |
| 800 | | href=formatter.href.browser(reponame or None))) |
| | 807 | href=formatter.href.browser(repos.reponame or None))) |
| 801 | 808 | |
| 802 | | all_repos = sorted( |
| 803 | | (reponame, info) for reponame, info in all_repos.iteritems() |
| 804 | | if info.get('hidden') not in _TRUE_VALUE |
| 805 | | and rm.get_repository(reponame).can_view(formatter.perm)) |
| | 809 | all_repos = dict((reponame, rm.get_repository(reponame)) |
| | 810 | for reponame in all_repos) |
| | 811 | all_repos = sorted((reponame, repos) for reponame, repos in all_repos |
| | 812 | if repos |
| | 813 | and repos.params.get('hidden') not in _TRUE_VALUE |
| | 814 | and repos.can_view(formatter.perm)) |
| 806 | 815 | |
| 807 | 816 | if format == 'list': |
| 808 | 817 | return tag.dl([ |
| 809 | | tag(tag.dt(repolink(reponame)), |
| 810 | | tag.dd(repoinfo.get('description'))) |
| 811 | | for reponame, repoinfo in all_repos]) |
| | 818 | tag(tag.dt(repolink(reponame, repos)), |
| | 819 | tag.dd(repos.params.get('description'))) |
| | 820 | for reponame, repos in all_repos]) |
| 812 | 821 | else: # compact |
| 813 | | return Markup(', ').join([repolink(reponame) |
| 814 | | for reponame, repoinfo in all_repos]) |
| | 822 | return Markup(', ').join([repolink(reponame, repos) |
| | 823 | for reponame, repos in all_repos]) |
| 815 | 824 | |
| 816 | 825 | |
| 817 | 826 | |
| … |
… |
|
| 820 | 829 | def __init__(self, env, context): |
| 821 | 830 | self.env = env |
| 822 | 831 | self.context = context |
| 823 | | self.reponame = context.resource.parent.id |
| | 832 | self.repos = env.get_repository(context.resource.parent.id) |
| 824 | 833 | self.path = context.resource.id |
| 825 | | self.repos = env.get_repository(self.reponame) |
| 826 | 834 | self.rev = context.resource.version |
| 827 | 835 | # maintain state |
| 828 | 836 | self.prev_chgset = None |
| … |
… |
|
| 876 | 884 | |
| 877 | 885 | # -- compute anchor and style once per revision |
| 878 | 886 | if rev not in self.chgset_data: |
| 879 | | chgset_href = self.context.href.changeset(rev, |
| 880 | | self.reponame or None, |
| 881 | | path) |
| | 887 | chgset_href = \ |
| | 888 | self.context.href.changeset(rev, self.repos.reponame or None, |
| | 889 | path) |
| 882 | 890 | short_author = chgset.author.split(' ', 1)[0] |
| 883 | 891 | title = shorten_line('%s: %s' % (short_author, chgset.message)) |
| 884 | 892 | anchor = tag.a('[%s]' % self.repos.short_rev(rev), # shortname |
diff --git a/trac/versioncontrol/web_ui/changeset.py b/trac/versioncontrol/web_ui/changeset.py
|
a
|
b
|
|
| 34 | 34 | from trac.resource import Resource, ResourceNotFound |
| 35 | 35 | from trac.search import ISearchSource, search_to_sql, shorten_result |
| 36 | 36 | from trac.timeline.api import ITimelineEventProvider |
| 37 | | from trac.util import embedded_numbers, content_disposition |
| | 37 | from trac.util import content_disposition, embedded_numbers, pathjoin |
| 38 | 38 | from trac.util.compat import any, set |
| 39 | 39 | from trac.util.datefmt import pretty_timedelta, utc |
| 40 | 40 | from trac.util.text import exception_to_unicode, to_unicode, \ |
| … |
… |
|
| 252 | 252 | try: |
| 253 | 253 | new_path = repos.normalize_path(new_path) |
| 254 | 254 | new = repos.normalize_rev(new) |
| | 255 | full_new_path = '/' + pathjoin(repos.reponame, new_path) |
| 255 | 256 | old_path = repos.normalize_path(old_path or new_path) |
| 256 | 257 | old = repos.normalize_rev(old or new) |
| | 258 | full_old_path = '/' + pathjoin(repos.reponame, old_path) |
| 257 | 259 | except NoSuchChangeset, e: |
| 258 | 260 | raise ResourceNotFound(e.message, _('Invalid Changeset Number')) |
| 259 | 261 | |
| … |
… |
|
| 269 | 271 | else: |
| 270 | 272 | restricted = old_path == new_path # (same path or not) |
| 271 | 273 | |
| 272 | | # -- redirect if changing the diff options |
| 273 | | if req.args.has_key('update'): |
| | 274 | # -- redirect if changing the diff options or alias requested |
| | 275 | if req.args.has_key('update') or reponame != repos.reponame: |
| | 276 | reponame = repos.reponame or None |
| 274 | 277 | if chgset: |
| 275 | 278 | if restricted: |
| 276 | | req.redirect(req.href.changeset(new, reponame or None, |
| 277 | | new_path)) |
| | 279 | req.redirect(req.href.changeset(new, reponame, new_path)) |
| 278 | 280 | else: |
| 279 | | req.redirect(req.href.changeset(new, reponame or None)) |
| | 281 | req.redirect(req.href.changeset(new, reponame)) |
| 280 | 282 | else: |
| 281 | | req.redirect(req.href.changeset(new, reponame or None, |
| | 283 | req.redirect(req.href.changeset(new, reponame, |
| 282 | 284 | new_path, old=old, |
| 283 | 285 | old_path=full_old_path)) |
| 284 | 286 | |
| … |
… |
|
| 300 | 302 | old_path = new_path |
| 301 | 303 | data = {'old_path': old_path, 'old_rev': old, |
| 302 | 304 | 'new_path': new_path, 'new_rev': new} |
| 303 | | data['reponame'] = reponame or None |
| 304 | | data['diff'] = diff_data |
| 305 | | data['wiki_format_messages'] = self.wiki_format_messages |
| | 305 | data.update({'repos': repos, 'reponame': repos.reponame or None, |
| | 306 | 'diff': diff_data, |
| | 307 | 'wiki_format_messages': self.wiki_format_messages}) |
| 306 | 308 | |
| 307 | 309 | if chgset: |
| 308 | 310 | chgset = repos.get_changeset(new) |
| … |
… |
|
| 340 | 342 | self._render_zip(req, filename, repos, data) |
| 341 | 343 | |
| 342 | 344 | # -- HTML format |
| 343 | | self._render_html(req, reponame, repos, chgset, restricted, xhr, data) |
| | 345 | self._render_html(req, repos, chgset, restricted, xhr, data) |
| 344 | 346 | |
| 345 | 347 | if chgset: |
| 346 | 348 | diff_params = 'new=%s' % new |
| 347 | 349 | else: |
| 348 | | diff_params = unicode_urlencode({'new_path': new_path, |
| 349 | | 'new': new, |
| 350 | | 'old_path': old_path, |
| 351 | | 'old': old}) |
| 352 | | add_link(req, 'alternate', '?format=diff&'+diff_params, |
| | 350 | diff_params = unicode_urlencode({'new_path': new_path, 'new': new, |
| | 351 | 'old_path': old_path, 'old': old}) |
| | 352 | add_link(req, 'alternate', '?format=diff&' + diff_params, |
| 353 | 353 | _('Unified Diff'), 'text/plain', 'diff') |
| 354 | | add_link(req, 'alternate', '?format=zip&'+diff_params, _('Zip Archive'), |
| 355 | | 'application/zip', 'zip') |
| | 354 | add_link(req, 'alternate', '?format=zip&' + diff_params, |
| | 355 | _('Zip Archive'), 'application/zip', 'zip') |
| 356 | 356 | add_script(req, 'common/js/diff.js') |
| 357 | 357 | add_stylesheet(req, 'common/css/changeset.css') |
| 358 | 358 | add_stylesheet(req, 'common/css/diff.css') |
| … |
… |
|
| 371 | 371 | |
| 372 | 372 | # Internal methods |
| 373 | 373 | |
| 374 | | def _render_html(self, req, reponame, repos, chgset, restricted, xhr, data): |
| | 374 | def _render_html(self, req, repos, chgset, restricted, xhr, data): |
| 375 | 375 | """HTML version""" |
| 376 | 376 | data['restricted'] = restricted |
| 377 | 377 | browser = BrowserModule(self.env) |
| | 378 | reponame = repos.reponame or None |
| 378 | 379 | |
| 379 | 380 | if chgset: # Changeset Mode (possibly restricted on a path) |
| 380 | 381 | path, rev = data['new_path'], data['new_rev'] |
| … |
… |
|
| 405 | 406 | title = _changeset_title(rev) |
| 406 | 407 | |
| 407 | 408 | # Support for revision properties (#2545) |
| 408 | | repos_resource = Resource('repository', reponame) |
| 409 | 409 | context = Context.from_request(req, 'changeset', chgset.rev, |
| 410 | | parent=repos_resource) |
| | 410 | parent=repos.resource) |
| 411 | 411 | data['context'] = context |
| 412 | 412 | revprops = chgset.get_properties() |
| 413 | 413 | data['properties'] = browser.render_properties('revprop', context, |
| … |
… |
|
| 419 | 419 | if prev: |
| 420 | 420 | prev_path, prev_rev = prev[:2] |
| 421 | 421 | if prev_rev: |
| 422 | | prev_href = req.href.changeset(prev_rev, |
| 423 | | reponame or None, |
| | 422 | prev_href = req.href.changeset(prev_rev, reponame, |
| 424 | 423 | prev_path) |
| 425 | 424 | else: |
| 426 | 425 | prev_path = prev_rev = None |
| 427 | 426 | else: |
| 428 | 427 | add_link(req, 'first', |
| 429 | | req.href.changeset(oldest_rev, reponame or None), |
| | 428 | req.href.changeset(oldest_rev, reponame), |
| 430 | 429 | _('Changeset %(id)s', id=oldest_rev)) |
| 431 | 430 | prev_path = data['old_path'] |
| 432 | 431 | prev_rev = repos.previous_rev(chgset.rev) |
| 433 | 432 | if prev_rev: |
| 434 | | prev_href = req.href.changeset(prev_rev, |
| 435 | | reponame or None) |
| | 433 | prev_href = req.href.changeset(prev_rev, reponame) |
| 436 | 434 | if prev_rev: |
| 437 | 435 | add_link(req, 'prev', prev_href, _changeset_title(prev_rev)) |
| 438 | 436 | youngest_rev = repos.youngest_rev |
| … |
… |
|
| 441 | 439 | next_rev = repos.next_rev(chgset.rev, path) |
| 442 | 440 | if next_rev: |
| 443 | 441 | if repos.has_node(path, next_rev): |
| 444 | | next_href = req.href.changeset(next_rev, |
| 445 | | reponame or None, |
| | 442 | next_href = req.href.changeset(next_rev, reponame, |
| 446 | 443 | path) |
| 447 | 444 | else: # must be a 'D'elete or 'R'ename, show full cset |
| 448 | | next_href = req.href.changeset(next_rev, |
| 449 | | reponame or None) |
| | 445 | next_href = req.href.changeset(next_rev, reponame) |
| 450 | 446 | else: |
| 451 | 447 | add_link(req, 'last', |
| 452 | | req.href.changeset(youngest_rev, |
| 453 | | reponame or None), |
| | 448 | req.href.changeset(youngest_rev, reponame), |
| 454 | 449 | _('Changeset %(id)s', id=youngest_rev)) |
| 455 | 450 | next_rev = repos.next_rev(chgset.rev) |
| 456 | 451 | if next_rev: |
| 457 | | next_href = req.href.changeset(next_rev, |
| 458 | | reponame or None) |
| | 452 | next_href = req.href.changeset(next_rev, reponame) |
| 459 | 453 | if next_rev: |
| 460 | 454 | add_link(req, 'next', next_href, _changeset_title(next_rev)) |
| 461 | 455 | |
| … |
… |
|
| 478 | 472 | return {'path': node.path, |
| 479 | 473 | 'rev': node.rev, |
| 480 | 474 | 'shortrev': repos.short_rev(node.rev), |
| 481 | | 'href': req.href.browser(reponame or None, |
| | 475 | 'href': req.href.browser(reponame, |
| 482 | 476 | node.created_path, |
| 483 | 477 | rev=node.created_rev, |
| 484 | 478 | annotate=annotated and 'blame' or \ |
| … |
… |
|
| 623 | 617 | filestats[change] += 1 |
| 624 | 618 | if change in Changeset.DIFF_CHANGES: |
| 625 | 619 | if chgset: |
| 626 | | href = req.href.changeset(new_node.rev, |
| 627 | | reponame or None, |
| | 620 | href = req.href.changeset(new_node.rev, reponame, |
| 628 | 621 | new_node.path) |
| 629 | 622 | title = _('Show the changeset %(id)s restricted to ' |
| 630 | 623 | '%(path)s', id=new_node.rev, |
| 631 | 624 | path=new_node.path) |
| 632 | 625 | else: |
| 633 | 626 | href = req.href.changeset( |
| 634 | | new_node.created_rev, reponame or None, |
| | 627 | new_node.created_rev, reponame, |
| 635 | 628 | new_node.created_path, |
| 636 | 629 | old=old_node.created_rev, |
| 637 | | old_path=posixpath.join(reponame, |
| 638 | | old_node.created_path)) |
| | 630 | old_path=pathjoin(repos.reponame, |
| | 631 | old_node.created_path)) |
| 639 | 632 | title = _('Show the %(range)s differences restricted ' |
| 640 | 633 | 'to %(path)s', |
| 641 | 634 | range='r%s:%s' % (old_node.rev, new_node.rev), |
| … |
… |
|
| 713 | 706 | else: |
| 714 | 707 | old_node_path = repos.normalize_path(old_node.path) |
| 715 | 708 | diff_old_path = repos.normalize_path(data['old_path']) |
| 716 | | new_path = posixpath.join(data['new_path'], |
| 717 | | old_node_path[len(diff_old_path)+1:]) |
| | 709 | new_path = pathjoin(data['new_path'], |
| | 710 | old_node_path[len(diff_old_path) + 1:]) |
| 718 | 711 | |
| 719 | 712 | if old_content != new_content: |
| 720 | 713 | options = data['diff']['options'] |
| … |
… |
|
| 845 | 838 | # only repository filter. |
| 846 | 839 | filters = [] |
| 847 | 840 | rm = RepositoryManager(self.env) |
| 848 | | repositories = rm.get_all_repositories() |
| | 841 | repositories = rm.get_real_repositories() |
| 849 | 842 | if len(repositories) > 1: |
| 850 | | visible_repos = set( |
| 851 | | name for name, info in repositories.items() |
| 852 | | if info.get('hidden') not in _TRUE_VALUES |
| 853 | | and rm.get_repository(name).can_view(req.perm)) |
| 854 | | default_is_aliased = any(info.get('alias') == '' and |
| 855 | | name in visible_repos |
| 856 | | for name, info in repositories.items()) |
| 857 | | default_is_alias = repositories.get('', {}).get('alias') \ |
| 858 | | in visible_repos |
| 859 | | for reponame in repositories.keys(): |
| 860 | | if reponame: |
| 861 | | label = reponame |
| 862 | | elif default_is_aliased or default_is_alias: |
| 863 | | continue |
| 864 | | else: |
| 865 | | label = _('(default)') |
| 866 | | if reponame in visible_repos: |
| 867 | | filters.append(('repo-' + reponame, |
| 868 | | u"\xa0\xa0-\xa0" + label)) |
| | 843 | filters = [ |
| | 844 | ('repo-' + repos.reponame, |
| | 845 | u"\xa0\xa0-\xa0" + (repos.reponame or _('(default)'))) |
| | 846 | for repos in repositories |
| | 847 | if repos.params.get('hidden') not in _TRUE_VALUES |
| | 848 | and repos.can_view(req.perm)] |
| 869 | 849 | filters.sort() |
| 870 | 850 | add_script(req, 'common/js/timeline_multirepos.js') |
| 871 | 851 | changeset_label = _('Changesets in all repositories') |
| … |
… |
|
| 893 | 873 | collapse_changesets = lambda c: c.rev |
| 894 | 874 | |
| 895 | 875 | uids_seen = {} |
| 896 | | def generate_changesets(reponame, repos): |
| 897 | | repos_resource = Resource('repository', reponame) |
| | 876 | def generate_changesets(repos): |
| 898 | 877 | for _, changesets in groupby(repos.get_changesets(start, stop), |
| 899 | 878 | key=collapse_changesets): |
| 900 | 879 | viewable_changesets = [] |
| 901 | 880 | for cset in changesets: |
| 902 | 881 | cset_resource = Resource('changeset', cset.rev, |
| 903 | | parent=repos_resource) |
| | 882 | parent=repos.resource) |
| 904 | 883 | if cset.can_view(req.perm): |
| 905 | | repos_for_uid = [reponame] |
| | 884 | repos_for_uid = [repos.reponame] |
| 906 | 885 | uid = repos.get_changeset_uid(cset.rev) |
| 907 | 886 | if uid: |
| 908 | 887 | # uid can be seen in multiple repositories |
| 909 | 888 | if uid in uids_seen: |
| 910 | | uids_seen[uid].append(reponame) |
| | 889 | uids_seen[uid].append(repos.reponame) |
| 911 | 890 | continue # already viewable, simply append |
| 912 | 891 | uids_seen[uid] = repos_for_uid |
| 913 | 892 | viewable_changesets.append((cset, cset_resource, |
| … |
… |
|
| 919 | 898 | show_location, show_files)) |
| 920 | 899 | |
| 921 | 900 | rm = RepositoryManager(self.env) |
| 922 | | for reponame in rm.get_all_repositories(): |
| 923 | | if all_repos or ('repo-' + reponame) in repo_filters: |
| | 901 | for repos in rm.get_real_repositories(): |
| | 902 | if all_repos or ('repo-' + repos.reponame) in repo_filters: |
| 924 | 903 | try: |
| 925 | | repos = rm.get_repository(reponame) |
| 926 | | for event in generate_changesets(reponame, repos): |
| | 904 | for event in generate_changesets(repos): |
| 927 | 905 | yield event |
| 928 | 906 | except TracError, e: |
| 929 | 907 | self.log.error("Timeline event provider for repository" |
| 930 | 908 | " '%s' failed: %r", |
| 931 | | reponame, exception_to_unicode(e)) |
| | 909 | repos.reponame, exception_to_unicode(e)) |
| 932 | 910 | |
| 933 | 911 | def render_timeline_event(self, context, field, event): |
| 934 | 912 | changesets, show_location, show_files = event[3] |
| … |
… |
|
| 1062 | 1040 | try: |
| 1063 | 1041 | changeset = repos.get_changeset(rev) |
| 1064 | 1042 | if changeset.can_view(formatter.perm): |
| 1065 | | href = formatter.href.changeset(rev, reponame or None, |
| | 1043 | href = formatter.href.changeset(rev, |
| | 1044 | repos.reponame or None, |
| 1066 | 1045 | path) |
| 1067 | 1046 | return tag.a(label, class_="changeset", |
| 1068 | 1047 | title=shorten_line(changeset.message), |
| … |
… |
|
| 1119 | 1098 | def get_search_results(self, req, terms, filters): |
| 1120 | 1099 | if not 'changeset' in filters: |
| 1121 | 1100 | return |
| | 1101 | rm = RepositoryManager(self.env) |
| | 1102 | repositories = dict((repos.params['id'], repos) |
| | 1103 | for repos in rm.get_real_repositories()) |
| 1122 | 1104 | db = self.env.get_db_cnx() |
| 1123 | 1105 | sql, args = search_to_sql(db, ['rev', 'message', 'author'], terms) |
| 1124 | 1106 | cursor = db.cursor() |
| 1125 | 1107 | cursor.execute("SELECT repos,rev,time,author,message " |
| 1126 | 1108 | "FROM revision WHERE " + sql, args) |
| 1127 | | for reponame, rev, ts, author, log in cursor: |
| 1128 | | repos = self.env.get_repository(reponame) |
| | 1109 | for id, rev, ts, author, log in cursor: |
| | 1110 | repos = repositories.get(id) |
| 1129 | 1111 | if not repos: |
| 1130 | 1112 | continue # revisions for a no longer active repository |
| 1131 | | cset = Resource('repository', repos.reponame).child('changeset', |
| 1132 | | rev) |
| | 1113 | cset = repos.resource.child('changeset', rev) |
| 1133 | 1114 | if 'CHANGESET_VIEW' in req.perm(cset): |
| 1134 | | yield (req.href.changeset(rev, reponame or None), |
| | 1115 | yield (req.href.changeset(rev, repos.reponame or None), |
| 1135 | 1116 | '[%s]: %s' % (rev, shorten_line(log)), |
| 1136 | 1117 | datetime.fromtimestamp(ts, utc), author, |
| 1137 | 1118 | shorten_result(log, terms)) |
| … |
… |
|
| 1159 | 1140 | |
| 1160 | 1141 | if repos: |
| 1161 | 1142 | entries = [(e.isdir, e.name, |
| 1162 | | '/' + posixpath.join(reponame, e.path)) |
| | 1143 | '/' + pathjoin(repos.reponame, e.path)) |
| 1163 | 1144 | for e in repos.get_node(path).get_entries() |
| 1164 | 1145 | if e.can_view(req.perm)] |
| 1165 | 1146 | if not reponame: |
| 1166 | | entries.extend((True, name, '/' + name) |
| 1167 | | for name in rm.get_all_repositories() |
| 1168 | | if rm.get_repository(name).can_view(req.perm)) |
| | 1147 | entries.extend((True, repos.reponame, '/' + repos.reponame) |
| | 1148 | for repos in rm.get_real_repositories() |
| | 1149 | if repos.can_view(req.perm)) |
| 1169 | 1150 | |
| 1170 | 1151 | elem = tag.ul( |
| 1171 | 1152 | [tag.li(isdir and tag.b(path) or path) |
| … |
… |
|
| 1192 | 1173 | old_rev = old_repos.normalize_rev(old_rev) |
| 1193 | 1174 | |
| 1194 | 1175 | # -- prepare rendering |
| 1195 | | data = {'new_path': posixpath.join(new_reponame, new_path), |
| | 1176 | data = {'new_path': '/' + pathjoin(new_repos.reponame, new_path), |
| 1196 | 1177 | 'new_rev': new_rev, |
| 1197 | | 'old_path': posixpath.join(old_reponame, old_path), |
| | 1178 | 'old_path': '/' + pathjoin(old_repos.reponame, old_path), |
| 1198 | 1179 | 'old_rev': old_rev} |
| 1199 | 1180 | |
| 1200 | 1181 | add_script(req, 'common/js/suggest.js') |
diff --git a/trac/versioncontrol/web_ui/log.py b/trac/versioncontrol/web_ui/log.py
|
a
|
b
|
|
| 25 | 25 | from trac.core import * |
| 26 | 26 | from trac.mimeview import Context |
| 27 | 27 | from trac.perm import IPermissionRequestor |
| 28 | | from trac.resource import Resource |
| 29 | 28 | from trac.util import Ranges |
| 30 | 29 | from trac.util.compat import any |
| 31 | 30 | from trac.util.html import html |
| … |
… |
|
| 82 | 81 | verbose = req.args.get('verbose') |
| 83 | 82 | limit = int(req.args.get('limit') or self.default_log_limit) |
| 84 | 83 | |
| 85 | | reponame, repos, path = RepositoryManager(self.env).\ |
| 86 | | get_repository_by_path(path) |
| | 84 | rm = RepositoryManager(self.env) |
| | 85 | reponame, repos, path = rm.get_repository_by_path(path) |
| | 86 | |
| | 87 | if not repos: |
| | 88 | raise ResourceNotFound(_("No repository '%(repo)s' found", |
| | 89 | repo=reponame)) |
| | 90 | |
| | 91 | if reponame != repos.reponame: # Redirect alias |
| | 92 | qs = req.query_string |
| | 93 | req.redirect(req.href.log(repos.reponame or None, path) |
| | 94 | + (qs and '?' + qs or '')) |
| 87 | 95 | |
| 88 | 96 | normpath = repos.normalize_path(path) |
| 89 | 97 | # if `revs` parameter is given, then we're restricted to the |
| … |
… |
|
| 198 | 206 | params.update(args) |
| 199 | 207 | if verbose: |
| 200 | 208 | params['verbose'] = verbose |
| 201 | | return req.href.log(reponame or None, path, **params) |
| | 209 | return req.href.log(repos.reponame or None, path, **params) |
| 202 | 210 | |
| 203 | 211 | if format in ('rss', 'changelog'): |
| 204 | 212 | info = [i for i in info if i['change']] # drop separators |
| … |
… |
|
| 247 | 255 | cs['actions'] = actions |
| 248 | 256 | extra_changes[rev] = cs |
| 249 | 257 | |
| 250 | | repos_resource = Resource('repository', reponame) |
| 251 | 258 | data = { |
| 252 | 259 | 'context': Context.from_request(req, 'source', path, |
| 253 | | parent=repos_resource), |
| 254 | | 'reponame': reponame or None, 'repos': repos, |
| | 260 | parent=repos.resource), |
| | 261 | 'reponame': repos.reponame or None, 'repos': repos, |
| 255 | 262 | 'path': path, 'rev': rev, 'stop_rev': stop_rev, |
| 256 | 263 | 'revranges': revranges, |
| 257 | 264 | 'mode': mode, 'verbose': verbose, 'limit' : limit, |
| … |
… |
|
| 265 | 272 | return 'revisionlog.txt', data, 'text/plain' |
| 266 | 273 | elif req.args.get('format') == 'rss': |
| 267 | 274 | data['context'] = Context.from_request(req, 'source', |
| 268 | | path, parent=repos_resource, |
| | 275 | path, parent=repos.resource, |
| 269 | 276 | absurls=True) |
| 270 | 277 | return 'revisionlog.rss', data, 'application/rss+xml' |
| 271 | 278 | |
| … |
… |
|
| 286 | 293 | add_stylesheet(req, 'common/css/diff.css') |
| 287 | 294 | add_stylesheet(req, 'common/css/browser.css') |
| 288 | 295 | |
| 289 | | path_links = get_path_links(req.href, reponame, path, rev) |
| | 296 | path_links = get_path_links(req.href, repos.reponame, path, rev) |
| 290 | 297 | if path_links: |
| 291 | 298 | data['path_links'] = path_links |
| 292 | 299 | if len(path_links) > 1: |
| … |
… |
|
| 301 | 308 | add_link(req, 'alternate', changelog_href, _('ChangeLog'), 'text/plain') |
| 302 | 309 | |
| 303 | 310 | add_ctxtnav(req, _('View Latest Revision'), |
| 304 | | href=req.href.browser(reponame or None, path)) |
| | 311 | href=req.href.browser(repos.reponame or None, path)) |
| 305 | 312 | if 'next' in req.chrome['links']: |
| 306 | 313 | next = req.chrome['links']['next'][0] |
| 307 | 314 | add_ctxtnav(req, tag.span(tag.a(_('Older Revisions'), |
| … |
… |
|
| 368 | 375 | revs = None |
| 369 | 376 | if 'LOG_VIEW' in formatter.perm: |
| 370 | 377 | if revranges: |
| 371 | | href = formatter.href.log(reponame or None, path or '/', |
| | 378 | href = formatter.href.log(repos.reponame or None, path or '/', |
| 372 | 379 | revs=str(revranges)) |
| 373 | 380 | else: |
| 374 | 381 | try: |
| 375 | 382 | rev = repos.normalize_rev(revs) |
| 376 | 383 | except NoSuchChangeset: |
| 377 | 384 | rev = None |
| 378 | | href = formatter.href.log(reponame or None, path or '/', |
| | 385 | href = formatter.href.log(repos.reponame or None, path or '/', |
| 379 | 386 | rev=rev) |
| 380 | 387 | if query and (revranges or revs): |
| 381 | 388 | query = '&' + query[1:] |
diff --git a/trac/versioncontrol/web_ui/tests/wikisyntax.py b/trac/versioncontrol/web_ui/tests/wikisyntax.py
|
a
|
b
|
|
| 3 | 3 | import unittest |
| 4 | 4 | |
| 5 | 5 | from trac.test import Mock |
| 6 | | from trac.wiki.tests import formatter |
| 7 | 6 | from trac.versioncontrol import NoSuchChangeset |
| 8 | 7 | from trac.versioncontrol.api import * |
| 9 | 8 | from trac.versioncontrol.web_ui import * |
| | 9 | from trac.wiki.tests import formatter |
| 10 | 10 | |
| 11 | 11 | |
| 12 | 12 | def _get_changeset(rev): |
| … |
… |
|
| 25 | 25 | raise NoSuchChangeset(rev) |
| 26 | 26 | |
| 27 | 27 | def _get_repository(reponame): |
| 28 | | return Mock(get_changeset=_get_changeset, youngest_rev='200', |
| | 28 | return Mock(reponame=reponame, youngest_rev='200', |
| | 29 | get_changeset=_get_changeset, |
| 29 | 30 | normalize_rev=_normalize_rev) |
| 30 | 31 | |
| 31 | 32 | def repository_setup(tc): |
diff --git a/trac/web/chrome.py b/trac/web/chrome.py
|
a
|
b
|
|
| 58 | 58 | from trac.mimeview import get_mimetype, Context |
| 59 | 59 | from trac.resource import * |
| 60 | 60 | from trac.util import compat, get_reporter_id, presentation, get_pkginfo, \ |
| 61 | | translation |
| | 61 | pathjoin, translation |
| 62 | 62 | from trac.util.compat import any, partial |
| 63 | 63 | from trac.util.html import escape, plaintext |
| 64 | 64 | from trac.util.text import pretty_size, obfuscate_email_address, \ |
| … |
… |
|
| 366 | 366 | 'ngettext': translation.ngettext, |
| 367 | 367 | 'paginate': presentation.paginate, |
| 368 | 368 | 'partial': partial, |
| | 369 | 'pathjoin': pathjoin, |
| 369 | 370 | 'plaintext': plaintext, |
| 370 | 371 | 'pprint': pprint.pformat, |
| 371 | 372 | 'pretty_size': pretty_size, |
diff --git a/tracopt/ticket/commit_updater.py b/tracopt/ticket/commit_updater.py
|
a
|
b
|
|
| 43 | 43 | from trac.config import BoolOption, Option |
| 44 | 44 | from trac.core import Component, implements |
| 45 | 45 | from trac.perm import PermissionCache |
| 46 | | from trac.resource import Resource |
| 47 | 46 | from trac.ticket import Ticket |
| 48 | 47 | from trac.ticket.notification import TicketNotifyEmail |
| 49 | 48 | from trac.ticket.web_ui import TicketModule |
| … |
… |
|
| 276 | 275 | reponame = args.get('repository') |
| 277 | 276 | rev = args.get('revision') |
| 278 | 277 | repos = RepositoryManager(self.env).get_repository(reponame) |
| 279 | | changeset = repos.get_changeset(rev) |
| | 278 | if repos: |
| | 279 | changeset = repos.get_changeset(rev) |
| | 280 | message = changeset.message |
| | 281 | rev = changeset.rev |
| | 282 | else: |
| | 283 | message = content |
| 280 | 284 | if formatter.context.resource.realm == 'ticket': |
| 281 | 285 | ticket_re = CommitTicketUpdater.ticket_re |
| 282 | 286 | if not any(int(tkt_id) == formatter.context.resource.id |
| 283 | | for tkt_id in ticket_re.findall(changeset.message)): |
| | 287 | for tkt_id in ticket_re.findall(message)): |
| 284 | 288 | return tag.p("(The changeset message doesn't reference this " |
| 285 | 289 | "ticket)", class_='hint') |
| 286 | 290 | if ChangesetModule(self.env).wiki_format_messages: |
| 287 | 291 | return tag.div(format_to_html(self.env, |
| 288 | | formatter.context('changeset', changeset.rev, |
| 289 | | parent=Resource('repository', reponame)), |
| 290 | | changeset.message, escape_newlines=True), class_='message') |
| | 292 | formatter.context('changeset', rev, parent=repos.resource), |
| | 293 | message, escape_newlines=True), class_='message') |
| 291 | 294 | else: |
| 292 | | return tag.pre(changeset.message, class_='message') |
| | 295 | return tag.pre(message, class_='message') |