summaryrefslogtreecommitdiff
path: root/app/services/commits/change_service.rb
blob: 25e22f14e60b0c3d9cb3c0827cf76a6607578d33 (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
77
78
79
80
81
module Commits
  class ChangeService < ::BaseService
    class ValidationError < StandardError; end
    class ChangeError < StandardError; end

    def execute
      @start_project = params[:start_project] || @project
      @start_branch = params[:start_branch]
      @target_branch = params[:target_branch]
      @commit = params[:commit]
      @create_merge_request = params[:create_merge_request].present?

      check_push_permissions unless @create_merge_request
      commit
    rescue Repository::CommitError, Gitlab::Git::Repository::InvalidBlobName, GitHooksService::PreReceiveError,
           ValidationError, ChangeError => ex
      error(ex.message)
    end

    private

    def commit
      raise NotImplementedError
    end

    def commit_change(action)
      raise NotImplementedError unless repository.respond_to?(action)

      if @create_merge_request
        into = @commit.public_send("#{action}_branch_name")
        tree_branch = @start_branch
      else
        into = tree_branch = @target_branch
      end

      tree_id = repository.public_send(
        "check_#{action}_content", @commit, tree_branch)

      if tree_id
        validate_target_branch(into) if @create_merge_request

        repository.public_send(
          action,
          current_user,
          @commit,
          into,
          tree_id,
          start_project: @start_project,
          start_branch_name: @start_branch)

        success
      else
        error_msg = "Sorry, we cannot #{action.to_s.dasherize} this #{@commit.change_type_title(current_user)} automatically.
                     A #{action.to_s.dasherize} may have already been performed with this #{@commit.change_type_title(current_user)}, or a more recent commit may have updated some of its content."
        raise ChangeError, error_msg
      end
    end

    def check_push_permissions
      allowed = ::Gitlab::UserAccess.new(current_user, project: project).can_push_to_branch?(@target_branch)

      unless allowed
        raise ValidationError.new('You are not allowed to push into this branch')
      end

      true
    end

    def validate_target_branch(new_branch)
      # Temporary branch exists and contains the change commit
      return if repository.find_branch(new_branch)

      result = ValidateNewBranchService.new(@project, current_user)
        .execute(new_branch)

      if result[:status] == :error
        raise ChangeError, "There was an error creating the source branch: #{result[:message]}"
      end
    end
  end
end