# Copyright (C) 2010-2024 RhodeCode GmbH # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU Affero General Public License, version 3 # (only), as published by the Free Software Foundation. # # 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. # # You should have received a copy of the GNU Affero General Public License # along with this program. If not, see . # # This program is dual-licensed. If you wish to learn more about the # RhodeCode Enterprise Edition, including its added features, Support services, # and proprietary license terms, please see https://rhodecode.com/licenses/ import logging from pyramid.httpexceptions import HTTPNotFound, HTTPFound from pyramid.renderers import render from pyramid.response import Response from rhodecode.apps._base import RepoAppView import rhodecode.lib.helpers as h from rhodecode.lib import ext_json from rhodecode.lib.auth import ( LoginRequired, HasRepoPermissionAnyDecorator) from rhodecode.lib.graphmod import _colored, _dagwalker from rhodecode.lib.helpers import RepoPage from rhodecode.lib.utils2 import str2bool from rhodecode.lib.str_utils import safe_int, safe_str from rhodecode.lib.vcs.exceptions import ( RepositoryError, CommitDoesNotExistError, CommitError, NodeDoesNotExistError, EmptyRepositoryError) log = logging.getLogger(__name__) DEFAULT_CHANGELOG_SIZE = 20 class RepoChangelogView(RepoAppView): def _get_commit_or_redirect(self, commit_id, redirect_after=True): """ This is a safe way to get commit. If an error occurs it redirects to tip with proper message :param commit_id: id of commit to fetch :param redirect_after: toggle redirection """ _ = self.request.translate try: return self.rhodecode_vcs_repo.get_commit(commit_id) except EmptyRepositoryError: if not redirect_after: return None h.flash(h.literal( _('There are no commits yet')), category='warning') raise HTTPFound( h.route_path('repo_summary', repo_name=self.db_repo_name)) except (CommitDoesNotExistError, LookupError): msg = _('No such commit exists for this repository') h.flash(msg, category='error') raise HTTPNotFound() except RepositoryError as e: h.flash(h.escape(safe_str(e)), category='error') raise HTTPNotFound() def _graph(self, repo, commits, prev_data=None, next_data=None): """ Generates a DAG graph for repo :param repo: repo instance :param commits: list of commits """ if not commits: return ext_json.str_json([]), ext_json.str_json([]) def serialize(commit, parents=True): data = dict( raw_id=commit.raw_id, idx=commit.idx, branch=None, ) if parents: data['parents'] = [ serialize(x, parents=False) for x in commit.parents] return data prev_data = prev_data or [] next_data = next_data or [] current = [serialize(x) for x in commits] commits = prev_data + current + next_data dag = _dagwalker(repo, commits) data = [[commit_id, vtx, edges, branch] for commit_id, vtx, edges, branch in _colored(dag)] return ext_json.str_json(data), ext_json.str_json(current) def _check_if_valid_branch(self, branch_name, repo_name, f_path): if branch_name not in self.rhodecode_vcs_repo.branches_all: h.flash(f'Branch {h.escape(safe_str(branch_name))} is not found.', category='warning') redirect_url = h.route_path( 'repo_commits_file', repo_name=repo_name, commit_id=branch_name, f_path=f_path or '') raise HTTPFound(redirect_url) def _load_changelog_data( self, c, collection, page, chunk_size, branch_name=None, dynamic=False, f_path=None, commit_id=None): def url_generator(page_num): query_params = { 'page': page_num } if branch_name: query_params.update({ 'branch': branch_name }) if f_path: # changelog for file return h.route_path( 'repo_commits_file', repo_name=c.rhodecode_db_repo.repo_name, commit_id=commit_id, f_path=f_path, _query=query_params) else: return h.route_path( 'repo_commits', repo_name=c.rhodecode_db_repo.repo_name, _query=query_params) c.total_cs = len(collection) c.showing_commits = min(chunk_size, c.total_cs) c.pagination = RepoPage(collection, page=page, item_count=c.total_cs, items_per_page=chunk_size, url_maker=url_generator) c.next_page = c.pagination.next_page c.prev_page = c.pagination.previous_page if dynamic: if self.request.GET.get('chunk') != 'next': c.next_page = None if self.request.GET.get('chunk') != 'prev': c.prev_page = None page_commit_ids = [x.raw_id for x in c.pagination] c.comments = c.rhodecode_db_repo.get_comments(page_commit_ids) c.statuses = c.rhodecode_db_repo.statuses(page_commit_ids) def load_default_context(self): c = self._get_local_tmpl_context(include_app_defaults=True) c.rhodecode_repo = self.rhodecode_vcs_repo return c @LoginRequired() @HasRepoPermissionAnyDecorator( 'repository.read', 'repository.write', 'repository.admin') def repo_changelog(self): c = self.load_default_context() commit_id = self.request.matchdict.get('commit_id') f_path = self._get_f_path(self.request.matchdict) show_hidden = str2bool(self.request.GET.get('evolve')) chunk_size = 20 c.branch_name = branch_name = self.request.GET.get('branch') or '' c.book_name = book_name = self.request.GET.get('bookmark') or '' c.f_path = f_path c.commit_id = commit_id c.show_hidden = show_hidden hist_limit = safe_int(self.request.GET.get('limit')) or None p = safe_int(self.request.GET.get('page', 1), 1) c.selected_name = branch_name or book_name if not commit_id and branch_name: self._check_if_valid_branch(branch_name, self.db_repo_name, f_path) c.changelog_for_path = f_path pre_load = self.get_commit_preload_attrs() partial_xhr = self.request.environ.get('HTTP_X_PARTIAL_XHR') try: if f_path: log.debug('generating changelog for path %s', f_path) # get the history for the file ! base_commit = self.rhodecode_vcs_repo.get_commit(commit_id) try: collection = base_commit.get_path_history( f_path, limit=hist_limit, pre_load=pre_load) if collection and partial_xhr: # for ajax call we remove first one since we're looking # at it right now in the context of a file commit collection.pop(0) except (NodeDoesNotExistError, CommitError): # this node is not present at tip! try: commit = self._get_commit_or_redirect(commit_id) collection = commit.get_path_history(f_path) except RepositoryError as e: h.flash(safe_str(e), category='warning') redirect_url = h.route_path( 'repo_commits', repo_name=self.db_repo_name) raise HTTPFound(redirect_url) collection = list(reversed(collection)) else: collection = self.rhodecode_vcs_repo.get_commits( branch_name=branch_name, show_hidden=show_hidden, pre_load=pre_load, translate_tags=False) self._load_changelog_data( c, collection, p, chunk_size, c.branch_name, f_path=f_path, commit_id=commit_id) except EmptyRepositoryError as e: h.flash(h.escape(safe_str(e)), category='warning') raise HTTPFound( h.route_path('repo_summary', repo_name=self.db_repo_name)) except HTTPFound: raise except (RepositoryError, CommitDoesNotExistError, Exception) as e: log.exception(safe_str(e)) h.flash(h.escape(safe_str(e)), category='error') if commit_id: # from single commit page, we redirect to main commits raise HTTPFound( h.route_path('repo_commits', repo_name=self.db_repo_name)) else: # otherwise we redirect to summary raise HTTPFound( h.route_path('repo_summary', repo_name=self.db_repo_name)) if partial_xhr or self.request.environ.get('HTTP_X_PJAX'): # case when loading dynamic file history in file view # loading from ajax, we don't want the first result, it's popped # in the code above html = render( 'rhodecode:templates/commits/changelog_file_history.mako', self._get_template_context(c), self.request) return Response(html) commit_ids = [] if not f_path: # only load graph data when not in file history mode commit_ids = c.pagination c.graph_data, c.graph_commits = self._graph( self.rhodecode_vcs_repo, commit_ids) return self._get_template_context(c) @LoginRequired() @HasRepoPermissionAnyDecorator( 'repository.read', 'repository.write', 'repository.admin') def repo_commits_elements(self): c = self.load_default_context() commit_id = self.request.matchdict.get('commit_id') f_path = self._get_f_path(self.request.matchdict) show_hidden = str2bool(self.request.GET.get('evolve')) chunk_size = 20 hist_limit = safe_int(self.request.GET.get('limit')) or None def wrap_for_error(err): html = '' \ 'ERROR: {}' \ ''.format(err) return Response(html) c.branch_name = branch_name = self.request.GET.get('branch') or '' c.book_name = book_name = self.request.GET.get('bookmark') or '' c.f_path = f_path c.commit_id = commit_id c.show_hidden = show_hidden c.selected_name = branch_name or book_name if branch_name and branch_name not in self.rhodecode_vcs_repo.branches_all: return wrap_for_error( safe_str(f'Branch: {branch_name} is not valid')) pre_load = self.get_commit_preload_attrs() if f_path: try: base_commit = self.rhodecode_vcs_repo.get_commit(commit_id) except (RepositoryError, CommitDoesNotExistError, Exception) as e: log.exception(safe_str(e)) raise HTTPFound( h.route_path('repo_commits', repo_name=self.db_repo_name)) collection = base_commit.get_path_history( f_path, limit=hist_limit, pre_load=pre_load) collection = list(reversed(collection)) else: collection = self.rhodecode_vcs_repo.get_commits( branch_name=branch_name, show_hidden=show_hidden, pre_load=pre_load, translate_tags=False) p = safe_int(self.request.GET.get('page', 1), 1) try: self._load_changelog_data( c, collection, p, chunk_size, dynamic=True, f_path=f_path, commit_id=commit_id) except EmptyRepositoryError as e: return wrap_for_error(safe_str(e)) except (RepositoryError, CommitDoesNotExistError, Exception) as e: log.exception('Failed to fetch commits') return wrap_for_error(safe_str(e)) prev_data = None next_data = None try: prev_graph = ext_json.json.loads(self.request.POST.get('graph') or '{}') except ext_json.json.JSONDecodeError: prev_graph = {} if self.request.GET.get('chunk') == 'prev': next_data = prev_graph elif self.request.GET.get('chunk') == 'next': prev_data = prev_graph commit_ids = [] if not f_path: # only load graph data when not in file history mode commit_ids = c.pagination c.graph_data, c.graph_commits = self._graph( self.rhodecode_vcs_repo, commit_ids, prev_data=prev_data, next_data=next_data) return self._get_template_context(c)