Timetombs

泛义的工具是文明的基础,而确指的工具却是愚人的器物

66h / 117a
,更新于 2025-01-05T12:19:33Z+08:00 by   1072b1b

[Git] 内容寻址文件系统

版权声明 - CC BY-NC-SA 4.0

不要感觉奇怪,为什么介绍Git的文章里面怎么会出现一个文件系统,还内容寻址,这都是什么啊?其实内容寻址文件系统(content-addressable filesystem)才是git的底层核心,而我们平时使用的commitbranchcheckout等等命令仅仅只是在上层包装成了VCS(version control system)的样子。

文件系统是用来在物理存储设备上为上层应用提供抽象的文件管理功能的。比如常见的fat、fat32、ntfs、ext4等等。

内容寻址指的是基于文件的内容来定位文件,git使用SHA11来唯一标识文件的。而非文件系统中常用的基于文件的位置(location),比如/bin/sh这样的路径位置。

那么Git的底层就就构建在这样一个基于SHA作为key的KV文件系统。它所有的秘密都藏在.git目录下,我们这就来一探究竟。

1 .git 目录

初始化一个新的仓库git init temp,然后查看一下默认的目录结构tree -p .git

.git
├── [-rw-r--r--]  HEAD
├── [-rw-r--r--]  config
├── [-rw-r--r--]  description
├── [drwxr-xr-x]  hooks
│   ├── [-rwxr-xr-x]  applypatch-msg.sample
│   ├── [-rwxr-xr-x]  commit-msg.sample
│   ├── [-rwxr-xr-x]  fsmonitor-watchman.sample
│   ├── [-rwxr-xr-x]  post-update.sample
│   ├── [-rwxr-xr-x]  pre-applypatch.sample
│   ├── [-rwxr-xr-x]  pre-commit.sample
│   ├── [-rwxr-xr-x]  pre-merge-commit.sample
│   ├── [-rwxr-xr-x]  pre-push.sample
│   ├── [-rwxr-xr-x]  pre-rebase.sample
│   ├── [-rwxr-xr-x]  pre-receive.sample
│   ├── [-rwxr-xr-x]  prepare-commit-msg.sample
│   ├── [-rwxr-xr-x]  push-to-checkout.sample
│   └── [-rwxr-xr-x]  update.sample
├── [drwxr-xr-x]  info
│   └── [-rw-r--r--]  exclude
├── [drwxr-xr-x]  objects
│   ├── [drwxr-xr-x]  info
│   └── [drwxr-xr-x]  pack
└── [drwxr-xr-x]  refs
    ├── [drwxr-xr-x]  heads
    └── [drwxr-xr-x]  tags

8 directories, 17 files
  1. HEAD文件:当前工作区的ref信息,文件内容默认是:ref: refs/heads/main,代表当前位于main分支。
  2. config文件:当前repository的配置信息。
  3. description文件:GitWeb 程序使用的描述说明,这里不关心。
  4. hooks目录:客户端或服务端的hook脚本。
  5. info/exclude文件:配置不希望被记录在.gitignore文件中的忽略模式。
  6. objects目录:存放核心对象(blob、tree、commit)数据的位置。
  7. refs目录:指向objects数据的指针数据。
  8. index文件:这个文件还未创建。用来保存暂存区的信息。

我们通过操作git的底层命令来直接生成一个commit,以此来介绍下其运行机制。

2 objects目录

objects目录存储着git的4个核心对象:blob、tree、commit和tag。

  1. blob:文件内容:存储源码、文本或者二进制格式文件。
  2. tree:文件信息:目录信息、文件名、文件权限属性信息以及文件名对应的blob对象的hash。
  3. commit:提交信息:parent commit、committer、auther、time、commit message以及tree对象的hash等信息。
  4. tag:标签信息:tagger、time tag message以及commit对象的hash。

2.1 blob object

首先使用git hash-object2手动生成一个blob对象,代表我们要保存的文件内容,这个命令会返回根据文件内容+附加信息生成的SHA1。

echo 'this is file content 1' | git hash-object -w -t blob --stdin
068b6574adc8d309c1ff2438ad82b63197144a63

tree -f .git/objects
.git/objects
├── .git/objects/06
│   └── .git/objects/06/8b6574adc8d309c1ff2438ad82b63197144a63
├── .git/objects/info
└── .git/objects/pack

