Правила полета на Git

🌍 EnglishEspañolРусский简体中文한국어Tiếng ViệtFrançais

Что это за “правила полета” такие?

Есть инструкции для астронавтов (а здесь подобное руководство для пользователей Git) о том, как быть, если что-то пошло не так.

Полетные правила - это с большим трудом полученный запас знаний, записанный в форме инструкций, в которых указан последовательный порядок действий в разных ситуациях и сказано, почему нужно делать именно так. По сути это крайне подробный, стандартный план действий в ситуациях, развивающихся по прописанным сценариям. […]

В НАСА фиксируют все наши промахи, аварии и методы решения проблем с начала 1960 х гг., когда наземные группы проекта Mercury начали составлять сборник “выученных уроков”. Теперь в этом сборнике перечислены тысячи сложных ситуаций и их решения – от отказа двигателей, поломки ручек люка до проблем с компьютером.

— Крис Хэдфилд, Руководство астронавта по жизни на Земле (перевод Дмитрия Лазарева).

Соглашения для этого документа

Для наглядности во всех примерах в этом документе используется измененное приглашение командной строки, чтобы показать текущую ветку и есть ли подготовленные изменения. Ветка заключена в кавычки, а символ * после ветки означает подготовленные изменения.

Приведенные команды работают на Git версии 2.13.0 и выше. Для обновления Вашей версии Git посетите вебсайт Git.

Заходите в чат https://gitter.im/k88hudson/git-flight-rules

Репозитории

Я хочу создать локальный репозиторий

Чтобы создать репозиторий Git в текущей папке:

(my-folder) $ git init

Я хочу клонировать удаленный репозиторий

Чтобы клонировать (копировать) удаленный репозиторий, скопируйте адрес репозиторий и запустите:

$ git clone [url]

Git сохранит копию в папке с названием репозитория. Убедитесь, что у Вас есть соединение с удаленным сервером, откуда Вы клонируете (в большинстве случаев это означает наличие выхода в Интернет).

Чтобы клонировать в папку с произвольным именем:

$ git clone [url] name-of-new-folder

Я неправильно задал удаленный репозиторий

Вот несколько возможных проблем:

Если Вы склонировали неправильный репозиторий, то просто удалите появившуюся папку и склонируйте правильный репозиторий.

Если Вы неправильно настроили origin для существующего локального репозиторий, то измените адрес origin на правильный с помощью:

$ git remote set-url origin [правильный url]

См. подробнее на StackOverflow.

Редактирование коммитов

Что я только что сохранил?

Допустим, Вы не глядя сохранили изменения с помощью git commit -a и теперь не уверены что именно сохранили. В таком случае Вы можете просмотреть последний коммит в HEAD с помощью:

(master)$ git show

Или

$ git log -n1 -p

Если Вы хотите просмотреть файл из определенного коммита, Вы можете сделать так (<commitid> - нужный Вам коммит):

$ git show <commitid>:filename

Я неправильно написал сообщение коммита

Если Вы неправильно сохранили коммит, но еще не сделали push, то для исправления сообщения коммита сделайте следующее:

$ git commit --amend --only

Это откроет текстовый редактор по-умолчанию, в котором Вы сможете исправить сообщение. С другой стороны Вы можете сделать это одной командой:

$ git commit --amend --only -m 'xxxxxxx'

Если Вы уже сделали push, то Вы по-прежнему можете исправить коммит, но после этого придется делать push с принудительной перезаписью, что не рекомендуется.

Я сделал коммит с неправильным именем автора и адресом электронной почты

Если это один коммит, то исправьте его с помощью amend

$ git commit --amend --no-edit --author "New Authorname <authoremail@mydomain.com>"

Или Вы можете сконфигурировать глобальные настройки автора git config --global author.(name|email), а затем выполнить

$ git commit --amend --reset-author --no-edit

Если Вам нужно изменить всю историю, то смотрите документацию для git filter-branch.

Я хочу удалить файл из предыдущего коммита

Чтобы удалить изменения файла из предыдущего коммита, сделайте следующее:

$ git checkout HEAD^ myfile
$ git add myfile
$ git commit --amend --no-edit

Когда в коммит был добавлен новый файл и Вы хотите его удалить (только из Git), выполните:

$ git rm --cached myfile
$ git commit --amend --no-edit

Это особенно полезно, когда у Вас открытый патч, а Вы сохранили ненужный файл и теперь нужно сделать принудительный push для обновления патча в удаленном репозитории. Опция --no-edit оставляет прежнее сообщение коммита без изменений.

Я хочу удалить последний коммит

Если хотите удалить опубликованные коммиты, воспользуйтесь приведенным ниже приемом. Однако, это бесповоротно изменит у Вас историю Git, а также испортит историю Git у любого, что уже стянул (pull) изменения из репозитория. Короче говоря, никогда так не делайте, если не уверены.

$ git reset HEAD^ --hard
$ git push --force-with-lease [remote] [branch]

Если Вы еще не опубликовали коммит, то просто сбросьте ветку в состояние перед Вашим последним коммитом (подготовленные изменения не пропадут):

(my-branch*)$ git reset --soft HEAD@{1}

Это работает, если Вы еще не сделали push. Если Вы уже сделали push, то единственный по-настоящему безопасный способ это git revert SHAofBadCommit. Это создаст новый коммит, который отменит все изменения предыдущего коммита. Или, если ветка, в которую вы делаете push безопасна для перезаписи (т.е. не предполагается, другие разработчики будут стягивать из нее изменения), то просто используйте git push --force-with-lease. Подробнее см. в этом пункте выше.

Удалить произвольный коммит

Здесь уместны те же предупреждения, что и в пункте выше. По возможности, никогда так не делайте.

$ git rebase --onto SHA1_OF_BAD_COMMIT^ SHA1_OF_BAD_COMMIT
$ git push --force-with-lease [remote] [branch]

Или сделайте интерактивное перебазирование и удалите строки ненужных коммитов.

Я пытаюсь опубликовать исправленный коммит, но получаю сообщение об ошибке

