git-annex/Logs/Transfer.hs

258 lines
8.5 KiB
Haskell
Raw Normal View History

{- git-annex transfer information files and lock files
2012-07-01 18:29:00 +00:00
-
- Copyright 2012 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Logs.Transfer where
import Common.Annex
import Annex.Perms
import Annex.Exception
2012-09-17 18:58:43 +00:00
import Annex.UUID
2012-07-01 18:29:00 +00:00
import qualified Git
import Types.Remote
import Types.Key
import Utility.Percentage
2012-07-01 18:29:00 +00:00
import System.Posix.Types
import Data.Time.Clock
import Data.Time.Clock.POSIX
import Data.Time
import System.Locale
import Control.Concurrent
2012-07-01 18:29:00 +00:00
{- Enough information to uniquely identify a transfer, used as the filename
- of the transfer information file. -}
data Transfer = Transfer
{ transferDirection :: Direction
2012-07-05 20:34:20 +00:00
, transferUUID :: UUID
, transferKey :: Key
}
deriving (Eq, Ord, Read, Show)
2012-07-01 18:29:00 +00:00
2012-07-02 17:49:27 +00:00
{- Information about a Transfer, stored in the transfer information file.
-
- Note that the associatedFile may not correspond to a file in the local
- git repository. It's some file, possibly relative to some directory,
- of some repository, that was acted on to initiate the transfer.
-}
2012-07-01 18:29:00 +00:00
data TransferInfo = TransferInfo
{ startedTime :: Maybe POSIXTime
, transferPid :: Maybe ProcessID
, transferTid :: Maybe ThreadId
2012-07-05 20:34:20 +00:00
, transferRemote :: Maybe Remote
2012-07-01 18:29:00 +00:00
, bytesComplete :: Maybe Integer
, associatedFile :: Maybe FilePath
, transferPaused :: Bool
2012-07-01 18:29:00 +00:00
}
deriving (Show, Eq, Ord)
2012-07-01 18:29:00 +00:00
2012-09-17 18:58:43 +00:00
stubTransferInfo :: TransferInfo
stubTransferInfo = TransferInfo Nothing Nothing Nothing Nothing Nothing Nothing False
2012-07-01 18:29:00 +00:00
data Direction = Upload | Download
deriving (Eq, Ord, Read, Show)
2012-07-01 18:29:00 +00:00
showLcDirection :: Direction -> String
showLcDirection Upload = "upload"
showLcDirection Download = "download"
2012-07-01 18:29:00 +00:00
readLcDirection :: String -> Maybe Direction
readLcDirection "upload" = Just Upload
readLcDirection "download" = Just Download
readLcDirection _ = Nothing
2012-07-01 18:29:00 +00:00
{- Transfers that will accomplish the same task. -}
equivilantTransfer :: Transfer -> Transfer -> Bool
equivilantTransfer t1 t2
| transferDirection t1 == Download && transferDirection t2 == Download &&
2012-08-29 19:32:57 +00:00
transferKey t1 == transferKey t2 = True
| otherwise = t1 == t2
percentComplete :: Transfer -> TransferInfo -> Maybe Percentage
percentComplete (Transfer { transferKey = key }) info =
percentage <$> keySize key <*> Just (fromMaybe 0 $ bytesComplete info)
upload :: UUID -> Key -> AssociatedFile -> Annex Bool -> Annex Bool
upload u key file a = runTransfer (Transfer Upload u key) file a
2012-07-01 19:04:29 +00:00
download :: UUID -> Key -> AssociatedFile -> Annex Bool -> Annex Bool
download u key file a = runTransfer (Transfer Download u key) file a
2012-07-01 19:04:29 +00:00
{- Runs a transfer action. Creates and locks the lock file while the
- action is running, and stores info in the transfer information
- file. Will throw an error if the transfer is already in progress.
-
- If the transfer action returns False, the transfer info is
- left in the failedTransferDir.
2012-07-01 18:29:00 +00:00
-}
runTransfer :: Transfer -> Maybe FilePath -> Annex Bool -> Annex Bool
runTransfer t file a = do
2012-07-01 19:04:29 +00:00
tfile <- fromRepo $ transferFile t
createAnnexDirectory $ takeDirectory tfile
2012-07-01 18:29:00 +00:00
mode <- annexFileMode
info <- liftIO $ TransferInfo
<$> (Just . utcTimeToPOSIXSeconds <$> getCurrentTime)
<*> pure Nothing -- pid not stored in file, so omitted for speed
<*> pure Nothing -- tid ditto
2012-07-01 18:29:00 +00:00
<*> pure Nothing -- not 0; transfer may be resuming
2012-07-05 20:34:20 +00:00
<*> pure Nothing
2012-07-01 18:29:00 +00:00
<*> pure file
<*> pure False
ok <- bracketIO (prep tfile mode info) (cleanup tfile) a
unless ok $ failed info
return ok
2012-07-01 18:29:00 +00:00
where
prep tfile mode info = do
fd <- openFd (transferLockFile tfile) ReadWrite (Just mode)
2012-07-01 18:29:00 +00:00
defaultFileFlags { trunc = True }
locked <- catchMaybeIO $
setLock fd (WriteLock, AbsoluteSeek, 0, 0)
when (locked == Nothing) $
error $ "transfer already in progress"
writeTransferInfoFile info tfile
return fd
cleanup tfile fd = do
void $ tryIO $ removeFile tfile
void $ tryIO $ removeFile $ transferLockFile tfile
closeFd fd
failed info = do
failedtfile <- fromRepo $ failedTransferFile t
createAnnexDirectory $ takeDirectory failedtfile
liftIO $ writeTransferInfoFile info failedtfile
2012-07-01 18:29:00 +00:00
{- If a transfer is still running, returns its TransferInfo. -}
checkTransfer :: Transfer -> Annex (Maybe TransferInfo)
2012-07-01 19:04:29 +00:00
checkTransfer t = do
2012-07-01 18:29:00 +00:00
mode <- annexFileMode
2012-07-01 19:04:29 +00:00
tfile <- fromRepo $ transferFile t
2012-07-01 18:29:00 +00:00
mfd <- liftIO $ catchMaybeIO $
openFd (transferLockFile tfile) ReadOnly (Just mode) defaultFileFlags
2012-07-01 18:29:00 +00:00
case mfd of
Nothing -> return Nothing -- failed to open file; not running
Just fd -> do
locked <- liftIO $
getLock fd (WriteLock, AbsoluteSeek, 0, 0)
liftIO $ closeFd fd
2012-07-01 18:29:00 +00:00
case locked of
Nothing -> return Nothing
2012-09-17 04:18:07 +00:00
Just (pid, _) -> liftIO $ catchDefaultIO Nothing $
readTransferInfoFile (Just pid) tfile
2012-07-01 18:29:00 +00:00
{- Gets all currently running transfers. -}
getTransfers :: Annex [(Transfer, TransferInfo)]
getTransfers = do
2012-08-23 17:42:13 +00:00
transfers <- catMaybes . map parseTransferFile . concat <$> findfiles
2012-07-01 18:29:00 +00:00
infos <- mapM checkTransfer transfers
return $ map (\(t, Just i) -> (t, i)) $
filter running $ zip transfers infos
where
2012-08-23 17:42:13 +00:00
findfiles = liftIO . mapM dirContentsRecursive
=<< mapM (fromRepo . transferDir)
[Download, Upload]
2012-07-01 18:29:00 +00:00
running (_, i) = isJust i
{- Gets failed transfers for a given remote UUID. -}
getFailedTransfers :: UUID -> Annex [(Transfer, TransferInfo)]
getFailedTransfers u = catMaybes <$> (liftIO . getpairs =<< concat <$> findfiles)
where
getpairs = mapM $ \f -> do
let mt = parseTransferFile f
mi <- readTransferInfoFile Nothing f
return $ case (mt, mi) of
(Just t, Just i) -> Just (t, i)
_ -> Nothing
findfiles = liftIO . mapM dirContentsRecursive
=<< mapM (fromRepo . failedTransferDir u)
[Download, Upload]
2012-09-17 18:58:43 +00:00
removeFailedTransfer :: Transfer -> Annex ()
removeFailedTransfer t = do
f <- fromRepo $ failedTransferFile t
liftIO $ void $ tryIO $ removeFile f
2012-07-01 18:29:00 +00:00
{- The transfer information file to use for a given Transfer. -}
transferFile :: Transfer -> Git.Repo -> FilePath
2012-08-23 17:42:13 +00:00
transferFile (Transfer direction u key) r = transferDir direction r
</> fromUUID u
2012-07-01 19:04:29 +00:00
</> keyFile key
2012-07-01 18:29:00 +00:00
{- The transfer information file to use to record a failed Transfer -}
failedTransferFile :: Transfer -> Git.Repo -> FilePath
failedTransferFile (Transfer direction u key) r = failedTransferDir u direction r
</> keyFile key
{- The transfer lock file corresponding to a given transfer info file. -}
transferLockFile :: FilePath -> FilePath
2012-07-17 21:22:00 +00:00
transferLockFile infofile = let (d,f) = splitFileName infofile in
combine d ("lck." ++ f)
2012-07-01 18:29:00 +00:00
{- Parses a transfer information filename to a Transfer. -}
parseTransferFile :: FilePath -> Maybe Transfer
parseTransferFile file
| "lck." `isPrefixOf` (takeFileName file) = Nothing
| otherwise = case drop (length bits - 3) bits of
2012-07-01 19:04:29 +00:00
[direction, u, key] -> Transfer
<$> readLcDirection direction
<*> pure (toUUID u)
2012-07-01 18:29:00 +00:00
<*> fileKey key
_ -> Nothing
where
bits = splitDirectories file
writeTransferInfoFile :: TransferInfo -> FilePath -> IO ()
writeTransferInfoFile info tfile = do
h <- openFile tfile WriteMode
fileEncoding h
hPutStr h $ writeTransferInfo info
hClose h
2012-07-01 18:29:00 +00:00
writeTransferInfo :: TransferInfo -> String
2012-07-01 19:04:29 +00:00
writeTransferInfo info = unlines
2012-07-01 18:29:00 +00:00
-- transferPid is not included; instead obtained by looking at
-- the process that locks the file.
2012-07-19 00:48:08 +00:00
[ maybe "" show $ startedTime info
2012-07-01 18:29:00 +00:00
-- bytesComplete is not included; changes too fast
2012-07-01 19:04:29 +00:00
, fromMaybe "" $ associatedFile info -- comes last; arbitrary content
2012-07-01 18:29:00 +00:00
]
readTransferInfoFile :: (Maybe ProcessID) -> FilePath -> IO (Maybe TransferInfo)
readTransferInfoFile mpid tfile = do
h <- openFile tfile ReadMode
fileEncoding h
hClose h `after` (readTransferInfo mpid <$> hGetContentsStrict h)
readTransferInfo :: (Maybe ProcessID) -> String -> Maybe TransferInfo
readTransferInfo mpid s =
2012-07-01 18:29:00 +00:00
case bits of
[time] -> TransferInfo
2012-07-19 00:48:08 +00:00
<$> (Just <$> parsePOSIXTime time)
<*> pure mpid
2012-07-01 18:29:00 +00:00
<*> pure Nothing
<*> pure Nothing
<*> pure Nothing
<*> pure (if null filename then Nothing else Just filename)
<*> pure False
2012-07-01 18:29:00 +00:00
_ -> Nothing
where
2012-07-01 19:04:29 +00:00
(bits, filebits) = splitAt 1 $ lines s
filename = join "\n" filebits
2012-07-19 00:48:08 +00:00
parsePOSIXTime :: String -> Maybe POSIXTime
parsePOSIXTime s = utcTimeToPOSIXSeconds
<$> parseTime defaultTimeLocale "%s%Qs" s
2012-08-23 17:42:13 +00:00
{- The directory holding transfer information files for a given Direction. -}
transferDir :: Direction -> Git.Repo -> FilePath
transferDir direction r = gitAnnexTransferDir r </> showLcDirection direction
{- The directory holding failed transfer information files for a given
- Direction and UUID -}
failedTransferDir :: UUID -> Direction -> Git.Repo -> FilePath
failedTransferDir u direction r = gitAnnexTransferDir r
</> "failed"
</> showLcDirection direction
</> fromUUID u