2011-04-28 21:21:45 +00:00
|
|
|
|
{- A remote that provides hooks to run shell commands.
|
|
|
|
|
-
|
|
|
|
|
- Copyright 2011 Joey Hess <joey@kitenet.net>
|
|
|
|
|
-
|
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
|
-}
|
|
|
|
|
|
|
|
|
|
module Remote.Hook (remote) where
|
|
|
|
|
|
|
|
|
|
import qualified Data.ByteString.Lazy.Char8 as L
|
|
|
|
|
import qualified Data.Map as M
|
|
|
|
|
import System.Exit
|
|
|
|
|
|
2011-10-05 20:02:51 +00:00
|
|
|
|
import Common.Annex
|
2011-06-02 01:56:04 +00:00
|
|
|
|
import Types.Remote
|
2011-06-30 17:16:57 +00:00
|
|
|
|
import qualified Git
|
2011-04-28 21:21:45 +00:00
|
|
|
|
import Config
|
2011-10-04 04:40:47 +00:00
|
|
|
|
import Annex.Content
|
2011-08-17 00:49:54 +00:00
|
|
|
|
import Remote.Helper.Special
|
|
|
|
|
import Remote.Helper.Encryptable
|
2011-04-28 21:21:45 +00:00
|
|
|
|
import Crypto
|
|
|
|
|
|
|
|
|
|
remote :: RemoteType Annex
|
|
|
|
|
remote = RemoteType {
|
|
|
|
|
typename = "hook",
|
|
|
|
|
enumerate = findSpecialRemotes "hooktype",
|
|
|
|
|
generate = gen,
|
|
|
|
|
setup = hookSetup
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gen :: Git.Repo -> UUID -> Maybe RemoteConfig -> Annex (Remote Annex)
|
|
|
|
|
gen r u c = do
|
|
|
|
|
hooktype <- getConfig r "hooktype" (error "missing hooktype")
|
|
|
|
|
cst <- remoteCost r expensiveRemoteCost
|
|
|
|
|
return $ encryptableRemote c
|
|
|
|
|
(storeEncrypted hooktype)
|
|
|
|
|
(retrieveEncrypted hooktype)
|
|
|
|
|
Remote {
|
|
|
|
|
uuid = u,
|
|
|
|
|
cost = cst,
|
|
|
|
|
name = Git.repoDescribe r,
|
|
|
|
|
storeKey = store hooktype,
|
|
|
|
|
retrieveKeyFile = retrieve hooktype,
|
|
|
|
|
removeKey = remove hooktype,
|
|
|
|
|
hasKey = checkPresent r hooktype,
|
|
|
|
|
hasKeyCheap = False,
|
2011-09-19 00:11:39 +00:00
|
|
|
|
config = Nothing,
|
|
|
|
|
repo = r
|
2011-04-28 21:21:45 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
hookSetup :: UUID -> RemoteConfig -> Annex RemoteConfig
|
|
|
|
|
hookSetup u c = do
|
2011-07-15 16:47:14 +00:00
|
|
|
|
let hooktype = fromMaybe (error "Specify hooktype=") $
|
2011-05-15 06:49:43 +00:00
|
|
|
|
M.lookup "hooktype" c
|
2011-04-28 21:21:45 +00:00
|
|
|
|
c' <- encryptionSetup c
|
|
|
|
|
gitConfigSpecialRemote u c' "hooktype" hooktype
|
|
|
|
|
return c'
|
|
|
|
|
|
|
|
|
|
hookEnv :: Key -> Maybe FilePath -> Maybe [(String, String)]
|
2011-04-29 18:04:20 +00:00
|
|
|
|
hookEnv k f = Just $ fileenv f ++ keyenv
|
2011-04-28 21:21:45 +00:00
|
|
|
|
where
|
|
|
|
|
env s v = ("ANNEX_" ++ s, v)
|
2011-04-29 18:04:20 +00:00
|
|
|
|
keyenv =
|
|
|
|
|
[ env "KEY" (show k)
|
2011-12-20 16:23:49 +00:00
|
|
|
|
, env "HASH_1" (hashbits !! 0)
|
|
|
|
|
, env "HASH_2" (hashbits !! 1)
|
2011-04-28 21:21:45 +00:00
|
|
|
|
]
|
2011-04-29 18:04:20 +00:00
|
|
|
|
fileenv Nothing = []
|
|
|
|
|
fileenv (Just file) = [env "FILE" file]
|
2011-12-20 16:23:49 +00:00
|
|
|
|
hashbits = map takeDirectory $ splitPath $ hashDirMixed k
|
2011-04-28 21:21:45 +00:00
|
|
|
|
|
|
|
|
|
lookupHook :: String -> String -> Annex (Maybe String)
|
|
|
|
|
lookupHook hooktype hook =do
|
2011-10-04 02:24:57 +00:00
|
|
|
|
g <- gitRepo
|
2011-04-28 21:21:45 +00:00
|
|
|
|
command <- getConfig g hookname ""
|
|
|
|
|
if null command
|
|
|
|
|
then do
|
|
|
|
|
warning $ "missing configuration for " ++ hookname
|
|
|
|
|
return Nothing
|
|
|
|
|
else return $ Just command
|
|
|
|
|
where
|
|
|
|
|
hookname = hooktype ++ "-" ++ hook ++ "-hook"
|
|
|
|
|
|
|
|
|
|
runHook :: String -> String -> Key -> Maybe FilePath -> Annex Bool -> Annex Bool
|
2011-05-15 06:49:43 +00:00
|
|
|
|
runHook hooktype hook k f a = maybe (return False) run =<< lookupHook hooktype hook
|
|
|
|
|
where
|
|
|
|
|
run command = do
|
2011-07-19 18:07:23 +00:00
|
|
|
|
showOutput -- make way for hook output
|
2011-04-28 21:21:45 +00:00
|
|
|
|
res <- liftIO $ boolSystemEnv
|
2011-05-15 06:49:43 +00:00
|
|
|
|
"sh" [Param "-c", Param command] $ hookEnv k f
|
2011-04-28 21:21:45 +00:00
|
|
|
|
if res
|
|
|
|
|
then a
|
|
|
|
|
else do
|
|
|
|
|
warning $ hook ++ " hook exited nonzero!"
|
|
|
|
|
return res
|
|
|
|
|
|
|
|
|
|
store :: String -> Key -> Annex Bool
|
|
|
|
|
store h k = do
|
2011-11-29 02:43:51 +00:00
|
|
|
|
src <- inRepo $ gitAnnexLocation k
|
2011-11-08 19:34:10 +00:00
|
|
|
|
runHook h "store" k (Just src) $ return True
|
2011-04-28 21:21:45 +00:00
|
|
|
|
|
|
|
|
|
storeEncrypted :: String -> (Cipher, Key) -> Key -> Annex Bool
|
|
|
|
|
storeEncrypted h (cipher, enck) k = withTmp enck $ \tmp -> do
|
2011-11-29 02:43:51 +00:00
|
|
|
|
src <- inRepo $ gitAnnexLocation k
|
2011-11-08 19:34:10 +00:00
|
|
|
|
liftIO $ withEncryptedContent cipher (L.readFile src) $ L.writeFile tmp
|
2011-04-28 21:21:45 +00:00
|
|
|
|
runHook h "store" enck (Just tmp) $ return True
|
|
|
|
|
|
|
|
|
|
retrieve :: String -> Key -> FilePath -> Annex Bool
|
|
|
|
|
retrieve h k f = runHook h "retrieve" k (Just f) $ return True
|
|
|
|
|
|
|
|
|
|
retrieveEncrypted :: String -> (Cipher, Key) -> FilePath -> Annex Bool
|
|
|
|
|
retrieveEncrypted h (cipher, enck) f = withTmp enck $ \tmp ->
|
2011-11-11 00:24:24 +00:00
|
|
|
|
runHook h "retrieve" enck (Just tmp) $ liftIO $ catchBoolIO $ do
|
2011-04-28 21:21:45 +00:00
|
|
|
|
withDecryptedContent cipher (L.readFile tmp) $ L.writeFile f
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
remove :: String -> Key -> Annex Bool
|
2011-07-15 16:47:14 +00:00
|
|
|
|
remove h k = runHook h "remove" k Nothing $ return True
|
2011-04-28 21:21:45 +00:00
|
|
|
|
|
2011-11-09 22:33:15 +00:00
|
|
|
|
checkPresent :: Git.Repo -> String -> Key -> Annex (Either String Bool)
|
2011-04-28 21:21:45 +00:00
|
|
|
|
checkPresent r h k = do
|
2011-07-19 18:07:23 +00:00
|
|
|
|
showAction $ "checking " ++ Git.repoDescribe r
|
2011-04-28 21:21:45 +00:00
|
|
|
|
v <- lookupHook h "checkpresent"
|
2011-11-11 00:24:24 +00:00
|
|
|
|
liftIO $ catchMsgIO $ check v
|
2011-04-28 21:21:45 +00:00
|
|
|
|
where
|
2011-07-15 16:47:14 +00:00
|
|
|
|
findkey s = show k `elem` lines s
|
2011-04-28 21:21:45 +00:00
|
|
|
|
env = hookEnv k Nothing
|
|
|
|
|
check Nothing = error "checkpresent hook misconfigured"
|
|
|
|
|
check (Just hook) = do
|
|
|
|
|
(frompipe, topipe) <- createPipe
|
|
|
|
|
pid <- forkProcess $ do
|
|
|
|
|
_ <- dupTo topipe stdOutput
|
|
|
|
|
closeFd frompipe
|
|
|
|
|
executeFile "sh" True ["-c", hook] env
|
|
|
|
|
closeFd topipe
|
|
|
|
|
fromh <- fdToHandle frompipe
|
|
|
|
|
reply <- hGetContentsStrict fromh
|
|
|
|
|
hClose fromh
|
|
|
|
|
s <- getProcessStatus True False pid
|
|
|
|
|
case s of
|
2011-07-15 16:47:14 +00:00
|
|
|
|
Just (Exited ExitSuccess) -> return $ findkey reply
|
2011-04-28 21:21:45 +00:00
|
|
|
|
_ -> error "checkpresent hook failed"
|