To https://github.com/yourusername/repo.git
! [rejected]        mybranch -> mybranch (non-fast-forward)
error: failed to push some refs to 'https://github.com/tanay1337/webmaker.org.git'
hint: Updates were rejected because the tip of your current branch is behind
hint: its remote counterpart. Integrate the remote changes (e.g.
hint: 'git pull ...') before pushing again.
hint: See the 'Note about fast-forwards' in 'git push --help' for details.

Напомним, что подобно перебазированию (см. ниже), исправление коммита (amend) заменяет старый коммит новым, поэтому Вы должны делать принудительный push (--force-with-lease) Ваших изменений, если хотите заменить уже опубликованные на удаленном репозитории коммиты. Будьте осторожны, когда так делаете – всегда проверяйте с какой веткой Вы проводите эти действия!

(my-branch)$ git push origin mybranch --force-with-lease

В общем, избегайте делать принудительный push. Лучше создать и опубликовать еще один коммит, чем переписывать измененные коммиты, т.к. это приведет к конфликтам истории у других разработчиков, которые работают с этой или дочерними ветками. --force-with-lease по-прежнему выдаст ошибку, если кто-то одновременно с Вами работает с данной веткой и Ваш принудительный push переписал бы чужие изменения.

Если Вы абсолютно уверены, что никто кроме Вас не работает с данной веткой или Вы хотите обновить вершину ветви в любом случае, то используйте --force (-f), но вообще этого следует избегать.

Я случайно сделал жесткий сброс (–hard) и теперь хочу вернуть свои изменения

Если Вы случайно сделали git reset --hard, то вы можете вернуть назад коммиты, т.к. Git несколько дней хранит все действия в журнале.

Замечание: Это относится только если ваша работа была сохранена, т.е. Вы сделали коммит или stash. git reset --hard удалит несохраненные изменения, так что пользуйтесь им с осторожностью. (Безопасная опция это git reset --keep.)

(master)$ git reflog

Вы увидете список Ваших предыдущих коммитов и коммит для сброса. Выберите SHA коммита, который хотите вернуть и снова сделайте сброс:

(master)$ git reset --hard SHA1234

И Вы сможете продолжить работу.

Я случайно опубликовал ненужное слияние

Если Вы случайно сделали слияние в основную ветку разработки до того, как были сделаны все необходимые изменения, Вы по-прежнему можете отменить слияние. Но есть одна загвоздка: Коммит слияния имеет более одного родителя (обычно два).

Команда для решения этой проблемы

(feature-branch)$ git revert -m 1 <commit>

где опция -m 1 говорит выбрать родителя номер 1 (ветка, в которую было сделано слияние) в качестве родителя для отката.

Заметка: номер родителя - это не идентификатор коммита. Допустим, коммит слияния имеет строчку Merge: 8e2ce2d 86ac2e7. Номер родителя - это порядковый номер (отсчет с 1) нужного родителя в этой строчке, первый идентификатор - номер 1, второй - номер 2 и т.д.

Я случайно закоммитил и опубликовал файлы с конфиденциальными данными

Если Вы случайно опубликовали файлы, содержащие конфиденциальные данные (пароли, ключи и пр.), Вы можете изменить последний коммит с помощью amend. Помните, что как только Вы опубликовали коммит, то Вы должны считать всё его содержание скомпромитированны. С помощью дальнейших шагов можно удалить конфиденциальную информацию из публичного репозиторий или из Вашей локальной копии, вы не сможете удалить свою конфиденциальную информацию у людей, которые могли успеть скопировать себе её. Если Вы закоммитили пароль, то незамедлительно поменяйте его. Если Вы закоммитили ключ, то незамедлительно сгенерируйте новый. Изменения опубликованного коммита недостаточно, потому что кто-нибудь мог скопировать исходный коммит до того, как Вы успели его исправить.

Как только удалите конфиденциальные данные из файла, запустите

(feature-branch)$ git add edited_file
(feature-branch)$ git commit --amend --no-edit
(feature-branch)$ git push --force-with-lease origin [branch]

Если Вы хотите удалить файл целиком, но оставить его в локальной копии, то запустите

(feature-branch)$ git rm --cached sensitive_file
echo sensitive_file >> .gitignore
(feature-branch)$ git add .gitignore
(feature-branch)$ git commit --amend --no-edit
(feature-branch)$ git push --force-with-lease origin [branch]

Другим способом хранения конфиденциальных данных является хранение в переменных окружения.

Если Вы хотите удалить файл целиком и не оставлять его в локальной копии, то запустите

(feature-branch)$ git rm sensitive_file
(feature-branch)$ git commit --amend --no-edit
(feature-branch)$ git push --force-with-lease origin [branch]

Если Вы успели сделать другие коммиты после коммита с конфиденциальными данными, то Вам нужно использовать rebase.

Мне нужно изменить содержимое коммита, который не является последним

Пусть Вы сделали несколько (например, три) коммитов и после этого поняли, что упустили что-то, что по смыслу относится к первому из этих трёх коммитов. Вы могли бы сделать новый коммит, содержащий эти изменения, и тогда у Вас была бы чистая кодовая база, но Ваши коммиты не были бы атомарными (т.е. связанные изменения не содержались бы в едином коммите). В такой ситуации Вы бы хотели изменить коммит, к которому относятся эти изменения, а следующие коммиты оставить как есть. В таком случае Вас может спасти git rebase.

Рассмотрим ситуацию, когда Вы хотите изменить третий с конца коммит.

(your-branch)$ git rebase -i HEAD~4

запустит интерактивный режим rebase, что позволит Вам отредактировать любые из Ваших последних трёх коммитов. Появится текстовый редактор с подобным содержанием:

pick 9e1d264 The third last commit
pick 4b6e19a The second to last commit
pick f4037ec The last commit

которое Вам нужно изменить так:

edit 9e1d264 The third last commit
pick 4b6e19a The second to last commit
pick f4037ec The last commit

Это говорит rebase, что Вы хотите изменить третий с конца коммит, а другие два оставить как есть. Теперь сохраните и закройте редактор. После этого Git начнет процедуру rebase. Он остановится на коммите, который Вы пометили для изменений, давая Вам возможность изменить этот коммит. Теперь Вы можете доделать то, что забыли первоначально. Отредактируйте и подготовьте изменения. После этого запустите

