{- git-annex command - - Copyright 2010-2015 Joey Hess - - Licensed under the GNU AGPL version 3 or higher. -} {-# LANGUAGE CPP #-} module Command.Fix where import Command import Config import qualified Annex import Annex.ReplaceFile import Annex.Content import Annex.Perms import qualified Annex.Queue import qualified Database.Keys #if ! defined(mingw32_HOST_OS) import Utility.Touch import System.Posix.Files #endif cmd :: Command cmd = noCommit $ withGlobalOptions [annexedMatchingOptions] $ command "fix" SectionMaintenance "fix up links to annexed content" paramPaths (withParams seek) seek :: CmdParams -> CommandSeek seek ps = unlessM crippledFileSystem $ do withFilesInGit (commandAction . (whenAnnexed $ start FixAll)) =<< workTreeItems ps data FixWhat = FixSymlinks | FixAll start :: FixWhat -> RawFilePath -> Key -> CommandStart start fixwhat file key = do currlink <- liftIO $ catchMaybeIO $ readSymbolicLink $ fromRawFilePath file wantlink <- calcRepo $ gitAnnexLink (fromRawFilePath file) key case currlink of Just l | l /= wantlink -> fixby $ fixSymlink (fromRawFilePath file) wantlink | otherwise -> stop Nothing -> case fixwhat of FixAll -> fixthin FixSymlinks -> stop where fixby = starting "fix" (mkActionItem (key, file)) fixthin = do obj <- calcRepo $ gitAnnexLocation key stopUnless (isUnmodified key (fromRawFilePath file) <&&> isUnmodified key obj) $ do thin <- annexThin <$> Annex.getGitConfig fs <- liftIO $ catchMaybeIO $ getFileStatus (fromRawFilePath file) os <- liftIO $ catchMaybeIO $ getFileStatus obj case (linkCount <$> fs, linkCount <$> os, thin) of (Just 1, Just 1, True) -> fixby $ makeHardLink (fromRawFilePath file) key (Just n, Just n', False) | n > 1 && n == n' -> fixby $ breakHardLink (fromRawFilePath file) key obj _ -> stop breakHardLink :: FilePath -> Key -> FilePath -> CommandPerform breakHardLink file key obj = do replaceFile file $ \tmp -> do mode <- liftIO $ catchMaybeIO $ fileMode <$> getFileStatus file unlessM (checkedCopyFile key obj tmp mode) $ error "unable to break hard link" thawContent tmp modifyContent obj $ freezeContent obj Database.Keys.storeInodeCaches key [file] next $ return True makeHardLink :: FilePath -> Key -> CommandPerform makeHardLink file key = do replaceFile file $ \tmp -> do mode <- liftIO $ catchMaybeIO $ fileMode <$> getFileStatus file linkFromAnnex key tmp mode >>= \case LinkAnnexFailed -> error "unable to make hard link" _ -> noop next $ return True fixSymlink :: FilePath -> FilePath -> CommandPerform fixSymlink file link = do liftIO $ do #if ! defined(mingw32_HOST_OS) -- preserve mtime of symlink mtime <- catchMaybeIO $ modificationTimeHiRes <$> getSymbolicLinkStatus file #endif createDirectoryIfMissing True (parentDir file) removeFile file createSymbolicLink link file #if ! defined(mingw32_HOST_OS) maybe noop (\t -> touch file t False) mtime #endif next $ cleanupSymlink file cleanupSymlink :: FilePath -> CommandCleanup cleanupSymlink file = do Annex.Queue.addCommand "add" [Param "--force", Param "--"] [file] return True