diff options
author | Joey Hess <joeyh@joeyh.name> | 2015-03-04 16:08:41 -0400 |
---|---|---|
committer | Joey Hess <joeyh@joeyh.name> | 2015-03-04 16:08:41 -0400 |
commit | 2c911300c50040f17d314c5c0891c82ecab4bb88 (patch) | |
tree | 4a50dc665f1b9c37cc9ea364752b85c7953298aa | |
parent | 179af68964152d182caeea9fd6c75858e4cfd2af (diff) |
fixup annex link target calculation when submodules are used in filesystems not supporting symlinks
-rw-r--r-- | Annex/Fixup.hs | 10 | ||||
-rw-r--r-- | Locations.hs | 22 | ||||
-rw-r--r-- | doc/submodules.mdwn | 5 |
3 files changed, 27 insertions, 10 deletions
diff --git a/Annex/Fixup.hs b/Annex/Fixup.hs index cda36461c..bb35454ee 100644 --- a/Annex/Fixup.hs +++ b/Annex/Fixup.hs @@ -60,12 +60,13 @@ fixupDirect r = return r - submodule is mounted. - - When the filesystem doesn't support symlinks, we cannot make .git - - into a symlink. In this case, we merely adjust the Repo so that + - into a symlink. But we don't need too, since the repo will use direct + - mode, In this case, we merely adjust the Repo so that - symlinks to objects that get checked in will be in the right form. -} fixupSubmodule :: Repo -> GitConfig -> IO Repo fixupSubmodule r@(Repo { location = l@(Local { worktree = Just w, gitdir = d }) }) c - | (".git" </> "modules") `isInfixOf` d = do + | needsSubmoduleFixup r = do when (coreSymlinks c) $ replacedotgit `catchNonAsync` \_e -> hPutStrLn stderr @@ -84,3 +85,8 @@ fixupSubmodule r@(Repo { location = l@(Local { worktree = Just w, gitdir = d }) maybe (error "unset core.worktree failed") (\_ -> return ()) =<< Git.Config.unset "core.worktree" r fixupSubmodule r _ = return r + +needsSubmoduleFixup :: Repo -> Bool +needsSubmoduleFixup (Repo { location = (Local { worktree = Just _, gitdir = d }) }) = + (".git" </> "modules") `isInfixOf` d +needsSubmoduleFixup _ = False diff --git a/Locations.hs b/Locations.hs index 9b10d3c20..7602a27e4 100644 --- a/Locations.hs +++ b/Locations.hs @@ -79,6 +79,7 @@ import Types.Difference import qualified Git import Git.FilePath import Annex.DirHashes +import Annex.Fixup {- Conventions: - @@ -126,9 +127,9 @@ annexLocation config key hasher = objectDir </> keyPath key (hasher $ objectHash - the actual location of the file's content. -} gitAnnexLocation :: Key -> Git.Repo -> GitConfig -> IO FilePath -gitAnnexLocation key r config = gitAnnexLocation' key r config (annexCrippledFileSystem config) doesFileExist -gitAnnexLocation' :: Key -> Git.Repo -> GitConfig -> Bool -> (FilePath -> IO Bool) -> IO FilePath -gitAnnexLocation' key r config crippled checker +gitAnnexLocation key r config = gitAnnexLocation' key r config (annexCrippledFileSystem config) doesFileExist (Git.localGitDir r) +gitAnnexLocation' :: Key -> Git.Repo -> GitConfig -> Bool -> (FilePath -> IO Bool) -> FilePath -> IO FilePath +gitAnnexLocation' key r config crippled checker gitdir {- Bare repositories default to hashDirLower for new - content, as it's more portable. - @@ -147,18 +148,27 @@ gitAnnexLocation' key r config crippled checker - present. -} | otherwise = return $ inrepo $ annexLocation config key hashDirMixed where - inrepo d = Git.localGitDir r </> d + inrepo d = gitdir </> d check locs@(l:_) = fromMaybe l <$> firstM checker locs check [] = error "internal" -{- Calculates a symlink to link a file to an annexed object. -} +{- Calculates a symlink target to link a file to an annexed object. -} gitAnnexLink :: FilePath -> Key -> Git.Repo -> GitConfig -> IO FilePath gitAnnexLink file key r config = do currdir <- getCurrentDirectory let absfile = fromMaybe whoops $ absNormPathUnix currdir file - loc <- gitAnnexLocation' key r config False (\_ -> return True) + let gitdir = getgitdir currdir + loc <- gitAnnexLocation' key r config False (\_ -> return True) gitdir toInternalGitPath <$> relPathDirToFile (parentDir absfile) loc where + getgitdir currdir + {- This special case is for git submodules on filesystems not + - supporting symlinks; generate link target that will + - work portably. -} + | coreSymlinks config == False && needsSubmoduleFixup r = + fromMaybe whoops $ absNormPathUnix currdir $ + Git.repoPath r </> ".git" + | otherwise = Git.localGitDir r whoops = error $ "unable to normalize " ++ file {- File used to lock a key's content. -} diff --git a/doc/submodules.mdwn b/doc/submodules.mdwn index fe5fc9a9e..74668085d 100644 --- a/doc/submodules.mdwn +++ b/doc/submodules.mdwn @@ -5,7 +5,9 @@ Git normally makes a `.git` **file** in a submodule, that points to the real git repository under `.git/modules/`. This presents problems for git-annex. So, when used in a submodule, git-annex will automatically replace the `.git` file with a symlink -pointing at the git repository. +pointing at the git repository. (When the filesystem doesn't support +symlinks, direct mode is used, and submodules are supported in that +setup too.) With that taken care of, git-annex should work ok in submodules. Although this is a new and somewhat experimental feature. @@ -18,4 +20,3 @@ Known problems: will refuse to delete it, complaining that the submodule "uses a .git directory". Workaround: Use `rm -rf` to delete the tree, and then `git commit`. -* This won't work on filesystems not supporting symlinks. |