debian-mirror-gitlab/lib/backup/repository.rb

246 lines
6.9 KiB
Ruby
Raw Normal View History

2018-12-05 23:21:45 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
require 'yaml'
module Backup
class Repository
2018-11-08 19:23:39 +05:30
attr_reader :progress
def initialize(progress)
@progress = progress
end
2018-05-09 12:01:36 +05:30
2020-10-24 23:57:45 +05:30
def dump(max_concurrency:, max_storage_concurrency:)
2014-09-02 18:07:02 +05:30
prepare
2020-10-24 23:57:45 +05:30
if max_concurrency <= 1 && max_storage_concurrency <= 1
return dump_consecutive
end
2014-09-02 18:07:02 +05:30
2020-10-24 23:57:45 +05:30
if Project.excluding_repository_storage(Gitlab.config.repositories.storages.keys).exists?
raise Error, 'repositories.storages in gitlab.yml is misconfigured'
end
2018-11-08 19:23:39 +05:30
2020-10-24 23:57:45 +05:30
semaphore = Concurrent::Semaphore.new(max_concurrency)
errors = Queue.new
2018-11-08 19:23:39 +05:30
2020-10-24 23:57:45 +05:30
threads = Gitlab.config.repositories.storages.keys.map do |storage|
Thread.new do
Rails.application.executor.wrap do
dump_storage(storage, semaphore, max_storage_concurrency: max_storage_concurrency)
rescue => e
errors << e
end
2018-11-08 19:23:39 +05:30
end
end
2020-10-24 23:57:45 +05:30
ActiveSupport::Dependencies.interlock.permit_concurrent_loads do
threads.each(&:join)
end
raise errors.pop unless errors.empty?
2018-11-08 19:23:39 +05:30
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
def backup_project(project)
path_to_project_bundle = path_to_bundle(project)
Gitlab::GitalyClient::RepositoryService.new(project.repository)
.create_bundle(path_to_project_bundle)
2018-11-18 11:00:15 +05:30
backup_custom_hooks(project)
rescue => e
progress_warn(project, e, 'Failed to backup repo')
2018-11-08 19:23:39 +05:30
end
2018-11-18 11:00:15 +05:30
def backup_custom_hooks(project)
FileUtils.mkdir_p(project_backup_path(project))
2018-11-08 19:23:39 +05:30
2018-11-18 11:00:15 +05:30
custom_hooks_path = custom_hooks_tar(project)
2018-11-08 19:23:39 +05:30
Gitlab::GitalyClient::RepositoryService.new(project.repository)
.backup_custom_hooks(custom_hooks_path)
end
def restore_custom_hooks(project)
2018-11-18 11:00:15 +05:30
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)
2018-05-09 12:01:36 +05:30
end
2014-09-02 18:07:02 +05:30
2018-05-09 12:01:36 +05:30
def restore
2014-09-02 18:07:02 +05:30
Project.find_each(batch_size: 1000) do |project|
2018-11-08 19:23:39 +05:30
progress.print " * #{project.full_path} ... "
2014-09-02 18:07:02 +05:30
2020-04-08 14:13:33 +05:30
restore_repo_success =
2018-11-08 19:23:39 +05:30
begin
2020-04-08 14:13:33 +05:30
try_restore_repository(project)
rescue => err
progress.puts "Error: #{err}".color(:red)
false
2018-11-08 19:23:39 +05:30
end
2017-08-17 22:00:37 +05:30
2018-11-08 19:23:39 +05:30
if restore_repo_success
2017-08-17 22:00:37 +05:30
progress.puts "[DONE]".color(:green)
2015-04-26 12:48:37 +05:30
else
2018-11-08 19:23:39 +05:30
progress.puts "[Failed] restoring #{project.full_path} repository".color(:red)
2015-04-26 12:48:37 +05:30
end
2014-09-02 18:07:02 +05:30
wiki = ProjectWiki.new(project)
2020-03-13 15:44:24 +05:30
wiki.repository.remove rescue nil
2017-08-17 22:00:37 +05:30
path_to_wiki_bundle = path_to_bundle(wiki)
2014-09-02 18:07:02 +05:30
2017-08-17 22:00:37 +05:30
if File.exist?(path_to_wiki_bundle)
2018-11-08 19:23:39 +05:30
progress.print " * #{wiki.full_path} ... "
begin
wiki.repository.create_from_bundle(path_to_wiki_bundle)
restore_custom_hooks(wiki)
2015-04-26 12:48:37 +05:30
2018-11-08 19:23:39 +05:30
progress.puts "[DONE]".color(:green)
rescue => e
progress.puts "[Failed] restoring #{wiki.full_path} wiki".color(:red)
progress.puts "Error #{e}".color(:red)
2014-09-02 18:07:02 +05:30
end
end
end
2019-02-15 15:39:39 +05:30
restore_object_pools
2014-09-02 18:07:02 +05:30
end
protected
2020-04-08 14:13:33 +05:30
def try_restore_repository(project)
path_to_project_bundle = path_to_bundle(project)
project.repository.remove rescue nil
if File.exist?(path_to_project_bundle)
project.repository.create_from_bundle(path_to_project_bundle)
restore_custom_hooks(project)
else
project.repository.create_repository
end
true
end
2014-09-02 18:07:02 +05:30
def path_to_bundle(project)
2017-09-10 17:25:29 +05:30
File.join(backup_repos_path, project.disk_path + '.bundle')
2017-08-17 22:00:37 +05:30
end
2018-11-18 11:00:15 +05:30
def project_backup_path(project)
File.join(backup_repos_path, project.disk_path)
end
2017-08-17 22:00:37 +05:30
2018-11-18 11:00:15 +05:30
def custom_hooks_tar(project)
File.join(project_backup_path(project), "custom_hooks.tar")
2014-09-02 18:07:02 +05:30
end
def backup_repos_path
2017-08-17 22:00:37 +05:30
File.join(Gitlab.config.backup.path, 'repositories')
end
2014-09-02 18:07:02 +05:30
def prepare
FileUtils.rm_rf(backup_repos_path)
2015-09-11 14:41:01 +05:30
FileUtils.mkdir_p(Gitlab.config.backup.path)
FileUtils.mkdir(backup_repos_path, mode: 0700)
2014-09-02 18:07:02 +05:30
end
2016-08-24 12:49:21 +05:30
private
2020-10-24 23:57:45 +05:30
def dump_consecutive
Project.find_each(batch_size: 1000) do |project|
dump_project(project)
end
end
def dump_storage(storage, semaphore, max_storage_concurrency:)
errors = Queue.new
queue = SizedQueue.new(1)
threads = Array.new(max_storage_concurrency) do
Thread.new do
Rails.application.executor.wrap do
while project = queue.pop
semaphore.acquire
begin
dump_project(project)
rescue => e
errors << e
break
ensure
semaphore.release
end
end
end
end
end
Project.for_repository_storage(storage).find_each(batch_size: 100) do |project|
break unless errors.empty?
queue.push(project)
end
raise errors.pop unless errors.empty?
ensure
queue.close
ActiveSupport::Dependencies.interlock.permit_concurrent_loads do
threads.each(&:join)
end
end
def dump_project(project)
progress.puts " * #{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
if !empty_repo?(project)
backup_project(project)
progress.puts " * #{display_repo_path(project)} ... " + "[DONE]".color(:green)
else
progress.puts " * #{display_repo_path(project)} ... " + "[SKIPPED]".color(:cyan)
end
wiki = ProjectWiki.new(project)
if !empty_repo?(wiki)
backup_project(wiki)
progress.puts " * #{display_repo_path(project)} ... " + "[DONE] Wiki".color(:green)
else
progress.puts " * #{display_repo_path(project)} ... " + "[SKIPPED] Wiki".color(:cyan)
end
end
2017-08-17 22:00:37 +05:30
def progress_warn(project, cmd, output)
progress.puts "[WARNING] Executing #{cmd}".color(:orange)
2018-03-17 18:26:18 +05:30
progress.puts "Ignoring error on #{display_repo_path(project)} - #{output}".color(:orange)
2017-08-17 22:00:37 +05:30
end
def empty_repo?(project_or_wiki)
2018-03-17 18:26:18 +05:30
project_or_wiki.repository.expire_emptiness_caches
project_or_wiki.repository.empty?
2017-08-17 22:00:37 +05:30
end
2018-03-17 18:26:18 +05:30
def display_repo_path(project)
project.hashed_storage?(:repository) ? "#{project.full_path} (#{project.disk_path})" : project.full_path
end
2019-02-15 15:39:39 +05:30
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
2014-09-02 18:07:02 +05:30
end
end