On Wed, 28 May 2008, Junio C Hamano wrote: > Daniel Barkalow <barkalow@xxxxxxxxxxxx> writes: > > > Ah, yes, CE_VALID. But it doesn't quite work as well as I'd like, because > > No, I do not think we should involve CE_VALID here. It means something > completely different. What I meant was that through "git status" the user > can tell there is an unexpected breakage in the work tree, _if_ we make > checkout to finish with "best effort" (and still report an error). I think there are two separate issues here: (1) you've gotten partway through a checkout, and something fails, and we want to leave things in as good shape as possible; (2) you're trying to check out a tree that can't be accurately represented on your filesystem. In case (2), there's a good chance that you want to leave the unrepresentable stuff alone or modify it in the index without going through the filesystem. Case (2) is very much like working on a filesystem without +x bits, where you'd really like to ignore what the filesystem reports and only diverge from what was read into the index from a tree object when the user makes explicit modifications to the index, rather than updating the index from the working directory state. > > I think the right test for this is if create_file() returns EEXIST, but > > readdir doesn't show anything. > > I think relying on EEXIST is too specific for this particular breakage, > even though such a test may catch it. It's not just EEXIST; it's EEXIST for a filename with no struct dirent. > A checkout may fail in the middle if a filesystem refuses to create a > pathname that has certain characters in it (e.g. dosen't NTFS refuse a > path with :|<"?*> in it, or is it just the Explorer UI layer rejecting > them?), or perhaps one leading directory may be unwritable. We would > want to catch and cope with such a brokage the same way. I think the leading directory thing is qualitatively different; you're probably not going to want to create a patch to rename all of the files in that directory to be in a directory that's not owned by a different user on your machine, while you are likely to try to get a project to use less odd names if there's a filename with a : in it or something. I'm not sure what error open() will give you for that. The other interesting case is when the filesystem is case insensative and the project contains files that differ only in case; again, the filesystem will report EEXIST on open for a path that readdir doesn't list. > The checkout "unpack-trees" codepath does: > > - Make sure things can be checked out safely with the internal data > before doing anything to the filesystem, i.e. no lost local changes, no > lost untracked files, etc. > > - For each path: > > - make room for it, removing directory at the place as necessary where > a blob must sit and removing an existing blob as needed; > > - create a new file or symlink; > > And currently I think we stop on any failure. The thing is, stopping on a > failure during the internal checking is fine --- no external damage has > been made yet. But once we started updating the work tree, we _are_ > committed and not aborting in the middle for a single failure would be the > saner thing to do. In addition, even after such a failure after we are > committed, we probably should update the HEAD and the index. > > "status" would then show the difference between what should have been > checked out and what is. It might be enough to improve the issue of "git > bisect hitting a checkout failure --- the work tree is half checked-out > state, and the index, the HEAD, and the work tree are in a very > inconsistent state". For case (1), things are in an inconsistant state; for case (2), the index and HEAD agree, and there are known gaps in the work tree. > We would probably signal such an error from git-checkout differently from > an early refusal that does not do anything, to tell the callers, such as > "git-bisect", that the checkout _has been_ already done, even though there > may be breakages in the work tree. > > > ... that notes the situation where you seem to have file A instead of > > file B, but fstat("B") returns A's inode, and marks the index to say that > > entry B is listed in the filesystem as A instead. > > I personally do not think such auto-substution is a way to go --- what > makes you trust inode information from such an untrustworthy filesystem > that does not do what it was told to do? I suspect that stopping at the > error site and not automatically making the damage yet larger by doing > such magic would keep the recovery procedure simpler. People seem to use Windows and OS X despite the filesystems being broken. There seems to be a space of filesystems which aren't corrupted, but do unexpected things with respect to filenames. If we can find appropriate filenames, the content is stored reliably (or the open O_EXCL is refused). We should be able to identify that the user isn't trying to make a change by way of the working directory when the difference we see is something that isn't clearly representable by the filesystem. > But I wouldn't keep people from experimenting. Perhaps the end result > could be even readable and mergeable, although I am quite pessimistic. I think, in any case, that it should be pretty clean to have a "filesystem is inadaquate" CE flag, which mean that we just ignore the filesystem. Finding things that are reported with the wrong name is probably harder. -Daniel *This .sig left intentionally blank* -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html