无法签出git子模块路径

问题描述 投票:31回答:8

使用git子模块时遇到问题。

每当我从上游存储库收到一个新的子模块引用时,执行git submodule update会得到以下结果:

fatal: reference is not a tree: dd208d46ecdd1ac0d2b2594a610fe4c9150fece1
Unable to checkout 'dd208d46ecdd1ac0d2b2594a610fe4c9150fece1' in submodule path 'submodule/path'

重要的是要注意子模块有几个遥控器,其中上游遥控器应该用于更新子模块参考树。我猜我的问题在那里,但我不确定。

我的设置如下:

Git项目

遥控器:

  1. origin(我的git fork)
  2. upstream(项目回购)

子模块“模块”,有遥控器:

  1. origin(我的git fork)
  2. upstream(项目回购)

有谁知道是什么导致了我的问题?

git git-submodules
8个回答
37
投票

在执行git submodule update时,git会尝试检出超级项目中保存的提交/树(在您的示例中,提交ID为dd208d4...

我认为你得到错误,因为在子模块中没有这样的对象存在。你必须确保它在那里。通常这意味着您必须首先从远程获取/拉取它。

可能你必须这样做

git submodule foreach git fetch
git submodule update

或者可能

git fetch --recurse-submodules

假设子模块已配置,以便它可以从远程origin获取丢失的提交。最后,您必须知道,从哪里可以获取丢失的提交,您必须得到它。

您可以通过执行以下操作来检查您是否拥有dd208d4...

cd ./module
git log dd208d46ecdd1ac0d2b2594a610fe4c9150fece1
git cat-file -p dd208d46ecdd1ac0d2b2594a610fe4c9150fece1
git ls-tree dd208d46ecdd1ac0d2b2594a610fe4c9150fece1

造成这种问题的一个可能原因是,从超级模块发布新提交的人没有从子模块发布必要的提交。他必须首先从子模块发布提交。


7
投票

确保已按下子模块

cd submodule-dir
git push

在我的情况下,我有:

  • 致力于子模块
  • 没推
  • 使用更新的子模块向父级提交
  • 推了父母

所以难怪它无法找到。

然后,如果您使用的是诸如GitHub之类的Web界面,您还可以转到子模块存储库网页,并仔细检查您需要的提交是否显示在那里。

push.recurseSubmodules on-demand

可以通过以下方式进一步自动推送:

git push --recurse-submodules=on-demand

它还根据需要推送子模块,或者从2.7开始:

git config push.recurseSubmodules on-demand
git push

1
投票

我有同样的问题,我决定为父项目添加一个新的提交并推送所有


1
投票

当我忘记推动我的一个子模块中的更改时,只是看到了这个问题

确保推送更改


0
投票

我的问题是,在我的repo的cat .gitmodules上,我指着子模块的repo错误的遥控器(我最初用原始遥控器克隆它,但后来切换到它的叉子; gitmodules文件从未更新以反映更改)。


0
投票

我的问题是我在build.gradle文件中的子模块中有未提交的更改(我认为这些更改是自动更改的)。他们出现在git diff。我只是做git checkout .重置子模块回购没有变化,然后git submodule update工作。


0
投票

另一种方法可能是,没有命令行git选项,但手动执行此操作。当子模块路径被移动/替换(但不正确)时,情况大多发生,从而仍然指向本地结账存储库中的旧引用。

1)找到存储库和子模块

ls -la .git/modules
   rm -f .git/modules/<module-with-issue>

2)删除旧的本地子模块配置

 gedit .git/config

(在此处删除子模块url条目)

它看起来像这样;

 *[submodule "module-with-issue"]
       url = ...*

3)现在,新获取和更新子模块git fetch git submodule update --recursive --init

注意:在尝试子模块更新之前,可能还需要删除存储库中本地签出的子模块文件夹。


0
投票

添加git rm --cached <submodule-directory>帮助我gitlab:

.prepare_deploy: &prepare_deploy
  before_script:
    - bundle install -j $(nproc) --path vendor
    - which ssh-agent || (apt-get update -y && apt-get install openssh-client -y)
    - mkdir -p ~/.ssh
    - chmod 700 ~/.ssh
    - eval $(ssh-agent -s)
    - echo -e "Host *\n\tStrictHostKeyChecking no\n\n" > ~/.ssh/config
    - (echo "$SSH_PRIVATE_KEY" | base64 --decode) > ~/.ssh/id_rsa
    - chmod 600 ~/.ssh/id_rsa
    - git rm --cached <submodule-directory>
    - git submodule sync --recursive
    - git submodule update --init --recursive
© www.soinside.com 2019 - 2024. All rights reserved.