Git之深入解析在没有合适的网络或者可共享仓库情况下的git bundle打包操作

网友投稿 578 2022-05-29

了解了管理或者维护 Git 仓库、实现代码控制所需的大多数日常命令和工作流程,尝试跟了踪和提交文件的基本操作,并且掌握了暂存区和轻量级地分支及合并的威力。如果想进一步对 Git 深入学习,可以学习一些 Git 更加强大的功能,这些功能可能并不会在日常操作中使用,但在某些时候可能还是会起到一定的关键性作用。

如果还不清楚 Git 的基础使用流程、分支的管理、托管服务器的技术以及分布式工作流程等相关的技术和能力,请参考博客:

Git之深入解析Git的安装流程与初次运行Git前的环境配置;

Git之深入解析本地仓库的基本操作·仓库的获取更新和提交历史的查看撤销以及标签别名的使用;

Git之深入解析Git的杀手级特性·分支管理与变基的开发工作流以及远程分支的跟踪;

Git之深入解析如何运行自己的Git仓库托管服务器;

Git之深入解析如何使用Git的分布式工作流程与如何管理多人开发贡献的项目。

虽然已经了解了网络传输 Git 数据的常用方法(如 HTTP,SSH 等),但还有另外一种不太常见却又十分有用的方式,Git 可以将它的数据“打包”到一个文件中。

来看看一个简单的例子,假设有一个包含两个提交的仓库:

$ git log commit 9a466c572fe88b195efd356c3f2bbeccdb504102 Author: Scott Chacon Date: Wed Mar 10 07:34:10 2010 -0800 second commit commit b1ec3248f39900d2a406049d762aa68e9641be25 Author: Scott Chacon Date: Wed Mar 10 07:34:01 2010 -0800 first commit

1

2

3

4

5

6

7

8

9

10

11

12

如果想把这个仓库发送给其他人但没有其他仓库的权限,或者就是懒得新建一个仓库,就可以用 git bundle create 命令来打包:

$ git bundle create repo.bundle HEAD master Counting objects: 6, done. Delta compression using up to 2 threads. Compressing objects: 100% (2/2), done. Writing objects: 100% (6/6), 441 bytes, done. Total 6 (delta 0), reused 0 (delta 0)

Git之深入解析在没有合适的网络或者可共享仓库情况下的git bundle打包操作

1

2

3

4

5

6

然后就会有一个名为 repo.bundle 的文件,该文件包含了所有重建该仓库 master 分支所需的数据。在使用 bundle 命令时,需要列出所有希望打包的引用或者提交的区间,如果希望这个仓库可以在别处被克隆,应该像例子中那样增加一个 HEAD 引用。我们可以将这个 repo.bundle 文件通过邮件或者U盘传给别人。

另一方面,假设别人传给我们一个 repo.bundle 文件并希望在这个项目上工作,可以从这个二进制文件中克隆出一个目录,就像从一个 URL 克隆一样:

$ git clone repo.bundle repo Cloning into 'repo'... ... $ cd repo $ git log --oneline 9a466c5 second commit b1ec324 first commit

1

2

3

4

5

6

7

如果在打包时没有包含 HEAD 引用,还需要在命令后指定一个 -b master 或者其他被引入的分支,否则 Git 不知道应该检出哪一个分支。

现在假设提交了 3 个修订,并且要用邮件或者U盘将新的提交放在一个包里传回去:

$ git log --oneline 71b84da last commit - second repo c99cf5b fourth commit - second repo 7011d3d third commit - second repo 9a466c5 second commit b1ec324 first commit

1

2

3

4

5

6

首先我们需要确认我们希望被打包的提交区间,和网络协议不太一样,网络协议会自动计算出所需传输的最小数据集,而我们需要手动计算。当然可以像上面那样将整个仓库打包,但最好仅仅打包变更的部分,就是我们刚刚在本地做的 3 个提交。

为了实现这个目标,我们需要计算出差别,有很多种方式去指明一个提交区间,我们可以使用 “origin/master…master”或者“master ^origin/master”之类的方法来获取那 3 个在 master 分支而不在原始仓库中的提交。可以用 log 命令来测试:

$ git log --oneline master ^origin/master 71b84da last commit - second repo c99cf5b fourth commit - second repo 7011d3d third commit - second repo

1

2

3

4

这样就获取到我们希望被打包的提交列表,让我们将这些提交打包,可以用 git bundle create 命令,加上想用的文件名,以及要打包的提交区间:

$ git bundle create commits.bundle master ^9a466c5 Counting objects: 11, done. Delta compression using up to 2 threads. Compressing objects: 100% (3/3), done. Writing objects: 100% (9/9), 775 bytes, done. Total 9 (delta 0), reused 0 (delta 0)

1

2

3

4

5

6

当她拿到这个包时,她可以在导入到仓库之前查看这个包里包含了什么内容,bundle verify 命令可以检查这个文件是否是一个合法的 Git 包,是否拥有共同的祖先来导入:

$ git bundle verify ../commits.bundle The bundle contains 1 ref 71b84daaf49abed142a373b6e5c59a22dc6560dc refs/heads/master The bundle requires these 1 ref 9a466c572fe88b195efd356c3f2bbeccdb504102 second commit ../commits.bundle is okay

1

2

3

4

5

6

如果打包工具仅仅把最后两个提交打包,而不是三个,原始的仓库是无法导入这个包的, 因为这个包缺失了必要的提交记录。这时候 verify 的输出类似:

$ git bundle verify ../commits-bad.bundle error: Repository lacks these prerequisite commits: error: 7011d3d8fc200abe0ad561c011c3852a4b7bbe95 third commit - second repo

1

2

3

而我们的第一个包是合法的,所以可以从这个包里提取出提交。如果想查看这边包里可以导入哪些分支,同样有一个命令可以列出这些顶端:

$ git bundle list-heads ../commits.bundle 71b84daaf49abed142a373b6e5c59a22dc6560dc refs/heads/master

1

2

verify 子命令同样可以知道哪些顶端,该功能的目的是查看哪些是可以被拉入的,所以可以使用 fetch 或者 pull 命令从包中导入提交。这里我们要从包中取出 master 分支到我们仓库中的 other-master 分支:

$ git fetch ../commits.bundle master:other-master From ../commits.bundle * [new branch] master -> other-master

1

2

3

可以看到已经将提交导入到 other-master 分支,以及在这期间我们自己在 master 分支上的提交:

$ git log --oneline --decorate --graph --all * 8255d41 (HEAD, master) third commit - first repo | * 71b84da (other-master) last commit - second repo | * c99cf5b fourth commit - second repo | * 7011d3d third commit - second repo |/ * 9a466c5 second commit * b1ec324 first commit

1

2

3

4

5

6

7

8

因此,当在没有合适的网络或者可共享仓库的情况下,git bundle 很适合用于共享或者网络类型的操作。

Git 网络

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:14.6 Linux常见服务类别及功能
下一篇:centos7——Linux系统下如何连接网络图文讲解教程
相关文章