diffs_controller.rb 4.4 KB
Newer Older
1 2
# frozen_string_literal: true

D
Douwe Maan 已提交
3 4 5 6 7
class Projects::MergeRequests::DiffsController < Projects::MergeRequests::ApplicationController
  include DiffHelper
  include RendersNotes

  before_action :apply_diff_view_cookie!
8 9
  before_action :commit, except: :diffs_batch
  before_action :define_diff_vars, except: :diffs_batch
10
  before_action :define_diff_comment_vars, except: [:diffs_batch, :diffs_metadata]
D
Douwe Maan 已提交
11 12

  def show
F
Felipe Artur 已提交
13
    render_diffs
D
Douwe Maan 已提交
14 15 16
  end

  def diff_for_path
F
Felipe Artur 已提交
17
    render_diffs
D
Douwe Maan 已提交
18 19
  end

20 21 22 23 24 25 26 27
  def diffs_batch
    return render_404 unless Feature.enabled?(:diffs_batch_load, @merge_request.project)

    diffable = @merge_request.merge_request_diff

    return render_404 unless diffable

    diffs = diffable.diffs_in_batch(params[:page], params[:per_page], diff_options: diff_options)
28 29 30
    positions = @merge_request.note_positions_for_paths(diffs.diff_file_paths, current_user)

    diffs.unfold_diff_files(positions.unfoldable)
31 32 33

    options = {
      merge_request: @merge_request,
34
      diff_view: diff_view,
35 36 37 38 39 40
      pagination_data: diffs.pagination_data
    }

    render json: PaginatedDiffSerializer.new(current_user: current_user).represent(diffs, options)
  end

41 42 43 44 45
  def diffs_metadata
    render json: DiffsMetadataSerializer.new(project: @merge_request.project)
                   .represent(@diffs, additional_attributes)
  end

D
Douwe Maan 已提交
46 47
  private

48 49 50 51
  def preloadable_mr_relations
    [{ source_project: :namespace }, { target_project: :namespace }]
  end

F
Felipe Artur 已提交
52 53
  def render_diffs
    @environment = @merge_request.environments_for(current_user).last
54

55
    @diffs.unfold_diff_files(note_positions.unfoldable)
56 57
    @diffs.write_cache

58 59 60 61 62 63
    request = {
      current_user: current_user,
      project: @merge_request.project,
      render: ->(partial, locals) { view_to_html_string(partial, locals) }
    }

64 65 66
    options = additional_attributes.merge(diff_view: diff_view)

    render json: DiffsSerializer.new(request).represent(@diffs, options)
F
Felipe Artur 已提交
67 68
  end

D
Douwe Maan 已提交
69
  def define_diff_vars
70
    @merge_request_diffs = @merge_request.merge_request_diffs.viewable.order_id_desc
71
    @compare = commit || find_merge_request_diff_compare
72 73 74 75 76
    return render_404 unless @compare

    @diffs = @compare.diffs(diff_options)
  end

77
  # rubocop: disable CodeReuse/ActiveRecord
M
micael.bergeron 已提交
78
  def commit
79 80
    return unless commit_id = params[:commit_id].presence
    return unless @merge_request.all_commits.exists?(sha: commit_id)
M
micael.bergeron 已提交
81 82 83

    @commit ||= @project.commit(commit_id)
  end
84
  # rubocop: enable CodeReuse/ActiveRecord
M
micael.bergeron 已提交
85

86
  # rubocop: disable CodeReuse/ActiveRecord
87
  def find_merge_request_diff_compare
D
Douwe Maan 已提交
88
    @merge_request_diff =
89 90
      if diff_id = params[:diff_id].presence
        @merge_request.merge_request_diffs.viewable.find_by(id: diff_id)
D
Douwe Maan 已提交
91 92 93 94
      else
        @merge_request.merge_request_diff
      end

95 96
    return unless @merge_request_diff

D
Douwe Maan 已提交
97 98
    @comparable_diffs = @merge_request_diffs.select { |diff| diff.id < @merge_request_diff.id }

99
    if @start_sha = params[:start_sha].presence
D
Douwe Maan 已提交
100 101 102 103 104 105 106 107
      @start_version = @comparable_diffs.find { |diff| diff.head_commit_sha == @start_sha }

      unless @start_version
        @start_sha = @merge_request_diff.head_commit_sha
        @start_version = @merge_request_diff
      end
    end

108 109 110 111 112
    if @start_sha
      @merge_request_diff.compare_with(@start_sha)
    else
      @merge_request_diff
    end
D
Douwe Maan 已提交
113
  end
114
  # rubocop: enable CodeReuse/ActiveRecord
D
Douwe Maan 已提交
115

F
Felipe Artur 已提交
116 117 118 119 120 121 122 123 124 125 126 127 128
  def additional_attributes
    {
      environment: @environment,
      merge_request: @merge_request,
      merge_request_diff: @merge_request_diff,
      merge_request_diffs: @merge_request_diffs,
      start_version: @start_version,
      start_sha: @start_sha,
      commit: @commit,
      latest_diff: @merge_request_diff&.latest?
    }
  end

D
Douwe Maan 已提交
129 130 131
  def define_diff_comment_vars
    @new_diff_note_attrs = {
      noteable_type: 'MergeRequest',
132 133
      noteable_id: @merge_request.id,
      commit_id: @commit&.id
D
Douwe Maan 已提交
134 135 136 137 138 139 140
    }

    @diff_notes_disabled = false

    @use_legacy_diff_notes = !@merge_request.has_complete_diff_refs?

    @grouped_diff_discussions = @merge_request.grouped_diff_discussions(@compare.diff_refs)
141
    @notes = prepare_notes_for_rendering(@grouped_diff_discussions.values.flatten.flat_map(&:notes), @merge_request)
D
Douwe Maan 已提交
142
  end
143

144 145 146 147
  def note_positions
    @note_positions ||= Gitlab::Diff::PositionCollection.new(renderable_notes.map(&:position))
  end

148 149 150 151 152
  def renderable_notes
    define_diff_comment_vars unless @notes

    @notes
  end
D
Douwe Maan 已提交
153
end
154 155

Projects::MergeRequests::DiffsController.prepend_if_ee('EE::Projects::MergeRequests::DiffsController')