Форум программистов «Весельчак У»
  *
Добро пожаловать, Гость. Пожалуйста, войдите или зарегистрируйтесь.
Вам не пришло письмо с кодом активации?

  • Рекомендуем проверить настройки временной зоны в вашем профиле (страница "Внешний вид форума", пункт "Часовой пояс:").
  • У нас больше нет рассылок. Если вам приходят письма от наших бывших рассылок mail.ru и subscribe.ru, то знайте, что это не мы рассылаем.
   Начало  
Наши сайты
Помощь Поиск Календарь Почта Войти Регистрация  
 
Страниц: [1]   Вниз
  Печать  
Автор Тема: GIT: сделать ветку и занести комит, не трогая work tree  (Прочитано 4465 раз)
0 Пользователей и 1 Гость смотрят эту тему.
RXL
Технический
Администратор

Offline Offline
Пол: Мужской

WWW
« : 02-04-2020 11:12 » 

Это памятка. Автоматизаторам в помощь.

Здесь я буду ветвить и комитить, не трогая ничего, кроме 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
Записан

... мы преодолеваем эту трудность без синтеза распределенных прототипов. (с) Жуков М.С.
Страниц: [1]   Вверх
  Печать  
 

Powered by SMF 1.1.21 | SMF © 2015, Simple Machines