git_push_service.rb 5.2 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 27 28 29 30 31 32
    if push_remove_branch?(ref, newrev)
      @push_commits = []
    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)
33

34 35 36 37
        # Ensure HEAD points to the default branch in case it is not master
        branch_name = Gitlab::Git.ref_name(ref)
        project.change_head(branch_name)

38 39 40 41
        # 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 已提交
42
        end
43 44 45 46 47 48 49
      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
50
        process_commit_messages(ref)
51
      end
52 53 54 55 56 57
    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)
    end
58

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

61 62 63
    EventCreateService.new.push(project, user, @push_data)
    project.execute_hooks(@push_data.dup, :push_hooks)
    project.execute_services(@push_data.dup, :push_hooks)
64 65 66 67
  end

  protected

68 69
  # 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.
70
  def process_commit_messages(ref)
71 72 73 74 75 76
    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.
77
      issues_to_close = commit.closes_issues(user)
78

79 80 81 82 83 84 85
      # 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)

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

91 92 93 94 95
      if project.default_issues_tracker?
        create_cross_reference_notes(commit, issues_to_close)
      end
    end
  end
96

97 98 99 100 101 102
  def create_cross_reference_notes(commit, issues_to_close)
    # 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, user) - issues_to_close
    refs.reject! { |r| commit.has_mentioned?(r) }
103

104 105 106 107 108
    if refs.present?
      author ||= commit_user(commit)

      refs.each do |r|
        Note.create_cross_reference_note(r, commit, author)
109 110 111 112
      end
    end
  end

D
Douwe Maan 已提交
113
  def build_push_data(oldrev, newrev, ref)
114 115
    Gitlab::PushDataBuilder.
      build(project, user, oldrev, newrev, ref, push_commits)
116 117
  end

118
  def push_to_existing_branch?(ref, oldrev)
119
    # Return if this is not a push to a branch (e.g. new commits)
D
Douwe Maan 已提交
120
    Gitlab::Git.branch_ref?(ref) && !Gitlab::Git.blank_ref?(oldrev)
121 122
  end

123
  def push_to_new_branch?(ref, oldrev)
124
    Gitlab::Git.branch_ref?(ref) && Gitlab::Git.blank_ref?(oldrev)
125 126
  end

127
  def push_remove_branch?(ref, newrev)
128
    Gitlab::Git.branch_ref?(ref) && Gitlab::Git.blank_ref?(newrev)
D
Dmitriy Zaporozhets 已提交
129 130
  end

131
  def push_to_branch?(ref)
132
    Gitlab::Git.branch_ref?(ref)
133 134
  end

135
  def is_default_branch?(ref)
136 137
    Gitlab::Git.branch_ref?(ref) &&
      (Gitlab::Git.ref_name(ref) == project.default_branch || project.default_branch.nil?)
138 139
  end

140
  def commit_user(commit)
D
Douwe Maan 已提交
141
    commit.author || user
142 143
  end
end