(your-branch)$ git commit --amend

это заставит Git пересоздать коммит, но оставить сообщение коммита прежним. После этого тяжелая часть работы завершена.

(your-branch)$ git rebase --continue

сделает оставшуюся работу за Вас.

Подготовка изменений (staging)

Мне нужно добавить подготовленные изменения в предыдущий коммит

(my-branch*)$ git commit --amend

Если Вы не хотите менять сообщение коммита, скажите git использовать прежнее сообщение:

(my-branch*)$ git commit --amend -C HEAD

Я хочу подготовить только часть файла, а не весь файл целиком

Обычно, если хотите подготовить часть файл, Вы запускаете:

$ git add --patch filename.x

-p - сокращение для --patch. Это откроет интерактивный режим. Вы сможете разбить коммит с помощью опции s, однако, если файл новый, то у Вас не будет такой возможности. При добавлении нового файла делайте так:

$ git add -N filename.x

Затем используйте опцию e для ручного выбора строк. Запустив git diff --cached или git diff --staged, Вы увидите какие строки вы подготовили по-сравнению с тем, что сохранено в рабочей копии.

Я хочу добавить изменения одного файла в два разных коммита

git add добавляет в коммит весь файл целиком. git add -p позволяет интерактивно выбрать изменения, которые Вы хотите добавить.

Я подготовил слишком много правок и теперь хочу разделить их на несколько отдельных коммитов

git reset -p откроет интерактивный диалог сброса. Это похоже на git add -p, за исключением того, что выбор “yes” уберёт правку из готовящегося коммита.

Я хочу подготовить свои неподготовленные правки и убрать из подготовки то, что уже подготовлено

Это сложно. Лучшее, что я смог придумать это отложить (stash) неподготовленные изменения. Затем сделать сброс. После этого вернуть отложенные изменения и добавить их.

$ git stash -k
$ git reset --hard
$ git stash pop
$ git add -A

Неподготовленные правки

Я хочу переместить мои неподготовленные правки в новую ветку

$ git checkout -b my-branch

Я хочу переместить неподготовленные правки в другую существующую ветку

$ git stash
$ git checkout my-branch
$ git stash pop

Я хочу отменить мои локальные несохраненные изменения (подготовленные и неподготовленные)

Если Вы хотите отменить все подготовленные и неподготовленные изменения, то можете сделать так:

(my-branch)$ git reset --hard
# or
(master)$ git checkout -f

Это уберет из индекса все подготовленные изменения:

$ git reset

Это удалит все локальные изменения, которые не были сохранены или добавлены в индекс (нужно запускать из корня репозитория):

$ git checkout .

Вы также можете отменить несохраненные изменения в определенном файле или папке:

$ git checkout [some_dir|file.txt]

Еще один способ отменить все несохраненные изменения (длиннее, зато работает в любой папке):

$ git reset --hard HEAD

Это удалит все локальные неотслеживаемые файлы, так что останутся только отслеживаемые:

$ git clean -fd

-x удалит также и игнорируемые файлы.

Я хочу отменить некоторые неподготовленные изменения

Когда Вы хотите избавиться от некоторых, но не всех изменений в Вашей рабочей копии.

Сделайте checkout ненужных изменений, оставив нужные.

$ git checkout -p
# Отвечайте `y` для всех фрагментов, которые Вы хотите выбросить

Другим подходом является использование stash. Отложите все хорошие изменения, сбросьте рабочую копию и верните отложенные хорошие изменения.

$ git stash -p
# Выберите фрагменты, которые Вы хотите сохранить
$ git reset --hard
$ git stash pop

В качестве альтернативы, отложите ненужные изменения, а затем выбросьте их.

$ git stash -p
# Выберите фрагменты, которые Вы не хотите сохранять
$ git stash drop

Я хочу отбросить неподготовленные изменения в некоторых файлах

Когда Вы хотите убрать изменения какого-то файла в Вашей рабочей копии.

$ git checkout myFile

Чтобы убрать изменения в нескольких файлах, перечислите их имена.

$ git checkout myFirstFile mySecondFile

Я хочу убрать все неподготовленные локальные изменения

Когда Вы хотите убрать все неподготовленные локальные изменения

$ git checkout .

Я хочу удалить все неотслеживаемые файлы

Когда Вы хотите удалить все неотслеживаемые файлы

$ git clean -f

Я хочу убрать заданный файл из подготовленного

Иногда один или несколько неотслеживаемых файлов ненароком оказываются в подготовленном. Чтобы убрать их из подготовленного:

$ git reset -- <filename>

Это уберёт файл из подготовленного и он снова будет выглядеть как неотслеживаемый.

Ветки

Я хочу получить список всех веток

Список локальных веток

$ git branch

Список удаленных веток

$ git branch -r

Список всех веток (локальных и удаленных)

$ git branch -a

Создать ветку на определенном коммите

$ git checkout -b <branch> <SHA1_OF_COMMIT>

Я стянул изменения (pull) из неправильной ветки или в неправильную ветку

Это очередная возможность воспользоваться git reflog, чтобы посмотреть куда указывала ваша HEAD перед неправильным pull.

(master)$ git reflog
ab7555f HEAD@{0}: pull origin wrong-branch: Fast-forward
c5bc55a HEAD@{1}: checkout: checkout message goes here

Просто сбросьте ветку обратно на требуемый коммит:

$ git reset --hard c5bc55a

Готово.

Я хочу отменить локальные коммиты, чтобы моя ветка стала такой же как на сервере

Подтвердите, что не хотите отправлять изменения на сервер.

git status покажет на сколько коммитов Вы опережаете источник:

(my-branch)$ git status
# On branch my-branch
# Your branch is ahead of 'origin/my-branch' by 2 commits.
#   (use "git push" to publish your local commits)
#

Один из способов сбросить до источника (чтобы иметь то же, что и в удаленном репозитории):

(master)$ git reset --hard origin/my-branch

Я сохранил коммит в ветку master вместо новой ветки

Создайте новую ветку, оставаясь на master:

(master)$ git branch my-branch

Сбросьте ветку master к предыдущему коммиту:

