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 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 55 56 57
      @push_data = post_receive_data(oldrev, newrev, ref)
      create_push_event(@push_data)
      project.execute_hooks(@push_data.dup, :push_hooks)
      project.execute_services(@push_data.dup)
58
    end
59 60 61 62
  end

  protected

D
Dmitriy Zaporozhets 已提交
63
  def create_push_event(push_data)
64
    Event.create!(
65 66 67 68 69 70 71
      project: project,
      action: Event::PUSHED,
      data: push_data,
      author_id: push_data[:user_id]
    )
  end

72 73
  # 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.
74
  def process_commit_messages(ref)
75 76 77 78 79 80 81 82
    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)

83 84 85 86 87 88 89
      # 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)

90 91 92
        issues_to_close.each do |issue|
          Issues::CloseService.new(project, author, {}).execute(issue, commit)
        end
93 94 95 96 97 98 99
      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) }
100 101 102 103 104 105 106

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

        refs.each do |r|
          Note.create_cross_reference_note(r, commit, author, project)
        end
107 108 109 110
      end
    end
  end

111
  def post_receive_data(oldrev, newrev, ref)
112 113
    Gitlab::PushDataBuilder.
      build(project, user, oldrev, newrev, ref, push_commits)
114 115
  end

116
  def push_to_existing_branch?(ref, oldrev)
117 118 119
    ref_parts = ref.split('/')

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

123
  def push_to_new_branch?(ref, oldrev)
124 125
    ref_parts = ref.split('/')

126
    ref_parts[1].include?('heads') && oldrev == Gitlab::Git::BLANK_SHA
127 128
  end

129
  def push_remove_branch?(ref, newrev)
D
Dmitriy Zaporozhets 已提交
130 131
    ref_parts = ref.split('/')

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

135
  def push_to_branch?(ref)
136
    ref.include?('refs/heads')
137 138
  end

139
  def is_default_branch?(ref)
140 141 142
    ref == "refs/heads/#{project.default_branch}"
  end

143
  def commit_user(commit)
144
    User.find_for_commit(commit.author_email, commit.author_name) || user
145 146
  end
end