Why does git break if '/' is the root of the repository?

We'd like to use git to maintain system configurations. Because sometimes configuration data exists outside of /etc, we've started doing something like this on our systems:

  # cd /
  # git init
  # git add etc
  # git add some/other/path
  # git commit -m 'initial import'

And so forth. This works, up to a point. As long as your cwd == '/', git behaves normally. However, if you try, for example, to run git from inside a subdirectory:

cd /etc
git status

You get garbage. In our case, thousands of lines of "deleted:" listings for files that clearly still exist. This behavior appears to be exclusive to running git in /; doing the same thing anywhere else works just fine.

I can "fix" the behavior like this:

GIT_WORK_TREE=/ git status

And hey, everything works the way Linus intended...but this is a pain. I don't want to set it in the environment unilaterally (because this would conflict with the use of git in other repositories), and I'd like to avoid a wrapper script. Do I have any other options?


This is a complete guess that you could use to investigate further, but I suspect that git's "find the .git directory" behavior is interacting with the fact that / is its own parent directory. Maybe the "stop at the root" logic has a fencepost-type error.


Setting the core.worktree option on the repository takes care of this nicely:

git config core.worktree /

This works much better than setting GIT_WORK_TREE in the environment. Yay!


This will be fixed in Git 2.4.1+ (Q2 2015).
See commit 84ccad8 by Jeff King ( peff ), and merged in 7502b23.

init : don't set core.worktree when initializing /.git

If you create a git repository in the root directory with " git init / ", we erroneously write a core.worktree entry.
This isn't wrong, in the sense that it's OK to set core.worktree when we don't need to. But it is unnecessarily surprising if you later move the .git directory to another path (which usually moves the relative working tree, but is foiled if there is an explicit worktree set).

The problem is that we check whether core.worktree is necessary by seeing if we can make the git_dir by concatenating "/.git" onto the working tree .
That would lead to " //.git " in this instance, but we actually have " /.git " (without the doubled slash).

(That is why core.worktree is incorrectly set here, when git init is done at the root folder / )

We can fix this by special-casing the root directory. I also split the logic out into its own function to make the conditional a bit more readable (and used skip_prefix, which I think makes it a little more obvious what is going on).

No tests, as we would need to be able to write to "/" to do so.
I did manually confirm that:

sudo git init /
cd /
git rev-parse --show-toplevel
git config core.worktree

still finds the top-level correctly (as " / "), and does not set any core.worktree variable .

链接地址: http://www.djcxy.com/p/18886.html

上一篇: 你可以改变.git文件夹的位置

下一篇: 如果'/'是版本库的根目录,为什么git会中断?