Как сделать резервную копию локального репозитория Git?
Я использую git в относительно небольшом проекте и обнаружил, что архивирование содержимого каталога.git может быть хорошим способом для резервного копирования проекта. Но это немного странно, потому что, когда я восстанавливаюсь, первое, что мне нужно сделать, это git reset --hard
,
Есть ли проблемы с резервным копированием git-репо таким способом? Кроме того, есть ли лучший способ сделать это (например, портативный формат Git или что-то подобное?)?
7 ответов
Я начал немного взламывать скрипт Yar, и в результате получился github, включая man-страницы и скрипт установки:
https://github.com/najamelan/git-backup
Установка:
git clone "https://github.com/najamelan/git-backup.git"
cd git-backup
sudo ./install.sh
Приветствую все предложения и тяну запрос на github.
#!/usr/bin/env ruby
#
# For documentation please sea man git-backup(1)
#
# TODO:
# - make it a class rather than a function
# - check the standard format of git warnings to be conform
# - do better checking for git repo than calling git status
# - if multiple entries found in config file, specify which file
# - make it work with submodules
# - propose to make backup directory if it does not exists
# - depth feature in git config (eg. only keep 3 backups for a repo - like rotate...)
# - TESTING
# allow calling from other scripts
def git_backup
# constants:
git_dir_name = '.git' # just to avoid magic "strings"
filename_suffix = ".git.bundle" # will be added to the filename of the created backup
# Test if we are inside a git repo
`git status 2>&1`
if $?.exitstatus != 0
puts 'fatal: Not a git repository: .git or at least cannot get zero exit status from "git status"'
exit 2
else # git status success
until File::directory?( Dir.pwd + '/' + git_dir_name ) \
or File::directory?( Dir.pwd ) == '/'
Dir.chdir( '..' )
end
unless File::directory?( Dir.pwd + '/.git' )
raise( 'fatal: Directory still not a git repo: ' + Dir.pwd )
end
end
# git-config --get of version 1.7.10 does:
#
# if the key does not exist git config exits with 1
# if the key exists twice in the same file with 2
# if the key exists exactly once with 0
#
# if the key does not exist , an empty string is send to stdin
# if the key exists multiple times, the last value is send to stdin
# if exaclty one key is found once, it's value is send to stdin
#
# get the setting for the backup directory
# ----------------------------------------
directory = `git config --get backup.directory`
# git config adds a newline, so remove it
directory.chomp!
# check exit status of git config
case $?.exitstatus
when 1 : directory = Dir.pwd[ /(.+)\/[^\/]+/, 1]
puts 'Warning: Could not find backup.directory in your git config file. Please set it. See "man git config" for more details on git configuration files. Defaulting to the same directroy your git repo is in: ' + directory
when 2 : puts 'Warning: Multiple entries of backup.directory found in your git config file. Will use the last one: ' + directory
else unless $?.exitstatus == 0 then raise( 'fatal: unknown exit status from git-config: ' + $?.exitstatus ) end
end
# verify directory exists
unless File::directory?( directory )
raise( 'fatal: backup directory does not exists: ' + directory )
end
# The date and time prefix
# ------------------------
prefix = ''
prefix_date = Time.now.strftime( '%F' ) + ' - ' # %F = YYYY-MM-DD
prefix_time = Time.now.strftime( '%H:%M:%S' ) + ' - '
add_date_default = true
add_time_default = false
prefix += prefix_date if git_config_bool( 'backup.prefix-date', add_date_default )
prefix += prefix_time if git_config_bool( 'backup.prefix-time', add_time_default )
# default bundle name is the name of the repo
bundle_name = Dir.pwd.split('/').last
# set the name of the file to the first command line argument if given
bundle_name = ARGV[0] if( ARGV[0] )
bundle_name = File::join( directory, prefix + bundle_name + filename_suffix )
puts "Backing up to bundle #{bundle_name.inspect}"
# git bundle will print it's own error messages if it fails
`git bundle create #{bundle_name.inspect} --all --remotes`
end # def git_backup
# helper function to call git config to retrieve a boolean setting
def git_config_bool( option, default_value )
# get the setting for the prefix-time from git config
config_value = `git config --get #{option.inspect}`
# check exit status of git config
case $?.exitstatus
# when not set take default
when 1 : return default_value
when 0 : return true unless config_value =~ /(false|no|0)/i
when 2 : puts 'Warning: Multiple entries of #{option.inspect} found in your git config file. Will use the last one: ' + config_value
return true unless config_value =~ /(false|no|0)/i
else raise( 'fatal: unknown exit status from git-config: ' + $?.exitstatus )
end
end
# function needs to be called if we are not included in another script
git_backup if __FILE__ == $0
Другой официальный способ - использовать git bundle.
Это создаст файл, который поддерживает git fetch
а также git pull
для того, чтобы обновить свой второй репо.
Полезно для инкрементного резервного копирования и восстановления.
Но если вам нужно сделать резервную копию всего (потому что у вас нет второго репо с уже существующим содержимым), резервное копирование будет более сложным, как упоминалось в моем другом ответе, после комментария Kent Fredric:
$ git bundle create /tmp/foo master
$ git bundle create /tmp/foo-all --all
$ git bundle list-heads /tmp/foo
$ git bundle list-heads /tmp/foo-all
(Это атомарная операция, в отличие от создания архива из .git
папка, как прокомментировал fantabolous)
Предупреждение: я бы не рекомендовал Pat Notz, которое заключается в клонировании репо.
Резервное копирование многих файлов всегда сложнее, чем резервное копирование или обновление... только один.
Если вы посмотрите историю изменений ответа Dan Rosenstark, вы увидите, что Yar сначала использовал clone --mirror
... с правкой:
Использование этого с Dropbox - полный беспорядок.
У вас будут ошибки синхронизации, и вы НЕ МОЖЕТЕ ЗАВЕРШИТЬ СПРАВОЧНИК В DROPBOX.
использованиеgit bundle
если вы хотите сделать резервную копию в свой дропбокс.
Текущее решение Yar использует git bundle
,
Я считаю так.
Для этого я создаю удаленный (пустой) репозиторий (на отдельном диске, USB-ключе, сервере резервного копирования или даже github), а затем использую push --mirror
чтобы этот удаленный репозиторий выглядел точно так же, как мой локальный (кроме удаленного репозитория).
Это подтолкнет все ссылки (ветви и теги), включая обновления без ускоренной перемотки. Я использую это для создания резервных копий моего локального хранилища.
Страница руководства описывает это так:
Вместо того, чтобы называть каждую ссылку нажатием, указывает, что все ссылки под
$GIT_DIR/refs/
(который включает, но не ограничиваетсяrefs/heads/
,refs/remotes/
, а такжеrefs/tags/
) отражаться в удаленном хранилище. Вновь созданные локальные ссылки будут отправлены на удаленный конец, локально обновленные ссылки будут принудительно обновлены на удаленном конце, а удаленные ссылки будут удалены с удаленного конца. Это значение по умолчанию, если опция конфигурацииremote.<remote>.mirror
установлено.
Я сделал псевдоним, чтобы сделать push:
git config --add alias.bak "push --mirror github"
Тогда я просто бегу git bak
всякий раз, когда я хочу сделать резервную копию.
[Просто оставив это здесь для моей справки.]
Мой пакетный скрипт называется git-backup
выглядит так
#!/usr/bin/env ruby
if __FILE__ == $0
bundle_name = ARGV[0] if (ARGV[0])
bundle_name = `pwd`.split('/').last.chomp if bundle_name.nil?
bundle_name += ".git.bundle"
puts "Backing up to bundle #{bundle_name}"
`git bundle create /data/Dropbox/backup/git-repos/#{bundle_name} --all`
end
Иногда я использую git backup
а иногда я использую git backup different-name
что дает мне большинство возможностей, которые мне нужны.
Нашел простой официальный способ после того, как пробирался сквозь стены текста выше, который заставил бы вас думать, что его нет.
Создайте полный комплект с:
$ git bundle create <filename> --all
Восстановите его с помощью:
$ git clone <filename> <folder>
Эта операция атомарна AFAIK. Проверьте официальные документы для деталей шероховатых.
Что касается "zip": пакеты git сжаты и удивительно малы по сравнению с размером папки.git.
Оба ответа на эти вопросы верны, но мне все еще не хватает полного, короткого решения для резервного копирования Github-репозитория в локальный файл. Суть доступна здесь, не стесняйтесь вилка или адаптироваться к вашим потребностям.
backup.sh:
#!/bin/bash
# Backup the repositories indicated in the command line
# Example:
# bin/backup user1/repo1 user1/repo2
set -e
for i in $@; do
FILENAME=$(echo $i | sed 's/\//-/g')
echo "== Backing up $i to $FILENAME.bak"
git clone git@github.com:$i $FILENAME.git --mirror
cd "$FILENAME.git"
git bundle create ../$FILENAME.bak --all
cd ..
rm -rf $i.git
echo "== Repository saved as $FILENAME.bak"
done
restore.sh:
#!/bin/bash
# Restore the repository indicated in the command line
# Example:
# bin/restore filename.bak
set -e
FOLDER_NAME=$(echo $1 | sed 's/.bak//')
git clone --bare $1 $FOLDER_NAME.git
Вы можете сделать резервную копию git-репо с помощью git-copy. git-copy сохранил новый проект как голое хранилище, это означает минимальную стоимость хранения.
git copy /path/to/project /backup/project.backup
Затем вы можете восстановить свой проект с git clone
git clone /backup/project.backup project
Пришел к этому вопросу через гугл.
Вот что я сделал самым простым способом.
git checkout branch_to_clone
затем создайте новую ветку git из этой ветки
git checkout -b new_cloned_branch
Switched to branch 'new_cloned_branch'
вернитесь в исходную ветку и продолжайте:
git checkout branch_to_clone
Предполагая, что вы облажались и нужно что-то восстановить из резервной ветки:
git checkout new_cloned_branch -- <filepath> #notice the space before and after "--"
Лучшая часть, если что-то напортачило, вы можете просто удалить исходную ветку и вернуться в резервную ветку!!