2018-07-17 12:50:37 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2013-01-17 10:35:57 -05:00
|
|
|
module Projects
|
2015-03-07 14:47:06 -05:00
|
|
|
class CreateService < BaseService
|
2019-04-09 11:38:58 -04:00
|
|
|
include ValidatesClassificationLabel
|
|
|
|
|
2013-01-18 13:21:13 -05:00
|
|
|
def initialize(user, params)
|
2019-08-06 09:05:18 -04:00
|
|
|
@current_user, @params = user, params.dup
|
|
|
|
@skip_wiki = @params.delete(:skip_wiki)
|
2018-07-03 09:09:58 -04:00
|
|
|
@initialize_with_readme = Gitlab::Utils.to_boolean(@params.delete(:initialize_with_readme))
|
2019-08-06 09:05:18 -04:00
|
|
|
@import_data = @params.delete(:import_data)
|
|
|
|
@relations_block = @params.delete(:relations_block)
|
2013-01-18 13:21:13 -05:00
|
|
|
end
|
|
|
|
|
2013-01-17 10:35:57 -05:00
|
|
|
def execute
|
2019-10-08 20:06:06 -04:00
|
|
|
if create_from_template?
|
2017-08-07 08:21:28 -04:00
|
|
|
return ::Projects::CreateFromTemplateService.new(current_user, params).execute
|
|
|
|
end
|
|
|
|
|
2014-06-26 16:24:17 -04:00
|
|
|
@project = Project.new(params)
|
2013-01-17 10:35:57 -05:00
|
|
|
|
2016-03-18 08:28:16 -04:00
|
|
|
# Make sure that the user is allowed to use the specified visibility level
|
2019-08-06 09:05:18 -04:00
|
|
|
if project_visibility.restricted?
|
|
|
|
deny_visibility_level(@project, project_visibility.visibility_level)
|
2016-03-18 20:04:53 -04:00
|
|
|
return @project
|
|
|
|
end
|
2013-11-06 10:13:21 -05:00
|
|
|
|
2017-01-23 13:41:30 -05:00
|
|
|
set_project_name_from_path
|
2013-01-17 10:35:57 -05:00
|
|
|
|
2014-06-26 16:24:17 -04:00
|
|
|
# get namespace id
|
|
|
|
namespace_id = params[:namespace_id]
|
2013-01-17 10:35:57 -05:00
|
|
|
|
|
|
|
if namespace_id
|
|
|
|
# Find matching namespace and check if it allowed
|
|
|
|
# for current user if namespace_id passed.
|
2014-06-26 16:24:17 -04:00
|
|
|
unless allowed_namespace?(current_user, namespace_id)
|
|
|
|
@project.namespace_id = nil
|
2013-01-17 10:35:57 -05:00
|
|
|
deny_namespace
|
|
|
|
return @project
|
|
|
|
end
|
|
|
|
else
|
|
|
|
# Set current user namespace if namespace_id is nil
|
2013-01-18 13:21:13 -05:00
|
|
|
@project.namespace_id = current_user.namespace_id
|
2013-01-17 10:35:57 -05:00
|
|
|
end
|
|
|
|
|
2019-08-06 09:05:18 -04:00
|
|
|
@relations_block&.call(@project)
|
2017-08-29 07:11:21 -04:00
|
|
|
yield(@project) if block_given?
|
|
|
|
|
2019-04-09 11:38:58 -04:00
|
|
|
validate_classification_label(@project, :external_authorization_classification_label)
|
|
|
|
|
2018-06-05 09:56:29 -04:00
|
|
|
# If the block added errors, don't try to save the project
|
|
|
|
return @project if @project.errors.any?
|
|
|
|
|
2013-01-28 10:22:45 -05:00
|
|
|
@project.creator = current_user
|
2013-01-17 10:35:57 -05:00
|
|
|
|
2019-08-06 09:05:18 -04:00
|
|
|
save_project_and_import_data
|
2014-03-12 11:14:48 -04:00
|
|
|
|
2016-06-15 11:31:00 -04:00
|
|
|
after_create_actions if @project.persisted?
|
2013-01-17 10:35:57 -05:00
|
|
|
|
2018-01-02 09:16:40 -05:00
|
|
|
import_schedule
|
2017-06-01 10:27:35 -04:00
|
|
|
|
2013-01-17 10:35:57 -05:00
|
|
|
@project
|
2017-04-13 19:09:17 -04:00
|
|
|
rescue ActiveRecord::RecordInvalid => e
|
|
|
|
message = "Unable to save #{e.record.type}: #{e.record.errors.full_messages.join(", ")} "
|
|
|
|
fail(error: message)
|
2015-12-08 23:52:13 -05:00
|
|
|
rescue => e
|
2018-06-06 04:56:50 -04:00
|
|
|
@project.errors.add(:base, e.message) if @project
|
2016-05-30 12:02:54 -04:00
|
|
|
fail(error: e.message)
|
2013-01-17 10:35:57 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
def deny_namespace
|
|
|
|
@project.errors.add(:namespace, "is not valid")
|
|
|
|
end
|
|
|
|
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2013-01-17 10:35:57 -05:00
|
|
|
def allowed_namespace?(user, namespace_id)
|
2014-01-19 13:55:59 -05:00
|
|
|
namespace = Namespace.find_by(id: namespace_id)
|
2014-05-28 12:02:26 -04:00
|
|
|
current_user.can?(:create_projects, namespace)
|
2013-01-17 10:35:57 -05:00
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2014-11-29 14:34:18 -05:00
|
|
|
|
|
|
|
def after_create_actions
|
2018-03-07 09:07:09 -05:00
|
|
|
log_info("#{@project.owner.name} created a new project \"#{@project.full_name}\"")
|
2015-02-13 06:01:28 -05:00
|
|
|
|
2018-01-03 03:10:43 -05:00
|
|
|
unless @project.gitlab_project_import?
|
2017-12-21 10:58:36 -05:00
|
|
|
@project.write_repository_config
|
2016-10-04 01:40:03 -04:00
|
|
|
@project.create_wiki unless skip_wiki?
|
2016-06-15 11:31:00 -04:00
|
|
|
end
|
2015-09-03 10:12:15 -04:00
|
|
|
|
2018-11-29 11:08:14 -05:00
|
|
|
@project.track_project_repository
|
2020-02-10 07:08:59 -05:00
|
|
|
@project.create_project_setting unless @project.project_setting
|
2018-11-29 11:08:14 -05:00
|
|
|
|
2015-02-13 06:01:28 -05:00
|
|
|
event_service.create_project(@project, current_user)
|
2014-11-29 14:34:18 -05:00
|
|
|
system_hook_service.execute_hooks_for(@project, :create)
|
|
|
|
|
2017-08-23 10:28:16 -04:00
|
|
|
setup_authorizations
|
2018-04-04 11:14:19 -04:00
|
|
|
|
|
|
|
current_user.invalidate_personal_projects_count
|
2018-07-03 09:09:58 -04:00
|
|
|
|
|
|
|
create_readme if @initialize_with_readme
|
2017-08-23 10:28:16 -04:00
|
|
|
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
|
2017-08-25 11:32:06 -04:00
|
|
|
if @project.group
|
|
|
|
@project.group.refresh_members_authorized_projects(blocking: false)
|
2017-08-23 10:28:16 -04:00
|
|
|
current_user.refresh_authorized_projects
|
|
|
|
else
|
2018-07-11 10:36:08 -04:00
|
|
|
@project.add_maintainer(@project.namespace.owner, current_user: current_user)
|
2014-11-29 14:34:18 -05:00
|
|
|
end
|
|
|
|
end
|
2016-05-11 09:08:27 -04:00
|
|
|
|
2018-07-03 09:09:58 -04:00
|
|
|
def create_readme
|
|
|
|
commit_attrs = {
|
|
|
|
branch_name: 'master',
|
|
|
|
commit_message: 'Initial commit',
|
|
|
|
file_path: 'README.md',
|
|
|
|
file_content: "# #{@project.name}\n\n#{@project.description}"
|
|
|
|
}
|
|
|
|
|
|
|
|
Files::CreateService.new(@project, current_user, commit_attrs).execute
|
|
|
|
end
|
|
|
|
|
2016-10-04 01:40:03 -04:00
|
|
|
def skip_wiki?
|
|
|
|
!@project.feature_available?(:wiki, current_user) || @skip_wiki
|
|
|
|
end
|
|
|
|
|
2019-08-06 09:05:18 -04:00
|
|
|
def save_project_and_import_data
|
2016-05-11 09:08:27 -04:00
|
|
|
Project.transaction do
|
2019-08-06 09:05:18 -04:00
|
|
|
@project.create_or_update_import_data(data: @import_data[:data], credentials: @import_data[:credentials]) if @import_data
|
2016-05-11 09:08:27 -04:00
|
|
|
|
2018-03-27 05:28:04 -04:00
|
|
|
if @project.save
|
|
|
|
unless @project.gitlab_project_import?
|
2020-02-11 13:08:58 -05:00
|
|
|
create_services_from_active_templates(@project)
|
2018-03-27 05:28:04 -04:00
|
|
|
@project.create_labels
|
|
|
|
end
|
|
|
|
|
|
|
|
unless @project.import?
|
|
|
|
raise 'Failed to create repository' unless @project.create_repository
|
|
|
|
end
|
2016-05-11 09:08:27 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-05-30 12:02:54 -04:00
|
|
|
|
|
|
|
def fail(error:)
|
|
|
|
message = "Unable to save project. Error: #{error}"
|
2018-03-27 11:33:29 -04:00
|
|
|
log_message = message.dup
|
2016-05-30 12:02:54 -04:00
|
|
|
|
2018-03-27 11:33:29 -04:00
|
|
|
log_message << " Project ID: #{@project.id}" if @project&.id
|
2019-07-10 15:26:47 -04:00
|
|
|
Rails.logger.error(log_message) # rubocop:disable Gitlab/RailsLogger
|
2016-05-30 12:02:54 -04:00
|
|
|
|
2019-04-10 06:03:37 -04:00
|
|
|
if @project && @project.persisted? && @project.import_state
|
|
|
|
@project.import_state.mark_as_failed(message)
|
2016-05-30 12:02:54 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
@project
|
|
|
|
end
|
2016-11-16 06:46:07 -05:00
|
|
|
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2020-02-11 13:08:58 -05:00
|
|
|
def create_services_from_active_templates(project)
|
|
|
|
Service.where(template: true, active: true).each do |template|
|
|
|
|
service = Service.build_from_template(project.id, template)
|
2016-11-16 06:46:07 -05:00
|
|
|
service.save!
|
|
|
|
end
|
|
|
|
end
|
2018-08-27 11:31:01 -04:00
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2017-01-23 13:41:30 -05:00
|
|
|
|
|
|
|
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
|
2018-01-02 09:16:40 -05:00
|
|
|
|
|
|
|
private
|
|
|
|
|
2019-10-08 20:06:06 -04:00
|
|
|
def create_from_template?
|
|
|
|
@params[:template_name].present? || @params[:template_project_id].present?
|
|
|
|
end
|
|
|
|
|
2018-01-02 09:16:40 -05:00
|
|
|
def import_schedule
|
|
|
|
if @project.errors.empty?
|
2018-11-27 04:41:27 -05:00
|
|
|
@project.import_state.schedule if @project.import? && !@project.bare_repository_import?
|
2018-01-02 09:16:40 -05:00
|
|
|
else
|
|
|
|
fail(error: @project.errors.full_messages.join(', '))
|
|
|
|
end
|
|
|
|
end
|
2019-08-06 09:05:18 -04:00
|
|
|
|
|
|
|
def project_visibility
|
|
|
|
@project_visibility ||= Gitlab::VisibilityLevelChecker
|
|
|
|
.new(current_user, @project, project_params: { import_data: @import_data })
|
|
|
|
.level_restricted?
|
|
|
|
end
|
2013-01-17 10:35:57 -05:00
|
|
|
end
|
|
|
|
end
|
2019-09-13 09:26:31 -04:00
|
|
|
|
|
|
|
Projects::CreateService.prepend_if_ee('EE::Projects::CreateService')
|