Gitでバージョン管理されたプロジェクトを扱うと、必ず .git
というディレクトリがあります。
このディレクトリを使ってGitがバージョン管理をしているらしい、ということはなんとなく知っています。でも、具体的になにが入っていて、どう管理されているんだろう?
このアドベントカレンダーは、そんな疑問からGitを深堀りしていきます。
オブジェクト
こちらとこちらの記事より、 .git/objects/
ディレクトリの中にはバイナリファイルが入っていることがわかりました。
また、 git cat-file
コマンドを利用することで、バイナリファイルのタイプと中身を見られることもご紹介しました。
(再掲)
$ git cat-file -t 9daeafb9864cf43055ae93beb0afd6c7d144bfa4 # タイプ
blob
$ git cat-file -p 9daeafb9864cf43055ae93beb0afd6c7d144bfa4 # 中身
test
$
$ git cat-file -t 58c608a3fa830548019591fc45f307554ca9a57f # タイプ
tree
$ git cat-file -p 58c608a3fa830548019591fc45f307554ca9a57f # 中身
100644 blob 9daeafb9864cf43055ae93beb0afd6c7d144bfa4 a.txt
$
$ git cat-file -t e452a0a52250236a24d1f63dd7e7fbb84d6e55c5 # タイプ
commit
$ git cat-file -p e452a0a52250236a24d1f63dd7e7fbb84d6e55c5 # 中身
tree 58c608a3fa830548019591fc45f307554ca9a57f
author user <user@machine> 1733321663 +0900
committer user <user@machine> 1733321663 +0900
commit test
$
以下では、これら3つのオブジェクトのタイプについて確認していきます。
(1) blobタイプ
blobタイプのオブジェクトは git add
コマンドを実行すると作成されました。
中にはファイルの中身だけが入っています。
ファイルの中身を書き換えたら、元のblobオブジェクトはそのまま残って、新しいblobオブジェクトが作成されます。
(2) treeタイプ
treeタイプのオブジェクトは git commit
コマンドを実行するとされました。
その中には以下の4つが記載されます。
- モード
- 「object」※3のオブジェクトタイプ
- コミットしたファイルを表すobjectディレクトリのハッシュ値
- ファイル名
なお、もしコミットしたファイルがディレクトリの中に入っていれば、以下のようになります。
- モード
- 「tree」※3のオブジェクトのタイプ
- ディレクトリを表すtreeオブジェクトのハッシュ値
- ディレクトリ名
1のモードは、「UNIXモードから取り入れた概念ですが、それほどの柔軟性はありません。」とのことです。(公式ドキュメントより)
blobファイルに対しては下記3つのモードがあるそうです。(ディレクトリについては詳しい説明がありませんでした)
- 100644:通常のファイル
- 100755:実行可能ファイル
- 120000:シンボリックリンク
(3) commitタイプ
commitのオブジェクトも、treeオブジェクトと同様に git commit
コマンドを実行するとされました。
中にはコミットに関するメタデータが入っています。
具体的には下記の5つの情報です。
- treeオブジェクトのハッシュ値(コミットが作成された時点のスナップショット)
- 作者
- コミッター
- 空行
- コミットメッセージ
commitオブジェクトのハッシュ値は git log
コマンドで確認できます。
$ git log
commit e452a0a52250236a24d1f63dd7e7fbb84d6e55c5 (HEAD -> main)
Author: user <user@machine>
Date: Wed Dec 4 23:14:23 2024 +0900
commit test
$
以上をまとめると、Gitは管理対象のファイルを下記の3つに分けて管理していることがわかります。
- ファイルの中身(blobオブジェクト)
- コミット時点のスナップショット(treeオブジェクト)
- コミットのメタデータ(commitオブジェクト)
そして、この3つの情報は、 git log
コマンド -> commitオブジェクト -> treeオブジェクト -> blobオブジェクトと辿っていくことが出来ます。
参考書籍・参考資料
- エンジニアのためのGitの教科書[上級編] Git内部の仕組みを理解する
- git 公式ドキュメント