Подготовка
Если вы никогда ранее не использовали git, для начала вам необходимо осуществить установку. Выполните следующие команды, чтобы git узнал ваше имя и электронную почту.
git config --global user.name "Orkhan Alishov" git config --global user.email "o.alishoff@gmail.com"
Также параметры установки окончаний строк (Windows):
git config --global core.autocrlf true git config --global core.safecrlf true
Создание проекта
Начните работу в пустом рабочем каталоге (например, work) с создания пустого каталога с именем "hello", затем войдите в него и создайте там файл с именем hello.html с таким содержанием:
mkdir hello cd hello touch hello.html
hello.html:
Hello, World
Теперь у вас есть каталог с одним файлом. Чтобы создать git репозиторий из этого каталога, выполните команду git init:
git init
Теперь давайте добавим в репозиторий страницу "Hello, World"
$ git add hello.html $ git commit -m "First Commit" [master (root-commit) 911e8c9] First Commit 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 hello.html
Проверка состояния
Используйте команду git status, чтобы проверить текущее состояние репозитория:
$ git status # On branch master nothing to commit (working directory clean)
Команда проверки состояния сообщит, что коммитить нечего. Это означает, что в репозитории хранится текущее состояние рабочего каталога, и нет никаких изменений, ожидающих записи.
Внесение изменений
Добавим кое-какие HTML-теги к нашему приветствию. Измените содержимое файла на:
<h1>Hello, World!</h1>
Теперь проверьте состояние рабочего каталога:
$ git status # On branch master # 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) # # modified: hello.html # no changes added to commit (use "git add" and/or "git commit -a")
Первое, что нужно заметить, это то, что git знает, что файл hello.html был изменен, но при этом эти изменения еще не зафиксированы в репозитории.
Также обратите внимание на то, что сообщение о состоянии дает вам подсказку о том, что нужно делать дальше. Если вы хотите добавить эти изменения в репозиторий, используйте команду git add. В противном случае используйте команду git сheckout для отмены изменений.
Индексация изменений
Теперь дайте команду git проиндексировать изменения. Проверьте состояние:
$ git add hello.html $ git status # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # modified: hello.html #
Изменения файла hello.html были проиндексированы. Это означает, что git теперь знает об изменении, но изменение пока не навсегда записано в репозиторий. Следующий коммит будет включать в себя проиндексированные изменения.
Если вы решили, что не хотите коммитить изменения, команда состояния напомнит вам о том, что с помощью команды git reset можно снять индексацию этих изменений.
Индексация и коммит
Отдельный шаг индексации в git позволяет вам продолжать вносить изменения в рабочий каталог, а затем, в момент, когда вы захотите взаимодействовать с версионным контролем, git позволит записать изменения в малых коммитах, которые фиксируют то, что вы сделали.
Предположим, что вы отредактировали три файла (a.html, b.html и c.html). Теперь вы хотите закоммитить все изменения, при этом чтобы изменения в a.html и b.html были одним коммитом, в то время как изменения в c.html логически не связаны с первыми двумя файлами и должны идти отдельным коммитом.
В теории, вы можете сделать следующее:
git add a.html git add b.html git commit -m "Changes for a and b" git add c.html git commit -m "Unrelated change to c"
Разделяя индексацию и коммит, вы имеете возможность с легкостью настроить, что идет в какой коммит.
Коммит изменений
Достаточно об индексации. Давайте сделаем коммит того, что мы проиндексировали, в репозиторий.
Когда вы ранее использовали git commit для коммита первоначальной версии файла hello.html в репозиторий, вы включили метку -m, которая делает комментарий в командной строке. Команда commit позволит вам интерактивно редактировать комментарии для коммита. Теперь давайте это проверим.
Если вы опустите метку -m из командной строки, git перенесет вас в редактор по вашему выбору. Редактор выбирается из следующего списка (в порядке приоритета):
У меня переменная EDITOR установлена в notepad.
Сделайте коммит сейчас и проверьте состояние:
git commit
Вы увидите в вашем редакторе:
| # Please enter the commit message for your changes. Lines starting # with '#' will be ignored, and an empty message aborts the commit. # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # modified: hello.html #
В первой строке введите комментарий: "Added h1 tag". Сохраните файл и выйдите из редактора. Вы увидите:
git commit Waiting for notepad... [master 569aa96] Added h1 tag 1 files changed, 1 insertions(+), 1 deletions(-)
В конце давайте еще раз проверим состояние:
$ git status # On branch master nothing to commit (working directory clean)
Рабочий каталог чистый, можете продолжить работу.
Изменения, а не файлы
Большинство систем версионного контроля работают с файлами. Вы добавляете файл в версионный контроль, а система отслеживает изменения файла с этого момента.
Git фокусируется на изменениях в файле, а не самом файле. Когда вы осуществляете команду git add file, вы не говорите git добавить файл в репозиторий. Скорее вы говорите, что git надо отметить текущее состояние файла, коммит которого будет произведен позже.
Измените страницу "Hello, World", чтобы она содержала стандартные теги <html> и <body>:
<html> <body> <h1>Hello, World!</h1> </body> </html>
Теперь добавьте это изменение в индекс git:
git add hello.html
Теперь добавьте заголовки HTML (секцию <head>) к странице "Hello, World":
<html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
Проверьте текущий статус:
$ git status # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # modified: hello.html # # 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) # # modified: hello.html #
Обратите внимание на то, что hello.html указан дважды в состоянии. Первое изменение (добавление стандартных тегов) проиндексировано и готово к коммиту. Второе изменение (добавление заголовков HTML) является непроиндексированным. Если бы вы делали коммит сейчас, заголовки не были бы сохранены в репозиторий. Давайте проверим.
Произведите коммит проиндексированного изменения (значение по умолчанию), а затем еще раз проверьте состояние:
$ git commit -m "Added standard HTML page tags" [master 8c32287] Added standard HTML page tags 1 files changed, 3 insertions(+), 1 deletions(-) $ git status # On branch master # 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) # # modified: hello.html # no changes added to commit (use "git add" and/or "git commit -a")
Состояние команды говорит о том, что hello.html имеет незафиксированные изменения, но уже не в буферной зоне.
Теперь добавьте второе изменение в индекс, а затем проверьте состояние с помощью команды git status:
$ git add . $ git status # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # modified: hello.html #
Второе изменение было проиндексировано и готово к коммиту.
Примечание: В качестве файла для добавления, мы использовали текущий каталог (.). Это самый краткий и удобный путь для добавления всех изменений в файлы текущего каталога и его подкаталоги. Но поскольку он добавляет все, не лишним будет проверить состояние перед запуском add, просто чтобы убедиться, что вы не добавили какой-то файл, который добавлять было не нужно.
Сделайте коммит второго изменения:
git commit -m "Added HTML header"
История
Получение списка произведенных изменений - функция команды git log.
$ git log commit fa3c1411aa09441695a9e645d4371e8d749da1dc Author: Orkhan Alyshov <orkhanalyshov@gmail.com> Date: Wed Mar 9 10:27:54 2011 -0500 Added HTML header commit 8c3228730ed03116815a5cc682e8105e7d981928 Author: Orkhan Alyshov <orkhanalyshov@gmail.com> Date: Wed Mar 9 10:27:54 2011 -0500 Added standard HTML page tags commit 43628f779cb333dd30d78186499f93638107f70b Author: Orkhan Alyshov <orkhanalyshov@gmail.com> Date: Wed Mar 9 10:27:54 2011 -0500 Added h1 tag commit 911e8c91caeab8d30ad16d56746cbd6eef72dc4c Author: Orkhan Alyshov <orkhanalyshov@gmail.com> Date: Wed Mar 9 10:27:54 2011 -0500 First Commit
Вот список всех четырех коммитов в репозиторий, которые мы успели совершить.
Вы полностью контролируете то, что отображает log. Мне, например, нравится однострочный формат:
$ git log --pretty=oneline fa3c1411aa09441695a9e645d4371e8d749da1dc Added HTML header 8c3228730ed03116815a5cc682e8105e7d981928 Added standard HTML page tags 43628f779cb333dd30d78186499f93638107f70b Added h1 tag 911e8c91caeab8d30ad16d56746cbd6eef72dc4c First Commit
Есть много вариантов выбора, какие элементы отображаются в логе. Поиграйте со следующими параметрами:
git log --pretty=oneline --max-count=2 git log --pretty=oneline --since='5 minutes ago' git log --pretty=oneline --until='5 minutes ago' git log --pretty=oneline --author=<your name> git log --pretty=oneline --all
Алиасы
Добавьте следующее в файл .gitconfig в вашем $HOME каталоге.
[alias] co = checkout ci = commit st = status br = branch hist = log --pretty=format:\"%h %ad | %s%d [%an]\" --graph --date=short type = cat-file -t dump = cat-file -p
Мы уже успели рассмотреть команды commit и status, в предыдущем уроке рассмотрели команду log и совсем скоро познакомимся с checkout. Главное, что стоит запомнить, так это то, что теперь вы можете вводить git st там, где раньше приходилось использовать git status. Аналогичным образом, пишем git co вместо git checkout и git ci вместо git commit. Что лучше всего, команда git hist позволит избежать ввода очень длинной команды log.
Мы добавили несколько алиасов для команд, которых мы еще не рассматривали. С командой git branch разберемся чуть позже, а команда git cat-file используется для исследования git, в чем мы вскоре убедимся.
Получение старых версий
Возвращаться назад в историю очень просто. Команда checkout скопирует любой снимок из репозитория в рабочий каталог.
$ git hist * fa3c141 2011-03-09 | Added HTML header (HEAD, master) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Примечание: Вы не забыли задать hist в вашем файле .gitconfig?
Изучите данные лога и найдите хэш для первого коммита. Он должен быть в последней строке данных git hist. Используйте этот хэш-код (достаточно первых 7 знаков) в команде ниже. Затем проверьте содержимое файла hello.html.
$ git checkout 911e8c9 Note: checking out '911e8c9'. You are in 'detached HEAD' state. You can look around, make experimental changes and commit them, and you can discard any commits you make in this state without impacting any branches by performing another checkout. If you want to create a new branch to retain commits you create, you may do so (now or later) by using -b with the checkout command again. Example: git checkout -b new_branch_name HEAD is now at 911e8c9... First Commit $ cat hello.html Hello, World
Обратите внимание на то, что содержимое файла hello.html является значением по умолчанию.
Примечание: Многие команды зависят от хэшевых значений в репозитории. Поскольку ваши хеш-значения будут отличаться от моих, когда вы видите что-то вроде <hash> или <treehash> в команде, подставьте необходимое значение хэш для вашего репозитория.
Вернитесь к последней версии в ветке master:
$ git checkout master Previous HEAD position was 911e8c9... First Commit Switched to branch 'master' $ cat hello.html <html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
"master" - имя ветки по умолчанию. Переключая имена веток, вы попадаете на последнюю версию выбранной ветки.
Создание тегов версий
Давайте назовем текущую версию страницы hello первой (v1):
git tag v1
Теперь текущая версия страницы называется v1.
Давайте создадим тег для версии, которая идет перед текущей версией и назовем его v1-beta. В первую очередь нам надо переключиться на предыдущую версию. Вместо поиска до хэш, мы будем использовать ^, обозначающее "родитель v1".
$ git checkout v1^ Note: checking out 'v1^'. You are in 'detached HEAD' state. You can look around, make experimental changes and commit them, and you can discard any commits you make in this state without impacting any branches by performing another checkout. If you want to create a new branch to retain commits you create, you may do so (now or later) by using -b with the checkout command again. Example: git checkout -b new_branch_name HEAD is now at 8c32287... Added standard HTML page tags $ cat hello.html <html> <body> <h1>Hello, World!</h1> </body> </html>
Эта версия c тегами <html> и <body>, но еще пока без <head>. Давайте сделаем ее версией v1-beta:
git tag v1-beta
Теперь попробуйте попереключаться между двумя отмеченными версиями:
$ git checkout v1 Previous HEAD position was 8c32287... Added standard HTML page tags HEAD is now at fa3c141... Added HTML header $ git checkout v1-beta Previous HEAD position was fa3c141... Added HTML header HEAD is now at 8c32287... Added standard HTML page tags
Вы можете увидеть, какие теги доступны, используя команду git tag:
$ git tag v1 v1-beta
Вы также можете посмотреть теги в логе:
$ git hist master --all * fa3c141 2011-03-09 | Added HTML header (v1, master) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (HEAD, v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Вы можете видеть теги (v1 и v1-beta) в логе вместе с именем ветки (master). Кроме того HEAD показывает коммит, на который вы переключились (на данный момент это v1-beta).
Отмена локальных изменений (до индексации)
Убедитесь, что вы находитесь на последнем коммите ветки master, прежде чем продолжить работу:
git checkout master
Иногда случается, что вы изменили файл в рабочем каталоге, и хотите отменить последние коммиты. С этим справится команда checkout.
Внесите изменение в файл hello.html в виде нежелательного комментария:
<html> <head> </head> <body> <h1>Hello, World!</h1> <!-- This is a bad comment. We want to revert it. --> </body> </html>
Сначала проверьте состояние рабочего каталога:
$ git status # On branch master # 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) # # modified: hello.html # no changes added to commit (use "git add" and/or "git commit -a")
Мы видим, что файл hello.html был изменен, но еще не проиндексирован.
Используйте команду checkout для переключения в версию файла hello.html в репозитории:
$ git checkout hello.html $ git status # On branch master nothing to commit (working directory clean) $ cat hello.html <html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
Команда status показывает нам, что не было произведено никаких изменений, не зафиксированных в рабочем каталоге. И "нежелательный комментарий" больше не является частью содержимого файла.
Отмена проиндексированных изменений (перед коммитом)
Внесите изменение в файл hello.html в виде нежелательного комментария:
<html> <head> <!-- This is an unwanted but staged comment --> </head> <body> <h1>Hello, World!</h1> </body> </html>
Проиндексируйте это изменение:
git add hello.html
Проверьте состояние нежелательного изменения:
$ git status # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # modified: hello.html #
Состояние показывает, что изменение было проиндексировано и готово к коммиту.
К счастью, вывод состояния показывает нам именно то, что мы должны сделать для отмены индексации изменения:
$ git reset HEAD hello.html Unstaged changes after reset: M hello.html
Команда reset сбрасывает буферную зону к HEAD. Это очищает буферную зону от изменений, которые мы только что проиндексировали.
Команда reset (по умолчанию) не изменяет рабочий каталог. Поэтому рабочий каталог все еще содержит нежелательный комментарий. Мы можем использовать команду checkout из предыдущего урока, чтобы удалить нежелательные изменения в рабочем каталоге.
Переключитесь на версию коммита:
$ git checkout hello.html $ git status # On branch master nothing to commit (working directory clean)
Наш рабочий каталог опять чист.
Отмена коммитов
Иногда вы понимаете, что новые коммиты являются неверными, и хотите их отменить. Есть несколько способов решения этого вопроса, здесь мы будем использовать самый безопасный. Мы отменим коммит путем создания нового коммита, отменяющего нежелательные изменения.
Измените файл hello.html на следующий:
<html> <head> </head> <body> <h1>Hello, World!</h1> <!-- This is an unwanted but committed change --> </body> </html>
Сделайте коммит:
git add hello.html git commit -m "Oops, we didn't want this commit"
Чтобы отменить коммит, нам необходимо сделать коммит, который удаляет изменения, сохраненные нежелательным коммитом:
$ git revert HEAD --no-edit [master 45fa96b] Revert "Oops, we didn't want this commit" 1 files changed, 1 insertions(+), 1 deletions(-)
Так как мы отменили самый последний произведенный коммит, мы смогли использовать HEAD в качестве аргумента для отмены. Мы можем отменить любой произвольной коммит в истории, указав его хэш-значение.
Примечание: команду --no-edit можно проигнорировать. Она была необходима для генерации выходных данных без открытия редактора.
Проверка лога показывает нежелательные и отмененные коммиты в наш репозиторий:
$ git hist * 45fa96b 2011-03-09 | Revert "Oops, we didn't want this commit" (HEAD, master) [Orkhan Alyshov] * 846b90c 2011-03-09 | Oops, we didn't want this commit [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Удаление коммиттов из ветки
Revert из предыдущего раздела является мощной командой, которая позволяет отменить любые коммиты в репозиторий. Однако, и оригинальный и отмененный коммиты видны в истории ветки (при использовании команды git log). Часто мы делаем коммит, и сразу понимаем, что это была ошибка. Было бы неплохо иметь команду возврата, которая позволила бы нам сделать вид, что неправильного коммита никогда и не было. Команда возврата даже предотвратила бы появление нежелательного коммита в истории git log.
Мы уже видели команду reset и использовали ее для согласования буферной зоны и выбранного коммита (мы использовали коммит HEAD в нашем предыдущем уроке).
При получении ссылки на коммит (т.е. хэш, ветка или имя тега), команда reset:
Давайте сделаем быструю проверку нашей истории коммитов:
$ git hist * 45fa96b 2011-03-09 | Revert "Oops, we didn't want this commit" (HEAD, master) [Orkhan Alyshov] * 846b90c 2011-03-09 | Oops, we didn't want this commit [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Мы видим, что два последних коммита в этой ветке - "Oops" и "Revert Oops". Давайте удалим их с помощью сброса. Но прежде чем удалить коммиты, давайте отметим последний коммит тегом, чтобы потом можно было его найти:
git tag oops
Глядя на историю лога (см. выше), мы видим, что коммит с тегом v1 является коммитом, предшествующим ошибочному коммиту. Давайте сбросим ветку до этой точки. Поскольку ветка имеет тег, мы можем использовать имя тега в команде сброса (если она не имеет тега, мы можем использовать хэш-значение):
$ git reset --hard v1 HEAD is now at fa3c141 Added HTML header $ git hist * fa3c141 2011-03-09 | Added HTML header (HEAD, v1, master) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Наша ветка master теперь указывает на коммит v1, а коммитов Oops и Revert Oops в ветке уже нет. Параметр --hard указывает, что рабочий каталог должен быть обновлен в соответствии с новым head ветки.
Что же случается с ошибочными коммитами? Оказывается, что коммиты все еще находятся в репозитории. На самом деле, мы все еще можем на них ссылаться. Помните, в начале этого урока мы создали для отмененного коммита тег "oops". Давайте посмотрим на все коммиты:
$ git hist --all * 45fa96b 2011-03-09 | Revert "Oops, we didn't want this commit" (oops) [Orkhan Alyshov] * 846b90c 2011-03-09 | Oops, we didn't want this commit [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (HEAD, v1, master) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Мы видим, что ошибочные коммиты не исчезли. Они все еще находятся в репозитории. Просто они отсутствуют в ветке master. Если бы мы не отметили их тегами, они по-прежнему находились бы в репозитории, но не было бы никакой возможности ссылаться на них, кроме как при помощи их хэш имен. Коммиты, на которые нет ссылок, остаются в репозитории до тех пор, пока не будет запущен сборщик мусора.
Удаление тега oops
Тег oops свою функцию выполнил. Давайте удалим его и коммиты, на которые он ссылался, сборщиком мусора:
$ git tag -d oops Deleted tag 'oops' (was 45fa96b) $ git hist --all * fa3c141 2011-03-09 | Added HTML header (HEAD, v1, master) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Внесение изменений в коммиты
Добавьте в страницу комментарий автора:
hello.html:
<!-- Author: Orkhan Alyshov --> <html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
git add hello.html git commit -m "Add an author comment"
После совершения коммита вы понимаете, что любой хороший комментарий должен включать электронную почту автора. Обновите страницу hello, включив в нее email:
hello.html:
<!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
Мы действительно не хотим создавать отдельный коммит только ради электронной почты. Давайте изменим предыдущий коммит, включив в него адрес электронной почты:
$ git add hello.html $ git commit --amend -m "Add an author/email comment" [master 6a78635] Add an author/email comment 1 files changed, 2 insertions(+), 1 deletions(-)
Просмотр истории:
$ git hist * 6a78635 2011-03-09 | Add an author/email comment (HEAD, master) [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Мы можем увидеть, что оригинальный коммит "author" заменен коммитом "author/email". Этого же эффекта можно достичь путем сброса последнего коммита в ветке, и повторного коммита новых изменений.
Перемещение файлов
- Первый способ:
Сейчас мы собираемся создать структуру нашего репозитория. Давайте перенесем страницу в каталог lib:
$ mkdir lib $ git mv hello.html lib $ git status # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # renamed: hello.html -> lib/hello.html #
Перемещая файлы с помощью git, мы информируем git о 2 вещах:
Оба эти факта сразу же проиндексированы и готовы к коммиту. Команда git status сообщает, что файл был перемещен.
- Второй способ:
Позитивной чертой git является то, что вы можете забыть о версионном контроле до того момента, когда вы готовы приступить к коммиту кода. Что бы случилось, если бы мы использовали командную строку операционной системы для перемещения файлов вместо команды git?
Оказывается, следующий набор команд идентичен нашим последним действиям. Работы здесь побольше, но результат тот же. Мы могли бы выполнить:
mkdir lib mv hello.html lib git add lib/hello.html git rm hello.html
Давайте сделаем коммит этого перемещения:
git commit -m "Moved hello.html to lib"
Подробнее о структуре
Давайте добавим файл index.html в наш репозиторий. Следующий файл отлично подойдет для этой цели.
index.html:
<html> <body> <iframe src="lib/hello.html" width="200" height="200" /> </body> </html>
Добавьте файл и сделайте коммит:
git add index.html git commit -m "Added index.html."
Теперь при открытии index.html, вы должны увидеть кусок страницы hello в маленьком окошке.
Git внутри: каталог .git
Настало время провести небольшое исследование. Для начала, из корневого каталога вашего проекта:
$ ls -C .git COMMIT_EDITMSG MERGE_RR config hooks info objects rr-cache HEAD ORIG_HEAD description index logs refs
Это магический каталог, в котором хранятся все "материалы" git. Давайте заглянем в каталог объектов:
$ ls -C .git/objects 09 24 28 45 59 6a 77 80 8c 97 af c4 e7 info 11 27 43 56 69 6b 78 84 91 9c b5 e4 fa pack
Вы должны увидеть кучу каталогов, имена которых состоят из 2 символов. Имена каталогов являются первыми двумя буквами хэша sha1 объекта, хранящегося в git.
$ ls -C .git/objects/09 6b74c56bfc6b40e754fc0725b8c70b2038b91e 9fb6f9d3a104feb32fcac22354c4d0e8a182c1
Смотрим в один из каталогов с именем из 2 букв. Вы увидите файлы с именами из 38 символов. Это файлы, содержащие объекты, хранящиеся в git. Они сжаты и закодированы, поэтому просмотр их содержимого нам мало чем поможет. Рассмотрим далее каталог .git внимательно.
$ cat .git/config [core] repositoryformatversion = 0 filemode = true bare = false logallrefupdates = true ignorecase = true [user] name = Orkhan Alyshov email = Orkhan@githowto.com
Это файл конфигурации, создающийся для каждого конкретного проекта. Записи в этом файле будут перезаписывать записи в файле .gitconfig вашего главного каталога, по крайней мере в рамках этого проекта.
$ ls .git/refs heads tags $ ls .git/refs/heads master $ ls .git/refs/tags v1 v1-beta $ cat .git/refs/tags/v1 fa3c1411aa09441695a9e645d4371e8d749da1dc
Вы должны узнавать файлы в подкаталоге тегов. Каждый файл соответствует тегу, ранее созданному с помощью команды git tag. Его содержание - это всего лишь хэш коммита, привязанный к тегу.
Каталог heads практически аналогичен, но используется для веток, а не тегов. На данный момент у нас есть только одна ветка, так что все, что вы увидите в этом каталоге – это ветка master.
$ cat .git/HEAD ref: refs/heads/master
Файл HEAD содержит ссылку на текущую ветку, в данный момент это должна быть ветка master.
Git внутри: работа непосредственно с объектами git
Давайте исследуем объекты git с помощью некоторых инструментов.
Поиск последнего коммита. Эта команда должна показать последний коммит в репозиторий. SHA1 хэш в вашей системе, вероятно, отличается от моего, но вы увидите что-то наподобие этого:
$ git hist --max-count=1 * 8029c07 2011-03-09 | Added index.html. (HEAD, master) [Orkhan Alyshov]
С помощью SHA1 хэша из коммита, указанного выше:
$ git cat-file -t 8029c07 commit $ git cat-file -p 8029c07 tree 096b74c56bfc6b40e754fc0725b8c70b2038b91e parent 567948ac55daa723807c0c16e34c76797efbcbed author Orkhan Alyshov <Orkhan@githowto.com> 1299684476 -0500 committer Orkhan Alyshov <Orkhan@githowto.com> 1299684476 -0500 Added index.html.
Примечание: Если вы задали алиасы "type" и "dump", как описано в уроке об алиасах, можете вводить команды git type и git dump вместо длинных команд.
Мы можем вывести дерево каталогов, ссылка на который идет в коммите. Это должно быть описание файлов (верхнего уровня) в нашем проекте (для конкретного коммита). Используйте SHA1 хэш из строки "дерева", из списка выше:
$ git cat-file -p 096b74c 100644 blob 28e0e9d6ea7e25f35ec64a43f569b550e8386f90 index.html 040000 tree e46f374f5b36c6f02fb3e9e922b79044f754d795 lib
Да, я вижу index.html и каталог lib.
Вывод каталога lib:
$ git cat-file -p e46f374 100644 blob c45f26b6fdc7db6ba779fc4c385d9d24fc12cf72 hello.html
Существует файл hello.html.
Вывод файла hello.html:
$ git cat-file -p c45f26b <!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
А вот и он. Мы вывели объекты коммитов, объекты деревьев и объекты блобов непосредственно из репозитория git. Это все, что есть - блобы, деревья и коммиты.
Создание ветки
Пора сделать наш hello world более выразительным. Так как это может занять некоторое время, лучше переместить эти изменения в отдельную ветку, чтобы изолировать их от изменений в ветке master.
Давайте назовем нашу новую ветку "style":
git checkout -b style git status
Примечание: git checkout -b <имяветки> является шорткатом для git branch <имяветки> за которым идет git checkout <имяветки>.
Обратите внимание, что команда git status сообщает о том, что вы находитесь в ветке "style".
Добавьте файл стилей style.css:
touch lib/style.css
lib/style.css:
h1 { color: red; }
git add lib/style.css git commit -m "Added css stylesheet"
Обновите файл hello.html, чтобы использовать стили style.css:
<!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> <link type="text/css" rel="stylesheet" media="all" href="style.css" /> </head> <body> <h1>Hello, World!</h1> </body> </html>
git add lib/hello.html git commit -m "Hello uses style.css"
Обновите файл index.html, чтобы он тоже использовал style.css:
<html> <head> <link type="text/css" rel="stylesheet" media="all" href="lib/style.css" /> </head> <body> <iframe src="lib/hello.html" width="200" height="200" /> </body> </html>
git add index.html git commit -m "Updated index.html"
Навигация по веткам
Теперь в вашем проекте есть две ветки:
$ git hist --all * 07a2a46 2011-03-09 | Updated index.html (HEAD, style) [Orkhan Alyshov] * 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] * 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] * 8029c07 2011-03-09 | Added index.html. (master) [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Просто используйте команду git checkout для переключения между ветками:
$ git checkout master Switched to branch 'master' $ cat lib/hello.html <!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> </head> <body> <h1>Hello, World!</h1> </body> </html>
Сейчас мы находимся на ветке Master. Это заметно по тому, что файл hello.html не использует стили style.css.
Вернемся к ветке style:
$ git checkout style Switched to branch 'style' $ cat lib/hello.html <!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> <link type="text/css" rel="stylesheet" media="all" href="style.css" /> </head> <body> <h1>Hello, World!</h1> </body> </html>
Изменения в ветке master
Пока вы меняли ветку style, кто-то решил обновить ветку master. Они добавили README:
Создайте файл README в ветке master:
git checkout master
README:
This is the Hello World example from the git tutorial.
Сделайте коммит изменений README в ветку master:
git add README git commit -m "Added README"
Просмотр отличающихся веток
Теперь у нас в репозитории есть две отличающиеся ветки. Используйте следующую лог-команду для просмотра веток и их отличий:
$ git hist --all * 6c0f848 2011-03-09 | Added README (HEAD, master) [Orkhan Alyshov] | * 07a2a46 2011-03-09 | Updated index.html (style) [Orkhan Alyshov] | * 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] | * 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] |/ * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Это наша первая возможность увидеть в действии --graph в git hist. Добавление опции --graph в git log вызывает построение дерева коммитов с помощью простых ASCII символов. Мы видим обе ветки (style и master), и то, что ветка master является текущей HEAD. Общим предшественником обеих веток является коммит "Added index.html". Метка --all гарантированно означает, что мы видим все ветки. По умолчанию показывается только текущая ветка.
Слияние
Слияние переносит изменения из двух веток в одну. Давайте вернемся к ветке style и сольем master с style:
$ git checkout style Switched to branch 'style' $ git merge master Merge made by recursive. README | 1 + 1 files changed, 1 insertions(+), 0 deletions(-) create mode 100644 README $ git hist --all * 5813a3f 2011-03-09 | Merge branch 'master' into style (HEAD, style) [Orkhan Alyshov] |\ | * 6c0f848 2011-03-09 | Added README (master) [Orkhan Alyshov] * | 07a2a46 2011-03-09 | Updated index.html [Orkhan Alyshov] * | 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] * | 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] |/ * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Путем периодического слияния ветки master с веткой style вы можете переносить из master любые изменения и поддерживать совместимость изменений style с изменениями в основной ветке. Однако, это делает графики коммитов действительно уродливыми. Позже мы рассмотрим возможность перебазирования, как альтернативы слиянию.
Создание конфликта
Вернитесь в ветку master и внесите следующие изменения:
git checkout master
lib/hello.html:
<!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> <!-- no style --> </head> <body> <h1>Hello, World! Life is great!</h1> </body> </html>
git add lib/hello.html git commit -m "Life is great!"
Просмотр веток:
$ git hist --all * 454ec68 2011-03-09 | Life is great! (HEAD, master) [Orkhan Alyshov] | * 5813a3f 2011-03-09 | Merge branch 'master' into style (style) [Orkhan Alyshov] | |\ | |/ |/| * | 6c0f848 2011-03-09 | Added README [Orkhan Alyshov] | * 07a2a46 2011-03-09 | Updated index.html [Orkhan Alyshov] | * 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] | * 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] |/ * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
После коммита "Added README" ветка master была объединена с веткой style, но в настоящее время в master есть дополнительный коммит, который не был слит с style.
Разрешение конфликтов
Теперь вернемся к ветке style и попытаемся объединить ее с новой веткой master:
$ git checkout style Switched to branch 'style' $ git merge master Auto-merging lib/hello.html CONFLICT (content): Merge conflict in lib/hello.html Automatic merge failed; fix conflicts and then commit the result.
Если вы откроете lib/hello.html, вы увидите:
<!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> <<<<<<< HEAD <link type="text/css" rel="stylesheet" media="all" href="style.css" /> ======= <!-- no style --> >>>>>>> master </head> <body> <h1>Hello,World! Life is great!</h1> </body> </html>
Первый раздел - версия во главе текущей ветки (style). Второй раздел - версия ветки master.
Вам необходимо вручную разрешить конфликт. Внесите изменения в lib/hello.html для достижения следующего результата:
<!-- Author: Orkhan Alyshov (Orkhan@githowto.com) --> <html> <head> <link type="text/css" rel="stylesheet" media="all" href="style.css" /> </head> <body> <h1>Hello, World! Life is great!</h1> </body> </html>
Сделайте коммит решения конфликта:
$ git add lib/hello.html $ git commit -m "Merged master fixed conflict." Recorded resolution for 'lib/hello.html'. [style 645c4e6] Merged master fixed conflict.
Перебазирование как альтернатива слиянию
Давайте рассмотрим различия между слиянием и перебазированием. Для того, чтобы это сделать, нам нужно вернуться в репозиторий в момент до первого слияния, а затем повторить те же действия, но с использованием перебазирования вместо слияния. Мы будем использовать команду reset для возврата веток к предыдущему состоянию.
Сброс ветки style
Давайте вернемся во времени на ветке style к точке перед тем, как мы слили ее с веткой master. Мы можем сбросить ветку к любому коммиту. По сути, это изменение указателя ветки на любую точку дерева коммитов. В этом случае мы хотим вернуться в ветке style в точку перед слиянием с master. Нам необходимо найти последний коммит перед слиянием:
$ git checkout style Already on 'style' $ git hist * 645c4e6 2011-03-09 | Merged master fixed conflict. (HEAD, style) [Orkhan Alyshov] |\ | * 454ec68 2011-03-09 | Life is great! (master) [Orkhan Alyshov] * | 5813a3f 2011-03-09 | Merge branch 'master' into style [Orkhan Alyshov] |\ \ | |/ | * 6c0f848 2011-03-09 | Added README [Orkhan Alyshov] * | 07a2a46 2011-03-09 | Updated index.html [Orkhan Alyshov] * | 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] * | 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] |/ * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Это немного трудно читать, но, глядя на данные, мы видим, что коммит "Updated index.html" был последним на ветке style перед слиянием. Давайте сбросим ветку style к этому коммиту:
$ git reset --hard 07a2a46 HEAD is now at 07a2a46 Updated index.html
Поищите лог ветки style. У нас в истории больше нет коммитов слияний:
$ git hist --all * 454ec68 2011-03-09 | Life is great! (master) [Orkhan Alyshov] * 6c0f848 2011-03-09 | Added README [Orkhan Alyshov] | * 07a2a46 2011-03-09 | Updated index.html (HEAD, style) [Orkhan Alyshov] | * 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] | * 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] |/ * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Сброс ветки master
Добавив интерактивный режим в ветку master, мы внесли изменения, конфликтующие с изменениями в ветке style. Давайте вернемся в ветке master в точку перед внесением конфликтующих изменений. Это позволяет нам продемонстрировать работу команды rebase, не беспокоясь о конфликтах:
$ git checkout master $ git hist * 454ec68 2011-03-09 | Life is great! (HEAD, master) [Orkhan Alyshov] * 6c0f848 2011-03-09 | Added README [Orkhan Alyshov] * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Коммит "Added README" идет непосредственно перед коммитом конфликтующего интерактивного режима. Мы сбросим ветку master к коммиту "Added README".
$ git reset --hard 6c0f848 $ git hist --all * 6c0f848 2011-03-09 | Added README (HEAD, master) [Orkhan Alyshov] | * 07a2a46 2011-03-09 | Updated index.html (style) [Orkhan Alyshov] | * 649d26c 2011-03-09 | Hello uses style.css [Orkhan Alyshov] | * 1f3cbd2 2011-03-09 | Added css stylesheet [Orkhan Alyshov] |/ * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Перебазирование
Итак, мы вернулись в точку до первого слияния и хотим перенести изменения из ветки master в нашу ветку style. На этот раз для переноса изменений из ветки master мы будем использовать команду rebase вместо слияния.
$ git checkout style Switched to branch 'style' $ git rebase master First, rewinding head to replay your work on top of it... Applying: Added css stylesheet Applying: Hello uses style.css Applying: Updated index.html $ git hist * 6e6c76a 2011-03-09 | Updated index.html (HEAD, style) [Orkhan Alyshov] * 1436f13 2011-03-09 | Hello uses style.css [Orkhan Alyshov] * 59da9a7 2011-03-09 | Added css stylesheet [Orkhan Alyshov] * 6c0f848 2011-03-09 | Added README (master) [Orkhan Alyshov] * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
- Слияние VS перебазирование
Конечный результат перебазирования очень похож на результат слияния. Ветка style в настоящее время содержит все свои изменения, а также все изменения ветки master. Однако, дерево коммитов значительно отличается. Дерево коммитов ветки style было переписано таким образом, что ветка master является частью истории коммитов. Это делает цепь коммитов линейной и гораздо более читабельной.
Когда использовать перебазирование, а когда слияние?
Не используйте перебазирование:
Учитывая приведенные выше рекомендации, я предпочитаю использовать rebase для кратковременных, локальных веток, а слияние для веток в публичном репозитории.
Слияние в ветку master
$ git checkout master Switched to branch 'master' $ git merge style Updating 6c0f848..6e6c76a Fast-forward index.html | 2 +- lib/style.css | 8 ++++++++ lib/hello.html | 6 ++++-- 3 files changed, 13 insertions(+), 3 deletions(-) create mode 100644 lib/style.css
Поскольку последний коммит ветки master прямо предшествует последнему коммиту ветки style, git может выполнить ускоренное слияние-перемотку. При быстрой перемотке вперед, git просто передвигает указатель вперед, таким образом указывая на тот же коммит, что и ветка style.
При быстрой перемотке конфликтов быть не может.
Несколько репозиториев
До сих пор мы работали с одним git репозиторием. Однако, git удается отлично работать с несколькими репозиториями. Эти дополнительные репозитории могут храниться локально, или доступ к ним может осуществляться через сетевое подключение. В следующем разделе мы создадим новый репозиторий с именем "cloned_hello". Мы покажем, как перемещать изменения из одного репозитория в другой и как разрешать конфликты, возникающие в результате работы с двумя репозиториями.
А пока что поработаем с локальными репозиториями (т.е. репозиториями, хранящимися на вашем локальном жестком диске), однако практически все, что вы узнаете в этом разделе, будет применяться к нескольким репозиториям, несмотря на то, хранятся ли они локально или являются публичными.
Примечание: Мы будем вносить изменения в обе копии наших репозиториев. Обращайте внимание на то, в каком репозитории вы находитесь на каждом шаге в следующих уроках.
Клонирование репозиториев
Если вы работаете в команде, последующие главы довольно важны в понимании, т.к. вы почти всегда будете работать с клонированными репозиториями.
Перейдите в рабочий каталог и сделайте клон вашего репозитория hello:
$ cd .. $ pwd /Users/Orkhan/Documents/Presentations/githowto/auto $ ls hello
В этот момент вы должны находиться в "рабочем" каталоге. Здесь должен быть единственный репозиторий под названием "hello".
Давайте создадим клон репозитория:
$ git clone hello cloned_hello Cloning into cloned_hello... done. $ ls cloned_hello hello
В вашем рабочем каталоге теперь должно быть два репозитория: оригинальный репозиторий "hello" и клонированный репозиторий "cloned_hello".
Просмотр клонированного репозитория
Давайте взглянем на клонированный репозиторий:
$ cd cloned_hello $ ls README index.html lib
Просмотрите историю репозитория:
$ git hist --all * 6e6c76a 2011-03-09 | Updated index.html (HEAD, origin/master, origin/style, origin/HEAD, master) [Orkhan Alyshov] * 1436f13 2011-03-09 | Hello uses style.css [Orkhan Alyshov] * 59da9a7 2011-03-09 | Added css stylesheet [Orkhan Alyshov] * 6c0f848 2011-03-09 | Added README [Orkhan Alyshov] * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
Вы увидите список всех коммитов в новый репозиторий, и он должен (более или менее) совпадать с историей коммитов в оригинальном репозитории. Единственная разница должна быть в названиях веток.
Вы увидите ветку master (HEAD) в списке истории. Вы также увидите ветки со странными именами (origin/master, origin/style и origin/HEAD). Мы поговорим о них чуть позже.
Что такое origin?
$ git remote origin
Мы видим, что клонированный репозиторий знает об имени по умолчанию удаленного репозитория. Давайте посмотрим, можем ли мы получить более подробную информацию об имени по умолчанию:
$ git remote show origin * remote origin Fetch URL: /Users/Orkhan/Documents/Presentations/githowto/auto/hello Push URL: /Users/Orkhan/Documents/Presentations/githowto/auto/hello HEAD branch (remote HEAD is ambiguous, may be one of the following): style master Remote branches: style tracked master tracked Local branch configured for 'git pull': master merges with remote master Local ref configured for 'git push': master pushes to master (up to date)
Мы видим, что "имя по умолчанию"("origin") удаленного репозитория - оригинальное hello. Удаленные репозитории обычно размещаются на отдельной машине, возможно, централизованном сервере. Однако, как мы видим здесь, они могут с тем же успехом указывать на репозиторий на той же машине. Нет ничего особенного в имени "origin", однако существует традиция использовать "origin" в качестве имени первичного централизованного репозитория (если таковой имеется).
Удаленные ветки
Давайте посмотрим на ветки, доступные в нашем клонированном репозитории:
$ git branch * master
Как мы видим, в списке только ветка master. Где ветка style? Команда git branch выводит только список локальных веток по умолчанию.
Для того, чтобы увидеть все ветки, попробуйте следующую команду:
$ git branch -a * master remotes/origin/HEAD -> origin/master remotes/origin/style remotes/origin/master
Git выводит все коммиты в оригинальный репозиторий, но ветки в удаленном репозитории не рассматриваются как локальные. Если мы хотим собственную ветку style, мы должны сами ее создать.
Изменение оригинального репозитория
Внесите изменения в оригинальный репозиторий hello. Сейчас мы находимся в репозитории hello.
Внесите следующие изменения в файл README:
This is the Hello World example from the git tutorial. (changed in original)
Теперь добавьте это изменение и сделайте коммит:
git add README git commit -m "Changed README in original repo"
Теперь в оригинальном репозитории есть более поздние изменения, которых нет в клонированной версии. Далее мы извлечем и сольем эти изменения в клонированный репозиторий.
Извлечение изменений
Сейчас мы находимся в репозитории cloned_hello.
$ git fetch From /Users/Orkhan/Documents/Presentations/githowto/auto/hello 6e6c76a..2faa4ea master -> origin/master $ git hist --all * 2faa4ea 2011-03-09 | Changed README in original repo (origin/master, origin/HEAD) [Orkhan Alyshov] * 6e6c76a 2011-03-09 | Updated index.html (HEAD, origin/style, master) [Orkhan Alyshov] * 1436f13 2011-03-09 | Hello uses style.css [Orkhan Alyshov] * 59da9a7 2011-03-09 | Added css stylesheet [Orkhan Alyshov] * 6c0f848 2011-03-09 | Added README [Orkhan Alyshov] * 8029c07 2011-03-09 | Added index.html. [Orkhan Alyshov] * 567948a 2011-03-09 | Moved hello.html to lib [Orkhan Alyshov] * 6a78635 2011-03-09 | Add an author/email comment [Orkhan Alyshov] * fa3c141 2011-03-09 | Added HTML header (v1) [Orkhan Alyshov] * 8c32287 2011-03-09 | Added standard HTML page tags (v1-beta) [Orkhan Alyshov] * 43628f7 2011-03-09 | Added h1 tag [Orkhan Alyshov] * 911e8c9 2011-03-09 | First Commit [Orkhan Alyshov]
На данный момент в репозитории есть все коммиты из оригинального репозитория, но они не интегрированы в локальные ветки клонированного репозитория. В истории выше найдите коммит "Changed README in original repo". Обратите внимание, что коммит включает в себя коммиты "origin/master" и "origin/HEAD". Теперь давайте посмотрим на коммит "Updated index.html". Вы увидите, что локальная ветка master указывает на этот коммит, а не на новый коммит, который мы только что извлекли. Выводом является то, что команда "git fetch" будет извлекать новые коммиты из удаленного репозитория, но не будет сливать их с вашими наработками в локальных ветках.
Мы можем продемонстрировать, что клонированный файл README не изменился:
$ cat README This is the Hello World example from the git tutorial.
Как видите, никаких изменений.
Слияние извлеченных изменений
Слейте извлеченные изменения в локальную ветку master:
$ git merge origin/master Updating 6e6c76a..2faa4ea Fast-forward README | 1 + 1 files changed, 1 insertions(+), 0 deletions(-)
Еще раз проверьте файл README. Сейчас мы должны увидеть изменения:
$ cat README This is the Hello World example from the git tutorial. (changed in original)
Вот и изменения. Хотя команда "git fetch" не сливает изменения, мы можем вручную слить изменения из удаленного репозитория.
Извлечение и слияние изменений
Команда git pull эквивалентна комбинации git fetch и git merge.
Мы не собираемся опять проходить весь процесс создания нового изменения и его извлечения, но мы хотим, чтобы вы знали, что выполнение:
git pull
действительно эквивалентно двум следующим шагам:
git fetch git merge origin/master
Чистые репозитории
Чистые репозитории (без рабочих каталогов) обычно используются для расшаривания.
Небольшое пояснение, что же все-таки означает "чистый репозиторий". Обычный git-репозиторий подразумевает, что вы будете использовать его как рабочую директорию, поэтому вместе с файлами проекта в актуальной версии, git хранит все служебные, "чисто-репозиториевские" файлы в поддиректории .git. В удаленных репозиториях нет смысла хранить рабочие файлы на диске (как это делается в рабочих копиях), а все что им действительно нужно - это дельты изменений и другие бинарные данные репозитория. Вот это и есть "чистый репозиторий".
Создайте чистый репозиторий:
$ git clone --bare hello hello.git Cloning into bare repository hello.git... done. $ ls hello.git HEAD config description hooks info objects packed-refs refs
Как правило, репозитории, оканчивающиеся на ".git" являются чистыми репозиториями. Мы видим, что в репозитории hello.git нет рабочего каталога. По сути, это есть не что иное, как каталог .git нечистого репозитория.
Добавление удаленного репозитория
Давайте добавим репозиторий hello.git к нашему оригинальному репозиторию:
cd hello git remote add shared ../hello.git
Отправка изменений
Так как чистые репозитории, как правило, расшариваются на каком-нибудь сетевом сервере, нам необходимо отправить наши изменения в другие репозитории. Начнем с создания изменения для отправки. Отредактируйте файл README и сделайте коммит:
This is the Hello World example from the git tutorial. (Changed in the original and pushed to shared)
git checkout master git add README git commit -m "Added shared comment to readme"
Теперь отправьте изменения в общий репозиторий:
$ git push shared master To ../hello.git 2faa4ea..79f507c master -> master
Извлечение общих изменений
Быстро переключитесь в клонированный репозиторий и извлеките изменения, только что отправленные в общий репозиторий:
cd ../cloned_hello
Сейчас мы находимся в репозитории cloned_hello.
git remote add shared ../hello.git git branch --track shared master git pull shared master cat README