big subcommand dispatch rework

not quite done.. head hurts
This commit is contained in:
Joey Hess 2010-11-01 17:01:27 -04:00
parent 59e49ae083
commit fefaa5cc48

View file

@ -27,82 +27,54 @@ import Core
import qualified Remotes import qualified Remotes
import qualified TypeInternals import qualified TypeInternals
{- A subcommand can take one of several kinds of input parameters. -} {- A subcommand runs in four stages. Each stage can return the next stage
data SubCmdInput = FilesInGit FilePath | FilesNotInGit FilePath |
FilesMissing FilePath | Description String | Keys String |
Tempfile FilePath | FilesToBeCommitted FilePath
{- A subcommand runs in three stages. Each stage can return the next stage
- to run. - to run.
- -
- 1. The start stage is run before anything is printed about the - 0. The parse stage takes the parameters passed to the subcommand,
- subcommand, is passed some input, and can early abort it - looks through the repo to find the ones that are relevant
- if the input does not make sense. It should run quickly and - to that subcommand (ie, new files to add), and returns a list of
- should not modify Annex state. - start stage actions to run. -}
- type SubCmdParse = [String] -> Git.Repo -> IO [SubCmdStart]
- 2. The perform stage is run after a message is printed about the subcommand {- 1. The start stage is run before anything is printed about the
- being run, and it should be where the bulk of the work happens. - subcommand, is passed some input, and can early abort it
- - if the input does not make sense. It should run quickly and
- 3. The cleanup stage is run only if the perform stage succeeds, and it - should not modify Annex state. -}
- returns the overall success/fail of the subcommand.
-}
type SubCmdStart = Annex (Maybe SubCmdPerform) type SubCmdStart = Annex (Maybe SubCmdPerform)
{- 2. The perform stage is run after a message is printed about the subcommand
- being run, and it should be where the bulk of the work happens. -}
type SubCmdPerform = Annex (Maybe SubCmdCleanup) type SubCmdPerform = Annex (Maybe SubCmdCleanup)
{- 3. The cleanup stage is run only if the perform stage succeeds, and it
- returns the overall success/fail of the subcommand. -}
type SubCmdCleanup = Annex Bool type SubCmdCleanup = Annex Bool
{- Runs a subcommand through its three stages. -}
doSubCmd :: String -> SubCmdStart -> Annex Bool
doSubCmd cmdname start = do
startres <- start :: Annex (Maybe SubCmdPerform)
case (startres) of
Nothing -> return True
Just perform -> do
--showStart cmdname param
performres <- perform :: Annex (Maybe SubCmdCleanup)
case (performres) of
Nothing -> do
showEndFail
return False
Just cleanup -> do
cleanupres <- cleanup
if (cleanupres)
then do
showEndOk
return True
else do
showEndFail
return False
data SubCommand = SubCommand { data SubCommand = SubCommand {
subcmdname :: String, subcmdname :: String,
subcmdaction :: (SubCmdInput -> SubCmdStart), subcmdparse :: SubCmdParse,
subcmdinput :: (String -> SubCmdInput),
subcmddesc :: String subcmddesc :: String
} }
subCmds :: [SubCommand] subCmds :: [SubCommand]
subCmds = [ subCmds = [
(SubCommand "add" addStart FilesNotInGit (SubCommand "add" (withFilesNotInGit addStart)
"add files to annex") "add files to annex")
, (SubCommand "get" getStart FilesInGit , (SubCommand "get" (withFilesInGit getStart)
"make content of annexed files available") "make content of annexed files available")
, (SubCommand "drop" dropStart FilesInGit , (SubCommand "drop" (withFilesInGit dropStart)
"indicate content of files not currently wanted") "indicate content of files not currently wanted")
, (SubCommand "move" moveStart FilesInGit , (SubCommand "move" (withFilesInGit moveStart)
"transfer content of files to/from another repository") "transfer content of files to/from another repository")
, (SubCommand "init" initStart Description , (SubCommand "init" (withDescription initStart)
"initialize git-annex with repository description") "initialize git-annex with repository description")
, (SubCommand "unannex" unannexStart FilesInGit , (SubCommand "unannex" (withFilesInGit unannexStart)
"undo accidential add command") "undo accidential add command")
, (SubCommand "fix" fixStart FilesInGit , (SubCommand "fix" (withFilesInGit fixStart)
"fix up symlinks to point to annexed content") "fix up symlinks to point to annexed content")
, (SubCommand "pre-commit" fixStart FilesToBeCommitted , (SubCommand "pre-commit" (withFilesToBeCommitted fixStart)
"fix up symlinks before they are committed") "fix up symlinks before they are committed")
, (SubCommand "fromkey" fromKeyStart FilesMissing , (SubCommand "fromkey" (withFilesMissing fromKeyStart)
"adds a file using a specific key") "adds a file using a specific key")
, (SubCommand "dropkey" dropKeyStart Keys , (SubCommand "dropkey" (withKeys dropKeyStart)
"drops annexed content for specified keys") "drops annexed content for specified keys")
, (SubCommand "setkey" setKeyStart Tempfile , (SubCommand "setkey" (withTempFile setKeyStart)
"sets annexed content for a key using a temp file") "sets annexed content for a key using a temp file")
] ]
@ -140,49 +112,66 @@ usage = usageInfo header options ++ "\nSubcommands:\n" ++ cmddescs
showcmd c = showcmd c =
(subcmdname c) ++ (subcmdname c) ++
(pad 11 (subcmdname c)) ++ (pad 11 (subcmdname c)) ++
(descSubCmdInput (subcmdinput c)) ++
(pad 13 (descSubCmdInput (subcmdinput c))) ++
(subcmddesc c) (subcmddesc c)
indent l = " " ++ l indent l = " " ++ l
pad n s = take (n - (length s)) $ repeat ' ' pad n s = take (n - (length s)) $ repeat ' '
{- Generate descriptions of wanted parameters for subcommands. -} {- Prepares a set of actions to run to perform a subcommand, based on
descSubCmdInput :: (String -> SubCmdInput) -> String
descSubCmdInput Description = "DESCRIPTION"
descSubCmdInput Keys = "KEY ..."
descSubCmdInput _ = "PATH ..."
{- Prepares a set of actions to run to handle a subcommand, based on
- the parameters passed to it. -} - the parameters passed to it. -}
prepSubCmd :: SubCommand -> Git.Repo -> [String] -> IO [Annex Bool] prepSubCmd :: SubCommand -> Git.Repo -> [String] -> IO [Annex Bool]
prepSubCmd SubCommand { subcmdname = name, subcmdaction = action, prepSubCmd SubCommand { subcmdname = name, subcmdparse = parse,
subcmdinput = input, subcmddesc = _ } repo params = do subcmddesc = _ } repo params = do
input <- findInput input params repo list <- parse params repo :: IO [SubCmdStart]
return $ map (doSubCmd name action) input return map (\a -> doSubCmd name a) list
{- Finds the type of parameters a subcommand wants, from among the passed {- Runs a subcommand through the perform and cleanup stages -}
- parameter list. -} doSubCmd :: String -> SubCmdPerform -> SubCmdCleanup
findInput :: (String -> SubCmdInput) -> [String] -> Git.Repo -> IO [SubCmdInput] doSubCmd cmdname perform = do
findInput FilesNotInGit params repo = do p <- perform
case (p) of
Nothing -> do
showEndFail
return False
Just cleanup -> do
c <- cleanup
if (c)
then do
showEndOk
return True
else do
showEndFail
return False
{- These functions parse a user's parameters into a list of SubCmdStart
actions to perform. -}
type ParseStrings = (String -> SubCmdStart) -> SubCmdParse
withFilesNotInGit :: ParseStrings
withFilesNotInGit a params repo = do
files <- mapM (Git.notInRepo repo) params files <- mapM (Git.notInRepo repo) params
return $ map FilesNotInGit $ notState $ foldl (++) [] files return $ map a $ notState $ foldl (++) [] files
findInput FilesInGit params repo = do withFilesInGit :: ParseStrings
withFilesInGit a params repo = do
files <- mapM (Git.inRepo repo) params files <- mapM (Git.inRepo repo) params
return $ map FilesInGit $ notState $ foldl (++) [] files return $ map a $ notState $ foldl (++) [] files
findInput FilesMissing params _ = do withFilesMissing :: ParseStrings
withFilesMissing a params _ = do
files <- liftIO $ filterM missing params files <- liftIO $ filterM missing params
return $ map FilesMissing $ notState $ files return $ map a $ notState files
where where
missing f = do missing f = do
e <- doesFileExist f e <- doesFileExist f
return $ not e return $ not e
findInput Description params _ = do withDescription :: ParseStrings
return $ map Description $ [unwords params] withDescription a params _ = do
findInput FilesToBeCommitted params repo = do return $ [a $ unwords params]
withFilesToBeCommitted :: ParseStrings
withFilesToBeCommitted a params repo = do
files <- mapM (Git.stagedFiles repo) params files <- mapM (Git.stagedFiles repo) params
return $ map FilesToBeCommitted $ notState $ foldl (++) [] files return $ map a $ notState $ foldl (++) [] files
findInput Keys params _ = return $ map Keys params withKeys :: ParseStrings
findInput Tempfile params _ = return $ map Tempfile params withKeys a params _ = return $ map a params
withTempFile :: ParseStrings
withTempFile a params _ = return $ map a params
{- filter out files from the state directory -} {- filter out files from the state directory -}
notState :: [FilePath] -> [FilePath] notState :: [FilePath] -> [FilePath]
@ -215,19 +204,19 @@ parseCmd argv state = do
{- The add subcommand annexes a file, storing it in a backend, and then {- The add subcommand annexes a file, storing it in a backend, and then
- moving it into the annex directory and setting up the symlink pointing - moving it into the annex directory and setting up the symlink pointing
- to its content. -} - to its content. -}
addStart :: SubCmdInput -> SubCmdStart addStart :: FilePath -> SubCmdStart
addStart (FilesNotInGit file) = notAnnexed file $ do addStart file = notAnnexed file $ do
s <- liftIO $ getSymbolicLinkStatus file s <- liftIO $ getSymbolicLinkStatus file
if ((isSymbolicLink s) || (not $ isRegularFile s)) if ((isSymbolicLink s) || (not $ isRegularFile s))
then return Nothing then return Nothing
else return $ Just $ addPerform file else return $ Just $ addPerform file
addPerform :: FilePath -> Annex (Maybe SubCmdCleanup) addPerform :: FilePath -> SubCmdPerform
addPerform file = do addPerform file = do
stored <- Backend.storeFileKey file stored <- Backend.storeFileKey file
case (stored) of case (stored) of
Nothing -> return Nothing Nothing -> return Nothing
Just (key, _) -> return $ Just $ addCleanup file key Just (key, _) -> return $ Just $ addCleanup file key
addCleanup :: FilePath -> Key -> Annex Bool addCleanup :: FilePath -> Key -> SubCmdCleanup
addCleanup file key = do addCleanup file key = do
logStatus key ValuePresent logStatus key ValuePresent
g <- Annex.gitRepo g <- Annex.gitRepo
@ -240,10 +229,10 @@ addCleanup file key = do
return True return True
{- The unannex subcommand undoes an add. -} {- The unannex subcommand undoes an add. -}
unannexStart :: SubCmdInput -> SubCmdStart unannexStart :: FilePath -> SubCmdStart
unannexStart (FilesInGit file) = isAnnexed file $ \(key, backend) -> do unannexStart file = isAnnexed file $ \(key, backend) -> do
return $ Just $ unannexPerform file key backend return $ Just $ unannexPerform file key backend
unannexPerform :: FilePath -> Key -> Backend -> Annex (Maybe SubCmdCleanup) unannexPerform :: FilePath -> Key -> Backend -> SubCmdPerform
unannexPerform file key backend = do unannexPerform file key backend = do
-- force backend to always remove -- force backend to always remove
Annex.flagChange "force" $ FlagBool True Annex.flagChange "force" $ FlagBool True
@ -251,7 +240,7 @@ unannexPerform file key backend = do
if (ok) if (ok)
then return $ Just $ unannexCleanup file key then return $ Just $ unannexCleanup file key
else return Nothing else return Nothing
unannexCleanup :: FilePath -> Key -> Annex Bool unannexCleanup :: FilePath -> Key -> SubCmdCleanup
unannexCleanup file key = do unannexCleanup file key = do
logStatus key ValueMissing logStatus key ValueMissing
g <- Annex.gitRepo g <- Annex.gitRepo
@ -264,13 +253,13 @@ unannexCleanup file key = do
return True return True
{- Gets an annexed file from one of the backends. -} {- Gets an annexed file from one of the backends. -}
getStart :: SubCmdInput -> Annex (Maybe SubCmdPerform) getStart :: FilePath -> SubCmdStart
getStart (FilesInGit file) = isAnnexed file $ \(key, backend) -> do getStart file = isAnnexed file $ \(key, backend) -> do
inannex <- inAnnex key inannex <- inAnnex key
if (inannex) if (inannex)
then return Nothing then return Nothing
else return $ Just $ getPerform key backend else return $ Just $ getPerform key backend
getPerform :: Key -> Backend -> Annex (Maybe SubCmdCleanup) getPerform :: Key -> Backend -> SubCmdPerform
getPerform key backend = do getPerform key backend = do
ok <- getViaTmp key (Backend.retrieveKeyFile backend key) ok <- getViaTmp key (Backend.retrieveKeyFile backend key)
if (ok) if (ok)
@ -279,19 +268,19 @@ getPerform key backend = do
{- Indicates a file's content is not wanted anymore, and should be removed {- Indicates a file's content is not wanted anymore, and should be removed
- if it's safe to do so. -} - if it's safe to do so. -}
dropStart :: SubCmdInput -> SubCmdStart dropStart :: FilePath -> SubCmdStart
dropStart (FilesInGit file) = isAnnexed file $ \(key, backend) -> do dropStart file = isAnnexed file $ \(key, backend) -> do
inbackend <- Backend.hasKey key inbackend <- Backend.hasKey key
if (not inbackend) if (not inbackend)
then return Nothing then return Nothing
else return $ Just $ dropPerform key backend else return $ Just $ dropPerform key backend
dropPerform :: Key -> Backend -> Annex (Maybe SubCmdCleanup) dropPerform :: Key -> Backend -> SubCmdPerform
dropPerform key backend = do dropPerform key backend = do
success <- Backend.removeKey backend key success <- Backend.removeKey backend key
if (success) if (success)
then return $ Just $ dropCleanup key then return $ Just $ dropCleanup key
else return Nothing else return Nothing
dropCleanup :: Key -> Annex Bool dropCleanup :: Key -> SubCmdCleanup
dropCleanup key = do dropCleanup key = do
logStatus key ValueMissing logStatus key ValueMissing
inannex <- inAnnex key inannex <- inAnnex key
@ -304,8 +293,8 @@ dropCleanup key = do
else return True else return True
{- Drops cached content for a key. -} {- Drops cached content for a key. -}
dropKeyStart :: SubCmdInput -> SubCmdStart dropKeyStart :: String -> SubCmdStart
dropKeyStart (Keys keyname) = do dropKeyStart keyname = do
backends <- Backend.list backends <- Backend.list
let key = genKey (backends !! 0) keyname let key = genKey (backends !! 0) keyname
present <- inAnnex key present <- inAnnex key
@ -315,26 +304,26 @@ dropKeyStart (Keys keyname) = do
else if (not force) else if (not force)
then error "dropkey is can cause data loss; use --force if you're sure you want to do this" then error "dropkey is can cause data loss; use --force if you're sure you want to do this"
else return $ Just $ dropKeyPerform key else return $ Just $ dropKeyPerform key
dropKeyPerform :: Key -> Annex (Maybe SubCmdCleanup) dropKeyPerform :: Key -> SubCmdPerform
dropKeyPerform key = do dropKeyPerform key = do
g <- Annex.gitRepo g <- Annex.gitRepo
let loc = annexLocation g key let loc = annexLocation g key
liftIO $ removeFile loc liftIO $ removeFile loc
return $ Just $ dropKeyCleanup key return $ Just $ dropKeyCleanup key
dropKeyCleanup :: Key -> Annex Bool dropKeyCleanup :: Key -> SubCmdCleanup
dropKeyCleanup key = do dropKeyCleanup key = do
logStatus key ValueMissing logStatus key ValueMissing
return True return True
{- Sets cached content for a key. -} {- Sets cached content for a key. -}
setKeyStart :: SubCmdInput -> SubCmdStart setKeyStart :: FilePath -> SubCmdStart
setKeyStart (Tempfile tmpfile) = do setKeyStart tmpfile = do
keyname <- Annex.flagGet "key" keyname <- Annex.flagGet "key"
when (null keyname) $ error "please specify the key with --key" when (null keyname) $ error "please specify the key with --key"
backends <- Backend.list backends <- Backend.list
let key = genKey (backends !! 0) keyname let key = genKey (backends !! 0) keyname
return $ Just $ setKeyPerform tmpfile key return $ Just $ setKeyPerform tmpfile key
setKeyPerform :: FilePath -> Key -> Annex (Maybe SubCmdCleanup) setKeyPerform :: FilePath -> Key -> SubCmdPerform
setKeyPerform tmpfile key = do setKeyPerform tmpfile key = do
g <- Annex.gitRepo g <- Annex.gitRepo
let loc = annexLocation g key let loc = annexLocation g key
@ -342,40 +331,37 @@ setKeyPerform tmpfile key = do
if (not ok) if (not ok)
then error "mv failed!" then error "mv failed!"
else return $ Just $ setKeyCleanup key else return $ Just $ setKeyCleanup key
setKeyCleanup :: Key -> Annex Bool setKeyCleanup :: Key -> SubCmdCleanup
setKeyCleanup key = do setKeyCleanup key = do
logStatus key ValuePresent logStatus key ValuePresent
return True return True
{- Fixes the symlink to an annexed file. -} {- Fixes the symlink to an annexed file. -}
fixStart :: SubCmdInput -> SubCmdStart fixStart :: FilePath -> SubCmdStart
fixStart (FilesInGit file) = fixStart' file fixStart file = isAnnexed file $ \(key, _) -> do
fixStart (FilesToBeCommitted file) = fixStart' file
fixStart' :: FilePath -> SubCmdStart
fixStart' file = isAnnexed file $ \(key, _) -> do
link <- calcGitLink file key link <- calcGitLink file key
l <- liftIO $ readSymbolicLink file l <- liftIO $ readSymbolicLink file
if (link == l) if (link == l)
then return Nothing then return Nothing
else return $ Just $ fixPerform file link else return $ Just $ fixPerform file link
fixPerform :: FilePath -> FilePath -> Annex (Maybe SubCmdCleanup) fixPerform :: FilePath -> FilePath -> SubCmdPerform
fixPerform file link = do fixPerform file link = do
liftIO $ createDirectoryIfMissing True (parentDir file) liftIO $ createDirectoryIfMissing True (parentDir file)
liftIO $ removeFile file liftIO $ removeFile file
liftIO $ createSymbolicLink link file liftIO $ createSymbolicLink link file
return $ Just $ fixCleanup file return $ Just $ fixCleanup file
fixCleanup :: FilePath -> Annex Bool fixCleanup :: FilePath -> SubCmdCleanup
fixCleanup file = do fixCleanup file = do
Annex.queue "add" [] file Annex.queue "add" [] file
return True return True
{- Stores description for the repository etc. -} {- Stores description for the repository etc. -}
initStart :: SubCmdInput -> SubCmdStart initStart :: String -> SubCmdStart
initStart (Description description) = do initStart description = do
when (null description) $ error $ when (null description) $ error $
"please specify a description of this repository\n" ++ usage "please specify a description of this repository\n" ++ usage
return $ Just $ initPerform description return $ Just $ initPerform description
initPerform :: String -> Annex (Maybe SubCmdCleanup) initPerform :: String -> SubCmdPerform
initPerform description = do initPerform description = do
g <- Annex.gitRepo g <- Annex.gitRepo
u <- getUUID g u <- getUUID g
@ -383,7 +369,7 @@ initPerform description = do
liftIO $ gitAttributes g liftIO $ gitAttributes g
liftIO $ gitPreCommitHook g liftIO $ gitPreCommitHook g
return $ Just $ initCleanup return $ Just $ initCleanup
initCleanup :: Annex Bool initCleanup :: SubCmdCleanup
initCleanup = do initCleanup = do
g <- Annex.gitRepo g <- Annex.gitRepo
logfile <- uuidLog logfile <- uuidLog
@ -392,8 +378,8 @@ initCleanup = do
return True return True
{- Adds a file pointing at a manually-specified key -} {- Adds a file pointing at a manually-specified key -}
fromKeyStart :: SubCmdInput -> SubCmdStart fromKeyStart :: FilePath -> SubCmdStart
fromKeyStart (FilesMissing file) = do fromKeyStart file = do
keyname <- Annex.flagGet "key" keyname <- Annex.flagGet "key"
when (null keyname) $ error "please specify the key with --key" when (null keyname) $ error "please specify the key with --key"
backends <- Backend.list backends <- Backend.list
@ -403,13 +389,13 @@ fromKeyStart (FilesMissing file) = do
unless (inbackend) $ error $ unless (inbackend) $ error $
"key ("++keyname++") is not present in backend" "key ("++keyname++") is not present in backend"
return $ Just $ fromKeyPerform file key return $ Just $ fromKeyPerform file key
fromKeyPerform :: FilePath -> Key -> Annex (Maybe SubCmdCleanup) fromKeyPerform :: FilePath -> Key -> SubCmdPerform
fromKeyPerform file key = do fromKeyPerform file key = do
link <- calcGitLink file key link <- calcGitLink file key
liftIO $ createDirectoryIfMissing True (parentDir file) liftIO $ createDirectoryIfMissing True (parentDir file)
liftIO $ createSymbolicLink link file liftIO $ createSymbolicLink link file
return $ Just $ fromKeyCleanup file return $ Just $ fromKeyCleanup file
fromKeyCleanup :: FilePath -> Annex Bool fromKeyCleanup :: FilePath -> SubCmdCleanup
fromKeyCleanup file = do fromKeyCleanup file = do
Annex.queue "add" [] file Annex.queue "add" [] file
return True return True
@ -418,8 +404,8 @@ fromKeyCleanup file = do
- -
- This only operates on the cached file content; it does not involve - This only operates on the cached file content; it does not involve
- moving data in the key-value backend. -} - moving data in the key-value backend. -}
moveStart :: SubCmdInput -> SubCmdStart moveStart :: FilePath -> SubCmdStart
moveStart (FilesInGit file) = do moveStart file = do
fromName <- Annex.flagGet "fromrepository" fromName <- Annex.flagGet "fromrepository"
toName <- Annex.flagGet "torepository" toName <- Annex.flagGet "torepository"
case (fromName, toName) of case (fromName, toName) of
@ -439,13 +425,13 @@ moveStart (FilesInGit file) = do
- A file's content can be moved even if there are insufficient copies to - A file's content can be moved even if there are insufficient copies to
- allow it to be dropped. - allow it to be dropped.
-} -}
moveToStart :: FilePath -> Annex (Maybe SubCmdPerform) moveToStart :: FilePath -> SubCmdStart
moveToStart file = isAnnexed file $ \(key, _) -> do moveToStart file = isAnnexed file $ \(key, _) -> do
ishere <- inAnnex key ishere <- inAnnex key
if (not ishere) if (not ishere)
then return Nothing -- not here, so nothing to do then return Nothing -- not here, so nothing to do
else return $ Just $ moveToPerform key else return $ Just $ moveToPerform key
moveToPerform :: Key -> Annex (Maybe SubCmdCleanup) moveToPerform :: Key -> SubCmdPerform
moveToPerform key = do moveToPerform key = do
-- checking the remote is expensive, so not done in the start step -- checking the remote is expensive, so not done in the start step
remote <- Remotes.commandLineRemote remote <- Remotes.commandLineRemote
@ -462,7 +448,7 @@ moveToPerform key = do
then return $ Just $ moveToCleanup remote key tmpfile then return $ Just $ moveToCleanup remote key tmpfile
else return Nothing -- failed else return Nothing -- failed
Right True -> return $ Just $ dropCleanup key Right True -> return $ Just $ dropCleanup key
moveToCleanup :: Git.Repo -> Key -> FilePath -> Annex Bool moveToCleanup :: Git.Repo -> Key -> FilePath -> SubCmdCleanup
moveToCleanup remote key tmpfile = do moveToCleanup remote key tmpfile = do
-- Tell remote to use the transferred content. -- Tell remote to use the transferred content.
ok <- Remotes.runCmd remote "git-annex" ["setkey", "--quiet", ok <- Remotes.runCmd remote "git-annex" ["setkey", "--quiet",
@ -487,14 +473,14 @@ moveToCleanup remote key tmpfile = do
- If the current repository already has the content, it is still removed - If the current repository already has the content, it is still removed
- from the other repository. - from the other repository.
-} -}
moveFromStart :: FilePath -> Annex (Maybe SubCmdPerform) moveFromStart :: FilePath -> SubCmdStart
moveFromStart file = isAnnexed file $ \(key, _) -> do moveFromStart file = isAnnexed file $ \(key, _) -> do
remote <- Remotes.commandLineRemote remote <- Remotes.commandLineRemote
l <- Remotes.keyPossibilities key l <- Remotes.keyPossibilities key
if (not $ null $ filter (\r -> Remotes.same r remote) l) if (not $ null $ filter (\r -> Remotes.same r remote) l)
then return $ Just $ moveFromPerform key then return $ Just $ moveFromPerform key
else return Nothing else return Nothing
moveFromPerform :: Key -> Annex (Maybe SubCmdCleanup) moveFromPerform :: Key -> SubCmdPerform
moveFromPerform key = do moveFromPerform key = do
remote <- Remotes.commandLineRemote remote <- Remotes.commandLineRemote
ishere <- inAnnex key ishere <- inAnnex key
@ -506,7 +492,7 @@ moveFromPerform key = do
if (ok) if (ok)
then return $ Just $ moveFromCleanup remote key then return $ Just $ moveFromCleanup remote key
else return Nothing -- fail else return Nothing -- fail
moveFromCleanup :: Git.Repo -> Key -> Annex Bool moveFromCleanup :: Git.Repo -> Key -> SubCmdCleanup
moveFromCleanup remote key = do moveFromCleanup remote key = do
ok <- Remotes.runCmd remote "git-annex" ["dropkey", "--quiet", "--force", ok <- Remotes.runCmd remote "git-annex" ["dropkey", "--quiet", "--force",
"--backend=" ++ (backendName key), "--backend=" ++ (backendName key),