class RepositoryForkWorker ForkError = Class.new(StandardError) include Sidekiq::Worker include Gitlab::ShellAdapter include DedicatedSidekiqQueue sidekiq_options status_expiration: StuckImportJobsWorker::IMPORT_JOBS_EXPIRATION def perform(project_id, forked_from_repository_storage_path, source_path, target_path) project = Project.find(project_id) return unless start_fork(project) Gitlab::Metrics.add_event(:fork_repository, source_path: source_path, target_path: target_path) result = gitlab_shell.fork_repository(forked_from_repository_storage_path, source_path, project.repository_storage_path, target_path) raise ForkError, "Unable to fork project #{project_id} for repository #{source_path} -> #{target_path}" unless result project.repository.after_import raise ForkError, "Project #{project_id} had an invalid repository after fork" unless project.valid_repo? project.import_finish rescue ForkError => ex fail_fork(project, ex.message) raise rescue => ex return unless project fail_fork(project, ex.message) raise ForkError, "#{ex.class} #{ex.message}" end private def start_fork(project) return true if project.import_start Rails.logger.info("Project #{project.full_path} was in inconsistent state (#{project.import_status}) while forking.") false end def fail_fork(project, message) Rails.logger.error(message) project.mark_import_as_failed(message) end end