git-annex/Command/Add.hs

87 lines
2.4 KiB
Haskell
Raw Normal View History

{- git-annex command
-
- Copyright 2010 Joey Hess <joey@kitenet.net>
-
- Licensed under the GNU GPL version 3 or higher.
-}
module Command.Add where
2011-10-05 20:02:51 +00:00
import Common.Annex
import Annex.Exception
import Command
import qualified Annex
2011-10-04 04:40:47 +00:00
import qualified Annex.Queue
import qualified Backend
2011-10-15 20:21:08 +00:00
import Logs.Location
2011-10-04 04:40:47 +00:00
import Annex.Content
2011-08-20 20:11:42 +00:00
import Utility.Touch
2011-09-15 20:57:02 +00:00
import Backend
command :: [Command]
command = [Command "add" paramPaths defaultChecks seek "add files to annex"]
2010-11-11 22:54:52 +00:00
{- Add acts on both files not checked into git yet, and unlocked files. -}
seek :: [CommandSeek]
2010-11-11 22:54:52 +00:00
seek = [withFilesNotInGit start, withFilesUnlocked start]
{- 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
- to its content. -}
start :: BackendFile -> CommandStart
2011-09-15 20:57:02 +00:00
start p@(_, file) = notAnnexed file $ do
s <- liftIO $ getSymbolicLinkStatus file
2011-07-15 16:47:14 +00:00
if isSymbolicLink s || not (isRegularFile s)
2011-05-15 06:02:46 +00:00
then stop
else do
showStart "add" file
next $ perform p
perform :: BackendFile -> CommandPerform
2011-09-15 20:57:02 +00:00
perform (backend, file) = do
k <- Backend.genKey file backend
case k of
2011-05-15 06:02:46 +00:00
Nothing -> stop
Just (key, _) -> do
handle (undo file key) $ moveAnnex key file
next $ cleanup file key True
{- On error, put the file back so it doesn't seem to have vanished.
- This can be called before or after the symlink is in place. -}
undo :: FilePath -> Key -> IOException -> Annex a
undo file key e = do
2011-07-15 16:47:14 +00:00
unlessM (inAnnex key) rethrow -- no cleanup to do
liftIO $ whenM (doesFileExist file) $ removeFile file
handle tryharder $ fromAnnex key file
logStatus key InfoMissing
rethrow
where
rethrow = throw e
-- fromAnnex could fail if the file ownership is weird
tryharder :: IOException -> Annex ()
tryharder _ = do
g <- gitRepo
liftIO $ renameFile (gitAnnexLocation g key) file
cleanup :: FilePath -> Key -> Bool -> CommandCleanup
cleanup file key hascontent = do
handle (undo file key) $ do
link <- calcGitLink file key
liftIO $ createSymbolicLink link file
when hascontent $ do
logStatus key InfoPresent
-- touch the symlink to have the same mtime as the
-- file it points to
s <- liftIO $ getFileStatus file
let mtime = modificationTime s
liftIO $ touch file (TimeSpec mtime) False
force <- Annex.getState Annex.force
if force
2011-10-04 04:40:47 +00:00
then Annex.Queue.add "add" [Param "-f", Param "--"] [file]
else Annex.Queue.add "add" [Param "--"] [file]
return True