On Thu, Nov 12, 2015 at 9:35 PM, Jeff King <peff@xxxxxxxx> wrote: > > Hrm. Do we want to make these workarounds work correctly? Or is the > final solution going to be that the first command you gave simply works, > and no workarounds are needed. If the latter, I wonder if we want to be > adding tests for the workarounds in the first place. I think we want to make the final solution just work. I dug into that and it is harder than expected. I may even call it a bug. The bug doesn't occur often as it is only triggered by things like rewriting history (forced pushes) or a short dpeth argument. So if you invoke "git clone --recursive", it will internally just delegate the submodule handling to "git submodule update --init --recursive", which then (as the submodule doesn't exist yet) will delegate the cloning to "git submodule--helper clone", which will then call git clone for the actual cloning. However in this whole chain of commands we never pass around the actual sha1 we need. The strategy is to clone first and then checkout the sha1, which the superprojects wants to see. The desired sha1 was hopefully included in the cloning, so we can check it out. But the sha1 may not be present if we have a very short depth argument, or if we rewrote history. In case of a short depth argument, consider the following history: ... <- A <- B A is the recorded sha1 in the superproject, whereas B is the HEAD in the remote you're cloning from. If cloning with depth=1, the most naive way would have been to pass on the depth argument down the command chain, but then we would end up cloning B with no further depth, and upon checkout we cannot find A. In case of the rewritten history, consider: .. < - C <- B \ A whereas A is the recorded sha1 in the superproject, but on a different branch (or even just a dangling commit. but used to be on master). B is the master branch. In case we pass on --depth to cloning the submodule, --single-branch is implied by --depth, so we would not clone A. In case of A being a dangling commit, we wouldn't even clone it without the depth argument. So I propose: * similar to fetch, we enable clone to obtain a specific sha1 from remote. * we explicitly pass the submodule sha1 as recorded in the superproject to the submodule fetch/clone in case we follow the exact sha1. In case of --remote or the branch field present in the superprojects .gitmodule file, we can just pass the branch name. Thanks, Stefan -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html