在Git 1.7.0中稀疏结账?

时间:2010-02-25 18:30:20

标签: git version-control sparse-checkout

使用Git 1.7.0中的新sparse checkout feature,是否可以像在SVN中那样获取子目录的内容?我找到了this example,但它保留了完整的目录结构。想象一下,我只想要'perl'目录的内容,而没有名为'perl'的实际目录。

- 编辑 -

示例:

我的git存储库包含以下路径

repo/.git/
repo/perl/
repo/perl/script1.pl
repo/perl/script2.pl
repo/images/
repo/images/image1.jpg
repo/images/image2.jpg
repo/doc/
repo/doc/readme.txt
repo/doc/help.txt

我想要的是能够从上面的存储库中产生这种布局:

repo/.git/
repo/script1.pl
repo/script2.pl

然而,使用当前的稀疏结账功能,似乎只能获得

repo/.git/
repo/perl/script1.pl
repo/perl/script2.pl

这不是我想要的。

7 个答案:

答案 0 :(得分:26)

您仍然需要克隆整个存储库,该存储库将包含所有文件。您可以使用--depth标记仅检索有限数量的历史记录。

克隆存储库后,读取树技巧将存储库的“视图”限制为仅.git/info/sparse-checkout文件中的文件或目录。

我写了一个快速脚本来帮助管理稀疏性,因为此刻它有点不友好:

#!/bin/sh
echo > .git/info/sparse-checkout
for i in "$@"
do
    echo "$i" >> .git/info/sparse-checkout
done
git read-tree -m -u HEAD

如果您将此脚本git-sparse.sh保存到通过调用git --exec-path报告的路径中,那么您可以运行git sparse foo/ bar/以仅“结帐”foo和bar目录,或{{1让一切恢复原状。

答案 1 :(得分:16)

简短的回答是否定的。 Git将所有文件视为一个单元。

我建议您将存储库分解为逻辑块。 perl,images和docs的单独一个。如果您还需要维护超级回购样式,则可以创建由Submodules组成的回购。

答案 2 :(得分:7)

富查克的答案很接近,但却错过了一步。您需要显式启用稀疏结帐:

  

git config core.sparsecheckout true

此博客文章概述了所有步骤:

  

http://blog.quilitz.de/2010/03/checkout-sub-directories-in-git-sparse-checkouts/comment-page-1/#comment-3146

答案 3 :(得分:5)

现在没有详细说明你为什么要这样做,你的问题可以(可能)通过符号链接/快捷方式轻松解决。

回答这个问题 - 不,并且有一个有意义的理由。即使使用“稀疏结账”,也可以下载回购的整个历史记录。澄清为什么这是必要的 - 否则跟踪重命名的文件将是一个痛苦的...脖子。想象一下,您将文件/repo_root/asd/file1.cpp移动到/repo_root/fgh/file1.cpp - 现在如果您只下载了/repo_root/fgh增量,则不会知道file1.cpp。所以这意味着你必须下载所有的增量。但是你有一个完整的存储库;不仅仅是文件夹的剪切,因此只有/rero_root/fgh文件夹本身不是回购。结帐时这可能听起来不重要,但是当你提交时,git可能不知道工作正常。

解决方法:如果你真的想要,你可以创建一个以这种方式调用git-checkout的脚本(对于sh shell,不应该很难生成windows的批处理): / p>

!/bin/sh
curDir=`pwd`
cd $2
git-checkout $1
cp -R $3/* $4
cd $curDir

这里的第一个参数是checkout的分支,第二个是repo当前存在的文件夹,第三个是你想要真正使用的子目录,第四个是你要复制它的位置。 / p>

警告:我的shell技能几乎不存在,所以在测试后使用它。重新创建这个脚本的反向并不难,复制回来的东西,以便它可以提交给仓库。

答案 4 :(得分:3)

git filter-branch --subdirectory-filter是您所需要的,请参阅Detach (move) subdirectory into separate Git repository

这是一个小的bash脚本。

这将首先制作原始仓库的工作副本,然后使用子目录过滤器过滤分支,以获得您想要的。

#!/bin/bash
#
# git-subdir.sh
#
git clone --no-hardlinks $1 $2

cd $2

git filter-branch --subdirectory-filter $2 --prune-empty --tag-name-filter cat HEAD -- --all

git reset --hard

git remote rm origin

refbak=$(git for-each-ref --format="%(refname)" refs/original/)

if [ -n "$refbak" ];then
    echo -n $refbak | xargs -n 1 git update-ref -d
fi

git reflog expire --expire=now --all

git repack -ad

git gc --aggressive --prune=now

用于问题中的示例,git-subdir.sh repo perl可行。

答案 5 :(得分:2)

您可以尝试编织 - 它会跟踪遥控器,同时将它们与路径匹配。 https://github.com/evilchelu/braid/wiki

答案 6 :(得分:0)

您尝试执行的操作似乎是重命名目录树,以使您的文件最终位于不同的位置。在我看来,你要做的是一个代码/项目管理的反模板,有两个方面:模块分类(java节点下的java位,perl节点下的perl),以及一个包含不同位置文件的项目从开发人员可视化的地方开始。由于git维护目录内容的哈希以查看更改内容,因此这也会破坏git。

Daemeon Reiydelle