2013-04-05 12:01:19 -04:00
|
|
|
require 'yaml'
|
|
|
|
|
|
|
|
module Backup
|
|
|
|
class Repository
|
2017-02-22 13:18:40 -05:00
|
|
|
# rubocop:disable Metrics/AbcSize
|
2013-04-05 12:01:19 -04:00
|
|
|
def dump
|
|
|
|
prepare
|
|
|
|
|
|
|
|
Project.find_each(batch_size: 1000) do |project|
|
2014-11-20 09:46:04 -05:00
|
|
|
$progress.print " * #{project.path_with_namespace} ... "
|
2016-08-11 13:37:39 -04:00
|
|
|
path_to_project_repo = path_to_repo(project)
|
|
|
|
path_to_project_bundle = path_to_bundle(project)
|
2013-04-05 12:01:19 -04:00
|
|
|
|
|
|
|
# Create namespace dir if missing
|
2017-02-14 09:03:24 -05:00
|
|
|
FileUtils.mkdir_p(File.join(backup_repos_path, project.namespace.full_path)) if project.namespace
|
2013-04-05 12:01:19 -04:00
|
|
|
|
2014-05-08 16:49:27 -04:00
|
|
|
if project.empty_repo?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts "[SKIPPED]".color(:cyan)
|
2013-04-05 12:01:19 -04:00
|
|
|
else
|
2016-08-11 13:37:39 -04:00
|
|
|
in_path(path_to_project_repo) do |dir|
|
|
|
|
FileUtils.mkdir_p(path_to_tars(project))
|
|
|
|
cmd = %W(tar -cf #{path_to_tars(project, dir)} -C #{path_to_project_repo} #{dir})
|
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
|
|
|
|
|
|
|
unless status.zero?
|
|
|
|
puts "[FAILED]".color(:red)
|
|
|
|
puts "failed: #{cmd.join(' ')}"
|
|
|
|
puts output
|
|
|
|
abort 'Backup failed'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
cmd = %W(#{Gitlab.config.git.bin_path} --git-dir=#{path_to_project_repo} bundle create #{path_to_project_bundle} --all)
|
2014-11-20 09:46:04 -05:00
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
2016-08-11 13:37:39 -04:00
|
|
|
|
2014-10-01 09:43:27 -04:00
|
|
|
if status.zero?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts "[DONE]".color(:green)
|
2014-10-01 09:43:27 -04:00
|
|
|
else
|
2016-06-01 18:37:15 -04:00
|
|
|
puts "[FAILED]".color(:red)
|
2014-11-20 09:46:04 -05:00
|
|
|
puts "failed: #{cmd.join(' ')}"
|
2014-10-01 09:43:27 -04:00
|
|
|
puts output
|
|
|
|
abort 'Backup failed'
|
|
|
|
end
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
2013-04-05 14:20:11 -04:00
|
|
|
|
2014-04-09 07:35:58 -04:00
|
|
|
wiki = ProjectWiki.new(project)
|
2016-08-11 13:37:39 -04:00
|
|
|
path_to_wiki_repo = path_to_repo(wiki)
|
|
|
|
path_to_wiki_bundle = path_to_bundle(wiki)
|
2013-04-05 14:20:11 -04:00
|
|
|
|
2016-08-11 13:37:39 -04:00
|
|
|
if File.exist?(path_to_wiki_repo)
|
2014-11-20 09:46:04 -05:00
|
|
|
$progress.print " * #{wiki.path_with_namespace} ... "
|
2014-10-20 04:52:29 -04:00
|
|
|
if wiki.repository.empty?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts " [SKIPPED]".color(:cyan)
|
2013-04-05 14:20:11 -04:00
|
|
|
else
|
2016-08-11 13:37:39 -04:00
|
|
|
cmd = %W(#{Gitlab.config.git.bin_path} --git-dir=#{path_to_wiki_repo} bundle create #{path_to_wiki_bundle} --all)
|
2014-11-20 09:46:04 -05:00
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
2014-10-01 09:43:27 -04:00
|
|
|
if status.zero?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts " [DONE]".color(:green)
|
2014-10-01 09:43:27 -04:00
|
|
|
else
|
2016-06-01 18:37:15 -04:00
|
|
|
puts " [FAILED]".color(:red)
|
2014-11-20 09:46:04 -05:00
|
|
|
puts "failed: #{cmd.join(' ')}"
|
2016-08-11 13:37:39 -04:00
|
|
|
puts output
|
2014-10-01 09:43:27 -04:00
|
|
|
abort 'Backup failed'
|
|
|
|
end
|
2013-04-05 14:20:11 -04:00
|
|
|
end
|
|
|
|
end
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def restore
|
2017-02-28 16:08:40 -05:00
|
|
|
Gitlab.config.repositories.storages.each do |name, repository_storage|
|
|
|
|
path = repository_storage['path']
|
2016-08-09 17:23:25 -04:00
|
|
|
next unless File.exist?(path)
|
2016-06-22 17:04:51 -04:00
|
|
|
|
2013-04-05 12:01:19 -04:00
|
|
|
# Move repos dir to 'repositories.old' dir
|
2016-06-22 17:04:51 -04:00
|
|
|
bk_repos_path = File.join(path, '..', 'repositories.old.' + Time.now.to_i.to_s)
|
|
|
|
FileUtils.mv(path, bk_repos_path)
|
2016-07-24 07:56:27 -04:00
|
|
|
# This is expected from gitlab:check
|
2016-08-29 10:33:19 -04:00
|
|
|
FileUtils.mkdir_p(path, mode: 02770)
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
Project.find_each(batch_size: 1000) do |project|
|
2014-11-30 11:24:05 -05:00
|
|
|
$progress.print " * #{project.path_with_namespace} ... "
|
2016-08-11 13:37:39 -04:00
|
|
|
path_to_project_repo = path_to_repo(project)
|
|
|
|
path_to_project_bundle = path_to_bundle(project)
|
2013-04-05 12:01:19 -04:00
|
|
|
|
2016-06-22 17:04:51 -04:00
|
|
|
project.ensure_dir_exist
|
2013-04-05 12:01:19 -04:00
|
|
|
|
2017-02-22 13:18:40 -05:00
|
|
|
cmd = if File.exist?(path_to_project_bundle)
|
|
|
|
%W(#{Gitlab.config.git.bin_path} clone --bare #{path_to_project_bundle} #{path_to_project_repo})
|
|
|
|
else
|
|
|
|
%W(#{Gitlab.config.git.bin_path} init --bare #{path_to_project_repo})
|
|
|
|
end
|
2014-11-13 07:09:47 -05:00
|
|
|
|
2016-08-11 13:37:39 -04:00
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
|
|
|
if status.zero?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts "[DONE]".color(:green)
|
2013-04-05 12:01:19 -04:00
|
|
|
else
|
2016-06-01 18:37:15 -04:00
|
|
|
puts "[FAILED]".color(:red)
|
2014-11-20 09:46:04 -05:00
|
|
|
puts "failed: #{cmd.join(' ')}"
|
2016-08-11 13:37:39 -04:00
|
|
|
puts output
|
2014-10-01 09:43:27 -04:00
|
|
|
abort 'Restore failed'
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
2013-04-05 14:20:11 -04:00
|
|
|
|
2016-08-11 13:37:39 -04:00
|
|
|
in_path(path_to_tars(project)) do |dir|
|
|
|
|
cmd = %W(tar -xf #{path_to_tars(project, dir)} -C #{path_to_project_repo} #{dir})
|
|
|
|
|
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
|
|
|
unless status.zero?
|
|
|
|
puts "[FAILED]".color(:red)
|
|
|
|
puts "failed: #{cmd.join(' ')}"
|
|
|
|
puts output
|
|
|
|
abort 'Restore failed'
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-04-09 07:35:58 -04:00
|
|
|
wiki = ProjectWiki.new(project)
|
2016-08-11 13:37:39 -04:00
|
|
|
path_to_wiki_repo = path_to_repo(wiki)
|
|
|
|
path_to_wiki_bundle = path_to_bundle(wiki)
|
2013-04-05 14:20:11 -04:00
|
|
|
|
2016-08-11 13:37:39 -04:00
|
|
|
if File.exist?(path_to_wiki_bundle)
|
2014-11-30 11:24:05 -05:00
|
|
|
$progress.print " * #{wiki.path_with_namespace} ... "
|
|
|
|
|
|
|
|
# If a wiki bundle exists, first remove the empty repo
|
|
|
|
# that was initialized with ProjectWiki.new() and then
|
|
|
|
# try to restore with 'git clone --bare'.
|
2016-08-11 13:37:39 -04:00
|
|
|
FileUtils.rm_rf(path_to_wiki_repo)
|
|
|
|
cmd = %W(#{Gitlab.config.git.bin_path} clone --bare #{path_to_wiki_bundle} #{path_to_wiki_repo})
|
2014-11-29 14:59:28 -05:00
|
|
|
|
2016-08-11 13:37:39 -04:00
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
|
|
|
if status.zero?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts " [DONE]".color(:green)
|
2014-11-30 11:24:05 -05:00
|
|
|
else
|
2016-06-01 18:37:15 -04:00
|
|
|
puts " [FAILED]".color(:red)
|
2014-11-30 11:24:05 -05:00
|
|
|
puts "failed: #{cmd.join(' ')}"
|
2016-08-11 13:37:39 -04:00
|
|
|
puts output
|
2014-11-30 11:24:05 -05:00
|
|
|
abort 'Restore failed'
|
|
|
|
end
|
2013-04-05 14:20:11 -04:00
|
|
|
end
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
2013-04-10 08:11:45 -04:00
|
|
|
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.print 'Put GitLab hooks in repositories dirs'.color(:yellow)
|
2016-06-22 17:04:51 -04:00
|
|
|
cmd = %W(#{Gitlab.config.gitlab_shell.path}/bin/create-hooks) + repository_storage_paths_args
|
2016-08-11 13:37:39 -04:00
|
|
|
|
|
|
|
output, status = Gitlab::Popen.popen(cmd)
|
|
|
|
if status.zero?
|
2016-06-01 18:37:15 -04:00
|
|
|
$progress.puts " [DONE]".color(:green)
|
2013-04-10 08:11:45 -04:00
|
|
|
else
|
2016-06-01 18:37:15 -04:00
|
|
|
puts " [FAILED]".color(:red)
|
2014-11-20 09:46:04 -05:00
|
|
|
puts "failed: #{cmd}"
|
2016-08-11 13:37:39 -04:00
|
|
|
puts output
|
2013-04-10 08:11:45 -04:00
|
|
|
end
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
2017-02-22 13:18:40 -05:00
|
|
|
# rubocop:enable Metrics/AbcSize
|
2013-04-05 12:01:19 -04:00
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
def path_to_repo(project)
|
2014-11-05 11:51:08 -05:00
|
|
|
project.repository.path_to_repo
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def path_to_bundle(project)
|
2016-08-11 13:37:39 -04:00
|
|
|
File.join(backup_repos_path, project.path_with_namespace + '.bundle')
|
|
|
|
end
|
|
|
|
|
|
|
|
def path_to_tars(project, dir = nil)
|
|
|
|
path = File.join(backup_repos_path, project.path_with_namespace)
|
|
|
|
|
|
|
|
if dir
|
|
|
|
File.join(path, "#{dir}.tar")
|
|
|
|
else
|
|
|
|
path
|
|
|
|
end
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def backup_repos_path
|
2016-08-11 13:37:39 -04:00
|
|
|
File.join(Gitlab.config.backup.path, 'repositories')
|
|
|
|
end
|
|
|
|
|
|
|
|
def in_path(path)
|
|
|
|
return unless Dir.exist?(path)
|
|
|
|
|
|
|
|
dir_entries = Dir.entries(path)
|
2017-02-24 07:18:07 -05:00
|
|
|
|
|
|
|
yield('custom_hooks') if dir_entries.include?('custom_hooks')
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def prepare
|
|
|
|
FileUtils.rm_rf(backup_repos_path)
|
2015-07-30 04:17:34 -04:00
|
|
|
# Ensure the parent dir of backup_repos_path exists
|
|
|
|
FileUtils.mkdir_p(Gitlab.config.backup.path)
|
|
|
|
# Fail if somebody raced to create backup_repos_path before us
|
|
|
|
FileUtils.mkdir(backup_repos_path, mode: 0700)
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
2013-11-05 09:00:48 -05:00
|
|
|
|
|
|
|
def silent
|
2017-02-22 13:18:40 -05:00
|
|
|
{ err: '/dev/null', out: '/dev/null' }
|
2013-11-05 09:00:48 -05:00
|
|
|
end
|
2016-06-22 17:04:51 -04:00
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
def repository_storage_paths_args
|
2017-02-28 16:08:40 -05:00
|
|
|
Gitlab.config.repositories.storages.values.map { |rs| rs['path'] }
|
2016-06-22 17:04:51 -04:00
|
|
|
end
|
2013-04-05 12:01:19 -04:00
|
|
|
end
|
|
|
|
end
|