我有一个Git存储库,其中有几个子模块。如何在git子模块init运行后列出所有子模块的名称?
git子模块foreach命令可以回显子模块的名称,但这只有在它们被签出后才有效,而在init步骤之后没有发生。在签出之前,链中还有更多的步骤需要执行,我不希望将子模块的名称硬连接到脚本中。
那么是否存在一个Git命令来获取所有当前已注册但尚未签出的子模块的名称?
我有一个Git存储库,其中有几个子模块。如何在git子模块init运行后列出所有子模块的名称?
git子模块foreach命令可以回显子模块的名称,但这只有在它们被签出后才有效,而在init步骤之后没有发生。在签出之前,链中还有更多的步骤需要执行,我不希望将子模块的名称硬连接到脚本中。
那么是否存在一个Git命令来获取所有当前已注册但尚未签出的子模块的名称?
当前回答
按名称列出所有子模块:
Git子模块——quiet foreach——递归echo $name
其他回答
Tino的评论显示了一个很好的替代方案:
git config --null --file .gitmodules --get-regexp "\.path\$" | \
sed -nz "s/^[^\\n]*\n//p" | \
xargs --null rm -v;
但是——null/-z选项不适用于BusyBox上的sed或xargs。然后,这是我能想到的最好的替代方案(xargs不会在空格上中断):
grep "^\s*path\s*=" .gitmodules | \
sed -e "s/.*=\s*\"*//" -e "s/\"*\s*\$//" | \
xargs -n 1 -I mark rm -v "mark";
但它有一个缺点,因为xargs调用时不带——null,而我们使用的是-I标记,它正在吞噬前导空白。要修复它,不要使用xargs并将名称写入文件:
grep "^\s*path\s*=" .gitmodules | \
sed -e "s/.*=\s*\"*//" -e "s/\"*\s*\$//" > tempfilefordanmxargsmotherbroked.txt;
while IFS= read -r line; do
echo ".$line."
done <"tempfilefordanmxargsmotherbroked.txt"
rm "tempfilefordanmxargsmotherbroked.txt"
更正:BusyBox上的xargs没有删除前导空白,但完整版本可以。
获取路径
grep url .gitmodules | sed 's/.*= //'
在回购中得到名字
grep path .gitmodules | sed 's/.*= //'
你可以使用:
git submodule | awk '{ print $2 }'
你可以使用与git子模块init使用自身相同的机制,即查看.gitmodules。该文件枚举每个子模块路径及其引用的URL。
例如,从存储库的根目录,cat .gitmodules将把内容打印到屏幕上(假设你有cat)。
因为.gitmodule文件有Git配置格式,你可以使用Git配置来解析这些文件:
git config --file .gitmodules --name-only --get-regexp path
将显示所有子模块条目,并与
git config --file .gitmodules --get-regexp path | awk '{ print $2 }'
您只能得到子模块路径本身。
Git配置允许指定配置文件。 gitmodules是一个配置文件。
因此,在“使用空格作为cut命令的分隔符”的帮助下:
git config --file=.gitmodules --get-regexp ^^submodule.*\.path$ | cut -d " " -f 2
它将只列出路径,每个声明的子模块一个路径。
Tino在评论中指出:
这对于包含空格的子模块无效。 子模块路径可能包含换行符,如 Git子模块添加https://github.com/hilbix/bashy.git“子模块” Git mv 'sub module' $'sub\nmodule'
作为一个更强大的替代方案,Tino提出:
Git config -z——file . Git modules——get-regexp ` \。路径$' | \ Sed -nz 's/^[^\n]*\n//p' | \ Tr '\0' '\n' 对于有换行符的路径(可以用git mv创建),去掉| tr '\0' '\n',使用类似于…| while IFS= " read -d " path;做……用于使用bash进行进一步处理。 这需要一个现代bash,它能理解read -d "(不要忘记-d和"之间的空格)。