git - получение ВСЕХ предыдущих версий определенного файла/папки
Я хочу получить всю предыдущую версию определенного файла в репозитории git.
Я вижу, что можно получить одну конкретную версию с командой checkout, но я хочу, чтобы все они были. И команда git clone с параметром depth не позволяет мне клонировать вложенную папку ("недействительное имя репозитория").
Вы знаете, если это возможно и как?
спасибо
Ответы
Ответ 1
OP хотел получить все версии, но ответы не доставлялись. Особенно, если файл содержит сотни исправлений (все предложения слишком ручной). Единственное полузарабочее решение было предложено @Tobias в комментариях, но предлагаемая петля bash создавала файлы в случайном порядке, а также генерировала сотни пустых файлов при использовании против наших репозиториев. Одна из причин заключалась в том, что "rev-list --all --объекты" будут перечислять разные объекты (включая деревья, но бесполезные для нашей цели).
Я начал с решения Tobias, добавил счетчики, немного почистил и в итоге изобрел колесо в виде скрипта bash, приведенного ниже.
Скрипт:
- извлекать все версии файлов в /tmp/all_versions_exported
- принять 1 аргумент - относительный путь к файлу внутри git repo
- дать результат имена файлов числовой префикс (сортируемый)
- указать проверенное имя файла в файлах результатов (рассказать яблоки отдельно от апельсинов :)
- указать дату фиксации в имени файла результата (см. пример вывода ниже)
- не создавать пустые файлы результатов
cat/usr/local/bin/git_export_all_file_versions
#!/bin/bash
# we'll write all git versions of the file to this folder:
EXPORT_TO=/tmp/all_versions_exported
# take relative path to the file to inspect
GIT_PATH_TO_FILE=$1
# ---------------- don't edit below this line --------------
USAGE="Please cd to the root of your git proj and specify path to file you with to inspect (example: $0 some/path/to/file)"
# check if got argument
if [ "${GIT_PATH_TO_FILE}" == "" ]; then
echo "error: no arguments given. ${USAGE}" >&2
exit 1
fi
# check if file exist
if [ ! -f ${GIT_PATH_TO_FILE} ]; then
echo "error: File '${GIT_PATH_TO_FILE}' does not exist. ${USAGE}" >&2
exit 1
fi
# extract just a filename from given relative path (will be used in result file names)
GIT_SHORT_FILENAME=$(basename $GIT_PATH_TO_FILE)
# create folder to store all revisions of the file
if [ ! -d ${EXPORT_TO} ]; then
echo "creating folder: ${EXPORT_TO}"
mkdir ${EXPORT_TO}
fi
## uncomment next line to clear export folder each time you run script
#rm ${EXPORT_TO}/*
# reset coutner
COUNT=0
# iterate all revisions
git rev-list --all --objects -- ${GIT_PATH_TO_FILE} | \
cut -d ' ' -f1 | \
while read h; do \
COUNT=$((COUNT + 1)); \
COUNT_PRETTY=$(printf "%04d" $COUNT); \
COMMIT_DATE='git show $h | head -3 | grep 'Date:' | awk '{print $4"-"$3"-"$6}''; \
if [ "${COMMIT_DATE}" != "" ]; then \
git cat-file -p ${h}:${GIT_PATH_TO_FILE} > ${EXPORT_TO}/${COUNT_PRETTY}.${COMMIT_DATE}.${h}.${GIT_SHORT_FILENAME};\
fi;\
done
# return success code
echo "result stored to ${EXPORT_TO}"
exit 0
Пример использования:
cd /home/myname/my-git-repo
git_export_all_file_versions docs/howto/readme.txt
result stored to /tmp/all_versions_exported
ls /tmp/all_versions_exported
0001.17-Oct-2016.ee0a1880ab815fd8f67bc4299780fc0b34f27b30.readme.txt
0002.3-Oct-2016.d305158b94bedabb758ff1bb5e1ad74ed7ccd2c3.readme.txt
0003.29-Sep-2016.7414a3de62529bfdd3cb1dd20ebc1a977793102f.readme.txt
0004.28-Sep-2016.604cc0a34ec689606f7d3b2b5bbced1eece7483d.readme.txt
0005.28-Sep-2016.198043c219c81d776c6d8a20e4f36bd6d8a57825.readme.txt
0006.9-Sep-2016.5aea5191d4b86aec416b031cb84c2b78603a8b0f.readme.txt
<and so on and on . . .>
edit: если вы видите такие ошибки:
fatal: Недействительное имя объекта 3e93eba38b31b8b81905ceaa95eb47bbaed46494: readme.txt
это означает, что вы запустили скрипт не из корневой папки вашего проекта git.
Ответ 2
Сценарий, предоставленный Дмитрием, действительно решает проблему, но у него было несколько проблем, которые привели меня к тому, чтобы адаптировать его, чтобы он был более подходящим для моих нужд. В частности:
- Использование
git show
ломалось из-за моих настроек формата даты по умолчанию. - Я хотел, чтобы результаты отсортировались по дате, а не по порядку обратной даты.
- Я хотел иметь возможность запустить его против файла, который был удален из репо.
- Мне не нужны все изменения во всех ветких; Я просто хотел, чтобы изменения были доступны для HEAD.
- Я хотел, чтобы это было ошибкой, если оно не было в git-репо.
- Мне не нужно было редактировать сценарий для настройки определенных параметров.
- То, как это работало, было неэффективным.
- Мне не нужна нумерация в именах выходных файлов. (Соответствующая форматная дата служит той же цели.)
- Мне нужны более безопасные "пути с пробелами"
Вы можете увидеть последнюю версию моих модификаций в моем реестре github или здесь версию на момент написания этой статьи:
#!/bin/sh
# based on script provided by Dmitry Shevkoplyas at http://stackoverflow.com/questions/12850030/git-getting-all-previous-version-of-a-specific-file-folder
set -e
if ! git rev-parse --show-toplevel >/dev/null 2>&1 ; then
echo "Error: you must run this from within a git working directory" >&2
exit 1
fi
if [ "$#" -lt 1 ] || [ "$#" -gt 2 ]; then
echo "Usage: $0 <relative path to file> [<output directory>]" >&2
exit 2
fi
FILE_PATH="$1"
EXPORT_TO=/tmp/all_versions_exported
if [ -n "$2" ]; then
EXPORT_TO="$2"
fi
FILE_NAME="$(basename "$FILE_PATH")"
if [ ! -d "$EXPORT_TO" ]; then
echo "Creating directory '$EXPORT_TO'"
mkdir -p "$EXPORT_TO"
fi
echo "Writing files to '$EXPORT_TO'"
git log --diff-filter=d --date-order --reverse --format="%ad %H" --date=iso-strict "$FILE_PATH" | grep -v '^commit' | \
while read LINE; do \
COMMIT_DATE='echo $LINE | cut -d ' ' -f 1'; \
COMMIT_SHA='echo $LINE | cut -d ' ' -f 2'; \
printf '.' ; \
git cat-file -p "$COMMIT_SHA:$FILE_PATH" > "$EXPORT_TO/$COMMIT_DATE.$COMMIT_SHA.$FILE_NAME" ; \
done
echo
exit 0
Пример вывода:
$ git_export_all_file_versions bin/git_export_all_file_versions /tmp/stackoverflow/demo
Creating directory '/tmp/stackoverflow/demo'
Writing files to '/tmp/stackoverflow/demo'
...
$ ls -1 /tmp/stackoverflow/demo/
2017-05-02T15:52:52-04:00.c72640ed968885c3cc86812a2e1aabfbc2bc3b2a.git_export_all_file_versions
2017-05-02T16:58:56-04:00.bbbcff388d6f75572089964e3dc8d65a3bdf7817.git_export_all_file_versions
2017-05-02T17:05:50-04:00.67cbdeab97cd62813cec58d8e16d7c386c7dae86.git_export_all_file_versions
Ответ 3
git rev-list --all --objects -- path/to/file.txt
перечисляет все капли, связанные с пути репо
Чтобы получить определенную версию файла
git cat-file -p commitid:path/to/file.txt
(commitid может быть чем угодно
- символическое ref (ветвь, имена тегов, удаленные тоже)
- хеш-фиксация
- спецификация ревизии, такая как HEAD ~ 3, branch1 @{4} и т.д.
Ответ 4
Иногда старые версии файла доступны только через git reflog
. Недавно у меня была ситуация, когда мне нужно было прорыть все коммиты, даже те, которые больше не были частью журнала из-за случайной перезаписи во время интерактивного перезагрузки.
Я написал этот скрипт Ruby для вывода всех предыдущих версий файла, чтобы найти потерянную фиксацию. Достаточно легко было вывести результат этого, чтобы отследить мой недостающий файл. Надеюсь, это поможет кому-то.
#!/usr/bin/env ruby
path_to_file = ""
'git reflog'.split("\n").each do |log|
puts commit = log.split(" ").first
puts 'git show #{commit}:#{path_to_file}'
puts
end
То же самое можно сделать с помощью git log
.
Ответ 5
Все версии файла уже находятся в git repo, когда вы клонируете его. Вы можете создавать ветки, связанные с проверкой конкретной фиксации:
git checkout -b branchname {commit#}
Это может быть достаточно для быстрого и грязного ручного сравнения изменений:
- выписка в ветки
- Скопировать в буфер редактора
Это может быть хорошо, если у вас есть только несколько версий, которые нужно учитывать, и не против немного ручных, хотя и встроенных команд git.
Для скриптовых решений уже есть несколько других решений, которые были предоставлены в других ответах.