git_push_service.rb 5.0 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 13 14 15 16 17
  # Next, this method:
  #  1. Creates the push event
  #  2. Ensures that the project satellite exists
  #  3. Updates merge requests
  #  4. Recognizes cross-references from commit messages
  #  5. Executes the project's web hooks
  #  6. Executes the project's services
18 19 20 21 22
  #
  def execute(project, user, oldrev, newrev, ref)
    @project, @user = project, user

    project.ensure_satellite_exists
23
    project.repository.expire_cache
24
    project.update_repository_size
25

26
    if push_to_branch?(ref)
D
Dmitriy Zaporozhets 已提交
27
      if push_remove_branch?(ref, newrev)
28
        @push_commits = []
D
Dmitriy Zaporozhets 已提交
29 30 31 32 33
      elsif push_to_new_branch?(ref, oldrev)
        # 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)
34 35 36

          # Set protection on the default branch if configured
          if (current_application_settings.default_branch_protection != PROTECTION_NONE)
M
Marco Wessel 已提交
37
            developers_can_push = current_application_settings.default_branch_protection == PROTECTION_DEV_CAN_PUSH ? true : false
38 39
            project.protected_branches.create({ name: project.default_branch, developers_can_push: developers_can_push })
          end
D
Dmitriy Zaporozhets 已提交
40 41 42 43 44 45 46
        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)
        end
        process_commit_messages(ref)
47 48 49 50 51
      elsif push_to_existing_branch?(ref, oldrev)
        # Collect data for this git push
        @push_commits = project.repository.commits_between(oldrev, newrev)
        project.update_merge_requests(oldrev, newrev, ref, @user)
        process_commit_messages(ref)
52 53
      end

D
Dmitriy Zaporozhets 已提交
54
      @push_data = post_receive_data(oldrev, newrev, ref)
55
      EventCreateService.new.push(project, user, @push_data)
D
Dmitriy Zaporozhets 已提交
56
      project.execute_hooks(@push_data.dup, :push_hooks)
57
      project.execute_services(@push_data.dup, :push_hooks)
58
    end
59 60 61 62
  end

  protected

63 64
  # 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.
65
  def process_commit_messages(ref)
66 67 68 69 70 71 72 73
    is_default_branch = is_default_branch?(ref)

    @push_commits.each do |commit|
      # 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.
      issues_to_close = commit.closes_issues(project)

74 75 76 77 78 79 80
      # Load commit author only if needed.
      # For push with 1k commits it prevents 900+ requests in database
      author = nil

      if issues_to_close.present? && is_default_branch
        author ||= commit_user(commit)

81 82 83
        issues_to_close.each do |issue|
          Issues::CloseService.new(project, author, {}).execute(issue, commit)
        end
84 85 86 87 88 89 90
      end

      # Create cross-reference notes for any other references. Omit any issues that were referenced in an
      # issue-closing phrase, or have already been mentioned from this commit (probably from this commit
      # being pushed to a different branch).
      refs = commit.references(project) - issues_to_close
      refs.reject! { |r| commit.has_mentioned?(r) }
91 92 93 94 95 96 97

      if refs.present?
        author ||= commit_user(commit)

        refs.each do |r|
          Note.create_cross_reference_note(r, commit, author, project)
        end
98 99 100 101
      end
    end
  end

102
  def post_receive_data(oldrev, newrev, ref)
103 104
    Gitlab::PushDataBuilder.
      build(project, user, oldrev, newrev, ref, push_commits)
105 106
  end

107
  def push_to_existing_branch?(ref, oldrev)
108 109 110
    ref_parts = ref.split('/')

    # Return if this is not a push to a branch (e.g. new commits)
111
    ref_parts[1].include?('heads') && oldrev != Gitlab::Git::BLANK_SHA
112 113
  end

114
  def push_to_new_branch?(ref, oldrev)
115 116
    ref_parts = ref.split('/')

117
    ref_parts[1].include?('heads') && oldrev == Gitlab::Git::BLANK_SHA
118 119
  end

120
  def push_remove_branch?(ref, newrev)
D
Dmitriy Zaporozhets 已提交
121 122
    ref_parts = ref.split('/')

123
    ref_parts[1].include?('heads') && newrev == Gitlab::Git::BLANK_SHA
D
Dmitriy Zaporozhets 已提交
124 125
  end

126
  def push_to_branch?(ref)
127
    ref.include?('refs/heads')
128 129
  end

130
  def is_default_branch?(ref)
131 132 133
    ref == "refs/heads/#{project.default_branch}"
  end

134
  def commit_user(commit)
135
    User.find_for_commit(commit.author_email, commit.author_name) || user
136 137
  end
end