summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--Backend/File.hs17
-rw-r--r--Command/Move.hs4
-rw-r--r--Remotes.hs29
3 files changed, 29 insertions, 21 deletions
diff --git a/Backend/File.hs b/Backend/File.hs
index 092cf7e73..f9a3dbfcb 100644
--- a/Backend/File.hs
+++ b/Backend/File.hs
@@ -16,7 +16,6 @@ module Backend.File (backend, checkKey) where
import Control.Monad.State
import System.Directory
-import Data.List (intersect)
import TypeInternals
import LocationLog
@@ -50,7 +49,8 @@ dummyStore _ _ = return True
- and copy it over to this one. -}
copyKeyFile :: Key -> FilePath -> Annex Bool
copyKeyFile key file = do
- remotes <- Remotes.keyPossibilities key
+ (trusted, untrusted) <- Remotes.keyPossibilities key
+ let remotes = trusted ++ untrusted
if null remotes
then do
showNote "not available"
@@ -92,16 +92,11 @@ checkRemoveKey key numcopiesM = do
if force || numcopiesM == Just 0
then return True
else do
- g <- Annex.gitRepo
- locations <- liftIO $ keyLocations g key
- trusted <- getTrusted
- let trustedcopies = length $ intersect locations trusted
- remotes <- Remotes.keyPossibilities key
- untrustedremotes <- reposWithoutUUID remotes trusted
+ (trusted, untrusted) <- Remotes.keyPossibilities key
numcopies <- getNumCopies numcopiesM
- if numcopies > length untrustedremotes
- then notEnoughCopies numcopies (length untrustedremotes) []
- else findcopies numcopies trustedcopies untrustedremotes []
+ if numcopies > length untrusted
+ then notEnoughCopies numcopies (length untrusted) []
+ else findcopies numcopies (length trusted) untrusted []
where
findcopies need have [] bad
| have >= need = return True
diff --git a/Command/Move.hs b/Command/Move.hs
index cb6525919..61242955e 100644
--- a/Command/Move.hs
+++ b/Command/Move.hs
@@ -110,8 +110,8 @@ toCleanup move remote key tmpfile = do
fromStart :: Bool -> SubCmdStartString
fromStart move file = isAnnexed file $ \(key, _) -> do
remote <- Remotes.commandLineRemote
- l <- Remotes.keyPossibilities key
- if null $ filter (\r -> Remotes.same r remote) l
+ (trusted, untrusted) <- Remotes.keyPossibilities key
+ if null $ filter (\r -> Remotes.same r remote) (trusted ++ untrusted)
then return Nothing
else do
showAction move file
diff --git a/Remotes.hs b/Remotes.hs
index 725348a6a..390531550 100644
--- a/Remotes.hs
+++ b/Remotes.hs
@@ -24,7 +24,7 @@ import qualified Data.Map as Map
import Data.String.Utils
import System.Directory hiding (copyFile)
import System.Posix.Directory
-import Data.List
+import Data.List (intersect, sortBy)
import Control.Monad (when, unless, filterM)
import Types
@@ -43,11 +43,11 @@ import qualified SysConfig
list :: [Git.Repo] -> String
list remotes = join ", " $ map Git.repoDescribe remotes
-{- Cost ordered list of remotes that the LocationLog indicate may have a key. -}
-keyPossibilities :: Key -> Annex [Git.Repo]
+{- Cost ordered lists of remotes that the LocationLog indicate may have a key.
+ - The first list is of remotes that are trusted to have the key; the
+ - second is of untrusted remotes that may have the key. -}
+keyPossibilities :: Key -> Annex ([Git.Repo], [Git.Repo])
keyPossibilities key = do
- g <- Annex.gitRepo
- uuids <- liftIO $ keyLocations g key
allremotes <- remotesByCost
-- To determine if a remote has a key, its UUID needs to be known.
-- The locally cached UUIDs of remotes can fall out of date if
@@ -56,7 +56,7 @@ keyPossibilities key = do
-- sure we only do it once per git-annex run.
remotesread <- Annex.flagIsSet "remotesread"
if remotesread
- then reposByUUID allremotes uuids
+ then partition allremotes
else do
-- We assume that it's cheap to read the config
-- of non-URL remotes, so that is done each time.
@@ -74,11 +74,24 @@ keyPossibilities key = do
_ <- mapM tryGitConfigRead todo
Annex.flagChange "remotesread" $ FlagBool True
keyPossibilities key
- else reposByUUID allremotes uuids
+ else partition allremotes
where
cachedUUID r = do
u <- getUUID r
- return $ null u
+ return $ null u
+ partition allremotes = do
+ g <- Annex.gitRepo
+ validuuids <- liftIO $ keyLocations g key
+ alltrusted <- getTrusted
+ -- get uuids trusted to have the key
+ -- note that validuuids is assumed to not have dups
+ let validtrusted = intersect validuuids alltrusted
+ -- remotes that match uuids that have the key
+ validremotes <- reposByUUID allremotes validuuids
+ -- partition out the trusted and untrusted remotes
+ trustedremotes <- reposByUUID validremotes validtrusted
+ untrustedremotes <- reposWithoutUUID validremotes alltrusted
+ return (trustedremotes, untrustedremotes)
{- Checks if a given remote has the content for a key inAnnex.
- If the remote cannot be accessed, returns a Left error.