-
1. Gitの基本
- 1.1 バージョン管理について
- 1.2 Gitの簡単な歴史
- 1.3 Gitとは何か?
- 1.4 コマンドライン
- 1.5 Gitのインストール
- 1.6 Gitの初回セットアップ
- 1.7 ヘルプの取得
- 1.8 まとめ
-
2. Gitの基本
- 2.1 Gitリポジトリの取得
- 2.2 リポジトリへの変更の記録
- 2.3 コミット履歴の表示
- 2.4 元に戻す
- 2.5 リモートでの作業
- 2.6 タグ付け
- 2.7 Gitエイリアス
- 2.8 まとめ
-
3. Gitのブランチ機能
- 3.1 ブランチの基本
- 3.2 基本的なブランチとマージ
- 3.3 ブランチ管理
- 3.4 ブランチングワークフロー
- 3.5 リモートブランチ
- 3.6 リベース
- 3.7 まとめ
-
4. サーバー上のGit
- 4.1 プロトコル
- 4.2 サーバー上でのGitの準備
- 4.3 SSH公開鍵の生成
- 4.4 サーバーのセットアップ
- 4.5 Git Daemon
- 4.6 Smart HTTP
- 4.7 GitWeb
- 4.8 GitLab
- 4.9 サードパーティホストオプション
- 4.10 まとめ
-
5. 分散Git
- 5.1 分散ワークフロー
- 5.2 プロジェクトへの貢献
- 5.3 プロジェクトの管理
- 5.4 まとめ
-
6. GitHub
- 6.1 アカウントのセットアップと設定
- 6.2 プロジェクトへの貢献
- 6.3 プロジェクトの管理
- 6.4 組織の管理
- 6.5 GitHubのスクリプト化
- 6.6 まとめ
-
7. Gitのツール
- 7.1 リビジョン選択
- 7.2 対話的なステージング
- 7.3 スタッシュとクリーニング
- 7.4 作業の署名
- 7.5 検索
- 7.6 履歴の書き換え
- 7.7 Resetの謎を解く
- 7.8 高度なマージ
- 7.9 Rerere
- 7.10 Gitを使ったデバッグ
- 7.11 Submodules
- 7.12 バンドリング
- 7.13 Replace
- 7.14 認証情報の保存
- 7.15 まとめ
-
8. Gitのカスタマイズ
- 8.1 Gitの設定
- 8.2 Git属性
- 8.3 Gitフック
- 8.4 Git強制ポリシーの例
- 8.5 まとめ
-
9. Gitと他のシステム
- 9.1 Gitをクライアントとして
- 9.2 Gitへの移行
- 9.3 まとめ
-
10. Gitの内部構造
- 10.1 PlumbingとPorcelain
- 10.2 Gitオブジェクト
- 10.3 Gitリファレンス
- 10.4 Packfiles
- 10.5 Refspec
- 10.6 転送プロトコル
- 10.7 メンテナンスとデータ復旧
- 10.8 環境変数
- 10.9 まとめ
-
A1. 付録A: 他の環境でのGit
- A1.1 グラフィカルインターフェース
- A1.2 Visual StudioでのGit
- A1.3 Visual Studio CodeでのGit
- A1.4 IntelliJ / PyCharm / WebStorm / PhpStorm / RubyMineでのGit
- A1.5 Sublime TextでのGit
- A1.6 BashでのGit
- A1.7 ZshでのGit
- A1.8 PowerShellでのGit
- A1.9 まとめ
-
A2. 付録B: アプリケーションへのGitの組み込み
- A2.1 コマンドラインGit
- A2.2 Libgit2
- A2.3 JGit
- A2.4 go-git
- A2.5 Dulwich
-
A3. 付録C: Gitコマンド
- A3.1 セットアップと設定
- A3.2 プロジェクトの取得と作成
- A3.3 基本的なスナップショット
- A3.4 ブランチとマージ
- A3.5 プロジェクトの共有と更新
- A3.6 調査と比較
- A3.7 デバッグ
- A3.8 パッチ適用
- A3.9 メール
- A3.10 外部システム
- A3.11 管理
- A3.12 Plumbingコマンド
7.13 Git Tools - Replace
Replace
これまで強調してきたように、Gitのオブジェクトデータベース内のオブジェクトは変更できませんが、Gitはデータベース内のオブジェクトを他のオブジェクトで「置き換えるふりをする」興味深い方法を提供します。
`replace` コマンドを使用すると、Git内のオブジェクトを指定して、「このオブジェクトを参照するたびに、それは`別の`オブジェクトであると見なす」ということができます。これは、例えば `git filter-branch` を使って履歴全体を再構築することなく、履歴内のあるコミットを別のコミットに置き換える場合に最も一般的に役立ちます。
たとえば、巨大なコード履歴があり、リポジトリを新規開発者向けの短い履歴と、データマイニングに関心のある人向けのより長く大きな履歴の2つに分割したいとします。新しい履歴の最初のコミットを古い履歴の最新のコミットに「置き換える」ことで、一方の履歴をもう一方に結合できます。これは、通常、履歴を結合するために新しい履歴のすべてのコミットを書き換える必要がないため(親がSHA-1に影響するため)、便利です。
これを試してみましょう。既存のリポジトリを、最近の履歴と過去の履歴の2つのリポジトリに分割し、`replace` を介して最近のリポジトリのSHA-1値を変更することなく、それらを再結合する方法を見ていきます。
5つの単純なコミットを含むシンプルなリポジトリを使用します。
$ git log --oneline
ef989d8 Fifth commit
c6e1e95 Fourth commit
9c68fdc Third commit
945704c Second commit
c1822cf First commit
これを2つの履歴ラインに分割したいとします。1つはコミット1からコミット4までのラインで、これが過去の履歴になります。もう1つのラインはコミット4と5だけで、これが最近の履歴になります。