命令解释:

  1. -w:指示git把生成的blob对象持久化到objects目录下,路径是SHA1的前2个字符作为目录名,后38个字符作为文件名。
  2. -t blob:指定object类型为blob。
  3. --stdin:指示git把从stdin3中读取的内容作为文件内容。

需要注意的是git hash-object并不是直接对内容计算SHA1,而是会添加一个blob 23\0的前缀,其中23是文件内容的byte长度。等效于:

echo 'blob 23\0this is file content 1' | shasum -a 1
068b6574adc8d309c1ff2438ad82b63197144a63  -

所以如果文件内容保持不变,那么无论你执行多少次git hash-object,都只会生成一个blob对象。那么比如我们的git仓库中有两个文件的文件内容完全相同,但是目录名或者文件名不同,是不是只需要存储一份就可以了呢?感兴趣的可以自己试一试。

答案是只需存储一份blob对象即可。

当你用cat尝试去读取文件内容时,你会发现无法读取,这是因为它本身是一个采用deflate压缩算法的zlib stream4,而非文本文件。

hexdump -C .git/objects/06/8b6574adc8d309c1ff2438ad82b63197144a63
00000000  78 01 4b ca c9 4f 52 30  32 66 28 c9 c8 2c 56 00  |x.K..OR02f(..,V.|
00000010  a2 b4 cc 9c 54 85 e4 fc  bc 92 d4 bc 12 05 43 2e  |....T.........C.|
00000020  00 a2 8a 0a 0f                                    |.....|
00000025

这时我们可以使用git cat-file5来读取它。

# -p 打印blob的内容
git cat-file -p 068b6574adc8d309c1ff2438ad82b63197144a63
this is file content 1

# -s 返回byte长度
git cat-file -s 068b6574adc8d309c1ff2438ad82b63197144a63
23

# -t 返回object的类型
git cat-file -t 068b6574adc8d309c1ff2438ad82b63197144a63
blob

2.2 tree object

有了blob存储的文件内容,那么目录名和文件名信息怎么存储呢?答案就是tree对象。这里我们使用git update-index6index file下文介绍)和git write-tree7来手工创建一个tree对象。

# 指定文件属性100644、blob的hash和目录名、文件名信息
git update-index --add --cacheinfo 100644 068b6574adc8d309c1ff2438ad82b63197144a63 test-dir/blob-file.txt

# 生成一个tree对象,返回hash
git write-tree
73e9fd0cc8f2199bc05ce95cbc0bef2b38e56345

# 查看一下这个tree对象(它代表的是test-dir这个文件夹),它包含另外一个tree对象
git cat-file -p 73e9fd0cc8f2199bc05ce95cbc0bef2b38e56345
040000 tree 9ec1a2094d5084786ba165358deaa8e68cba8314    test-dir

# 查看另一个tree对象,这次它指向的就是上面创建的blob文件,并且带上了文件名。
git cat-file -p 9ec1a2094d5084786ba165358deaa8e68cba8314
100644 blob 068b6574adc8d309c1ff2438ad82b63197144a63    blob-file.txt

# 再来看一下目前所有的对象
tree .git/objects
.git/objects
├── 06
│   └── 8b6574adc8d309c1ff2438ad82b63197144a63
├── 73
│   └── e9fd0cc8f2199bc05ce95cbc0bef2b38e56345
├── 9e
│   └── c1a2094d5084786ba165358deaa8e68cba8314
├── info
└── pack

2.3 commit object

至此有了代表文件内容的blob对象和代表directory和file信息的tree对象,那么是时候创建一个commit了。这里我们使用git commit-tree8命令。

# 指定一个commit消息和tree对象,返回了commit对象hash
echo 'first commit form manual blob tree and commit' | git commit-tree 73e9fd0cc8f2199bc05ce95cbc0bef2b38e56345
bf84aa3517c5a51b50289f9ce17d7757b96a39dc

# 查看一下这个commit对象的信息
git cat-file -p bf84aa3517c5a51b50289f9ce17d7757b96a39dc
tree 73e9fd0cc8f2199bc05ce95cbc0bef2b38e56345
author lnh <lnhdyx@outlook.com> 1619526360 +0800
committer lnh <lnhdyx@outlook.com> 1619526360 +0800

first commit form manual blob tree and commit

2.4 tag object

这里我们使用git tag9命令。

