module Darcs.UI.RunCommand ( runTheCommand ) where
import Prelude ()
import Darcs.Prelude
import Data.List ( intercalate )
import Control.Monad ( unless, when )
import System.Console.GetOpt( ArgOrder( Permute, RequireOrder ),
OptDescr( Option ),
getOpt )
import System.Exit ( ExitCode ( ExitSuccess ), exitWith )
import Darcs.UI.Options ( (^), odesc, oparse, parseFlags, optDescr, (?) )
import Darcs.UI.Options.All
( stdCmdActions, StdCmdAction(..)
, anyVerbosity, verbosity, Verbosity(..), network, NetworkOptions(..)
, HooksConfig(..), hooks )
import Darcs.UI.Defaults ( applyDefaults )
import Darcs.UI.External ( viewDoc )
import Darcs.UI.Flags ( DarcsFlag (NewRepo), matchAny, fixRemoteRepos )
import Darcs.UI.Commands
( CommandArgs( CommandOnly, SuperCommandOnly, SuperCommandSub )
, CommandControl
, DarcsCommand
, commandName
, commandCommand
, commandPrereq
, commandExtraArgHelp
, commandExtraArgs
, commandArgdefaults
, commandCompleteArgs
, commandOptions
, commandParseOptions
, wrappedCommandName
, disambiguateCommands
, getSubcommands
, extractCommands
, superName
)
import Darcs.UI.Commands.GZCRCs ( doCRCWarnings )
import Darcs.UI.Commands.Clone ( makeRepoName, cloneToSSH )
import Darcs.UI.Usage
( getCommandHelp
, getCommandMiniHelp
, subusage
)
import Darcs.Patch.Match ( checkMatchSyntax )
import Darcs.Repository.Prefs ( getGlobal, getPreflist )
import Darcs.Repository.Test ( runPosthook, runPrehook )
import Darcs.Util.AtExit ( atexit )
import Darcs.Util.Download ( setDebugHTTP, disableHTTPPipelining )
import Darcs.Util.Exception ( die )
import Darcs.Util.Global ( setDebugMode, setTimingsMode )
import Darcs.Util.Path ( AbsolutePath, getCurrentDirectory, toPath, ioAbsoluteOrRemote, makeAbsolute )
import Darcs.Util.Progress ( setProgressMode )
import Darcs.Util.Text ( chompTrailingNewline, quote )
runTheCommand :: [CommandControl] -> String -> [String] -> IO ()
runTheCommand commandControlList cmd args =
either die rtc $ disambiguateCommands commandControlList cmd args
where
rtc (CommandOnly c, as) = runCommand Nothing c as
rtc (SuperCommandOnly c, as) = runRawSupercommand c as
rtc (SuperCommandSub c s, as) = runCommand (Just c) s as
runCommand :: Maybe (DarcsCommand pf1) -> DarcsCommand pf2 -> [String] -> IO ()
runCommand _ _ args
| "-all" `elem` args =
die "Are you sure you didn't mean --all rather than -all?"
runCommand msuper cmd args = do
old_wd <- getCurrentDirectory
let options = commandOptions old_wd cmd
case fixupMsgs $ getOpt Permute options args of
(cmdline_flags,orig_extra,getopt_errs) -> do
prereq_errors <- commandPrereq cmd cmdline_flags
new_wd <- getCurrentDirectory
user_defs <- getGlobal "defaults"
repo_defs <- getPreflist "defaults"
let (flags,flag_errors) =
applyDefaults (fmap commandName msuper) cmd old_wd user_defs repo_defs cmdline_flags
case parseFlags stdCmdActions flags of
Just Help -> viewDoc $ getCommandHelp msuper cmd
Just ListOptions -> do
setProgressMode False
possible_args <- commandCompleteArgs cmd (new_wd, old_wd) flags orig_extra
mapM_ putStrLn $ optionList options ++ possible_args
Just Disable ->
die $ "Command "++commandName cmd++" disabled with --disable option!"
Nothing -> case prereq_errors of
Left complaint -> die $
"Unable to " ++ quote ("darcs " ++ superName msuper ++ commandName cmd) ++
" here.\n\n" ++ complaint
Right () -> case getopt_errs ++ flag_errors of
[] -> do
extra <- commandArgdefaults cmd flags old_wd orig_extra
case extraArgumentsError extra cmd msuper of
Nothing -> runWithHooks cmd (new_wd, old_wd) flags extra
Just msg -> die msg
es -> die (intercalate "\n" es)
fixupMsgs :: (a, b, [String]) -> (a, b, [String])
fixupMsgs (fs,as,es) = (fs,as,map (("command line: "++).chompTrailingNewline) es)
runWithHooks :: DarcsCommand pf
-> (AbsolutePath, AbsolutePath)
-> [DarcsFlag] -> [String] -> IO ()
runWithHooks cmd (new_wd, old_wd) flags extra = do
checkMatchSyntax $ matchAny ? flags
oparse (anyVerbosity ^ network) setGlobalVariables flags
let hooksCfg = parseFlags hooks flags
let verb = parseFlags verbosity flags
preHookExitCode <- runPrehook (pre hooksCfg) verb new_wd
if preHookExitCode /= ExitSuccess
then exitWith preHookExitCode
else do fixedFlags <- fixRemoteRepos old_wd flags
phDir <- getPosthookDir new_wd cmd fixedFlags extra
let parsedFlags = commandParseOptions cmd fixedFlags
commandCommand cmd (new_wd, old_wd) parsedFlags extra
postHookExitCode <- runPosthook (post hooksCfg) verb phDir
exitWith postHookExitCode
setGlobalVariables :: Bool -> Bool -> Verbosity -> Bool -> NetworkOptions -> IO ()
setGlobalVariables debug debugHttp verb timings net = do
when timings setTimingsMode
when debug setDebugMode
when debugHttp setDebugHTTP
when (verb == Quiet) $ setProgressMode False
when (noHttpPipelining net) disableHTTPPipelining
unless (verb == Quiet) $ atexit $ doCRCWarnings (verb == Verbose)
getPosthookDir :: AbsolutePath -> DarcsCommand pf -> [DarcsFlag] -> [String] -> IO AbsolutePath
getPosthookDir new_wd cmd flags extra | commandName cmd `elem` ["get","clone"] = do
case extra of
[inrepodir, outname] -> getPosthookDir new_wd cmd (NewRepo outname:flags) [inrepodir]
[inrepodir] ->
case cloneToSSH flags of
Nothing -> do
repodir <- toPath <$> ioAbsoluteOrRemote inrepodir
reponame <- makeRepoName False flags repodir
return $ makeAbsolute new_wd reponame
_ -> return new_wd
_ -> die "You must provide 'clone' with either one or two arguments."
getPosthookDir new_wd _ _ _ = return new_wd
extraArgumentsError :: [String]
-> DarcsCommand pf1
-> Maybe (DarcsCommand pf2)
-> Maybe String
extraArgumentsError extra cmd msuper
| extraArgsCmd < 0 = Nothing
| extraArgsInput > extraArgsCmd = Just badArg
| extraArgsInput < extraArgsCmd = Just missingArg
| otherwise = Nothing
where
extraArgsInput = length extra
extraArgsCmd = commandExtraArgs cmd
badArg = "Bad argument: `" ++ unwords extra ++
"'\n" ++ getCommandMiniHelp msuper cmd
missingArg = "Missing argument: " ++ nthArg (length extra + 1) ++
"\n" ++ getCommandMiniHelp msuper cmd
nthArg n = nthOf n (commandExtraArgHelp cmd)
nthOf 1 (h:_) = h
nthOf n (_:hs) = nthOf (n1) hs
nthOf _ [] = "UNDOCUMENTED"
optionList :: [OptDescr DarcsFlag] -> [String]
optionList = concatMap names
where
names (Option sos los _ desc) =
map (short desc) sos ++ map (long desc) los
short d o = '-' : o : ";" ++ d
long d o = "--" ++ o ++ ";" ++ d
runRawSupercommand :: DarcsCommand pf -> [String] -> IO ()
runRawSupercommand super [] =
die $ "Command '"++ commandName super ++"' requires a subcommand!\n\n"
++ subusage super
runRawSupercommand super args = do
cwd <- getCurrentDirectory
case fixupMsgs $ getOpt RequireOrder (map (optDescr cwd) (odesc stdCmdActions)) args of
(flags,_,getopt_errs) -> case parseFlags stdCmdActions flags of
Just Help ->
viewDoc $ getCommandHelp Nothing super
Just ListOptions -> do
putStrLn "--help"
mapM_ (putStrLn . wrappedCommandName) (extractCommands $ getSubcommands super)
Just Disable -> do
die $ "Command " ++ commandName super ++
" disabled with --disable option!"
Nothing -> die $ case getopt_errs of
[] -> "Invalid subcommand!\n\n" ++ subusage super
_ -> intercalate "\n" getopt_errs