Deal with NFS problem that caused a failure to remove a directory when removing content from the annex.

I was able to reproduce this on linux using the kernel's nfs server and
mounting localhost:/. Determined that removing the directory fails when
the just-deleted file in it was locked. Considered dropping the lock
before removing the directory, but this would complicate parts of the code
that should not need to worry about locking. So instead, ignore the failure
to remove the directory in this case.

While I was at it, made it attempt to remove both levels of hash
directories, in case they're empty.
This commit is contained in:
Joey Hess 2012-02-24 16:30:47 -04:00
parent c9b48520cc
commit bd66f962d3
3 changed files with 39 additions and 10 deletions

View file

@ -244,20 +244,33 @@ withObjectLoc key a = do
let dir = parentDir file let dir = parentDir file
a (dir, file) a (dir, file)
cleanObjectLoc :: Key -> Annex ()
cleanObjectLoc key = do
file <- inRepo $ gitAnnexLocation key
liftIO $ removeparents file (3 :: Int)
where
removeparents _ 0 = return ()
removeparents file n = do
let dir = parentDir file
maybe (return ()) (const $ removeparents dir (n-1))
=<< catchMaybeIO (removeDirectory dir)
{- Removes a key's file from .git/annex/objects/ -} {- Removes a key's file from .git/annex/objects/ -}
removeAnnex :: Key -> Annex () removeAnnex :: Key -> Annex ()
removeAnnex key = withObjectLoc key $ \(dir, file) -> liftIO $ do removeAnnex key = withObjectLoc key $ \(dir, file) -> do
allowWrite dir liftIO $ do
removeFile file allowWrite dir
removeDirectory dir removeFile file
cleanObjectLoc key
{- Moves a key's file out of .git/annex/objects/ -} {- Moves a key's file out of .git/annex/objects/ -}
fromAnnex :: Key -> FilePath -> Annex () fromAnnex :: Key -> FilePath -> Annex ()
fromAnnex key dest = withObjectLoc key $ \(dir, file) -> liftIO $ do fromAnnex key dest = withObjectLoc key $ \(dir, file) -> do
allowWrite dir liftIO $ do
allowWrite file allowWrite dir
moveFile file dest allowWrite file
removeDirectory dir moveFile file dest
cleanObjectLoc key
{- Moves a key out of .git/annex/objects/ into .git/annex/bad, and {- Moves a key out of .git/annex/objects/ into .git/annex/bad, and
- returns the file it was moved to. -} - returns the file it was moved to. -}
@ -270,7 +283,7 @@ moveBad key = do
createDirectoryIfMissing True (parentDir dest) createDirectoryIfMissing True (parentDir dest)
allowWrite (parentDir src) allowWrite (parentDir src)
moveFile src dest moveFile src dest
removeDirectory (parentDir src) cleanObjectLoc key
logStatus key InfoMissing logStatus key InfoMissing
return dest return dest

2
debian/changelog vendored
View file

@ -33,6 +33,8 @@ git-annex (3.20120124) UNRELEASED; urgency=low
* rekey: New plumbing level command, can be used to change the keys used * rekey: New plumbing level command, can be used to change the keys used
for files en masse. for files en masse.
* Store web special remote url info in a more efficient location. * Store web special remote url info in a more efficient location.
* Deal with NFS problem that caused a failure to remove a directory
when removing content from the annex.
-- Joey Hess <joeyh@debian.org> Tue, 24 Jan 2012 16:21:55 -0400 -- Joey Hess <joeyh@debian.org> Tue, 24 Jan 2012 16:21:55 -0400

View file

@ -43,3 +43,17 @@ I'm on an nfs mounted filesystem (some netapp somewhere). This is repeatable, e
I suspect git-annex is just too fast and optimistic for big slow nfs directories. I suspect git-annex is just too fast and optimistic for big slow nfs directories.
> git-annex locks files while it is operating on their content
> to avoid race conditions with other git-annex processes.
> Quite likely this problem (which I can reproduce) is due to
> NFS having bad (non-POSIX) locking semantics.
>
> Probably the
> lock is represented on the NFS server as some form of lock file
> next to the file being locked, and so when that file is deleted, with
> the lock still held, the directory, which should then be empty, still
> contains this lock file.
>
> So, this can be worked around by it not failing when the directory
> unexpectedly cannot be removed. I've made that change. [[done]]
> --[[Joey]]