# 当使用 -a 或者 -s 参数时,才会生成tag对象
git tag -a v0.1 bf84aa3517c5a51b50289f9ce17d7757b96a39dc -m 'test tag'

# 查看这个tag的hash
cat .git/refs/tags/v0.1
8be7fa8832efbcabc48625ee9d651b6cd9f20858

# 查看一下v0.1这个tag
# 或者根据tag的hash
git cat-file -p v0.1
git cat-file -p 8be7fa8832efbcabc48625ee9d651b6cd9f20858
object bf84aa3517c5a51b50289f9ce17d7757b96a39dc
type commit
tag v0.1
tagger lnh <lnhdyx@outlook.com> 1619574383 +0800

test tag

3 refs目录

commit的hash是很不方便记忆和直接操作使用的,我们通常需要指定一些有意义的名字。refs目录就是用来存储这些名字和commit hash的ref用的。主要有一下三个目录:

  1. heads目录:分支ref信息。
  2. tags目录:标签ref信息。
  3. remotes目录:远程的分支ref和标签ref信息。

3.1 head ref

这里我们使用git update-ref10命令来让test分支指向上面的commit。

本质上是在refs/heads目录下添加一个以分支名test命名的文件,内容指向一个commit的hash。

# 让test分支指向上面创建的commit
git update-ref refs/heads/test bf84aa3517c5a51b50289f9ce17d7757b96a39dc

# 查看一下存储的文件
cat .git/refs/heads/test
bf84aa3517c5a51b50289f9ce17d7757b96a39dc

# -d 删除一个分支,仅仅只是删除ref,而不会删除commit
git branch -d main
Deleted branch main (was bf84aa3).

3.2 tag ref

除了像上文2.4 tag object那样生成一个tag对象的办法外,git还可以只添加一个tag的ref而不生成taf对象(本质和上面的head ref一样,向refs/tags目录写入一个文件即可)。

# 添加一个不生成tag对象的tag ref
git tag no-object-tag 8be7fa8832efbcabc48625ee9d651b6cd9f20858
# 或者
echo '8be7fa8832efbcabc48625ee9d651b6cd9f20858' > .git/refs/tags/no-object-tag

# 查看一下存储的文件
cat .git/refs/tags/no-object-tag
bf84aa3517c5a51b50289f9ce17d7757b96a39dc

# -d 删除一个tag,也仅仅只是删除ref,而不会删除commit
git tag -d no-object-tag
Deleted tag 'no-object-tag' (was 8be7fa8)

3.3 remotes目录

记录者远程的3.1 head ref3.2 tag ref4 HEAD文件

4 HEAD文件

HEAD文件存储着当前工作区的ref信息,可以是指向branch、tag或者某一个commit hash。可以使用底层命令git symbolic-ref11命令来让操作这个文件。

# 更新一下HEAD文件
git symbolic-ref HEAD refs/heads/test
# 查看HEAD文件,至此,一个commit算是完整的提交到分支上去了。
cat .git/HEAD
ref: refs/heads/test

# 那么我们看一下当前状态。咦,怎么提示我删除了一个文件呢?
# 这是因为我们上面的commit没有经过当前的工作区和暂存区,而是直接存储到了本地的repo中。
$ git status
On branch test
Changes not staged for commit:
  (use "git add/rm <file>..." to update what will be committed)
  (use "git restore <file>..." to discard changes in working目录)
        deleted:    test-dir/blob-file.txt

no changes added to commit (use "git add" and/or "git commit -a")

# 我们清空一下暂存区即可。
git checkout .

# 然后你就看到我们刚刚手工commit的文件内容已经出现了。
tree
.
└── test-dir
    └── blob-file.txt
# 查看一下文件内容
cat test-dir/blob-file.txt
this is file content 1

5 index文件

2.2 tree object创建tree时,我们先向index文件中更新了一些信息,然后才基于index创建出了第一个tree对象。这个index的文件中存储的是暂存区的数据。所有需要添加到仓库区(.git目录)中的数据都必须先进入到index中,这个文件也是一个二进制文件12。可是使用git ls-files13命令来查看暂存区的内容。

git ls-files --stage
100644 068b6574adc8d309c1ff2438ad82b63197144a63 0       test-dir/blob-file.txt

6 summary

通过借助底层命令手动创建一个commit分析了一下git的底层存储机制:blob、tree、commit、tag以及ref和index文件。如有错误之处欢迎指正!

7 reference

上一篇 : [Git] .gitattributes