diff options
author | Joey Hess <joeyh@joeyh.name> | 2015-11-13 13:22:45 -0400 |
---|---|---|
committer | Joey Hess <joeyh@joeyh.name> | 2015-11-13 13:22:45 -0400 |
commit | 9f3d75f06c2d6f4ed6b86fb23f8dec52d8d53b85 (patch) | |
tree | ef979b520871bfedfc95e4353380d1358d0b061c /Utility/LockFile | |
parent | ac0ca1b21f4670b0ab03df44a132caed64818978 (diff) |
add stat check
I have a strace taken on a lustre filesystem on which link() returned 0,
but didn't actually succeed, since the file already existed.
One of the linux man pages recommended using link followed by checking like
this. I was reading it yesterday, but cannot find it now.
Diffstat (limited to 'Utility/LockFile')
-rw-r--r-- | Utility/LockFile/PidLock.hs | 29 |
1 files changed, 28 insertions, 1 deletions
diff --git a/Utility/LockFile/PidLock.hs b/Utility/LockFile/PidLock.hs index d1ee6502a..4caf5a06b 100644 --- a/Utility/LockFile/PidLock.hs +++ b/Utility/LockFile/PidLock.hs @@ -92,7 +92,7 @@ tryLock lockfile = trySideLock lockfile $ \sidelock -> do nukeFile tmp return Nothing let tooklock = return $ Just $ LockHandle lockfile fd sidelock - ifM (isJust <$> catchMaybeIO (createLink tmp lockfile)) + ifM (linkToLock tmp lockfile) ( do nukeFile tmp tooklock @@ -111,6 +111,33 @@ tryLock lockfile = trySideLock lockfile $ \sidelock -> do _ -> failedlock ) +-- Linux man pages recommend linking a pid lock into place, +-- as the most portable atomic operation that will fail if +-- it already exists. However, on some network filesystems, +-- link will return success sometimes despite having failed, +-- so we have to stat both files to check if it actually worked. +linkToLock :: FilePath -> FilePath -> IO Bool +linkToLock src dest = ifM (isJust <$> catchMaybeIO (createLink src dest)) + ( catchDefaultIO False checklink + , return False + ) + where + checklink = do + x <- getSymbolicLinkStatus src + y <- getSymbolicLinkStatus dest + return $ and + [ deviceID x == deviceID y + , fileID x == fileID y + , fileMode x == fileMode y + , linkCount x == linkCount y + , fileOwner x == fileOwner y + , fileGroup x == fileGroup y + , specialDeviceID x == specialDeviceID y + , fileSize x == fileSize y + , modificationTime x == modificationTime y + , isRegularFile x == isRegularFile y + ] + -- | Waits as necessary to take a lock. -- -- Uses a 1 second wait-loop. |