On 6/8/2022 10:32 AM, Phillip Wood wrote: > Hi Stolee > > On 07/06/2022 21:42, Derrick Stolee via GitGitGadget wrote: >> [...] >> As an example, here is my in-progress bundle URI RFC split into subtopics as >> they appear during the TODO list of a git rebase -i --update-refs: >> >> pick 2d966282ff3 docs: document bundle URI standard >> pick 31396e9171a remote-curl: add 'get' capability >> pick 54c6ab70f67 bundle-uri: create basic file-copy logic >> pick 96cb2e35af1 bundle-uri: add support for http(s):// and file:// >> pick 6adaf842684 fetch: add --bundle-uri option >> pick 6c5840ed77e fetch: add 'refs/bundle/' to log.excludeDecoration >> label for-update-refs/refs/heads/bundle-redo/fetch >> [...] update-refs >> [...] >> Based on some of the discussion, it seemed like one way to do this would be >> to have an 'update-ref ' command that would take the place of these 'label' >> commands. However, this would require two things that make it a bit awkward: >> >> 1. We would need to replicate the storage of those positions during the >> rebase. 'label' already does this pretty well. I've added the >> "for-update-refs/" label to help here. > > I'm afraid I don't think that using a label with a name constructed from > a magic prefix and the full refname is a good user interface. > > (i) Having labels behave differently based on their name is confusing. The label commands do as advertised, but the 'update-refs' command does something with the set of refs based on their names, yes. We would need to store this information during the rebase _somewhere_, and refs/rewritten/ seems natural. > (ii) The length of the label string is cumbersome for users. Rebase already > has a reputation for being unfriendly and hard to use, this will not improve > that. "update-ref bundle-redo/fetch" is much simpler. I agree that your proposed replacement for "label for-update-refs/..." would look cleaner. I don't think that that is enough to justify hiding information from the user. > (iii) It means that we no longer store the oid of each branch when creating> the todo list and so cannot check if it is safe to update it at the end of the > rebase (just using the current value of the branch ref at the end of a long > running operation like rebase is not sufficient to be safe). The operation we are doing necessarily requires a force-update, since we are modifying history. The safety you are caring about here is about the case where the user modifies one of these refs between the initial 'git rebase --update-refs' and the final 'git rebase --continue' that actually writes the refs. You want to prevent that final update from succeeding because another change to those refs could be lost. I'm less concerned about this case, because the user is requesting that we update the refs pointing to this set of commits. But, I'm glad you brought it up. One way to prevent this situation would be to extend the idea of "what branch is being rebased?" (REBASE_HEAD) to "which branches are being rewritten?" (REBASE_HEADS?). If the worktree kept the full list somewhere in the $GIT_DIR, then other Git commands could notice that those refs are currently being overwritten and those updates should fail (similarly to how we currently fail to update a branch being rebased). This ties into your later point: >> 2. If we want to close out all of the refs as the rebase is finishing, then >> that "step" becomes invisible to the user (and a bit more complicated to >> insert). Thus, the 'update-refs' step performs this action. If the user >> wants to do things after that step, then they can do so by editing the >> TODO list. > > I'm not sure what the use case is for making the update-refs step visible to > the user - it seems to be added complexity for them to deal with. We don't do > that for the branch that is being rebased so why do we need to do it for the > other branches? As for the implementation we could just update the branches > at the end of the loop in pick_commits() where we update the branch and run the > post-rewrite hook etc. there is no need for an entry in the todo list. I concede that allowing the user to move the 'update-refs' command around is not super important. The biggest concern I originally had with this idea was that there was no connection between the "--update-refs" option in the first "git rebase" command and the final "git rebase --continue" command. That is, other than which refs are in refs/rewritten/*. HOWEVER, using refs/rewritten/* is likely buggy: if we had two rebases happening at the same time (operating on a disjoint set of refs), then how do we differentiate which refs make sense for us to update as we complete this rebase? So, I'm coming to the conclusion that using refs/rewritten/* is problematic and I should use something closer to REBASE_HEAD as a way to describe which refs are being updated. In that context, your 'update-ref <ref>' command makes a lot more sense. The user can decide to move those around _or_ remove them; it won't change their protection under "REBASE_HEADS" but would prevent them from being rewritten. While I think on this, I'll send my branch_checked_out() patches in a separate thread, since those have grown in complexity since this version. Thanks, -Stolee