さて、過去の履歴を作成するのは簡単です。履歴にブランチを作成し、そのブランチを新しいリモートリポジトリの`master`ブランチにプッシュするだけです。
$ git branch history c6e1e95
$ git log --oneline --decorate
ef989d8 (HEAD, master) Fifth commit
c6e1e95 (history) Fourth commit
9c68fdc Third commit
945704c Second commit
c1822cf First commit

これで、新しい`history`ブランチを新しいリポジトリの`master`ブランチにプッシュできます。
$ git remote add project-history https://github.com/schacon/project-history
$ git push project-history history:master
Counting objects: 12, done.
Delta compression using up to 2 threads.
Compressing objects: 100% (4/4), done.
Writing objects: 100% (12/12), 907 bytes, done.
Total 12 (delta 0), reused 0 (delta 0)
Unpacking objects: 100% (12/12), done.
To git@github.com:schacon/project-history.git
* [new branch] history -> master
これで履歴が公開されました。次に難しいのは、最近の履歴を小さくするために短縮することです。一方のコミットをもう一方の同等のコミットで置き換えることができるように、重複が必要です。そのため、これをコミット4と5だけに短縮します(コミット4が重複します)。
$ git log --oneline --decorate
ef989d8 (HEAD, master) Fifth commit
c6e1e95 (history) Fourth commit
9c68fdc Third commit
945704c Second commit
c1822cf First commit
この場合、履歴を拡張する方法の手順を記したベースコミットを作成すると便利です。これにより、他の開発者が短縮された履歴の最初のコミットに到達したときに、さらに履歴が必要な場合に何をすべきかを知ることができます。そこで、最初のコミットオブジェクトを指示付きのベースポイントとして作成し、残りのコミット(4と5)をその上にリベースします。
そのためには、分割するポイントを選択する必要があります。ここでは3番目のコミット、SHA-1でいうと`9c68fdc`がそれにあたります。したがって、私たちのベースコミットはそのツリーに基づいて作成されます。`commit-tree`コマンドを使用してベースコミットを作成できます。これはツリーを受け取り、新しい親なしコミットオブジェクトのSHA-1を返します。
$ echo 'Get history from blah blah blah' | git commit-tree 9c68fdc^{tree}
622e88e9cbfbacfb75b5279245b9fb38dfea10cf
注
|
`commit-tree`コマンドは、「plumbing(配管)」コマンドと呼ばれる一連のコマンドの1つです。これらは通常、直接使用されることを意図していませんが、**他の**Gitコマンドによってより小さなジョブを実行するために使用されます。今回のような奇妙なことを行う場合、これらのコマンドは非常に低レベルなことを可能にしますが、日常的な使用を意図したものではありません。plumbingコマンドの詳細については、PlumbingとPorcelainで読むことができます。 |

