gitlab-org--gitlab-foss/lib/backup/repository.rb

178 lines
5.1 KiB
Ruby
Raw Normal View History

# frozen_string_literal: true
2013-04-05 12:01:19 -04:00
require 'yaml'
module Backup
class Repository
include Gitlab::ShellAdapter
attr_reader :progress
def initialize(progress)
@progress = progress
end
2013-04-05 12:01:19 -04:00
def dump
prepare
Project.find_each(batch_size: 1000) do |project|
progress.print " * #{display_repo_path(project)} ... "
if project.hashed_storage?(:repository)
FileUtils.mkdir_p(File.dirname(File.join(backup_repos_path, project.disk_path)))
else
FileUtils.mkdir_p(File.join(backup_repos_path, project.namespace.full_path)) if project.namespace
end
2013-04-05 12:01:19 -04:00
2018-06-13 05:19:17 -04:00
if !empty_repo?(project)
backup_project(project)
2018-06-18 10:42:39 -04:00
progress.puts "[DONE]".color(:green)
2013-04-05 12:01:19 -04:00
else
2018-06-13 05:19:17 -04:00
progress.puts "[SKIPPED]".color(:cyan)
2013-04-05 12:01:19 -04:00
end
2013-04-05 14:20:11 -04:00
wiki = ProjectWiki.new(project)
2013-04-05 14:20:11 -04:00
2018-06-18 10:42:39 -04:00
if !empty_repo?(wiki)
2018-06-13 05:19:17 -04:00
backup_project(wiki)
2018-06-18 10:42:39 -04:00
progress.puts "[DONE] Wiki".color(:green)
2018-06-13 05:19:17 -04:00
else
progress.puts "[SKIPPED] Wiki".color(:cyan)
2013-04-05 14:20:11 -04:00
end
2013-04-05 12:01:19 -04:00
end
end
def prepare_directories
Gitlab.config.repositories.storages.each do |name, _repository_storage|
Gitlab::GitalyClient::StorageService.new(name).delete_all_repositories
end
end
2018-06-13 05:19:17 -04:00
def backup_project(project)
path_to_project_bundle = path_to_bundle(project)
Gitlab::GitalyClient::RepositoryService.new(project.repository)
.create_bundle(path_to_project_bundle)
backup_custom_hooks(project)
rescue => e
progress_warn(project, e, 'Failed to backup repo')
end
def backup_custom_hooks(project)
FileUtils.mkdir_p(project_backup_path(project))
2018-06-13 05:19:17 -04:00
custom_hooks_path = custom_hooks_tar(project)
2018-06-13 05:19:17 -04:00
Gitlab::GitalyClient::RepositoryService.new(project.repository)
.backup_custom_hooks(custom_hooks_path)
end
def restore_custom_hooks(project)
return unless Dir.exist?(project_backup_path(project))
return if Dir.glob("#{project_backup_path(project)}/custom_hooks*").none?
custom_hooks_path = custom_hooks_tar(project)
Gitlab::GitalyClient::RepositoryService.new(project.repository)
.restore_custom_hooks(custom_hooks_path)
end
def restore
prepare_directories
2013-04-05 12:01:19 -04:00
Project.find_each(batch_size: 1000) do |project|
2018-05-11 18:09:26 -04:00
progress.print " * #{project.full_path} ... "
path_to_project_bundle = path_to_bundle(project)
2017-08-15 22:49:54 -04:00
project.ensure_storage_path_exists
2013-04-05 12:01:19 -04:00
restore_repo_success = nil
if File.exist?(path_to_project_bundle)
begin
project.repository.create_from_bundle(path_to_project_bundle)
restore_custom_hooks(project)
restore_repo_success = true
rescue => e
restore_repo_success = false
progress.puts "Error: #{e}".color(:red)
end
else
restore_repo_success = gitlab_shell.create_project_repository(project)
end
if restore_repo_success
2017-06-07 10:50:21 -04:00
progress.puts "[DONE]".color(:green)
2013-04-05 12:01:19 -04:00
else
2018-05-11 18:09:26 -04:00
progress.puts "[Failed] restoring #{project.full_path} repository".color(:red)
2013-04-05 12:01:19 -04:00
end
2013-04-05 14:20:11 -04:00
wiki = ProjectWiki.new(project)
path_to_wiki_bundle = path_to_bundle(wiki)
2013-04-05 14:20:11 -04:00
if File.exist?(path_to_wiki_bundle)
2018-05-11 18:09:26 -04:00
progress.print " * #{wiki.full_path} ... "
begin
2018-05-11 18:09:26 -04:00
wiki.repository.create_from_bundle(path_to_wiki_bundle)
2018-06-21 07:56:00 -04:00
restore_custom_hooks(wiki)
2018-05-11 18:09:26 -04:00
progress.puts "[DONE]".color(:green)
rescue => e
2018-05-11 18:09:26 -04:00
progress.puts "[Failed] restoring #{wiki.full_path} wiki".color(:red)
progress.puts "Error #{e}".color(:red)
end
2013-04-05 14:20:11 -04:00
end
2013-04-05 12:01:19 -04:00
end
restore_object_pools
2013-04-05 12:01:19 -04:00
end
protected
def path_to_bundle(project)
File.join(backup_repos_path, project.disk_path + '.bundle')
end
def project_backup_path(project)
File.join(backup_repos_path, project.disk_path)
end
def custom_hooks_tar(project)
File.join(project_backup_path(project), "custom_hooks.tar")
2013-04-05 12:01:19 -04:00
end
def backup_repos_path
File.join(Gitlab.config.backup.path, 'repositories')
end
2013-04-05 12:01:19 -04:00
def prepare
FileUtils.rm_rf(backup_repos_path)
FileUtils.mkdir_p(Gitlab.config.backup.path)
FileUtils.mkdir(backup_repos_path, mode: 0700)
2013-04-05 12:01:19 -04:00
end
2013-11-05 09:00:48 -05:00
private
def progress_warn(project, cmd, output)
2017-06-07 10:50:21 -04:00
progress.puts "[WARNING] Executing #{cmd}".color(:orange)
progress.puts "Ignoring error on #{display_repo_path(project)} - #{output}".color(:orange)
end
2017-06-07 10:50:21 -04:00
def empty_repo?(project_or_wiki)
2017-12-07 10:33:30 -05:00
project_or_wiki.repository.expire_emptiness_caches
project_or_wiki.repository.empty?
end
def display_repo_path(project)
project.hashed_storage?(:repository) ? "#{project.full_path} (#{project.disk_path})" : project.full_path
end
def restore_object_pools
PoolRepository.includes(:source_project).find_each do |pool|
progress.puts " - Object pool #{pool.disk_path}..."
pool.source_project ||= pool.member_projects.first.root_of_fork_network
pool.state = 'none'
pool.save
pool.schedule
end
end
2013-04-05 12:01:19 -04:00
end
end