2012-07-03 14:58:40 +00:00
|
|
|
{- git-annex assistant transfer watching thread
|
|
|
|
-
|
|
|
|
- Copyright 2012 Joey Hess <joey@kitenet.net>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
module Assistant.Threads.TransferWatcher where
|
|
|
|
|
2012-07-20 23:29:59 +00:00
|
|
|
import Assistant.Common
|
2012-07-03 14:58:40 +00:00
|
|
|
import Assistant.DaemonStatus
|
2013-10-26 20:54:49 +00:00
|
|
|
import Assistant.TransferSlots
|
2012-07-03 14:58:40 +00:00
|
|
|
import Logs.Transfer
|
|
|
|
import Utility.DirWatcher
|
2013-03-11 02:24:13 +00:00
|
|
|
import Utility.DirWatcher.Types
|
2012-08-27 18:04:06 +00:00
|
|
|
import qualified Remote
|
2012-07-03 14:58:40 +00:00
|
|
|
|
2012-10-19 20:59:18 +00:00
|
|
|
import Control.Concurrent
|
2013-11-23 21:21:04 +00:00
|
|
|
import qualified Data.Map as M
|
2012-10-19 20:59:18 +00:00
|
|
|
|
2012-07-03 14:58:40 +00:00
|
|
|
{- This thread watches for changes to the gitAnnexTransferDir,
|
|
|
|
- and updates the DaemonStatus's map of ongoing transfers. -}
|
2012-10-29 17:09:58 +00:00
|
|
|
transferWatcherThread :: NamedThread
|
2013-01-26 06:09:33 +00:00
|
|
|
transferWatcherThread = namedThread "TransferWatcher" $ do
|
2012-10-29 17:09:58 +00:00
|
|
|
dir <- liftAnnex $ gitAnnexTransferDir <$> gitRepo
|
|
|
|
liftIO $ createDirectoryIfMissing True dir
|
|
|
|
let hook a = Just <$> asIO2 (runHandler a)
|
|
|
|
addhook <- hook onAdd
|
|
|
|
delhook <- hook onDel
|
|
|
|
modifyhook <- hook onModify
|
|
|
|
errhook <- hook onErr
|
2012-07-03 14:58:40 +00:00
|
|
|
let hooks = mkWatchHooks
|
2012-10-29 17:09:58 +00:00
|
|
|
{ addHook = addhook
|
|
|
|
, delHook = delhook
|
|
|
|
, modifyHook = modifyhook
|
|
|
|
, errHook = errhook
|
2012-07-03 14:58:40 +00:00
|
|
|
}
|
2012-10-29 17:09:58 +00:00
|
|
|
void $ liftIO $ watchDir dir (const False) hooks id
|
|
|
|
debug ["watching for transfers"]
|
2012-07-03 14:58:40 +00:00
|
|
|
|
2012-10-29 17:09:58 +00:00
|
|
|
type Handler = FilePath -> Assistant ()
|
2012-07-03 14:58:40 +00:00
|
|
|
|
|
|
|
{- Runs an action handler.
|
|
|
|
-
|
|
|
|
- Exceptions are ignored, otherwise a whole thread could be crashed.
|
|
|
|
-}
|
2012-10-29 17:09:58 +00:00
|
|
|
runHandler :: Handler -> FilePath -> Maybe FileStatus -> Assistant ()
|
|
|
|
runHandler handler file _filestatus =
|
2012-12-13 04:45:27 +00:00
|
|
|
either (liftIO . print) (const noop) =<< tryIO <~> handler file
|
2012-07-03 14:58:40 +00:00
|
|
|
|
|
|
|
{- Called when there's an error with inotify. -}
|
|
|
|
onErr :: Handler
|
2013-10-03 02:59:07 +00:00
|
|
|
onErr = error
|
2012-07-03 14:58:40 +00:00
|
|
|
|
2012-07-06 20:30:55 +00:00
|
|
|
{- Called when a new transfer information file is written. -}
|
2012-07-03 14:58:40 +00:00
|
|
|
onAdd :: Handler
|
2012-10-29 17:09:58 +00:00
|
|
|
onAdd file = case parseTransferFile file of
|
2012-07-03 14:58:40 +00:00
|
|
|
Nothing -> noop
|
2012-10-29 17:09:58 +00:00
|
|
|
Just t -> go t =<< liftAnnex (checkTransfer t)
|
|
|
|
where
|
|
|
|
go _ Nothing = noop -- transfer already finished
|
|
|
|
go t (Just info) = do
|
2013-03-01 19:23:59 +00:00
|
|
|
debug [ "transfer starting:", describeTransfer t info ]
|
2013-04-02 20:39:11 +00:00
|
|
|
r <- liftAnnex $ Remote.remoteFromUUID $ transferUUID t
|
2012-10-30 19:39:15 +00:00
|
|
|
updateTransferInfo t info { transferRemote = r }
|
2012-07-03 14:58:40 +00:00
|
|
|
|
2012-09-20 21:24:40 +00:00
|
|
|
{- Called when a transfer information file is updated.
|
|
|
|
-
|
|
|
|
- The only thing that should change in the transfer info is the
|
|
|
|
- bytesComplete, so that's the only thing updated in the DaemonStatus. -}
|
|
|
|
onModify :: Handler
|
2013-10-03 02:59:07 +00:00
|
|
|
onModify file = case parseTransferFile file of
|
|
|
|
Nothing -> noop
|
|
|
|
Just t -> go t =<< liftIO (readTransferInfoFile Nothing file)
|
2012-10-31 06:34:03 +00:00
|
|
|
where
|
|
|
|
go _ Nothing = noop
|
|
|
|
go t (Just newinfo) = alterTransferInfo t $
|
|
|
|
\i -> i { bytesComplete = bytesComplete newinfo }
|
2012-09-20 21:24:40 +00:00
|
|
|
|
|
|
|
{- This thread can only watch transfer sizes when the DirWatcher supports
|
|
|
|
- tracking modificatons to files. -}
|
|
|
|
watchesTransferSize :: Bool
|
|
|
|
watchesTransferSize = modifyTracked
|
|
|
|
|
2012-07-18 23:13:56 +00:00
|
|
|
{- Called when a transfer information file is removed. -}
|
2012-07-03 14:58:40 +00:00
|
|
|
onDel :: Handler
|
2012-10-29 17:09:58 +00:00
|
|
|
onDel file = case parseTransferFile file of
|
2012-07-03 14:58:40 +00:00
|
|
|
Nothing -> noop
|
2012-07-20 23:29:59 +00:00
|
|
|
Just t -> do
|
2012-10-29 17:09:58 +00:00
|
|
|
debug [ "transfer finishing:", show t]
|
2012-10-30 19:39:15 +00:00
|
|
|
minfo <- removeTransfer t
|
2012-10-19 20:59:18 +00:00
|
|
|
|
2013-11-23 21:21:04 +00:00
|
|
|
-- Run transfer hook.
|
2013-11-24 04:26:20 +00:00
|
|
|
m <- transferHook <$> getDaemonStatus
|
|
|
|
maybe noop (\hook -> void $ liftIO $ forkIO $ hook t)
|
|
|
|
(M.lookup (transferKey t) m)
|
2013-11-23 21:21:04 +00:00
|
|
|
|
2012-10-29 17:09:58 +00:00
|
|
|
finished <- asIO2 finishedTransfer
|
|
|
|
void $ liftIO $ forkIO $ do
|
2012-10-19 20:59:18 +00:00
|
|
|
{- XXX race workaround delay. The location
|
2012-12-13 04:45:27 +00:00
|
|
|
- log needs to be updated before finishedTransfer
|
|
|
|
- runs. -}
|
2012-10-19 20:59:18 +00:00
|
|
|
threadDelay 10000000 -- 10 seconds
|
2012-10-29 17:09:58 +00:00
|
|
|
finished t minfo
|