module Darcs.UI.Commands.Remove ( remove, rm, unadd ) where
import Prelude ()
import Darcs.Prelude
import Control.Monad ( when, foldM )
import Darcs.UI.Commands
( DarcsCommand(..)
, withStdOpts, nodefaults
, commandAlias, commandStub
, putWarning, putInfo
, amInHashedRepository
)
import Darcs.UI.Commands.Util ( expandDirs )
import Darcs.UI.Completion ( knownFileArgs )
import Darcs.UI.Flags
( DarcsFlag, useCache, dryRun, umask, diffAlgorithm, fixSubPaths, quiet )
import Darcs.UI.Options ( (^), odesc, ocheck, onormalise, defaultFlags, parseFlags, (?) )
import qualified Darcs.UI.Options.All as O
import Darcs.Repository.Flags ( UpdateWorking (..) )
import Darcs.Repository
( Repository
, withRepoLock
, RepoJob(..)
, addToPending
, readRecordedAndPending
, readUnrecorded
)
import Darcs.Repository.Diff( treeDiff )
import Darcs.Patch ( RepoPatch, PrimOf, PrimPatch, adddir, rmdir, addfile, rmfile,
listTouchedFiles )
import Darcs.Patch.Apply( ApplyState )
import Darcs.Patch.Witnesses.Ordered ( FL(..), (+>+), nullFL )
import Darcs.Patch.Witnesses.Sealed ( Sealed(..), Gap(..), FreeLeft, unFreeLeft )
import Darcs.Repository.Prefs ( filetypeFunction, FileType )
import Darcs.Util.Tree( Tree, TreeItem(..), find, modifyTree, expand, list )
import Darcs.Util.Path( anchorPath, AnchoredPath, fn2fp, SubPath, sp2fn
, AbsolutePath, floatPath )
import Darcs.Util.Printer ( text, vcat )
removeDescription :: String
removeDescription = "Remove files from version control."
removeHelp :: String
removeHelp =
"The `darcs remove` command exists primarily for symmetry with `darcs\n" ++
"add`, as the normal way to remove a file from version control is\n" ++
"simply to delete it from the working tree. This command is only\n" ++
"useful in the unusual case where one wants to record a removal patch\n" ++
"WITHOUT deleting the copy in the working tree (which can be re-added).\n" ++
"\n" ++
"Note that applying a removal patch to a repository (e.g. by pulling\n" ++
"the patch) will ALWAYS affect the working tree of that repository.\n"
remove :: DarcsCommand [DarcsFlag]
remove = DarcsCommand
{ commandProgramName = "darcs"
, commandName = "remove"
, commandHelp = removeHelp
, commandDescription = removeDescription
, commandExtraArgs = 1
, commandExtraArgHelp = ["<FILE or DIRECTORY> ..."]
, commandCommand = removeCmd
, commandPrereq = amInHashedRepository
, commandCompleteArgs = knownFileArgs
, commandArgdefaults = nodefaults
, commandAdvancedOptions = odesc removeAdvancedOpts
, commandBasicOptions = odesc removeBasicOpts
, commandDefaults = defaultFlags removeOpts
, commandCheckOptions = ocheck removeOpts
, commandParseOptions = onormalise removeOpts
}
where
removeBasicOpts = O.repoDir ^ O.recursive
removeAdvancedOpts = O.umask
removeOpts = removeBasicOpts `withStdOpts` removeAdvancedOpts
removeCmd :: (AbsolutePath, AbsolutePath) -> [DarcsFlag] -> [String] -> IO ()
removeCmd fps opts relargs = do
when (null relargs) $
fail "Nothing specified, nothing removed."
origfiles <- fixSubPaths fps relargs
when (null origfiles) $
fail "No valid arguments were given."
withRepoLock (dryRun ? opts) (useCache ? opts) YesUpdateWorking (umask ? opts) $
RepoJob $ \repository -> do
args <- if parseFlags O.recursive opts
then reverse `fmap` expandDirs False origfiles
else return origfiles
Sealed p <- makeRemovePatch opts repository args
when (nullFL p && not (null origfiles) && not (quiet opts)) $
fail "No files were removed."
addToPending repository YesUpdateWorking p
putInfo opts $ vcat $ map text $ ["Will stop tracking:"] ++ listTouchedFiles p
makeRemovePatch :: (RepoPatch p, ApplyState p ~ Tree)
=> [DarcsFlag] -> Repository rt p wR wU wT
-> [SubPath] -> IO (Sealed (FL (PrimOf p) wU))
makeRemovePatch opts repository files =
do recorded <- expand =<< readRecordedAndPending repository
unrecorded <- readUnrecorded repository $ Just files
ftf <- filetypeFunction
result <- foldM removeOnePath (ftf,recorded,unrecorded, []) $ map (floatPath . fn2fp . sp2fn) files
case result of
(_, _, _, patches) -> return $
unFreeLeft $ foldr (joinGap (+>+)) (emptyGap NilFL) $ reverse patches
where removeOnePath (ftf, recorded, unrecorded, patches) f = do
let recorded' = modifyTree recorded f Nothing
unrecorded' = modifyTree unrecorded f Nothing
local <- makeRemoveGap opts ftf recorded unrecorded unrecorded' f
return $ case local of
Just gap -> (ftf, recorded', unrecorded', gap : patches)
_ -> (ftf, recorded, unrecorded, patches)
makeRemoveGap :: PrimPatch prim => [DarcsFlag] -> (FilePath -> FileType)
-> Tree IO -> Tree IO -> Tree IO -> AnchoredPath
-> IO (Maybe (FreeLeft (FL prim)))
makeRemoveGap opts ftf recorded unrecorded unrecorded' f =
case (find recorded f, find unrecorded f) of
(Just (SubTree _), Just (SubTree unrecordedChildren)) ->
if not $ null (list unrecordedChildren)
then skipAndWarn "it is not empty"
else return $ Just $ freeGap (rmdir f_fp :>: NilFL)
(Just (File _), Just (File _)) -> do
Just `fmap` treeDiff (diffAlgorithm ? opts) ftf unrecorded unrecorded'
(Just (File _), _) ->
return $ Just $ freeGap (addfile f_fp :>: rmfile f_fp :>: NilFL)
(Just (SubTree _), _) ->
return $ Just $ freeGap (adddir f_fp :>: rmdir f_fp :>: NilFL)
(_, _) -> skipAndWarn "it is not tracked by darcs"
where f_fp = anchorPath "" f
skipAndWarn reason =
do putWarning opts . text $ "Can't remove " ++ f_fp
++ " (" ++ reason ++ ")"
return Nothing
rmDescription :: String
rmDescription = "Help newbies find `darcs remove'."
rmHelp :: String
rmHelp =
"The `darcs rm' command does nothing.\n" ++
"\n" ++
"The normal way to remove a file from version control is simply to\n" ++
"delete it from the working tree. To remove a file from version\n" ++
"control WITHOUT affecting the working tree, see `darcs remove'.\n"
rm :: DarcsCommand [DarcsFlag]
rm = commandStub "rm" rmHelp rmDescription remove
unadd :: DarcsCommand [DarcsFlag]
unadd = commandAlias "unadd" Nothing remove