summaryrefslogtreecommitdiff
path: root/lib/gitlab/conflict/file_collection.rb
blob: e621a76a7de48327541f11adc66b6381092fa23a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
module Gitlab
  module Conflict
    class FileCollection
      attr_reader :merge_request, :our_commit, :their_commit

      def initialize(merge_request)
        @merge_request = merge_request
        @our_commit = merge_request.source_branch_head.raw.raw_commit
        @their_commit = merge_request.target_branch_head.raw.raw_commit
      end

      def repository
        merge_request.project.repository
      end

      def merge_index
        @merge_index ||= repository.rugged.merge_commits(our_commit, their_commit)
      end

      def resolve_conflicts!(resolutions, commit_message, user:)
        rugged = repository.rugged
        committer = repository.user_to_committer(user)
        commit_message ||= default_commit_message

        files.each do |file|
          file.resolve!(resolutions, index: merge_index, rugged: rugged)
        end

        new_tree = merge_index.write_tree(rugged)

        Rugged::Commit.create(rugged,
                              author: committer,
                              committer: committer,
                              tree: new_tree,
                              message: commit_message,
                              parents: [our_commit, their_commit].map(&:oid),
                              update_ref: Gitlab::Git::BRANCH_REF_PREFIX + merge_request.source_branch)
      end

      def files
        @files ||= merge_index.conflicts.map do |conflict|
          their_path = conflict[:theirs][:path]
          our_path = conflict[:ours][:path]

          Gitlab::Conflict::File.new(merge_index.merge_file(our_path),
                                     conflict,
                                     diff_refs: merge_request.diff_refs,
                                     repository: repository)
        end
      end

      def as_json(opts = nil)
        {
          target_branch: merge_request.target_branch,
          source_branch: merge_request.source_branch,
          commit_sha: merge_request.diff_head_sha,
          commit_message: default_commit_message,
          files: files
        }
      end

      def default_commit_message
        conflict_filenames = merge_index.conflicts.map do |conflict|
          "#   #{conflict[:ours][:path]}"
        end

        <<EOM.chomp
Merge branch '#{merge_request.source_branch}' into '#{merge_request.target_branch}'

# Conflicts:
#{conflict_filenames.join("\n")}
EOM
      end
    end
  end
end