これでベースコミットができたので、残りの履歴を`git rebase --onto`でその上にリベースできます。`--onto`引数は、`commit-tree`から取得したSHA-1で、リベースポイントは3番目のコミット(保持したい最初のコミット`9c68fdc`の親)になります。
$ git rebase --onto 622e88 9c68fdc
First, rewinding head to replay your work on top of it...
Applying: fourth commit
Applying: fifth commit

これで、最近の履歴を、必要に応じて履歴全体を再構築する方法の指示を含む使い捨てのベースコミットの上に書き換えました。その新しい履歴を新しいプロジェクトにプッシュすると、人々がそのリポジトリをクローンしたときに、最新の2つのコミットと、指示付きのベースコミットのみが表示されるようになります。
では、ここで役割を切り替えて、プロジェクトを初めてクローンする人で、全履歴が必要な場合を考えます。この切り詰められたリポジトリをクローンした後で履歴データを取得するには、履歴リポジトリ用の2番目のリモートを追加してフェッチする必要があります。
$ git clone https://github.com/schacon/project
$ cd project
$ git log --oneline master
e146b5f Fifth commit
81a708d Fourth commit
622e88e Get history from blah blah blah
$ git remote add project-history https://github.com/schacon/project-history
$ git fetch project-history
From https://github.com/schacon/project-history
* [new branch] master -> project-history/master
これで、共同作業者は`master`ブランチに最近のコミット、`project-history/master`ブランチに過去のコミットを持つことになります。
$ git log --oneline master
e146b5f Fifth commit
81a708d Fourth commit
622e88e Get history from blah blah blah
$ git log --oneline project-history/master
c6e1e95 Fourth commit
9c68fdc Third commit
945704c Second commit
c1822cf First commit
それらを結合するには、置き換えたいコミットと、それで置き換えたいコミットを引数にして、`git replace`を呼び出すだけです。つまり、`master`ブランチの「4番目の」コミットを、`project-history/master`ブランチの「4番目の」コミットに置き換えたいのです。
$ git replace 81a708d c6e1e95
さて、`master`ブランチの履歴を見ると、このようになっているように見えます。
$ git log --oneline master
e146b5f Fifth commit
81a708d Fourth commit
9c68fdc Third commit
945704c Second commit
c1822cf First commit
素晴らしいでしょう? upstreamのすべてのSHA-1を変更することなく、履歴内の1つのコミットをまったく別のコミットに置き換えることができ、通常のツール(`bisect`、`blame`など)は期待通りに動作します。

興味深いことに、置き換えた`c6e1e95`コミットデータを使用しているにもかかわらず、SHA-1としては`81a708d`と表示されます。`cat-file`のようなコマンドを実行しても、置き換えられたデータが表示されます。
$ git cat-file -p 81a708d
tree 7bc544cf438903b65ca9104a1e30345eee6c083d
parent 9c68fdceee073230f19ebb8b5e7fc71b479c0252
author Scott Chacon <schacon@gmail.com> 1268712581 -0700
committer Scott Chacon <schacon@gmail.com> 1268712581 -0700
fourth commit
`81a708d`の実際の親は、ここで示されている`9c68fdce`ではなく、プレースホルダーコミット(`622e88e`)であったことを思い出してください。
もう一つ興味深いのは、このデータが私たちの参照に保持されていることです。
$ git for-each-ref
e146b5f14e79d4935160c0e83fb9ebe526b8da0d commit refs/heads/master
c6e1e95051d41771a649f3145423f8809d1a74d4 commit refs/remotes/history/master
e146b5f14e79d4935160c0e83fb9ebe526b8da0d commit refs/remotes/origin/HEAD
e146b5f14e79d4935160c0e83fb9ebe526b8da0d commit refs/remotes/origin/master
c6e1e95051d41771a649f3145423f8809d1a74d4 commit refs/replace/81a708dd0e167a3f691541c7a6463343bc457040
これは、私たちの置き換えを他の人と簡単に共有できることを意味します。なぜなら、これをサーバーにプッシュでき、他の人が簡単にダウンロードできるからです。これは、ここで説明した履歴の結合シナリオではあまり役に立ちません(結局誰もが両方の履歴をダウンロードすることになるので、なぜ分けるのか?)。しかし、他の状況では役立つことがあります。