summaryrefslogtreecommitdiff
path: root/Logs
diff options
context:
space:
mode:
authorGravatar Joey Hess <joey@kitenet.net>2012-09-24 13:16:50 -0400
committerGravatar Joey Hess <joey@kitenet.net>2012-09-24 13:18:16 -0400
commit3887432c54e1da6d66d364bb7f153a3c6a1cace8 (patch)
tree9941b1cd365ffadd13bdb84d33e4b07e4bb3f1e4 /Logs
parent364b40e5fcad5e6221df75b49953c838407a4546 (diff)
fixes for transfer resume
Fix resuming of downloads, which do not have a transfer info file to read. When checking upload progress, use the MVar, rather than re-reading the info file. Catch exceptions in the transfer action. Required a tryAnnex.
Diffstat (limited to 'Logs')
-rw-r--r--Logs/Transfer.hs43
1 files changed, 26 insertions, 17 deletions
diff --git a/Logs/Transfer.hs b/Logs/Transfer.hs
index 016571d23..3b68eeeb7 100644
--- a/Logs/Transfer.hs
+++ b/Logs/Transfer.hs
@@ -100,9 +100,10 @@ download u key file shouldretry a = runTransfer (Transfer Download u key) file s
runTransfer :: Transfer -> Maybe FilePath -> RetryDecider -> (MeterUpdate -> Annex Bool) -> Annex Bool
runTransfer t file shouldretry a = do
info <- liftIO $ startTransferInfo file
- (meter, tfile) <- mkProgressUpdater t info
+ (meter, tfile, metervar) <- mkProgressUpdater t info
mode <- annexFileMode
- ok <- retry tfile info $ bracketIO (prep tfile mode info) (cleanup tfile) (a meter)
+ ok <- retry info metervar $
+ bracketIO (prep tfile mode info) (cleanup tfile) (a meter)
unless ok $ failed info
return ok
where
@@ -123,26 +124,34 @@ runTransfer t file shouldretry a = do
failedtfile <- fromRepo $ failedTransferFile t
createAnnexDirectory $ takeDirectory failedtfile
liftIO $ writeTransferInfoFile info failedtfile
- retry tfile oldinfo run = do
- ok <- run
- if ok
- then return ok
- else do
- v <- liftIO $ readTransferInfoFile Nothing tfile
- case v of
- Nothing -> return ok
- Just newinfo -> if shouldretry oldinfo newinfo
- then retry tfile newinfo run
- else return ok
+ retry oldinfo metervar run = do
+ v <- tryAnnex run
+ case v of
+ Right b -> return b
+ Left _ -> do
+ b <- getbytescomplete metervar
+ let newinfo = oldinfo { bytesComplete = Just b }
+ if shouldretry oldinfo newinfo
+ then retry newinfo metervar run
+ else return False
+ getbytescomplete metervar
+ | transferDirection t == Upload =
+ liftIO $ readMVar metervar
+ | otherwise = do
+ f <- fromRepo $ gitAnnexTmpLocation (transferKey t)
+ liftIO $ catchDefaultIO 0 $
+ fromIntegral . fileSize
+ <$> getFileStatus f
{- Generates a callback that can be called as transfer progresses to update
- - the transfer info file. Also returns the file it'll be updating. -}
-mkProgressUpdater :: Transfer -> TransferInfo -> Annex (MeterUpdate, FilePath)
+ - the transfer info file. Also returns the file it'll be updating, and a
+ - MVar that can be used to read the number of bytesComplete. -}
+mkProgressUpdater :: Transfer -> TransferInfo -> Annex (MeterUpdate, FilePath, MVar Integer)
mkProgressUpdater t info = do
tfile <- fromRepo $ transferFile t
createAnnexDirectory $ takeDirectory tfile
mvar <- liftIO $ newMVar 0
- return (liftIO . updater tfile mvar, tfile)
+ return (liftIO . updater tfile mvar, tfile, mvar)
where
updater tfile mvar bytes = modifyMVar_ mvar $ \oldbytes -> do
if (bytes - oldbytes >= mindelta)
@@ -268,7 +277,7 @@ writeTransferInfo info = unlines
]
readTransferInfoFile :: (Maybe ProcessID) -> FilePath -> IO (Maybe TransferInfo)
-readTransferInfoFile mpid tfile = do
+readTransferInfoFile mpid tfile = catchDefaultIO Nothing $ do
h <- openFile tfile ReadMode
fileEncoding h
hClose h `after` (readTransferInfo mpid <$> hGetContentsStrict h)