git-annex/Backend/File.hs

151 lines
4.3 KiB
Haskell
Raw Normal View History

2010-10-15 20:42:36 +00:00
{- git-annex pseudo-backend
-
- This backend does not really do any independant data storage,
- it relies on the file contents in .git/annex/ in this repo,
- and other accessible repos.
-
- This is an abstract backend; getKey has to be implemented to complete
- it.
-}
2010-10-10 17:47:04 +00:00
2010-10-14 07:50:28 +00:00
module Backend.File (backend) where
2010-10-10 17:47:04 +00:00
2010-10-14 01:28:47 +00:00
import Control.Monad.State
2010-10-13 20:21:50 +00:00
import System.IO
import System.Cmd
2010-10-19 05:19:56 +00:00
import System.Cmd.Utils
2010-10-13 20:21:50 +00:00
import Control.Exception
2010-10-19 17:39:53 +00:00
import List
import Maybe
2010-10-16 20:20:49 +00:00
2010-10-18 06:06:27 +00:00
import TypeInternals
import LocationLog
import Locations
2010-10-14 06:41:54 +00:00
import qualified Remotes
2010-10-14 06:36:41 +00:00
import qualified GitRepo as Git
2010-10-14 19:31:44 +00:00
import Utility
2010-10-14 20:13:43 +00:00
import Core
2010-10-14 21:37:20 +00:00
import qualified Annex
import UUID
import qualified Backend
2010-10-10 17:47:04 +00:00
backend = Backend {
2010-10-15 23:32:56 +00:00
name = mustProvide,
getKey = mustProvide,
2010-10-10 19:41:35 +00:00
storeFileKey = dummyStore,
retrieveKeyFile = copyKeyFile,
removeKey = checkRemoveKey,
2010-10-14 19:31:44 +00:00
hasKey = checkKeyFile
2010-10-10 17:47:04 +00:00
}
2010-10-15 23:32:56 +00:00
mustProvide = error "must provide this field"
2010-10-15 20:42:36 +00:00
{- Storing a key is a no-op. -}
2010-10-14 01:28:47 +00:00
dummyStore :: FilePath -> Key -> Annex (Bool)
dummyStore file key = return True
2010-10-14 18:14:19 +00:00
2010-10-14 19:31:44 +00:00
{- Just check if the .git/annex/ file for the key exists. -}
checkKeyFile :: Key -> Annex Bool
2010-10-14 23:36:11 +00:00
checkKeyFile k = inAnnex k
2010-10-14 19:31:44 +00:00
{- Try to find a copy of the file in one of the remotes,
- and copy it over to this one. -}
2010-10-14 01:28:47 +00:00
copyKeyFile :: Key -> FilePath -> Annex (Bool)
copyKeyFile key file = do
2010-10-14 06:41:54 +00:00
remotes <- Remotes.withKey key
2010-10-14 21:37:20 +00:00
if (0 == length remotes)
2010-10-19 17:39:53 +00:00
then do
showNote "not available"
2010-10-19 17:39:53 +00:00
showLocations key
return False
2010-10-17 17:13:49 +00:00
else trycopy remotes remotes
where
2010-10-17 17:13:49 +00:00
trycopy full [] = do
showNote "not available"
showTriedRemotes full
showLocations key
2010-10-17 17:13:49 +00:00
return False
trycopy full (r:rs) = do
2010-10-14 02:59:43 +00:00
-- annexLocation needs the git config to have been
-- read for a remote, so do that now,
-- if it hasn't been already
result <- Remotes.tryGitConfigRead r
case (result) of
Left err -> trycopy full rs
Right r' -> do
2010-10-17 17:13:49 +00:00
showNote $ "copying from " ++ (Git.repoDescribe r ) ++ "..."
2010-10-19 05:46:07 +00:00
liftIO $ copyFromRemote r' key file
2010-10-19 05:46:07 +00:00
{- Tries to copy a file from a remote. -}
copyFromRemote :: Git.Repo -> Key -> FilePath -> IO Bool
copyFromRemote r key file = do
2010-10-14 06:36:41 +00:00
if (Git.repoIsLocal r)
2010-10-14 02:59:43 +00:00
then getlocal
else getremote
2010-10-13 20:21:50 +00:00
where
2010-10-19 05:46:07 +00:00
getlocal = boolSystem "cp" ["-a", location, file]
getremote = return False -- TODO implement get from remote
2010-10-14 23:36:11 +00:00
location = annexLocation r key
2010-10-19 17:39:53 +00:00
showLocations :: Key -> Annex ()
showLocations key = do
g <- Annex.gitRepo
u <- getUUID g
uuids <- liftIO $ keyLocations g key
let uuidsf = filter (\v -> v /= u) uuids
ppuuids <- prettyPrintUUIDs uuidsf
if (0 < length uuidsf)
then showLongNote $ "Try making some of these repositories available:\n" ++ ppuuids
else showLongNote $ "No other repository is known to contain the file."
showTriedRemotes remotes =
showLongNote $ "I was unable to access these remotes: " ++
(Remotes.list remotes)
{- Checks remotes to verify that enough copies of a key exist to allow
- for a key to be safely removed (with no data loss), and fails with an
- error if not. -}
checkRemoveKey :: Key -> Annex (Bool)
checkRemoveKey key = do
force <- Annex.flagIsSet "force"
if (force)
then return True
else do
g <- Annex.gitRepo
remotes <- Remotes.withKey key
let numcopies = read $ Git.configGet g config "1"
if (numcopies > length remotes)
then notEnoughCopies numcopies (length remotes) []
else findcopies numcopies 0 remotes []
where
2010-10-17 22:52:09 +00:00
config = "annex.numcopies"
findcopies need have [] bad =
if (have >= need)
then return True
else notEnoughCopies need have bad
findcopies need have (r:rs) bad = do
all <- Annex.supportedBackends
result <- liftIO $ ((try $ remoteHasKey r all)::IO (Either SomeException Bool))
case (result) of
Right True -> findcopies need (have+1) rs bad
Right False -> findcopies need have rs bad
Left _ -> findcopies need have rs (r:bad)
remoteHasKey r all = do
-- To check if a remote has a key, construct a new
-- Annex monad and query its backend.
a <- Annex.new r all
(result, _) <- Annex.run a (Backend.hasKey key)
return result
notEnoughCopies need have bad = do
2010-10-17 17:13:49 +00:00
unsafe
2010-10-19 17:39:53 +00:00
showLongNote $
"Could only verify the existence of " ++
(show have) ++ " out of " ++ (show need) ++
2010-10-19 17:39:53 +00:00
" necessary copies"
if (0 /= length bad) then showTriedRemotes bad else return ()
2010-10-19 17:39:53 +00:00
showLocations key
hint
2010-10-17 17:13:49 +00:00
return False
2010-10-19 17:39:53 +00:00
unsafe = showNote "unsafe"
hint = showLongNote $ "(Use --force to override this check, or adjust annex.numcopies.)"