Это памятка. Автоматизаторам в помощь.
Здесь я буду ветвить и комитить, не трогая ничего, кроме index. На входе имя родительской ветки и имя новой ветки (ее я заранее проверил, что еще нет). И да, ветки удаленные, не локальные. Локальная есть, но как временная. Конкретно моя задача: сделать ветку в модуле, сделать аналогичную ветку в родительском репозитории и занести имя ветки в .gitmodules. Изменение .gitmodules и подготовку оставлю «за кадром».
И так, есть склонированный проект в любом состоянии. Испортим только index. Да и тот можно было бы не портить, но у меня нет такой нужды.
На входе:
$mother_branch — точка ветвления
$new_branch — создаваемая ветка
$file_path — путь к файлу
$commit_message — текст комментария для комита
Прежде всего надо подтянуть данные с remote:
git fetch origin $mother_branch
Получить хеш ветки:
$commit_hash=$(git show-ref --hash $mother_branch)
Получить хеш дерева:
$tree_hash=$(git cat-file -p $commit_hash | grep '^tree ' | sed -E 's/^tree\s+(\S+).*/\1/)
Теперь будем менять index. Надо его почистить, на всякий случай:
git reset
Читаем дерево в index:
git read-tree $tree_hash
Экспортируем файл, не портя оригинальный:
$tmp_file=$(git checkout-index --temp $file_path | sed -E 's/^(\S+).*/\1/')
В этот момент модифицируем файл $tmp_path.
Создаем из файла объект с занесением в кеш объектов:
$new_file_hash=$(git hash-object -w $tmp_file)
Добавляем в index:
git update-index --add --cacheinfo 100644,$new_file_hash,$file_path
Создаем объект нового дерева:
$new_tree_hash=$(git write-tree)
Создаем комит, связанный с родительским:
$new_commit_hash=$(git commit-tree -p $commit_hash -m $commit_message $new_tree_hash)
Чистим за собой index:
git reset
Удаляем временный файл:
rm -f $tmp_file
Теперь создадим ветку:
git branch $new_branch $new_commit_hash
Отправим ее на remote:
git push origin $new_branch
Удалим локальную ветку за ненадобностью:
git branch -D $new_branch