git_push_service.rb 5.1 KB
Newer Older
1
class GitPushService
2
  attr_accessor :project, :user, :push_data, :push_commits
3 4
  include Gitlab::CurrentSettings
  include Gitlab::Access
5 6 7 8 9 10

  # This method will be called after each git update
  # and only if the provided user and project is present in GitLab.
  #
  # All callbacks for post receive action should be placed here.
  #
11 12
  # Next, this method:
  #  1. Creates the push event
13 14 15 16
  #  2. Updates merge requests
  #  3. Recognizes cross-references from commit messages
  #  4. Executes the project's web hooks
  #  5. Executes the project's services
17 18 19 20
  #
  def execute(project, user, oldrev, newrev, ref)
    @project, @user = project, user

21 22 23
    branch_name = Gitlab::Git.ref_name(ref)

    project.repository.expire_cache(branch_name)
24

25
    if push_remove_branch?(ref, newrev)
26 27
      project.repository.expire_has_visible_content_cache

28 29
      @push_commits = []
    elsif push_to_new_branch?(ref, oldrev)
30 31
      project.repository.expire_has_visible_content_cache

32 33 34 35
      # Re-find the pushed commits.
      if is_default_branch?(ref)
        # Initial push to the default branch. Take the full history of that branch as "newly pushed".
        @push_commits = project.repository.commits(newrev)
36

37 38 39
        # Ensure HEAD points to the default branch in case it is not master
        project.change_head(branch_name)

40 41 42 43
        # Set protection on the default branch if configured
        if (current_application_settings.default_branch_protection != PROTECTION_NONE)
          developers_can_push = current_application_settings.default_branch_protection == PROTECTION_DEV_CAN_PUSH ? true : false
          project.protected_branches.create({ name: project.default_branch, developers_can_push: developers_can_push })
D
Dmitriy Zaporozhets 已提交
44
        end
45 46 47 48 49 50 51
      else
        # Use the pushed commits that aren't reachable by the default branch
        # as a heuristic. This may include more commits than are actually pushed, but
        # that shouldn't matter because we check for existing cross-references later.
        @push_commits = project.repository.commits_between(project.default_branch, newrev)

        # don't process commits for the initial push to the default branch
52
        process_commit_messages(ref)
53
      end
54 55 56 57 58
    elsif push_to_existing_branch?(ref, oldrev)
      # Collect data for this git push
      @push_commits = project.repository.commits_between(oldrev, newrev)
      process_commit_messages(ref)
    end
59

60 61 62 63
    # Update merge requests that may be affected by this push. A new branch
    # could cause the last commit of a merge request to change.
    project.update_merge_requests(oldrev, newrev, ref, @user)

64
    @push_data = build_push_data(oldrev, newrev, ref)
D
Douwe Maan 已提交
65

66 67 68
    EventCreateService.new.push(project, user, @push_data)
    project.execute_hooks(@push_data.dup, :push_hooks)
    project.execute_services(@push_data.dup, :push_hooks)
69
    CreateCommitBuildsService.new.execute(project, @user, @push_data)
70
    ProjectCacheWorker.perform_async(project.id)
71 72 73 74
  end

  protected

75 76
  # Extract any GFM references from the pushed commit messages. If the configured issue-closing regex is matched,
  # close the referenced Issue. Create cross-reference Notes corresponding to any other referenced Mentionables.
77
  def process_commit_messages(ref)
78 79
    is_default_branch = is_default_branch?(ref)

80 81
    authors = Hash.new do |hash, commit|
      email = commit.author_email
82
      next hash[email] if hash.has_key?(email)
83

84 85
      hash[email] = commit_user(commit)
    end
86

87
    @push_commits.each do |commit|
88 89 90
      # Keep track of the issues that will be actually closed because they are on a default branch.
      # Hence, when creating cross-reference notes, the not-closed issues (on non-default branches)
      # will also have cross-reference.
91 92 93 94 95 96 97 98 99
      closed_issues = []

      if is_default_branch
        # Close issues if these commits were pushed to the project's default branch and the commit message matches the
        # closing regex. Exclude any mentioned Issues from cross-referencing even if the commits are being pushed to
        # a different branch.
        closed_issues = commit.closes_issues(user)
        closed_issues.each do |issue|
          Issues::CloseService.new(project, authors[commit], {}).execute(issue, commit)
100
        end
101 102
      end

103
      commit.create_cross_references!(authors[commit], closed_issues)
104 105 106
    end
  end

D
Douwe Maan 已提交
107
  def build_push_data(oldrev, newrev, ref)
108 109
    Gitlab::PushDataBuilder.
      build(project, user, oldrev, newrev, ref, push_commits)
110 111
  end

112
  def push_to_existing_branch?(ref, oldrev)
113
    # Return if this is not a push to a branch (e.g. new commits)
D
Douwe Maan 已提交
114
    Gitlab::Git.branch_ref?(ref) && !Gitlab::Git.blank_ref?(oldrev)
115 116
  end

117
  def push_to_new_branch?(ref, oldrev)
118
    Gitlab::Git.branch_ref?(ref) && Gitlab::Git.blank_ref?(oldrev)
119 120
  end

121
  def push_remove_branch?(ref, newrev)
122
    Gitlab::Git.branch_ref?(ref) && Gitlab::Git.blank_ref?(newrev)
D
Dmitriy Zaporozhets 已提交
123 124
  end

125
  def push_to_branch?(ref)
126
    Gitlab::Git.branch_ref?(ref)
127 128
  end

129
  def is_default_branch?(ref)
130 131
    Gitlab::Git.branch_ref?(ref) &&
      (Gitlab::Git.ref_name(ref) == project.default_branch || project.default_branch.nil?)
132 133
  end

134
  def commit_user(commit)
D
Douwe Maan 已提交
135
    commit.author || user
136 137
  end
end