2013-03-19 22:46:29 +00:00
|
|
|
{- A pool of "git-annex transferkeys" processes
|
|
|
|
-
|
|
|
|
- Copyright 2013 Joey Hess <joey@kitenet.net>
|
|
|
|
-
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
-}
|
|
|
|
|
|
|
|
module Assistant.TransferrerPool where
|
|
|
|
|
|
|
|
import Assistant.Common
|
|
|
|
import Assistant.Types.TransferrerPool
|
|
|
|
import Logs.Transfer
|
2013-12-01 18:56:37 +00:00
|
|
|
import Utility.Batch
|
2013-11-12 18:54:02 +00:00
|
|
|
|
2013-03-19 22:46:29 +00:00
|
|
|
import qualified Command.TransferKeys as T
|
|
|
|
|
|
|
|
import Control.Concurrent.STM
|
2013-12-11 03:19:18 +00:00
|
|
|
import System.Process (create_group, std_in, std_out)
|
2013-03-19 22:46:29 +00:00
|
|
|
import Control.Exception (throw)
|
|
|
|
import Control.Concurrent
|
|
|
|
|
|
|
|
{- Runs an action with a Transferrer from the pool. -}
|
2013-12-01 19:37:51 +00:00
|
|
|
withTransferrer :: FilePath -> BatchCommandMaker -> TransferrerPool -> (Transferrer -> IO a) -> IO a
|
|
|
|
withTransferrer program batchmaker pool a = do
|
|
|
|
t <- maybe (mkTransferrer program batchmaker) (checkTransferrer program batchmaker)
|
2013-03-19 22:46:29 +00:00
|
|
|
=<< atomically (tryReadTChan pool)
|
|
|
|
v <- tryNonAsync $ a t
|
|
|
|
unlessM (putback t) $
|
|
|
|
void $ forkIO $ stopTransferrer t
|
|
|
|
either throw return v
|
|
|
|
where
|
|
|
|
putback t = atomically $ ifM (isEmptyTChan pool)
|
|
|
|
( do
|
|
|
|
writeTChan pool t
|
|
|
|
return True
|
|
|
|
, return False
|
|
|
|
)
|
|
|
|
|
|
|
|
{- Requests that a Transferrer perform a Transfer, and waits for it to
|
|
|
|
- finish. -}
|
|
|
|
performTransfer :: Transferrer -> Transfer -> AssociatedFile -> IO Bool
|
|
|
|
performTransfer transferrer t f = catchBoolIO $ do
|
|
|
|
T.sendRequest t f (transferrerWrite transferrer)
|
|
|
|
T.readResponse (transferrerRead transferrer)
|
|
|
|
|
2013-12-11 03:19:18 +00:00
|
|
|
{- Starts a new git-annex transferkeys process, setting up handles
|
2013-03-19 22:46:29 +00:00
|
|
|
- that will be used to communicate with it. -}
|
2013-12-01 19:37:51 +00:00
|
|
|
mkTransferrer :: FilePath -> BatchCommandMaker -> IO Transferrer
|
|
|
|
mkTransferrer program batchmaker = do
|
2013-12-01 18:56:37 +00:00
|
|
|
{- It runs as a batch job. -}
|
2013-12-11 03:19:18 +00:00
|
|
|
let (program', params') = batchmaker (program, [Param "transferkeys"])
|
2013-03-19 22:46:29 +00:00
|
|
|
{- It's put into its own group so that the whole group can be
|
|
|
|
- killed to stop a transfer. -}
|
2013-12-11 03:19:18 +00:00
|
|
|
(Just writeh, Just readh, _, pid) <- createProcess
|
|
|
|
(proc program' $ toCommand params')
|
|
|
|
{ create_group = True
|
|
|
|
, std_in = CreatePipe
|
|
|
|
, std_out = CreatePipe
|
|
|
|
}
|
|
|
|
fileEncoding readh
|
|
|
|
fileEncoding writeh
|
2013-03-19 22:46:29 +00:00
|
|
|
return $ Transferrer
|
2013-12-11 03:19:18 +00:00
|
|
|
{ transferrerRead = readh
|
|
|
|
, transferrerWrite = writeh
|
2013-03-19 22:46:29 +00:00
|
|
|
, transferrerHandle = pid
|
|
|
|
}
|
|
|
|
|
|
|
|
{- Checks if a Transferrer is still running. If not, makes a new one. -}
|
2013-12-01 19:37:51 +00:00
|
|
|
checkTransferrer :: FilePath -> BatchCommandMaker -> Transferrer -> IO Transferrer
|
|
|
|
checkTransferrer program batchmaker t =
|
|
|
|
maybe (return t) (const $ mkTransferrer program batchmaker)
|
|
|
|
=<< getProcessExitCode (transferrerHandle t)
|
2013-03-19 22:46:29 +00:00
|
|
|
|
|
|
|
{- Closing the fds will stop the transferrer. -}
|
|
|
|
stopTransferrer :: Transferrer -> IO ()
|
|
|
|
stopTransferrer t = do
|
|
|
|
hClose $ transferrerRead t
|
|
|
|
hClose $ transferrerWrite t
|
|
|
|
void $ waitForProcess $ transferrerHandle t
|