{- git-annex key/value storage backends - - git-annex uses a key/value abstraction layer to allow files contents to be - stored in different ways. In theory, any key/value storage system could be - used to store the file contents, and git-annex would then retrieve them - as needed and put them in `.git/annex/`. - - When a file is annexed, a key is generated from its content and/or metadata. - This key can later be used to retrieve the file's content (its value). This - key generation must be stable for a given file content, name, and size. - - The mapping from filename to its key is stored in the .git-annex directory, - in a file named `$filename.$backend` - - Multiple pluggable backends are supported, and more than one can be used - to store different files' contents in a given repository. - -} module Backend ( lookupBackend, storeFile, retrieveFile, lookupKey, dropFile ) where import System.Directory import Data.String.Utils import Locations import GitRepo import Utility import Types {- Attempts to store a file in one of the backends, and returns - its key. -} storeFile :: State -> FilePath -> IO (Maybe Key) storeFile state file = storeFile' (backends state) state file storeFile' [] _ _ = return Nothing storeFile' (b:bs) state file = do try <- (getKey b) state (gitRelative (repo state) file) case (try) of Nothing -> nextbackend Just key -> do stored <- (storeFileKey b) state file key if (not stored) then nextbackend else do recordKey state b file key return $ Just key where nextbackend = storeFile' bs state file {- Attempts to retrieve an file from one of the backends, saving it to - a specified location. -} retrieveFile :: State -> FilePath -> FilePath -> IO Bool retrieveFile state file dest = do result <- lookupBackend state file case (result) of Nothing -> return False Just b -> do key <- lookupKey state b file (retrieveKeyFile b) state key dest {- Drops the key for a file from the backend that has it. -} dropFile :: State -> FilePath -> IO (Maybe Key) dropFile state file = do result <- lookupBackend state file case (result) of Nothing -> return Nothing Just b -> do key <- lookupKey state b file (removeKey b) state key removeFile $ backendFile state b file return $ Just key {- Looks up the backend used for an already annexed file. -} lookupBackend :: State -> FilePath -> IO (Maybe Backend) lookupBackend state file = lookupBackend' (backends state) state file lookupBackend' [] _ _ = return Nothing lookupBackend' (b:bs) state file = do present <- checkBackend b state file if present then return $ Just b else lookupBackend' bs state file {- Name of state file that holds the key for an annexed file, - using a given backend. -} backendFile :: State -> Backend -> FilePath -> String backendFile state backend file = gitStateDir (repo state) ++ (gitRelative (repo state) file) ++ "." ++ (name backend) {- Checks if a file is available via a given backend. -} checkBackend :: Backend -> State -> FilePath -> IO (Bool) checkBackend backend state file = doesFileExist $ backendFile state backend file {- Looks up the key a backend uses for an already annexed file. -} lookupKey :: State -> Backend -> FilePath -> IO Key lookupKey state backend file = do k <- readFile (backendFile state backend file) return $ chomp k where chomp s = if (endswith "\n" s) then (reverse . (drop 1) . reverse) s else s {- Records the key a backend uses for an annexed file. -} recordKey :: State -> Backend -> FilePath -> Key -> IO () recordKey state backend file key = do createDirectoryIfMissing True (parentDir record) writeFile record $ key ++ "\n" where record = backendFile state backend file