2013-01-17 10:35:57 -05:00
|
|
|
module Projects
|
2015-03-07 14:47:06 -05:00
|
|
|
class CreateService < BaseService
|
2013-01-18 13:21:13 -05:00
|
|
|
def initialize(user, params)
|
|
|
|
@current_user, @params = user, params.dup
|
|
|
|
end
|
|
|
|
|
2013-01-17 10:35:57 -05:00
|
|
|
def execute
|
2015-04-24 15:37:12 -04:00
|
|
|
forked_from_project_id = params.delete(:forked_from_project_id)
|
2016-05-10 05:10:51 -04:00
|
|
|
import_data = params.delete(:import_data)
|
2016-10-04 01:40:03 -04:00
|
|
|
@skip_wiki = params.delete(:skip_wiki)
|
|
|
|
|
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
|
2016-03-20 16:03:53 -04:00
|
|
|
unless Gitlab::VisibilityLevel.allowed_for?(current_user, params[:visibility_level])
|
2016-03-18 20:04:53 -04:00
|
|
|
deny_visibility_level(@project)
|
|
|
|
return @project
|
|
|
|
end
|
2013-11-06 10:13:21 -05:00
|
|
|
|
2016-09-19 14:28:41 -04:00
|
|
|
unless allowed_fork?(forked_from_project_id)
|
|
|
|
@project.errors.add(:forked_from_project_id, 'is forbidden')
|
|
|
|
return @project
|
|
|
|
end
|
|
|
|
|
2014-12-22 06:27:48 -05:00
|
|
|
# Set project name from path
|
2014-12-22 10:09:48 -05:00
|
|
|
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
|
2014-12-22 06:27:48 -05:00
|
|
|
@project.name = @project.path.dup
|
2014-12-22 10:09:48 -05:00
|
|
|
elsif @project.name.present?
|
|
|
|
# For compatibility - set path from name
|
|
|
|
# TODO: remove this in 8.0
|
|
|
|
@project.path = @project.name.dup.parameterize
|
2014-12-22 06:27:48 -05:00
|
|
|
end
|
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
|
|
|
|
|
2013-01-28 10:22:45 -05:00
|
|
|
@project.creator = current_user
|
2013-01-17 10:35:57 -05:00
|
|
|
|
2015-04-24 15:37:12 -04:00
|
|
|
if forked_from_project_id
|
|
|
|
@project.build_forked_project_link(forked_from_project_id: forked_from_project_id)
|
|
|
|
end
|
|
|
|
|
2016-05-12 11:16:35 -04:00
|
|
|
save_project_and_import_data(import_data)
|
2014-03-12 11:14:48 -04:00
|
|
|
|
2016-05-13 07:12:21 -04:00
|
|
|
@project.import_start if @project.import?
|
2016-05-11 05:37:49 -04:00
|
|
|
|
2016-06-15 11:31:00 -04:00
|
|
|
after_create_actions if @project.persisted?
|
2013-01-17 10:35:57 -05:00
|
|
|
|
2016-05-30 12:02:54 -04:00
|
|
|
if @project.errors.empty?
|
|
|
|
@project.add_import_job if @project.import?
|
|
|
|
else
|
2016-05-31 08:53:33 -04:00
|
|
|
fail(error: @project.errors.full_messages.join(', '))
|
2016-05-30 12:02:54 -04:00
|
|
|
end
|
2013-01-17 10:35:57 -05:00
|
|
|
@project
|
2015-12-08 23:52:13 -05:00
|
|
|
rescue => e
|
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
|
|
|
|
|
2016-09-19 14:28:41 -04:00
|
|
|
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
|
|
|
|
|
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
|
2014-11-29 14:34:18 -05:00
|
|
|
|
|
|
|
def after_create_actions
|
|
|
|
log_info("#{@project.owner.name} created a new project \"#{@project.name_with_namespace}\"")
|
2015-02-13 06:01:28 -05:00
|
|
|
|
2016-06-15 11:31:00 -04:00
|
|
|
unless @project.gitlab_project_import?
|
2016-10-04 01:40:03 -04:00
|
|
|
@project.create_wiki unless skip_wiki?
|
2016-06-15 11:31:00 -04:00
|
|
|
@project.build_missing_services
|
2015-07-25 16:43:18 -04:00
|
|
|
|
2016-06-15 11:31:00 -04:00
|
|
|
@project.create_labels
|
|
|
|
end
|
2015-09-03 10:12:15 -04: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)
|
|
|
|
|
2016-06-15 11:31:00 -04:00
|
|
|
unless @project.group || @project.gitlab_project_import?
|
2016-01-20 13:55:47 -05:00
|
|
|
@project.team << [current_user, :master, current_user]
|
2014-11-29 14:34:18 -05:00
|
|
|
end
|
|
|
|
end
|
2016-05-11 09:08:27 -04:00
|
|
|
|
2016-10-04 01:40:03 -04:00
|
|
|
def skip_wiki?
|
|
|
|
!@project.feature_available?(:wiki, current_user) || @skip_wiki
|
|
|
|
end
|
|
|
|
|
2016-05-12 11:16:35 -04:00
|
|
|
def save_project_and_import_data(import_data)
|
2016-05-11 09:08:27 -04:00
|
|
|
Project.transaction do
|
|
|
|
@project.create_or_update_import_data(data: import_data[:data], credentials: import_data[:credentials]) if import_data
|
|
|
|
|
2016-05-12 12:27:20 -04:00
|
|
|
if @project.save && !@project.import?
|
2016-05-11 09:08:27 -04:00
|
|
|
raise 'Failed to create repository' unless @project.create_repository
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-05-30 12:02:54 -04:00
|
|
|
|
|
|
|
def fail(error:)
|
|
|
|
message = "Unable to save project. Error: #{error}"
|
|
|
|
message << "Project ID: #{@project.id}" if @project && @project.id
|
|
|
|
|
|
|
|
Rails.logger.error(message)
|
|
|
|
|
|
|
|
if @project && @project.import?
|
|
|
|
@project.errors.add(:base, message)
|
|
|
|
@project.mark_import_as_failed(message)
|
|
|
|
end
|
|
|
|
|
|
|
|
@project
|
|
|
|
end
|
2013-01-17 10:35:57 -05:00
|
|
|
end
|
|
|
|
end
|