2011-03-30 18:56:31 +00:00
|
|
|
|
{- A "remote" that is just a filesystem directory.
|
2011-03-30 17:18:46 +00:00
|
|
|
|
-
|
|
|
|
|
- Copyright 2011 Joey Hess <joey@kitenet.net>
|
|
|
|
|
-
|
|
|
|
|
- Licensed under the GNU GPL version 3 or higher.
|
|
|
|
|
-}
|
|
|
|
|
|
|
|
|
|
module Remote.Directory (remote) where
|
|
|
|
|
|
2011-04-16 22:22:52 +00:00
|
|
|
|
import qualified Data.ByteString.Lazy.Char8 as L
|
2011-03-30 17:18:46 +00:00
|
|
|
|
import qualified Data.Map as M
|
|
|
|
|
|
2011-10-05 20:02:51 +00:00
|
|
|
|
import Common.Annex
|
2011-10-04 02:24:57 +00:00
|
|
|
|
import Utility.CopyFile
|
2011-06-02 01:56:04 +00:00
|
|
|
|
import Types.Remote
|
2011-06-30 17:16:57 +00:00
|
|
|
|
import qualified Git
|
2011-03-30 18:32:08 +00:00
|
|
|
|
import Config
|
2011-09-23 22:13:24 +00:00
|
|
|
|
import Utility.FileMode
|
2011-08-17 00:49:54 +00:00
|
|
|
|
import Remote.Helper.Special
|
|
|
|
|
import Remote.Helper.Encryptable
|
2011-04-16 22:22:52 +00:00
|
|
|
|
import Crypto
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-12-31 08:11:39 +00:00
|
|
|
|
remote :: RemoteType
|
2011-03-30 17:18:46 +00:00
|
|
|
|
remote = RemoteType {
|
|
|
|
|
typename = "directory",
|
2011-03-30 18:00:54 +00:00
|
|
|
|
enumerate = findSpecialRemotes "directory",
|
2011-03-30 17:18:46 +00:00
|
|
|
|
generate = gen,
|
2011-03-30 18:00:54 +00:00
|
|
|
|
setup = directorySetup
|
2011-03-30 17:18:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-12-31 08:11:39 +00:00
|
|
|
|
gen :: Git.Repo -> UUID -> Maybe RemoteConfig -> Annex Remote
|
2011-04-16 22:22:52 +00:00
|
|
|
|
gen r u c = do
|
2011-03-30 18:32:08 +00:00
|
|
|
|
dir <- getConfig r "directory" (error "missing directory")
|
2011-03-30 19:15:46 +00:00
|
|
|
|
cst <- remoteCost r cheapRemoteCost
|
2011-04-17 04:40:23 +00:00
|
|
|
|
return $ encryptableRemote c
|
2011-04-17 01:41:14 +00:00
|
|
|
|
(storeEncrypted dir)
|
|
|
|
|
(retrieveEncrypted dir)
|
|
|
|
|
Remote {
|
|
|
|
|
uuid = u,
|
|
|
|
|
cost = cst,
|
|
|
|
|
name = Git.repoDescribe r,
|
|
|
|
|
storeKey = store dir,
|
|
|
|
|
retrieveKeyFile = retrieve dir,
|
2012-01-20 17:23:11 +00:00
|
|
|
|
retrieveKeyFileCheap = retrieveCheap dir,
|
2011-04-17 01:41:14 +00:00
|
|
|
|
removeKey = remove dir,
|
|
|
|
|
hasKey = checkPresent dir,
|
|
|
|
|
hasKeyCheap = True,
|
2012-02-14 07:49:48 +00:00
|
|
|
|
whereisKey = Nothing,
|
2011-09-19 00:11:39 +00:00
|
|
|
|
config = Nothing,
|
2011-12-31 07:27:37 +00:00
|
|
|
|
repo = r,
|
|
|
|
|
remotetype = remote
|
2011-04-17 01:41:14 +00:00
|
|
|
|
}
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-04-15 19:09:36 +00:00
|
|
|
|
directorySetup :: UUID -> RemoteConfig -> Annex RemoteConfig
|
2011-03-30 18:00:54 +00:00
|
|
|
|
directorySetup u c = do
|
2011-03-30 17:18:46 +00:00
|
|
|
|
-- verify configuration is sane
|
2011-07-15 16:47:14 +00:00
|
|
|
|
let dir = fromMaybe (error "Specify directory=") $
|
2011-05-15 06:49:43 +00:00
|
|
|
|
M.lookup "directory" c
|
2012-01-24 19:28:13 +00:00
|
|
|
|
liftIO $ unlessM (doesDirectoryExist dir) $
|
|
|
|
|
error $ "Directory does not exist: " ++ dir
|
2011-04-16 20:29:28 +00:00
|
|
|
|
c' <- encryptionSetup c
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-03-30 18:32:08 +00:00
|
|
|
|
-- The directory is stored in git config, not in this remote's
|
|
|
|
|
-- persistant state, so it can vary between hosts.
|
2011-04-16 20:29:28 +00:00
|
|
|
|
gitConfigSpecialRemote u c' "directory" dir
|
|
|
|
|
return $ M.delete "directory" c'
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-11-29 03:20:31 +00:00
|
|
|
|
{- Locations to try to access a given Key in the Directory. -}
|
2011-11-22 22:20:55 +00:00
|
|
|
|
locations :: FilePath -> Key -> [FilePath]
|
2011-12-02 18:56:48 +00:00
|
|
|
|
locations d k = map (d </>) (keyPaths k)
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-11-22 22:20:55 +00:00
|
|
|
|
withCheckedFile :: (FilePath -> IO Bool) -> FilePath -> Key -> (FilePath -> IO Bool) -> IO Bool
|
|
|
|
|
withCheckedFile _ [] _ _ = return False
|
|
|
|
|
withCheckedFile check d k a = go $ locations d k
|
|
|
|
|
where
|
|
|
|
|
go [] = return False
|
|
|
|
|
go (f:fs) = do
|
|
|
|
|
use <- check f
|
|
|
|
|
if use
|
|
|
|
|
then a f
|
|
|
|
|
else go fs
|
|
|
|
|
|
|
|
|
|
withStoredFile :: FilePath -> Key -> (FilePath -> IO Bool) -> IO Bool
|
|
|
|
|
withStoredFile = withCheckedFile doesFileExist
|
|
|
|
|
|
2011-04-17 01:41:14 +00:00
|
|
|
|
store :: FilePath -> Key -> Annex Bool
|
|
|
|
|
store d k = do
|
2011-11-29 02:43:51 +00:00
|
|
|
|
src <- inRepo $ gitAnnexLocation k
|
2011-11-22 22:20:55 +00:00
|
|
|
|
liftIO $ catchBoolIO $ storeHelper d k $ copyFileExternal src
|
2011-04-17 01:41:14 +00:00
|
|
|
|
|
|
|
|
|
storeEncrypted :: FilePath -> (Cipher, Key) -> Key -> Annex Bool
|
|
|
|
|
storeEncrypted d (cipher, enck) k = do
|
2011-11-29 02:43:51 +00:00
|
|
|
|
src <- inRepo $ gitAnnexLocation k
|
2011-11-22 22:20:55 +00:00
|
|
|
|
liftIO $ catchBoolIO $ storeHelper d enck $ encrypt src
|
2011-03-30 18:56:31 +00:00
|
|
|
|
where
|
2011-04-17 01:41:14 +00:00
|
|
|
|
encrypt src dest = do
|
2011-04-19 19:26:50 +00:00
|
|
|
|
withEncryptedContent cipher (L.readFile src) $ L.writeFile dest
|
2011-04-17 01:41:14 +00:00
|
|
|
|
return True
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-11-22 22:20:55 +00:00
|
|
|
|
storeHelper :: FilePath -> Key -> (FilePath -> IO Bool) -> IO Bool
|
2011-12-02 18:56:48 +00:00
|
|
|
|
storeHelper d key a = do
|
2011-12-15 22:11:42 +00:00
|
|
|
|
let dest = Prelude.head $ locations d key
|
2012-01-16 20:28:07 +00:00
|
|
|
|
let tmpdest = dest ++ ".tmp"
|
2011-12-02 18:56:48 +00:00
|
|
|
|
let dir = parentDir dest
|
|
|
|
|
createDirectoryIfMissing True dir
|
|
|
|
|
allowWrite dir
|
2012-01-16 20:28:07 +00:00
|
|
|
|
ok <- a tmpdest
|
2011-12-02 18:56:48 +00:00
|
|
|
|
when ok $ do
|
2012-01-16 20:28:07 +00:00
|
|
|
|
renameFile tmpdest dest
|
2011-12-02 18:56:48 +00:00
|
|
|
|
preventWrite dest
|
|
|
|
|
preventWrite dir
|
|
|
|
|
return ok
|
2011-04-17 01:41:14 +00:00
|
|
|
|
|
2012-01-20 17:23:11 +00:00
|
|
|
|
retrieve :: FilePath -> Key -> FilePath -> Annex Bool
|
|
|
|
|
retrieve d k f = liftIO $ withStoredFile d k $ \file -> copyFileExternal file f
|
|
|
|
|
|
|
|
|
|
retrieveCheap :: FilePath -> Key -> FilePath -> Annex Bool
|
|
|
|
|
retrieveCheap d k f = liftIO $ withStoredFile d k $ \file ->
|
|
|
|
|
catchBoolIO $ createSymbolicLink file f >> return True
|
2011-04-17 01:41:14 +00:00
|
|
|
|
|
|
|
|
|
retrieveEncrypted :: FilePath -> (Cipher, Key) -> FilePath -> Annex Bool
|
|
|
|
|
retrieveEncrypted d (cipher, enck) f =
|
2011-11-22 22:20:55 +00:00
|
|
|
|
liftIO $ withStoredFile d enck $ \file -> catchBoolIO $ do
|
|
|
|
|
withDecryptedContent cipher (L.readFile file) $ L.writeFile f
|
2011-04-17 04:57:29 +00:00
|
|
|
|
return True
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-03-30 18:32:08 +00:00
|
|
|
|
remove :: FilePath -> Key -> Annex Bool
|
2011-11-22 22:20:55 +00:00
|
|
|
|
remove d k = liftIO $ withStoredFile d k $ \file -> catchBoolIO $ do
|
|
|
|
|
let dir = parentDir file
|
2011-04-17 04:57:29 +00:00
|
|
|
|
allowWrite dir
|
|
|
|
|
removeFile file
|
|
|
|
|
removeDirectory dir
|
|
|
|
|
return True
|
2011-03-30 17:18:46 +00:00
|
|
|
|
|
2011-11-09 22:33:15 +00:00
|
|
|
|
checkPresent :: FilePath -> Key -> Annex (Either String Bool)
|
2011-11-22 22:20:55 +00:00
|
|
|
|
checkPresent d k = liftIO $ catchMsgIO $ withStoredFile d k $
|
|
|
|
|
const $ return True -- withStoredFile checked that it exists
|