debian-mirror-gitlab/app/services/projects/create_service.rb

202 lines
6 KiB
Ruby
Raw Normal View History

2018-11-18 11:00:15 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
module Projects
class CreateService < BaseService
def initialize(user, params)
@current_user, @params = user, params.dup
2018-11-08 19:23:39 +05:30
@skip_wiki = @params.delete(:skip_wiki)
@initialize_with_readme = Gitlab::Utils.to_boolean(@params.delete(:initialize_with_readme))
2014-09-02 18:07:02 +05:30
end
def execute
2017-09-10 17:25:29 +05:30
if @params[:template_name]&.present?
return ::Projects::CreateFromTemplateService.new(current_user, params).execute
end
2015-09-11 14:41:01 +05:30
forked_from_project_id = params.delete(:forked_from_project_id)
2016-06-02 11:05:42 +05:30
import_data = params.delete(:import_data)
2016-11-03 12:29:30 +05:30
2014-09-02 18:07:02 +05:30
@project = Project.new(params)
2016-06-02 11:05:42 +05:30
# Make sure that the user is allowed to use the specified visibility level
2017-08-17 22:00:37 +05:30
unless Gitlab::VisibilityLevel.allowed_for?(current_user, @project.visibility_level)
2015-04-26 12:48:37 +05:30
deny_visibility_level(@project)
return @project
2014-09-02 18:07:02 +05:30
end
2016-10-01 15:18:49 +05:30
unless allowed_fork?(forked_from_project_id)
@project.errors.add(:forked_from_project_id, 'is forbidden')
return @project
end
2017-08-17 22:00:37 +05:30
set_project_name_from_path
2014-09-02 18:07:02 +05:30
# get namespace id
namespace_id = params[:namespace_id]
if namespace_id
# Find matching namespace and check if it allowed
# for current user if namespace_id passed.
unless allowed_namespace?(current_user, namespace_id)
@project.namespace_id = nil
deny_namespace
return @project
end
else
# Set current user namespace if namespace_id is nil
@project.namespace_id = current_user.namespace_id
end
2018-03-17 18:26:18 +05:30
yield(@project) if block_given?
2018-10-15 14:42:47 +05:30
# If the block added errors, don't try to save the project
return @project if @project.errors.any?
2014-09-02 18:07:02 +05:30
@project.creator = current_user
2015-09-11 14:41:01 +05:30
if forked_from_project_id
@project.build_forked_project_link(forked_from_project_id: forked_from_project_id)
end
2016-06-02 11:05:42 +05:30
save_project_and_import_data(import_data)
2014-09-02 18:07:02 +05:30
2015-04-26 12:48:37 +05:30
after_create_actions if @project.persisted?
2018-03-17 18:26:18 +05:30
import_schedule
2017-09-10 17:25:29 +05:30
2014-09-02 18:07:02 +05:30
@project
2017-08-17 22:00:37 +05:30
rescue ActiveRecord::RecordInvalid => e
message = "Unable to save #{e.record.type}: #{e.record.errors.full_messages.join(", ")} "
fail(error: message)
2015-12-23 02:04:40 +05:30
rescue => e
2018-11-08 19:23:39 +05:30
@project.errors.add(:base, e.message) if @project
fail(error: e.message)
2014-09-02 18:07:02 +05:30
end
protected
def deny_namespace
@project.errors.add(:namespace, "is not valid")
end
2016-10-01 15:18:49 +05:30
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
2014-09-02 18:07:02 +05:30
def allowed_namespace?(user, namespace_id)
namespace = Namespace.find_by(id: namespace_id)
current_user.can?(:create_projects, namespace)
end
2015-04-26 12:48:37 +05:30
def after_create_actions
2018-03-27 19:54:05 +05:30
log_info("#{@project.owner.name} created a new project \"#{@project.full_name}\"")
2015-04-26 12:48:37 +05:30
2016-06-22 15:30:34 +05:30
unless @project.gitlab_project_import?
2018-03-17 18:26:18 +05:30
@project.write_repository_config
2016-11-03 12:29:30 +05:30
@project.create_wiki unless skip_wiki?
2016-06-22 15:30:34 +05:30
end
2015-09-25 12:07:36 +05:30
2015-04-26 12:48:37 +05:30
event_service.create_project(@project, current_user)
system_hook_service.execute_hooks_for(@project, :create)
2018-03-17 18:26:18 +05:30
setup_authorizations
2018-05-09 12:01:36 +05:30
current_user.invalidate_personal_projects_count
2018-11-08 19:23:39 +05:30
create_readme if @initialize_with_readme
2018-03-17 18:26:18 +05:30
end
2017-08-17 22:00:37 +05:30
2018-03-17 18:26:18 +05:30
# 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
if @project.group
@project.group.refresh_members_authorized_projects(blocking: false)
current_user.refresh_authorized_projects
else
2018-11-18 11:00:15 +05:30
@project.add_maintainer(@project.namespace.owner, current_user: current_user)
2018-03-17 18:26:18 +05:30
end
2016-06-02 11:05:42 +05:30
end
2015-04-26 12:48:37 +05:30
2018-11-08 19:23:39 +05:30
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-11-03 12:29:30 +05:30
def skip_wiki?
!@project.feature_available?(:wiki, current_user) || @skip_wiki
end
2016-06-02 11:05:42 +05:30
def save_project_and_import_data(import_data)
Project.transaction do
@project.create_or_update_import_data(data: import_data[:data], credentials: import_data[:credentials]) if import_data
2018-05-09 12:01:36 +05:30
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
2016-06-02 11:05:42 +05:30
end
end
2015-04-26 12:48:37 +05:30
end
def fail(error:)
message = "Unable to save project. Error: #{error}"
2018-05-09 12:01:36 +05:30
log_message = message.dup
2018-05-09 12:01:36 +05:30
log_message << " Project ID: #{@project.id}" if @project&.id
Rails.logger.error(log_message)
2018-05-09 12:01:36 +05:30
if @project
2018-10-15 14:42:47 +05:30
@project.mark_import_as_failed(message) if @project.persisted? && @project.import?
end
@project
end
2017-08-17 22:00:37 +05:30
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
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-03-17 18:26:18 +05:30
private
def import_schedule
if @project.errors.empty?
@project.import_schedule if @project.import? && !@project.bare_repository_import?
else
fail(error: @project.errors.full_messages.join(', '))
end
end
2014-09-02 18:07:02 +05:30
end
end