create_service.rb 5.46 KB
Newer Older
1
module Projects
2
  class CreateService < BaseService
3 4 5 6
    def initialize(user, params)
      @current_user, @params = user, params.dup
    end

7
    def execute
8 9 10 11
      if @params[:template_name]&.present?
        return ::Projects::CreateFromTemplateService.new(current_user, params).execute
      end

12
      forked_from_project_id = params.delete(:forked_from_project_id)
13
      import_data = params.delete(:import_data)
14 15
      @skip_wiki = params.delete(:skip_wiki)

16
      @project = Project.new(params)
17

18
      # Make sure that the user is allowed to use the specified visibility level
19
      unless Gitlab::VisibilityLevel.allowed_for?(current_user, @project.visibility_level)
Felipe's avatar
Felipe committed
20 21 22
        deny_visibility_level(@project)
        return @project
      end
23

24 25 26 27 28
      unless allowed_fork?(forked_from_project_id)
        @project.errors.add(:forked_from_project_id, 'is forbidden')
        return @project
      end

29
      set_project_name_from_path
30

31 32
      # get namespace id
      namespace_id = params[:namespace_id]
33 34 35 36

      if namespace_id
        # Find matching namespace and check if it allowed
        # for current user if namespace_id passed.
37 38
        unless allowed_namespace?(current_user, namespace_id)
          @project.namespace_id = nil
39 40 41 42 43
          deny_namespace
          return @project
        end
      else
        # Set current user namespace if namespace_id is nil
44
        @project.namespace_id = current_user.namespace_id
45 46
      end

47 48
      yield(@project) if block_given?

49 50 51
      # If the block added errors, don't try to save the project
      return @project if @project.errors.any?

52
      @project.creator = current_user
53

54 55 56 57
      if forked_from_project_id
        @project.build_forked_project_link(forked_from_project_id: forked_from_project_id)
      end

James Lopez's avatar
James Lopez committed
58
      save_project_and_import_data(import_data)
59

60
      after_create_actions if @project.persisted?
61

James Lopez's avatar
James Lopez committed
62
      import_schedule
63

64
      @project
65 66 67
    rescue ActiveRecord::RecordInvalid => e
      message = "Unable to save #{e.record.type}: #{e.record.errors.full_messages.join(", ")} "
      fail(error: message)
68
    rescue => e
69
      fail(error: e.message)
70 71 72 73 74 75 76 77
    end

    protected

    def deny_namespace
      @project.errors.add(:namespace, "is not valid")
    end

78 79 80 81 82 83 84
    def allowed_fork?(source_project_id)
      return true if source_project_id.nil?

      source_project = Project.find_by(id: source_project_id)
      current_user.can?(:fork_project, source_project)
    end

85
    def allowed_namespace?(user, namespace_id)
86
      namespace = Namespace.find_by(id: namespace_id)
87
      current_user.can?(:create_projects, namespace)
88
    end
89 90

    def after_create_actions
91
      log_info("#{@project.owner.name} created a new project \"#{@project.full_name}\"")
92

93
      unless @project.gitlab_project_import?
94
        @project.write_repository_config
95
        @project.create_wiki unless skip_wiki?
96
      end
Valeriy's avatar
Valeriy committed
97

98
      event_service.create_project(@project, current_user)
99 100
      system_hook_service.execute_hooks_for(@project, :create)

101
      setup_authorizations
102 103

      current_user.invalidate_personal_projects_count
104 105 106 107 108 109
    end

    # Refresh the current user's authorizations inline (so they can access the
    # project immediately after this request completes), and any other affected
    # users in the background
    def setup_authorizations
110 111
      if @project.group
        @project.group.refresh_members_authorized_projects(blocking: false)
112 113
        current_user.refresh_authorized_projects
      else
114
        @project.add_master(@project.namespace.owner, current_user: current_user)
115 116
      end
    end
James Lopez's avatar
James Lopez committed
117

118 119 120 121
    def skip_wiki?
      !@project.feature_available?(:wiki, current_user) || @skip_wiki
    end

James Lopez's avatar
James Lopez committed
122
    def save_project_and_import_data(import_data)
James Lopez's avatar
James Lopez committed
123 124 125
      Project.transaction do
        @project.create_or_update_import_data(data: import_data[:data], credentials: import_data[:credentials]) if import_data

126 127 128 129 130 131 132 133 134
        if @project.save
          unless @project.gitlab_project_import?
            create_services_from_active_templates(@project)
            @project.create_labels
          end

          unless @project.import?
            raise 'Failed to create repository' unless @project.create_repository
          end
James Lopez's avatar
James Lopez committed
135 136 137
        end
      end
    end
138 139 140

    def fail(error:)
      message = "Unable to save project. Error: #{error}"
141
      log_message = message.dup
142

143 144
      log_message << " Project ID: #{@project.id}" if @project&.id
      Rails.logger.error(log_message)
145

146
      if @project
147
        @project.errors.add(:base, message)
148
        @project.mark_import_as_failed(message) if @project.persisted? && @project.import?
149 150 151 152
      end

      @project
    end
153 154 155 156 157 158 159

    def create_services_from_active_templates(project)
      Service.where(template: true, active: true).each do |template|
        service = Service.build_from_template(project.id, template)
        service.save!
      end
    end
160 161 162 163 164 165 166 167 168 169 170 171 172 173

    def set_project_name_from_path
      # Set project name from path
      if @project.name.present? && @project.path.present?
        # if both name and path set - everything is ok
      elsif @project.path.present?
        # Set project name from path
        @project.name = @project.path.dup
      elsif @project.name.present?
        # For compatibility - set path from name
        # TODO: remove this in 8.0
        @project.path = @project.name.dup.parameterize
      end
    end
James Lopez's avatar
James Lopez committed
174 175 176 177 178

    private

    def import_schedule
      if @project.errors.empty?
179
        @project.import_schedule if @project.import? && !@project.bare_repository_import?
James Lopez's avatar
James Lopez committed
180 181 182 183
      else
        fail(error: @project.errors.full_messages.join(', '))
      end
    end
184 185
  end
end