在没有全局忽略的情况下递归忽略SVN中的文件或目录模式的最佳方法

时间:2011-05-24 21:48:12

标签: svn bash version-control

很多时候,作为编译或打开项目文件的结果,会创建临时文件。我希望SVN忽略(而不是版本)创建它们的这些文件而不必全局忽略它们,因为不同的程序/进程可能在其他地方创建它们并且它们可能需要版本控制(相同的扩展名不同的文件类型)。使用Bash和SVN做到这一点的最佳方法是什么?我创建的以下bash脚本是否足够?

# Recursively find, remove, and set svn:ignore on files or folders. 
# For safety, you must perform an svn status and an svn commit yourself.
#
# $1 is the pattern to ignore
# $2 is the starting directory (optional)
# $3 is a flag to remove the ignored files locally (optional)
#
# Example: recursive_svn_ignore "*.bak" . --remove
#
recursive_svn_ignore () 
{
    cd "$2";
    svn info > /dev/null 2>&1;
    if [ "$?" -ne "0" ]; then
      echo "Skipping `pwd` because it is not under version control.";
      return 1;
    fi;
    # Check if pattern matches
    ls -l $1 > /dev/null 2>&1;
    if [ "$?" = "0" ]; then
        for f in "$1"; do
            if [ "$3" = "--remove" ]; then
              # Remove file from working directory and repository 
              svn rm --force -q $f; 
            else
              # Remove file from repository only
              svn rm --force --keep-local $f;
            fi;
        done
        echo "Adding "$1" in `pwd` to svn:ignore list";
        svn propget svn:ignore . > svnignore.tempfile;
        echo "$1" >> svnignore.tempfile;
        svn propset -q svn:ignore -F svnignore.tempfile .;
        rm svnignore.tempfile;
    fi;
    # Recurse
    for d in *
    do
        if [ -d "$d" ]; then
            ( recursive_svn_ignore "$1" "$d" )
        fi;
    done
}

正如@ david-van-brink所建议的,使用find通过消除递归简化了很多事情。这就是我想出来的。

#!/bin/bash
# Recursively find and set svn:ignore on files or folders.
# If the file was already version it will be removed from the repository
# and keep the local copy (can be overridden).
# This does not perform the final commit so you can review the changes
# using svn status.
#
# $1 pattern to ignore
# $2 remove the ignored files locally as well (optional)
#
# Example: find_svn_ignore "*.bak" . --remove
#
for a in `find . -name $1`
do
        svn info ${a%/*} > /dev/null 2>&1;
        if [ "$?" -ne "0" ]; then
                echo "Skipping ${a%/*} because it is not under version control.";
                continue;
        fi;
        echo "Ignoring ${a##*/} in ${a%/*}";
        svn propget svn:ignore ${a%/*} > svnignore.tempfile;
        echo "$1" >> svnignore.tempfile;
        svn propset -q svn:ignore -F svnignore.tempfile ${a%/*};
        rm svnignore.tempfile;
        if [ "$2" = "--remove" ]; then
                # Remove file from working directory and repository 
                svn rm --force -q ${a##*/};
        else   
                # Remove file from repository only
                svn rm --force --keep-local ${a##*/};
        fi;
done

2 个答案:

答案 0 :(得分:2)

使用“find”可能更好。类似的东西:

for a in `find . -name '*.bak'`
 do
     echo something $a
     echo something else to $a
 done

或者将模式作为参数传递给函数。 用你的工作逻辑取代我的“回声”。它只是避免了实际的递归,以这种方式使用“find”。


当然,确实你应该调整你的流量来生成其他地方的文件,或者至少只在一个“输出”文件夹中,尽可能整洁,所以你不需要忽略不断增长的文件列表。

但是,有些工具并不像我们想的那么整洁,唉。

答案 1 :(得分:1)

也许你还能做些什么。

为什么不将禁止带有某些后缀的文件放入Subversion存储库?

我有hook script就是这么做的。你给它一个glob模式或正则表达式,如果一个文件被添加到存储库中,其名称与该glob模式或正则表达式匹配,则不会发生提交。开发人员会收到一条错误消息,并且必须在提交完成之前删除添加的文件。

这可以做几件事:

  • 开发人员将自己添加忽略模式。
  • 他们将开始将构建对象移动到其他目录
  • 他们确保他们的构建脚本中的清理函数真正清理了编译创建的所有垃圾。

否则,您将不得不一遍又一遍地运行脚本。在某个地方,一些开发人员会检查他们不应该做的事情,而且 你的错误 ,因为你采取了自己来阻止这个从发生。