(master)$ git reset --hard HEAD^

HEAD^ - это сокращение для HEAD^1. Это подставляет первого предка HEAD, подобно этому HEAD^2 подставляет второго предка коммита (слияния могут иметь 2 предков).

Заметьте, что HEAD^2 это не то же самое, что HEAD~2 (для подробностей см. эту ссылку).

Если не хотите использовать HEAD^, то найдите хэш коммита, на который Вы хотите установить ветку (git log может помочь в этом). Затем сделайте сброс к этому хэшу. С помощью git push удостоверьтесь, что эти изменения отражены в удаленном репозитории.

К примеру, ветка master обязана находится на коммите с хэшем a13b85e:

(master)$ git reset --hard a13b85e
HEAD is now at a13b85e

Перейти на новую ветку для продолжения работы:

(master)$ git checkout my-branch

Я хочу сохранить файл целиком из другого ref-ish

Скажем, у Вас рабочий spike (см. заметку) на сотни изменений. Всё работает. Теперь Вы сохраняете эту работу в другую ветку:

(solution)$ git add -A && git commit -m "Добавлены все изменения из этого рывка в один большой коммит."

Когда Вы хотите поместить его в ветку (например feature или develop), Вы хотите сохранять по целым файлам. А также Вы хотите разбить большой коммит на несколько небольших.

Скажем, Вы имеете:

Вы можете выполнить это, перенеся содержимое файла в Вашу ветку:

(develop)$ git checkout solution -- file1.txt

Это скопирует содержимое данного файла из ветки solution в ветку develop:

# On branch develop
# Your branch is up-to-date with 'origin/develop'.
# Changes to be committed:
#  (use "git reset HEAD <file>..." to unstage)
#
#        modified:   file1.txt

Теперь сделайте коммит как обычно.

Заметка: Spike-решения делаются для анализа или решения проблемы. Эти решения используют, чтобы оценить проблему, и отбрасывают сразу же, как только все получают ясное представление о проблеме. ~ Wikipedia.

Я сделал несколько коммитов в одной ветке, а нужно было сохранять их в разных ветках

Скажем, Вы в ветке master. Запустив git log, Вы увидите, что сделали два коммита:

(master)$ git log

commit e3851e817c451cc36f2e6f3049db528415e3c114
Author: Alex Lee <alexlee@example.com>
Date:   Tue Jul 22 15:39:27 2014 -0400

    Bug #21 - Added CSRF protection

commit 5ea51731d150f7ddc4a365437931cd8be3bf3131
Author: Alex Lee <alexlee@example.com>
Date:   Tue Jul 22 15:39:12 2014 -0400

    Bug #14 - Fixed spacing on title

commit a13b85e984171c6e2a1729bb061994525f626d14
Author: Aki Rose <akirose@example.com>
Date:   Tue Jul 21 01:12:48 2014 -0400

    First commit

