152 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Ruby
		
	
	
	
| # Controller for a specific Commit
 | |
| #
 | |
| # Not to be confused with CommitsController, plural.
 | |
| class Projects::CommitController < Projects::ApplicationController
 | |
|   include CreatesCommit
 | |
|   include DiffForPath
 | |
|   include DiffHelper
 | |
| 
 | |
|   # Authorize
 | |
|   before_action :require_non_empty_project
 | |
|   before_action :authorize_download_code!, except: [:cancel_builds, :retry_builds]
 | |
|   before_action :authorize_update_build!, only: [:cancel_builds, :retry_builds]
 | |
|   before_action :authorize_read_commit_status!, only: [:builds]
 | |
|   before_action :commit
 | |
|   before_action :define_commit_vars, only: [:show, :diff_for_path, :builds]
 | |
|   before_action :define_status_vars, only: [:show, :builds]
 | |
|   before_action :define_note_vars, only: [:show, :diff_for_path]
 | |
|   before_action :authorize_edit_tree!, only: [:revert, :cherry_pick]
 | |
| 
 | |
|   def show
 | |
|     apply_diff_view_cookie!
 | |
| 
 | |
|     respond_to do |format|
 | |
|       format.html
 | |
|       format.diff  { render text: @commit.to_diff }
 | |
|       format.patch { render text: @commit.to_patch }
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def diff_for_path
 | |
|     render_diff_for_path(@diffs, @commit.diff_refs, @project)
 | |
|   end
 | |
| 
 | |
|   def builds
 | |
|   end
 | |
| 
 | |
|   def cancel_builds
 | |
|     ci_builds.running_or_pending.each(&:cancel)
 | |
| 
 | |
|     redirect_back_or_default default: builds_namespace_project_commit_path(project.namespace, project, commit.sha)
 | |
|   end
 | |
| 
 | |
|   def retry_builds
 | |
|     ci_builds.latest.failed.each do |build|
 | |
|       if build.retryable?
 | |
|         Ci::Build.retry(build, current_user)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     redirect_back_or_default default: builds_namespace_project_commit_path(project.namespace, project, commit.sha)
 | |
|   end
 | |
| 
 | |
|   def branches
 | |
|     @branches = @project.repository.branch_names_contains(commit.id)
 | |
|     @tags = @project.repository.tag_names_contains(commit.id)
 | |
|     render layout: false
 | |
|   end
 | |
| 
 | |
|   def revert
 | |
|     assign_change_commit_vars(@commit.revert_branch_name)
 | |
| 
 | |
|     return render_404 if @target_branch.blank?
 | |
| 
 | |
|     create_commit(Commits::RevertService, success_notice: "The #{@commit.change_type_title} has been successfully reverted.",
 | |
|                                           success_path: successful_change_path, failure_path: failed_change_path)
 | |
|   end
 | |
| 
 | |
|   def cherry_pick
 | |
|     assign_change_commit_vars(@commit.cherry_pick_branch_name)
 | |
| 
 | |
|     return render_404 if @target_branch.blank?
 | |
| 
 | |
|     create_commit(Commits::CherryPickService, success_notice: "The #{@commit.change_type_title} has been successfully cherry-picked.",
 | |
|                                               success_path: successful_change_path, failure_path: failed_change_path)
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def successful_change_path
 | |
|     return referenced_merge_request_url if @commit.merged_merge_request
 | |
| 
 | |
|     namespace_project_commits_url(@project.namespace, @project, @target_branch)
 | |
|   end
 | |
| 
 | |
|   def failed_change_path
 | |
|     return referenced_merge_request_url if @commit.merged_merge_request
 | |
| 
 | |
|     namespace_project_commit_url(@project.namespace, @project, params[:id])
 | |
|   end
 | |
| 
 | |
|   def referenced_merge_request_url
 | |
|     namespace_project_merge_request_url(@project.namespace, @project, @commit.merged_merge_request)
 | |
|   end
 | |
| 
 | |
|   def commit
 | |
|     @commit ||= @project.commit(params[:id])
 | |
|   end
 | |
| 
 | |
|   def pipelines
 | |
|     @pipelines ||= project.pipelines.where(sha: commit.sha)
 | |
|   end
 | |
| 
 | |
|   def ci_builds
 | |
|     @ci_builds ||= Ci::Build.where(pipeline: pipelines)
 | |
|   end
 | |
| 
 | |
|   def define_commit_vars
 | |
|     return git_not_found! unless commit
 | |
| 
 | |
|     opts = diff_options
 | |
|     opts[:ignore_whitespace_change] = true if params[:format] == 'diff'
 | |
| 
 | |
|     @diffs = commit.diffs(opts)
 | |
|     @notes_count = commit.notes.count
 | |
|   end
 | |
| 
 | |
|   def define_note_vars
 | |
|     @grouped_diff_discussions = commit.notes.grouped_diff_discussions
 | |
|     @notes = commit.notes.non_diff_notes.fresh
 | |
| 
 | |
|     Banzai::NoteRenderer.render(
 | |
|       @grouped_diff_discussions.values.flat_map(&:notes) + @notes,
 | |
|       @project,
 | |
|       current_user,
 | |
|     )
 | |
| 
 | |
|     @note = @project.build_commit_note(commit)
 | |
| 
 | |
|     @noteable = @commit
 | |
|     @comments_target = {
 | |
|       noteable_type: 'Commit',
 | |
|       commit_id: @commit.id
 | |
|     }
 | |
|   end
 | |
| 
 | |
|   def define_status_vars
 | |
|     @statuses = CommitStatus.where(pipeline: pipelines)
 | |
|     @builds = Ci::Build.where(pipeline: pipelines)
 | |
|   end
 | |
| 
 | |
|   def assign_change_commit_vars(mr_source_branch)
 | |
|     @commit = project.commit(params[:id])
 | |
|     @target_branch = params[:target_branch]
 | |
|     @mr_source_branch = mr_source_branch
 | |
|     @mr_target_branch = @target_branch
 | |
|     @commit_params = {
 | |
|       commit: @commit,
 | |
|       create_merge_request: params[:create_merge_request].present? || different_project?
 | |
|     }
 | |
|   end
 | |
| end
 |