Initially I had the following (simplified) repo structure:
MyRepo external1/MySub (git submodule) .gitsubmodules
Where
$ cat .gitsubmodules [submodule "external1/MySub"] path = external1/MySub url = user@repo:/remoterepo/externals/MySub.git
I then wanted to move MySubmodule to another directory in the same git repository, e.g., external2 so the structure is as follows:
MyRepo external2/MySub (git submodule) .gitsubmodules
What I did, and thought is enough, I simply moved (via OS mv
) the external1/MySub directory to external2/MySub and edited the .gitsubmodules file to:
$ cat .gitsubmodules [submodule "external2/MySub"] path = external2/MySub url = user@repo:/remoterepo/externals/MySub.git
After this change, I get the following error:
$ git status fatal: Could not chdir to '../../../../../repo/external/MySub': No such file or directory fatal: 'git status --porcelain' failed in submodule repo/external2/MySub
What am I missing? Any other changes need to be done for such move to work?
(I'm using git version 1.8.3rc3 on Windows 8.1)
Git Submodules Moving a submodule If needed, create the parent directory of the new location of the submodule ( mkdir -p new/path/to ). Move all content from the old to the new directory ( mv -vi old/path/to/module new/path/to/submodule ). Make sure Git tracks this directory ( git add new/path/to ).
Porcelain Format Version 1 Version 1 porcelain format is similar to the short format, but is guaranteed not to change in a backwards-incompatible way between Git versions or based on user configuration. This makes it ideal for parsing by scripts.
If you track branches in your submodules, you can update them via the --remote parameter of the git submodule update command. This pulls in new commits into the main repository and its submodules. It also changes the working directories of the submodules to the commit of the tracked branch.
The . gitmodules file, located in the top-level directory of a Git working tree, is a text file with a syntax matching the requirements of git-config[1]. The file contains one subsection per submodule, and the subsection value is the name of the submodule.
I simply moved the external1/MySub directory to external2/MySub
Move as in mv
(unix command), or git mv
(git command)?
You need to move the special entry of the index representing the submodule for the parent repo.
Normally, (reading git mv
), this should have been enough:
git mv external1/MySub external2/MySub
Moving a submodule using a gitfile (which means they were cloned with a Git version 1.7.8 or newer) will update the
gitfile
andcore.worktree
setting to make the submodule work in the new location.
It also will attempt to update thesubmodule.<name>.path
setting in thegitmodules
file and stage that file (unless -n is used).
That works best with the latest 1.8.5 git (like msysgit for Windows, or latest package for Unix)
Suseika adds in the comments
If you get message
fatal: renaming '%submodule%' failed: No such file or directory
it is probably because you're adding a directory level, e.g. moving submodule "
math
" to "libs/math
".git mv
doesn't create missing [intermediate] directories, you shouldmkdir
them yourself.
git 2.9 (June 2016) will improve git mv
for submodule:
See commit a127331 (19 Apr 2016) by Stefan Beller (stefanbeller
).
(Merged by Junio C Hamano -- gitster
-- in commit 9cb50a3, 29 Apr 2016)
mv
: allow moving nested submodules"
git mv old new
" did not adjust the path for a submodule that lives as a subdirectory insideold/
directory correctly.submodules however need to update their link to the git directory as well as updates to the
.gitmodules
file.
Git 2.12 (Q1 2017) offers to move nested submodules to the parent repo:
There is a new submodule helper "git submodule absorbgitdirs
" to make it easier to move embedded .git/
directory for submodules in a superproject to .git/modules/
(and point the latter with the former that is turned into a "gitdir:
" file) has been added.
See commit 7c4be45 (27 Dec 2016), commit f6f8586, commit 47e83eb, commit 1a248cf (12 Dec 2016), and commit 6f94351, commit 89c8626, commit 90c0011, commit 6f94351, commit 89c8626, commit 90c0011 (08 Dec 2016) by Stefan Beller (stefanbeller
).
(Merged by Junio C Hamano -- gitster
-- in commit da2b74e, 10 Jan 2017)
submodule: add
absorb-git-dir
functionWhen a submodule has its
git
dir inside the working dir, the submodule support for checkout that we plan to add in a later patch will fail.Add functionality to migrate the git directory to be absorbed into the superprojects git directory.
The newly added code in this patch is structured such that other areas of Git can also make use of it. The code in the submodule--helper is a mere wrapper and option parser for the function
absorb_git_dir_into_superproject
, that takes care of embedding the submodules git directory into the superprojects git dir. That function makes use of the more abstract function for this use caserelocate_gitdir
, which can be used by e.g. the worktree code eventually to move around a git directory.
Note: there is still (Git 2.14.x/2.15, Q4 2017) a bug linked to submodule moving: see "Git: moving submodules recursively (nested submodules)".
Git 2.15.x/2.16 (Q1 2018) will make moving submodule more robust, since "git fetch --recurse-submodules
" now knows that submodules can be moved around in the superproject in addition to getting updated, and finds the ones that need to be fetched accordingly.
See commit 4b4aced, commit c68f837 (16 Oct 2017), and commit 01ce122 (06 Oct 2017) by Heiko Voigt (hvoigt
).
(Merged by Junio C Hamano -- gitster
-- in commit b4d658b, 06 Nov 2017)
implement fetching of moved submodules
We store the changed submodules paths to calculate which submodule needs fetching. This does not work for moved submodules since their paths do not stay the same in case of a moved submodules.
In case of new submodules we do not have a path in the current checkout, since they just appeared in this fetch.It is more general to collect the submodule names for changes instead of their paths to include the above cases. If we do not have a configuration for a gitlink we rely on constructing a default name from the path if a git repository can be found at its path. We skip non-configured gitlinks whose default name collides with a configured one.
Note that, before Git 2.19 (Q3 2018), the code to try seeing if a fetch is necessary in a submodule during a fetch with --recurse-submodules
got confused when the path to the submodule was changed in the range of commits in the superproject, sometimes showing "(null)
".
This has been corrected.
See commit c3749f6, commit 5fc8475 (14 Jun 2018) by Stefan Beller (stefanbeller
).
(Merged by Junio C Hamano -- gitster
-- in commit 085d2ab, 28 Jun 2018)
submodule
: fixNULL
correctness in renamed broken submodulesWhen fetching with recursing into submodules, the fetch logic inspects the superproject which submodules actually need to be fetched.
This is tricky for submodules that were renamed in the fetched range of commits. This was implemented in c68f837 (implement fetching of moved submodules, 2017-10-16, Git v2.16.0), and this patch fixes a mistake in the logic there.
I changed a few more things manually and it actually worked with git 1.8.3rc3 (unnecessary with git 1.8.5):
Fix gitdir
path in .git submodule file repo/external2/MySub/.git
:
-gitdir: ../../../.git/modules/external1/MySub +gitdir: ../../../.git/modules/external2/MySub
Rename submodule directory under repo/.git/modules/
directory
$ mv repo/.git/modules/external1 repo/.git/modules/external2
Fix worktree
path in repo/.git/modules/external2/MySub/config
-worktree = ../../../../../external1/MySub +worktree = ../../../../../external2/MySub
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With