git-annex/Command.hs

145 lines
4.3 KiB
Haskell
Raw Normal View History

2011-09-15 20:57:02 +00:00
{- git-annex command infrastructure
-
2011-09-19 05:37:04 +00:00
- Copyright 2010-2011 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Command (
command,
noRepo,
2012-09-16 00:46:38 +00:00
noCommit,
withOptions,
next,
stop,
stopUnless,
prepCommand,
doCommand,
whenAnnexed,
ifAnnexed,
notBareRepo,
isBareRepo,
numCopies,
2011-11-11 05:52:58 +00:00
autoCopies,
autoCopiesWith,
checkAuto,
2011-11-11 05:52:58 +00:00
module ReExported
) where
2011-10-05 20:02:51 +00:00
import Common.Annex
2010-11-04 17:28:49 +00:00
import qualified Backend
import qualified Annex
import qualified Git
import qualified Remote
2011-11-11 05:52:58 +00:00
import Types.Command as ReExported
import Types.Option as ReExported
2011-11-11 05:52:58 +00:00
import Seek as ReExported
import Checks as ReExported
import Usage as ReExported
2011-10-15 20:21:08 +00:00
import Logs.Trust
import Config
import Annex.CheckAttr
{- Generates a normal command -}
command :: String -> String -> [CommandSeek] -> String -> Command
command = Command [] Nothing commonChecks False
2012-09-16 00:46:38 +00:00
{- Indicates that a command doesn't need to commit any changes to
- the git-annex branch. -}
noCommit :: Command -> Command
noCommit c = c { cmdnocommit = True }
{- Adds a fallback action to a command, that will be run if it's used
- outside a git repository. -}
noRepo :: IO () -> Command -> Command
noRepo a c = c { cmdnorepo = Just a }
2010-11-04 17:28:49 +00:00
{- Adds options to a command. -}
withOptions :: [Option] -> Command -> Command
withOptions o c = c { cmdoptions = o }
2011-05-15 06:02:46 +00:00
{- For start and perform stages to indicate what step to run next. -}
next :: a -> Annex (Maybe a)
next a = return $ Just a
{- Or to indicate nothing needs to be done. -}
stop :: Annex (Maybe a)
stop = return Nothing
{- Stops unless a condition is met. -}
stopUnless :: Annex Bool -> Annex (Maybe a) -> Annex (Maybe a)
stopUnless c a = ifM c ( a , stop )
2011-10-31 00:04:15 +00:00
{- Prepares to run a command via the check and seek stages, returning a
- list of actions to perform to run the command. -}
prepCommand :: Command -> [String] -> Annex [CommandCleanup]
prepCommand Command { cmdseek = seek, cmdcheck = c } params = do
2011-11-11 05:52:58 +00:00
mapM_ runCheck c
2011-10-31 00:04:15 +00:00
map doCommand . concat <$> mapM (\s -> s params) seek
2010-11-04 17:28:49 +00:00
{- Runs a command through the start, perform and cleanup stages -}
doCommand :: CommandStart -> CommandCleanup
2011-05-15 16:25:58 +00:00
doCommand = start
where
2011-10-31 00:04:15 +00:00
start = stage $ maybe skip perform
2011-07-05 18:58:33 +00:00
perform = stage $ maybe failure cleanup
cleanup = stage $ status
2011-09-19 05:37:04 +00:00
stage = (=<<)
2011-10-31 00:04:15 +00:00
skip = return True
2011-09-19 05:37:04 +00:00
failure = showEndFail >> return False
status r = showEndResult r >> return r
2010-11-04 17:28:49 +00:00
{- Modifies an action to only act on files that are already annexed,
- and passes the key and backend on to it. -}
2011-12-31 08:11:39 +00:00
whenAnnexed :: (FilePath -> (Key, Backend) -> Annex (Maybe a)) -> FilePath -> Annex (Maybe a)
whenAnnexed a file = ifAnnexed file (a file) (return Nothing)
2011-12-31 08:11:39 +00:00
ifAnnexed :: FilePath -> ((Key, Backend) -> Annex a) -> Annex a -> Annex a
ifAnnexed file yes no = maybe no yes =<< Backend.lookupFile file
notBareRepo :: Annex a -> Annex a
notBareRepo a = do
2011-10-29 22:47:53 +00:00
whenM isBareRepo $
error "You cannot run this subcommand in a bare repository."
a
isBareRepo :: Annex Bool
isBareRepo = fromRepo Git.repoIsLocalBare
numCopies :: FilePath -> Annex (Maybe Int)
numCopies file = readish <$> checkAttr "annex.numcopies" file
{- Used for commands that have an auto mode that checks the number of known
- copies of a key.
-
- In auto mode, first checks that the number of known
- copies of the key is > or < than the numcopies setting, before running
- the action. Also checks any preferred content settings.
-}
autoCopies :: FilePath -> Key -> (Int -> Int -> Bool) -> CommandStart -> CommandStart
autoCopies file key vs a = Annex.getState Annex.auto >>= go
where
go False = a
go True = do
numcopiesattr <- numCopies file
needed <- getNumCopies numcopiesattr
(_, have) <- trustPartition UnTrusted =<< Remote.keyLocations key
if length have `vs` needed then a else stop
autoCopiesWith :: FilePath -> Key -> (Int -> Int -> Bool) -> (Maybe Int -> CommandStart) -> CommandStart
autoCopiesWith file key vs a = do
numcopiesattr <- numCopies file
Annex.getState Annex.auto >>= auto numcopiesattr
2011-11-19 19:27:10 +00:00
where
auto numcopiesattr False = a numcopiesattr
auto numcopiesattr True = do
needed <- getNumCopies numcopiesattr
(_, have) <- trustPartition UnTrusted =<< Remote.keyLocations key
if length have `vs` needed
then a numcopiesattr
else stop
checkAuto :: Annex Bool -> Annex Bool
checkAuto checker = ifM (Annex.getState Annex.auto)
( checker , return True )