Обратим внимание на ссылки на каждый баг (e3851e8 для #21, 5ea5173 для #14).

Во-первых, сбросим ветку master на правильный коммит (a13b85e):

(master)$ git reset --hard a13b85e
HEAD is now at a13b85e

Теперь, мы может создать новую ветку для бага #21:

(master)$ git checkout -b 21
(21)$

Теперь сделаем cherry-pick коммита для бага #21 на верх нашей ветки. Это значит, что мы помещаем этот и только этот коммит напрямую на вершину ветки, какой бы она ни была.

(21)$ git cherry-pick e3851e8

На этом этапе есть вероятность конфликтов. О том как разрешить конфликты см. в главе Здесь были конфликты в разделе интерактивное перебазирование выше.

Теперь давайте создадим новую ветку для бага #14, которая также основана на master

(21)$ git checkout master
(master)$ git checkout -b 14
(14)$

И наконец, сделаем cherry-pick коммита для бага #14:

(14)$ git cherry-pick 5ea5173

Я хочу удалить локальные ветки, которые были удалены в upstream

Как только Вы слили пулл-реквест на GitHub, Вам предлагают удалить слитую ветку из Вашего форка. Если Вы не планируете продолжать работу в этой ветке, то для поддержания рабочей копии в чистоте Вы можете удалить локальные копии ненужных веток, чтобы не путаться в них.

$ git fetch -p upstream

где upstream - удаленная ветка, из которой Вы хотите получить изменения.

Я нечаянно удалил мою ветку

Если Вы регулярно отправляете изменения в удаленное хранилище, большую часть времени Вы в безопасности. Но в один прекрасный момент Вы всё же можете случайно удалить Ваши ветки. Скажем, мы создали ветку и создали новый файл:

(master)$ git checkout -b my-branch
(my-branch)$ git branch
(my-branch)$ touch foo.txt
(my-branch)$ ls
README.md foo.txt

Давайте добавим его и сохраним.

(my-branch)$ git add .
(my-branch)$ git commit -m 'foo.txt added'
(my-branch)$ foo.txt added
 1 files changed, 1 insertions(+)
 create mode 100644 foo.txt
(my-branch)$ git log

commit 4e3cd85a670ced7cc17a2b5d8d3d809ac88d5012
Author: siemiatj <siemiatj@example.com>
Date:   Wed Jul 30 00:34:10 2014 +0200

    foo.txt added

commit 69204cdf0acbab201619d95ad8295928e7f411d5
Author: Kate Hudson <katehudson@example.com>
Date:   Tue Jul 29 13:14:46 2014 -0400

    Fixes #6: Force pushing after amending commits

Теперь мы переключаемся обратно на master и ‘нечаянно’ удаляем нашу ветку.

(my-branch)$ git checkout master
Switched to branch 'master'
Your branch is up-to-date with 'origin/master'.
(master)$ git branch -D my-branch
Deleted branch my-branch (was 4e3cd85).
(master)$ echo О, нет, моя ветка удалена!
О, нет, моя ветка удалена!

На этом этапе Вы должны быть знакомы с ‘reflog’ (расширенный журнал). Он хранит историю всех действий в репозитории.

(master)$ git reflog
69204cd HEAD@{0}: checkout: moving from my-branch to master
4e3cd85 HEAD@{1}: commit: foo.txt added
69204cd HEAD@{2}: checkout: moving from master to my-branch

Как мы можем видеть, у нас есть хэш коммита из удаленной ветки. Посмотрим, можем ли мы восстановить удаленную ветку.

(master)$ git checkout -b my-branch-help
Switched to a new branch 'my-branch-help'
(my-branch-help)$ git reset --hard 4e3cd85
HEAD is now at 4e3cd85 foo.txt added
(my-branch-help)$ ls
README.md foo.txt

Вуаля! Мы вернули наш удаленный файл обратно. git reflog также бывает полезен, когда перебазирование срабатывает не так, как Вы хотели.

Я хочу удалить ветку

Чтобы удалить ветку на удаленном репозитории:

(master)$ git push origin --delete my-branch

Вы также можете сделать:

(master)$ git push origin :my-branch

Чтобы удалить локальную ветку:

(master)$ git branch -d my-branch

Чтобы удалить локальную ветку, которая не была слита с отслеживаемой веткой (заданной с помощью --track или --set-upstream) или с HEAD:

(master)$ git branch -D my-branch

Я хочу удалить несколько веток

Скажем, Вы хотите удалить все ветки, начинающиеся с fix/:

(master)$ git branch | grep 'fix/' | xargs git branch -d

Я хочу переименовать ветку

Чтобы переименовать текущую (локальную) ветку:

(master)$ git branch -m new-name

Чтобы переименовать другую (локальную) ветку:

(master)$ git branch -m old-name new-name

Я хочу перейти на удаленную ветку, над которой работает кто-то еще

Во-первых, получим все ветки из удаленного репозитория:

(master)$ git fetch --all

Скажем, Вы хотите перейти на daves из удаленного репозитория.

(master)$ git checkout --track origin/daves
Branch daves set up to track remote branch daves from origin.
Switched to a new branch 'daves'

(--track - это сокращение для git checkout -b [branch] [remotename]/[branch])

Это создаст Вам локальную копию ветки daves и после push обновления также появятся в удаленном репозитории.

Я хочу создать новую удаленную ветку из текущей локальной

$ git push <remote> HEAD

Если Вы хотите, чтобы текущая локальная ветка отслеживала соответствующую удаленную (upstream) ветку, тогда выполните следующее:

$ git push -u <remote> HEAD

В режиме upstream или в режиме simple (по-умолчанию в Git 2.0) параметра push.default, следующая команда отправит текущую ветку в удаленную ветку, которая была ранее зарегистрирована с помощью -u :

$ git push

Поведение других режимов git push описано в документации на push.default.

Я хочу настроить локальную ветку на отслеживание удаленной (upstream) ветки

Вы можете настроить текущую локальную ветку на отслеживание удаленной (upstream) ветки используя:

$ git branch --set-upstream-to [remotename]/[branch]
# или для краткости:
$ git branch -u [remotename]/[branch]

Для настройки отслеживаемой удаленной ветки на другую локальную ветку:

$ git branch -u [remotename]/[branch] [local-branch]

Я хочу настроить HEAD на отслеживание основной удаленной ветки

При просмотре удаленных веток можно увидеть какую удаленную ветку отслеживает HEAD. Может оказаться, что это не та ветка что нужно.

$ git branch -r
  origin/HEAD -> origin/gh-pages
  origin/master

Чтобы origin/HEAD отслеживала origin/master, выполните команду:

$ git remote set-head origin --auto
origin/HEAD set to master

Я сделал изменения в неправильной ветке

Вы сделали несохраненные изменения, а потом поняли, что находитесь не в той ветке. Отложите эти изменения, а затем примените их к нужной ветке:

(wrong_branch)$ git stash
(wrong_branch)$ git checkout <correct_branch>
(correct_branch)$ git stash apply

Перебазирование (rebase) и слияние (merge)

Я хочу отменить перебазирование/слияние

Вы можете слить/перебазировать Вашу ветку с неправильной веткой. А также бывают случаи, когда Вы не можете предугадать успешно ли завершится процесс перебазирования/слияния. Git сохраняет исходный указатель HEAD в переменную ORIG_HEAD перед тем как приступить к опасным операциям, так что вернуть ветку на состояние до перебазирования/слияния просто.

(my-branch)$ git reset --hard ORIG_HEAD

Я сделал перебазирование, но я не хочу делать принудительный push

К сожалению, вы должны сделать принудительный push, если хотите, чтобы изменения были отражены на удаленной ветке. Это потому что у вас изменена история. Удаленная ветка не примет изменения, если не сделать принудительный push. Это одна из основных причин, по которым большинство людей основывает свой рабочий процесс на слиянии вместо перебазирования - большие команды могут столкнуться с проблемами, если разработчики будут делать принудительный push. Используйте это с осторожностью. Безопасный способ использовать перебазирование - это не отражать Ваши изменения напрямую на удаленную ветку, а вместо этого делать следующее:

(master)$ git checkout my-branch
(my-branch)$ git rebase -i master
(my-branch)$ git checkout master
(master)$ git merge --ff-only my-branch

Чтобы узнать больше, см. эту SO ветку.

Я хочу объединить коммиты

Предположим, что Вы работаете в ветке, которая стала или станет пулл-реквестом в master. В простейшем случае когда всё, что Вы хотите сделать - это объединить все коммиты в один единственный коммит и Вам не важны временные метки, Вы можете сделать сброс и заново сделать коммит. Убедитесь, что ветка master обновлена и Ваши изменения сохранены, затем:

(my-branch)$ git reset --soft master
(my-branch)$ git commit -am "New awesome feature"

Если Вы хотите больше контроля, а также сохранить метки времени, Вам нужно сделать кое-что, называемое интерактивным перебазированием:

(my-branch)$ git rebase -i master

Если Вы не работаете с другой веткой, можете делать перебазирование относительно Вашей HEAD. Например, если Вы хотите объединить последние два коммита, Вам нужно делать перебазирование относительно HEAD~2. Для последних 3 - HEAD~3 и т.д.

(master)$ git rebase -i HEAD~2

После того, как запустите интерактивное перебазирование, Вы увидите нечто подобное в Вашем текстовом редакторе:

pick a9c8a1d Some refactoring
pick 01b2fd8 New awesome feature
pick b729ad5 fixup
pick e3851e8 another fix

# Rebase 8074d12..b729ad5 onto 8074d12
#
# Commands:
#  p, pick = use commit
#  r, reword = use commit, but edit the commit message
#  e, edit = use commit, but stop for amending
#  s, squash = use commit, but meld into previous commit
#  f, fixup = like "squash", but discard this commit's log message
#  x, exec = run command (the rest of the line) using shell
#
# These lines can be re-ordered; they are executed from top to bottom.
#
# If you remove a line here THAT COMMIT WILL BE LOST.
#
# However, if you remove everything, the rebase will be aborted.
#
# Note that empty commits are commented out

Все строки, начинающиеся с # являются комментариями и не оказывают влияния на перебазирование.

Теперь замените команды pick на команды, перечисленные ниже. Также Вы можете удалить коммиты, удалив соответствующие строки.

Например, если Вы хотите оставить только старейший коммит и объединить все последующие коммиты во второй коммит, Вам нужно рядом с каждым коммитом кроме первого и второго вместо pick написать f:

pick a9c8a1d Some refactoring
pick 01b2fd8 New awesome feature
f b729ad5 fixup
f e3851e8 another fix

Если Вы хотите объединить эти коммиты и переименовать коммит, Вам нужно дополнительно добавить r рядом со вторым коммитом или просто используйте s вместо f:

pick a9c8a1d Some refactoring
pick 01b2fd8 New awesome feature
s b729ad5 fixup
s e3851e8 another fix

Теперь Вы можете переименовать коммит в следующем запросе, который появится.

Newer, awesomer features

# Please enter the commit message for your changes. Lines starting
# with '#' will be ignored, and an empty message aborts the commit.
# rebase in progress; onto 8074d12
# You are currently editing a commit while rebasing branch 'master' on '8074d12'.
#
# Changes to be committed:
#   modified:   README.md
#

Если всё успешно, Вы увидите что-то вроде:

(master)$ Successfully rebased and updated refs/heads/master.

Безопасная стратегия слияния

--no-commit производит слияние, но не делает коммит результата, давая пользователю возможность проверить и доработать результат слияния перед коммитом. no-ff сохраняет доказательства того, что когда-то существовала ветка feature, сохраняя цельность истории проекта.

(master)$ git merge --no-ff --no-commit my-branch

Мне нужно слить ветку в единственный коммит

(master)$ git merge --squash my-branch

Я хочу объединить только неопубликованные коммиты

Иногда у Вас бывает несколько временных коммитов, которые Вы хотите объединить перед публикацией в upstream. Вы не хотите ненароком объединить свои коммиты с уже опубликованными, потому что кто-то уже мог работать с ними.

(master)$ git rebase -i @{u}

Это выполнит интерактивное перебазирование со списком еще не опубликованных коммитов и Вы сможете безопасно упорядочить/исправить/объединить коммиты из списка.

Мне нужно прервать слияние

Иногда слияние может создавать проблемы в некоторых файлах. В таких случаях мы можем воспользоваться опцией abort для прерывания текущего процесса разрешения конфликтов и попробовать вернуться к состоянию перед слиянием.

(my-branch)$ git merge --abort

Эта команда доступна начиная с версии Git >= 1.7.4

Мне нужно обновить родительский коммит моей ветки

Скажем, у меня есть ветка master, ветка feature-1, выросшая из master, и ветка feature-2, выросшая из feature-1. Если я сделаю коммит в feature-1, тогда родительский коммит ветки feature-2 будет не тем, что нужно (это должна быть верхушка feature-1, поскольку мы ответвились от неё). Мы можем исправить это с помощью git rebase --onto.

(feature-2)$ git rebase --onto feature-1 <первый коммит из feature-2, который Вы не хотите переносить> feature-2

Это помогает в тех случаях, когда у Вас одна функция основывается на другой функции, которая еще не слита, и все исправления feature-1 должны быть отражены в feature-2.

Проверить, что все коммиты ветви были слиты

Для проверки того, что все коммиты ветки слиты в другую ветку, Вам нужно сравнить вершины (или любые коммиты) этих ветвей:

(master)$ git log --graph --left-right --cherry-pick --oneline HEAD...feature/120-on-scroll

Это расскажет Вам обо всех коммитах, которые есть в одной, но отсутствуют в другой ветке и выдаст список всех различий ветвей. Или можете сделать так:

(master)$ git log master ^feature/120-on-scroll --no-merges

Возможные проблемы интерактивного перебазирования

Экран перебазирования говорит ‘noop’

Если Вы видите это:

noop

Это значит, что Вы пытаетесь сделать перебазирование в ветку, которая уже имеет идентичный коммит или она впереди текущей ветки. Вы может попробовать:

Здесь были конфликты

Если Вам не удается успешно завершить перебазирование, то, возможно, Вам придется разрешать конфликты.

Для начала запустите git status, чтобы увидеть какие файлы содержат конфликты:

(my-branch)$ git status
On branch my-branch
Changes not staged for commit:
  (use "git add <file>..." to update what will be committed)
  (use "git checkout -- <file>..." to discard changes in working directory)

  both modified:   README.md

В данном примере README.md содержит конфликты. Откройте файл и взгляните на следующее:

   <<<<<<< HEAD
   some code
   =========
   some code
   >>>>>>> new-commit

Вам нужно выбрать между кодом, который был добавлен в Вашем новом коммите (в данном примере, это все от средней линии до new-commit) и Вашей HEAD.

Если Вы хотите сохранить версию из какой-то одной ветки, то используйте --ours или --theirs:

(master*)$ git checkout --ours README.md

Если слияние более сложное, можете воспользоваться визуальным редактором различий:

(master*)$ git mergetool -t opendiff

После разрешения всех конфликтов и тестирования кода подготовьте файлы, которые Вы изменили git add, а затем продолжите перебазирование с помощью git rebase --continue

(my-branch)$ git add README.md
(my-branch)$ git rebase --continue

Если после разрешения всех конфликтов, Вы получили точно такое же дерево, какое было перед коммитом, то вместо этого Вам нужно сделать git rebase --skip.

В любой момент Вы можете остановить перебазирование и вернуть ветку в начальное состояние, выполнив:

(my-branch)$ git rebase --abort

Отложенные изменения (stash)

Отложить все правки

Чтобы отложить все правки в рабочем каталоге

$ git stash

Если Вы хотите отложить заодно и неотслеживаемые файлы, добавьте опцию -u.

$ git stash -u

Отложить заданные файлы

Отложить только один файл из рабочей папки

$ git stash push working-directory-path/filename.ext

Отложить несколько файлов из рабочей папки

$ git stash push working-directory-path/filename1.ext working-directory-path/filename2.ext

Отложить с сообщением

$ git stash save <message>

Применить заданный stash из списка

Во-первых, проверьте список отложенных изменений, используя

$ git stash list

Затем примените заданный stash из списка, используя

$ git stash apply "stash@{n}"

Здесь ‘n’ показывает позицию stash-а в стеке. Верхний stash имеет позицию 0.

Поиск

Я хочу найти строку в коммитах

Чтобы найти коммиты с заданной строкой, используйте следующее:

$ git log -S "string to find"

Общие параметры:

Я хочу искать по автору или сохранившему изменения (committer)

Найти все коммиты по автору или сохранившему изменения:

$ git log --author=<name or email>
$ git log --committer=<name or email>

Не забывайте, что автор и сохранивший изменения - это не всегда один и тот же человек. --author - это тот, кто написал код, а --committer - тот, кто сохранил код, написанный автором.

Я хочу получить список коммитов, содержащих заданный файл

Чтобы найти все коммиты, содержащие заданный файл, Вы можете использовать:

$ git log -- <path to file>

Обычно Вы задаете точный путь, но можете использовать подстановочные знаки:

$ git log -- **/*.js

При использовании подстановочных знаков используйте --name-status для просмотра списка файлов, сохраненных в каждом коммите:

$ git log --name-status -- **/*.js

Я хочу посмотреть историю коммитов для отдельной функции

Для отслеживания эволюции отдельной функции Вы можете использовать:

$ git log -L :FunctionName:FilePath

Заметьте, что Вы можете добавить и другие опции команды git log, вроде диапазона ревизий и выборки коммитов.

Найти метки для заданного коммита

Чтобы найти все метки для заданного коммита:

$ git tag --contains <commitid>

Субмодули

Клонировать все субмодули

$ git clone --recursive git://github.com/foo/bar.git

Если уже клонированы:

$ git submodule update --init --recursive

Удалить субмодуль

Создание субмодуля довольно прямолинейно, чего не скажешь об удалении. Команды, которые Вам нужны:

$ git submodule deinit submodulename
$ git rm submodulename
$ git rm --cached submodulename
$ rm -rf .git/modules/submodulename

Разное

Восстановить удаленный файл

Сначала нужно найти последний коммит, где файл еще существует:

$ git rev-list -n 1 HEAD -- filename

Затем взять версию файла из коммита непосредственно перед удалением:

git checkout deletingcommitid^ -- filename

Удалить метку

$ git tag -d <tag_name>
$ git push <remote> :refs/tags/<tag_name>

Восстановить удаленную метку

Если хотите восстановить метку, которая была удалена, Вы можете сделать следующее: во-первых, Вам нужно найти недостижимую метку:

$ git fsck --unreachable | grep tag

Запомните для себя хэш метки. Затем восстановите удаленную метку, используя команду git update-ref:

$ git update-ref refs/tags/<tag_name> <hash>

Ваша метка была восстановлена.

Удаленный патч

Если кто-то прислал Вам пулл-реквест на GitHub, но потом удалил свой форк, то вы не сможете клонировать его репозиторий или использовать git am, поскольку .diff, .patch URL’ы становятся недоступными. Но Вы можете сделать checkout самого пулл-реквеста используя специальные GitHub’s refs. Для получения содержимого PR#1 в новую ветку с названием pr_1:

$ git fetch origin refs/pull/1/head:pr_1
From github.com:foo/bar
 * [new ref]         refs/pull/1/head -> pr_1

Экспорт репозитория в Zip-файл

$ git archive --format zip --output /full/path/to/zipfile.zip master

Как опубликовать ветку и метку, если они имеют одинаковые имена

Если в удаленном репозитории есть метка с таким же именем, как и ветка , то после выполнения команды `$ git push ` Вы получите следующую ошибку:

$ git push origin <branch>
error: dst refspec same matches more than one.
error: failed to push some refs to '<git server>'

Исправить это можно, указав Вы хотите опубликовать именно ветку.

$ git push origin refs/heads/<branch-name>

Если Вы хотите опубликовать метку с таким же названием, как у ветки, используйте сходную команду.

$ git push origin refs/tags/<tag-name>

Отслеживание файлов

Я хочу изменить регистр в имени файла, не меняя содержимое файла

(master)$ git mv --force myfile MyFile

Я хочу переписать локальные файлы при выполнении git pull

(master)$ git fetch --all
(master)$ git reset --hard origin/master

Я хочу удалить файл из git, но оставить сам файл

(master)$ git rm --cached log.txt

Я хочу откатить файл до заданной ревизии

Полагая, что хэш желаемого коммита c5f567:

(master)$ git checkout c5f567 -- file1/to/restore file2/to/restore

Если вы хотите откатить файл к состоянию на 1 коммит раньше, чем c5f567, задайте хэш коммита как c5f567~1:

(master)$ git checkout c5f567~1 -- file1/to/restore file2/to/restore

Я хочу получить список изменений в заданном файле из разных коммитов или веток

Полагая, что Вы хотите сравнить последний коммит с файлом из коммита c5f567:

$ git diff HEAD:path_to_file/file c5f567:path_to_file/file

Аналогично для веток:

$ git diff master:path_to_file/file staging:path_to_file/file

Я хочу, чтобы Git игнорировал изменения в определенном файле

Это бывает нужно для шаблонов конфигураций или других файлов, предназначенных для добавления учетных записей, которые не нужно коммитить.

$ git update-index --assume-unchanged file-to-ignore

Обратите внимание, что это не удаляет файл из контроля версий - он всего лишь игнорируется локально. Чтобы отменить это и сказать Git снова отслеживать изменения, выполните обратную команду:

$ git update-index --no-assume-unchanged file-to-stop-ignoring

Конфигурация

Я хочу добавить псевдонимы для некоторых команд Git

В OS X и Linux Ваш файл конфигурации Git хранится в ~/.gitconfig. В качестве примера я добавил некоторые псевдонимы, которые сам использую для краткости (а также некоторые из моих типичных опечаток), в раздел [alias] как показано ниже:

[alias]
    a = add
    amend = commit --amend
    c = commit
    ca = commit --amend
    ci = commit -a
    co = checkout
    d = diff
    dc = diff --changed
    ds = diff --staged
    extend = commit --amend -C HEAD
    f = fetch
    loll = log --graph --decorate --pretty=oneline --abbrev-commit
    m = merge
    one = log --pretty=oneline
    outstanding = rebase -i @{u}
    reword = commit --amend --only
    s = status
    unpushed = log @{u}
    wc = whatchanged
    wip = rebase -i @{u}
    zap = fetch -p
    day = log --reverse --no-merges --branches=* --date=local --since=midnight --author=\"$(git config --get user.name)\"
    delete-merged-branches = "!f() { git checkout --quiet master && git branch --merged | grep --invert-match '\\*' | xargs -n 1 git branch --delete; git checkout --quiet @{-1}; }; f"

Я хочу добавить в свой репозиторий пустую папку

Вы не можете этого сделать! Git просто не поддерживает этого, но есть уловка. Вы можете создать файл .gitignore в папке со следующим содержанием:

 # Игнорировать всё в этой папке
 *
 # Кроме этого файла
 !.gitignore

Другой общеиспользуемый способ - это создать в папке пустой файл с названием .gitkeep.

$ mkdir mydir
$ touch mydir/.gitkeep

Вы можете назвать его просто .keep , в этом случае вторая строка выше будет touch mydir/.keep

Я хочу сохранить имя пользователя и пароль для репозитория

У Вас может быть репозиторий, требующий авторизации. В этом случае вы можете сохранить на время имя пользователя и пароль, и Вам не потребуется вводить их при каждом вызове push или pull. Помощник по учетным записям сделает это за Вас.

$ git config --global credential.helper cache
# Включает кэширование памяти учетных записей
$ git config --global credential.helper 'cache --timeout=3600'
# Задает таймаут для кэша 1 час (задается в секундах)

Найти помощника по учетным записям:

$ git help -a | grep credential
# Покажет доступных помощников по учетным записям

Специфическое кэширование учетных записей для некоторых ОС:

$ git config --global credential.helper osxkeychain
# Для OSX
$ git config --global credential.helper manager
# Git for Windows 2.7.3+
$ git config --global credential.helper gnome-keyring
# Ubuntu и другие дистрибутивы, основанные на GNOME

Вероятно можно найти и других помощников по учетным записям для других дистрибутивов и ОС.

Я хочу, чтобы Git игнорировал изменения разрешений и прав файлов

$ git config core.fileMode false

Если Вы хотите задать это поведение по-умолчанию для всех авторизованных пользователей, тогда используйте:

$ git config --global core.fileMode false

Я хочу задать данные пользователя в глобальных настройках

Чтобы настроить информацию о пользователе, используемую во всех локальных репозиториях, в частности имя, отображаемое в истории изменений:

$ git config --global user.name “[firstname lastname]”

Чтобы настроить адрес электронной почты, который будет связан с каждой записью в историю:

git config --global user.email “[valid-email]”

Я хочу настроить цветовую схему для командной строки Git

Для наглядности можно настроить автоматическую цветовую схему командной строки Git:

$ git config --global color.ui auto

Я не представляю что я сделал неправильно

Итак, Вы в затруднении - Вы сбросили что-то или Вы слили неправильную ветку, или Вы отправили изменения с принудительной перезаписью и теперь Вы не можете найти свои коммиты. Вы знаете, что в какой-то момент было всё в порядке и Вы хотите вернуться к этому состоянию.

Как раз для этого и нужен git reflog. reflog отслеживает все изменения вершины ветки, даже если на эту вершину не указывает ни единая ветвь или метка. В принципе, всякий раз при изменении HEAD, в reflog добавляется новая запись. К сожалению, это работает только с локальными репозиториями и отслеживаются только движения (а не, например, изменения файла, которые не были никуда записаны).

(master)$ git reflog
0a2e358 HEAD@{0}: reset: moving to HEAD~2
0254ea7 HEAD@{1}: checkout: moving from 2.2 to master
c10f740 HEAD@{2}: checkout: moving from master to 2.2

reflog выше показывает переход с ветки master на ветку 2.2 и обратно. Затем происходит жесткий сброс на старый коммит. Самое последнее действие находится вверху с надписью HEAD@{0}.

Если Вы случайно переместитесь назад, то reflog будет содержать коммит (0254ea7), на который указывала ветка master до того, как Вы случайно выбросили 2 коммита.

$ git reset --hard 0254ea7

С помощью git reset можно вернуть ветку master обратно на коммит, на котором она была прежде. Это обеспечивает безопасность при случайном изменении истории.

(взято из Источник).

Сокращения для команд Git

Git Bash

Как только Вы освоитесь с командами, описанными выше, возможно, Вам захочется написать сокращения для Git Bash. Это позволит Вас работать гораздо быстрее, выполняя сложные задачи с помощью весьма коротких команд.

alias sq=squash

function squash() {
    git rebase -i HEAD~$1
}

Скопируйте эти команды в Ваш .bashrc или .bash_profile.

PowerShell в Windows

Если Вы пользуетесь PowerShell на Windows, то и здесь Вы можете настроить псевдонимы и функции. Добавьте эти команды в Ваш профиль, путь к которому указан в переменной $profile. Узнать больше о профилях можно на портале документации Microsoft About Profiles.

Set-Alias sq Squash-Commits

function Squash-Commits {
  git rebase -i HEAD~$1
}

Другие ресурсы

Книги

Учебники

Скрипты и инструменты

Графические клиенты