From patchwork Fri Sep 23 18:55:15 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Patchwork-Submitter: Stefan Xenos X-Patchwork-Id: 12986888 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 1D98DC04A95 for ; Fri, 23 Sep 2022 18:55:58 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232831AbiIWSz4 (ORCPT ); Fri, 23 Sep 2022 14:55:56 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60094 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S231340AbiIWSzd (ORCPT ); Fri, 23 Sep 2022 14:55:33 -0400 Received: from mail-wm1-x329.google.com (mail-wm1-x329.google.com [IPv6:2a00:1450:4864:20::329]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id A5846106F67 for ; Fri, 23 Sep 2022 11:55:29 -0700 (PDT) Received: by mail-wm1-x329.google.com with SMTP id o5so790528wms.1 for ; Fri, 23 Sep 2022 11:55:29 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=zLmoBr2pEmkQK5muEffV0GOwf/y2vVRrl+oQB8WmJi0=; b=oDlUXAaS2FjgnOqPS5MQHBBEEpxd7AgBb3XQTJPeTVsrmONVXBesjoe+hEPF+azYvM vC5q9PGs9oOjA3S35J25EtV90Wl2GwOHmyIu+pSWGZc2KLeyz9Soukz2t3LZcNSgyPkV ojJm6wddlu9pgs1H5wkfvLtGU0GwB6BU6RS82taUNM24/GXphfZ4UqEsTF3JIwqn3kjt e5AvgTlTsDcm9V3m1o3CNiuzD8bGOPGDAYOTAXI1rnnatcQD49iAWkWJ/M7pPNm2QsBm EWJVvo1JXi55na3DzT9a9zHH/aWKNSvQKOsQvwQyvRpNpID0P4yo5rN1cXL7p4XmwyjC rphQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=zLmoBr2pEmkQK5muEffV0GOwf/y2vVRrl+oQB8WmJi0=; b=mWMdyLolCxqcM8bIC8L1HIblQPw/P1or8otFPGJJ+pzOXSTD6Q+KjZ5Pd3c5itmBmg 3vMHxYITTmDzhyw9pzc4qVW3heeU6ccpN4UC+vIHcKAFkmqJ6U06onsLv8HwjnEMK0kp k3r019xrJP3WRBpjceEAyLXilmuMTfXnQkofFCjDfFd1Ypb3kTeLsLWi9s1C1EY0A4nZ QmicrVJ6QRB+nd0oEodrm1ow8o7GeHPZq+LsxRMlfpvLclqfus2oaoZ9nZLPF3VTKYWm XfNGjP+ii2U3C2801Qp4641g8rgy/Pz8xi6FcnRcX2CBGMyjEF4kYt64eSrmFU4+V4sl tkSw== X-Gm-Message-State: ACrzQf1UlyDfPMNOJr6B8rf2CQpwUocuCNu6hmqkKI8pW7tKpOut6e3C 5JwXv3fLr/Jf/qOMShFIo5txOYtka+E= X-Google-Smtp-Source: AMsMyM4WSlNsASE7d7wGokCXHzYXgRVsce8D6KYpcB1kuad9MpvHllrTzs2Mc2/1024p92YxpAY9fA== X-Received: by 2002:a05:600c:5128:b0:3b5:d6:eb8f with SMTP id o40-20020a05600c512800b003b500d6eb8fmr7082838wms.65.1663959326854; Fri, 23 Sep 2022 11:55:26 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id e11-20020adfef0b000000b0022ae0965a8asm8257676wro.24.2022.09.23.11.55.26 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:26 -0700 (PDT) Message-Id: In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:15 +0000 Subject: [PATCH 01/10] technical doc: add a design doc for the evolve command Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Stefan Xenos Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Stefan Xenos From: Stefan Xenos This document describes what a change graph for git would look like, the behavior of the evolve command, and the changes planned for other commands. It was originally proposed in 2018, see https://public-inbox.org/git/20181115005546.212538-1-sxenos@google.com/ Signed-off-by: Stefan Xenos Signed-off-by: Chris Poucet --- Documentation/technical/evolve.txt | 1051 ++++++++++++++++++++++++++++ 1 file changed, 1051 insertions(+) create mode 100644 Documentation/technical/evolve.txt diff --git a/Documentation/technical/evolve.txt b/Documentation/technical/evolve.txt new file mode 100644 index 00000000000..68ee2457e52 --- /dev/null +++ b/Documentation/technical/evolve.txt @@ -0,0 +1,1051 @@ +Evolve +====== + +Objective +========= +Create an "evolve" command to help users craft a high quality commit history. +Users can improve commits one at a time and in any order, then run git evolve to +rewrite their recent history to ensure everything is up-to-date. We track +amendments to a commit over time in a change graph. Users can share their +progress with others by exchanging their change graphs using the standard push, +fetch, and format-patch commands. + +Status +====== +This proposal has not been implemented yet. + +Background +========== +Imagine you have three sequential changes up for review and you receive feedback +that requires editing all three changes. We'll define the word "change" +formally later, but for the moment let's say that a change is a work-in-progress +whose final version will be submitted as a commit in the future. + +While you're editing one change, more feedback arrives on one of the others. +What do you do? + +The evolve command is a convenient way to work with chains of commits that are +under review. Whenever you rebase or amend a commit, the repository remembers +that the old commit is obsolete and has been replaced by the new one. Then, at +some point in the future, you can run "git evolve" and the correct sequence of +rebases will occur in the correct order such that no commit has an obsolete +parent. + +Part of making the "evolve" command work involves tracking the edits to a commit +over time, which is why we need an change graph. However, the change +graph will also bring other benefits: + +- Users can view the history of a change directly (the sequence of amends and + rebases it has undergone, orthogonal to the history of the branch it is on). +- It will be possible to quickly locate and list all the changes the user + currently has in progress. +- It can be used as part of other high-level commands that combine or split + changes. +- It can be used to decorate commits (in git log, gitk, etc) that are either + obsolete or are the tip of a work in progress. +- By pushing and pulling the change graph, users can collaborate more + easily on changes-in-progress. This is better than pushing and pulling the + commits themselves since the change graph can be used to locate a more + specific merge base, allowing for better merges between different versions of + the same change. +- It could be used to correctly rebase local changes and other local branches + after running git-filter-branch. +- It can replace the change-id footer used by gerrit. + +Goals +----- +Legend: Goals marked with P0 are required. Goals marked with Pn should be +attempted unless they interfere with goals marked with Pn-1. + +P0. All commands that modify commits (such as the normal commit --amend or + rebase command) should mark the old commit as being obsolete and replaced by + the new one. No additional commands should be required to keep the + change graph up-to-date. +P0. Any commit that may be involved in a future evolve command should not be + garbage collected. Specifically: + - Commits that obsolete another should not be garbage collected until + user-specified conditions have occurred and the change has expired from + the reflog. User specified conditions for removing changes include: + - The user explicitly deleted the change. + - The change was merged into a specific branch. + - Commits that have been obsoleted by another should not be garbage + collected if any of their replacements are still being retained. +P0. A commit can be obsoleted by more than one replacement (called divergence). +P0. Users must be able to resolve divergence (convergence). +P1. Users should be able to share chains of obsolete changes in order to + collaborate on WIP changes. +P2. Such sharing should be at the user’s option. That is, it should be possible + to directly share a change without also sharing the file states or commit + comments from the obsolete changes that led up to it, and the choice not to + share those commits should not require changing any commit hashes. +P2. It should be possible to discard part or all of the change graph + without discarding the commits themselves that are already present in + branches and the reflog. +P2. Provide sufficient information to replace gerrit's Change-Id footers. + +Similar technologies +-------------------- +There are some other technologies that address the same end-user problem. + +Rebase -i can be used to solve the same problem, but users can't easily switch +tasks midway through an interactive rebase or have more than one interactive +rebase going on at the same time. It can't handle the case where you have +multiple changes sharing the same parent when that parent needs to be rebased +and won't let you collaborate with others on resolving a complicated interactive +rebase. You can think of rebase -i as a top-down approach and the evolve command +as the bottom-up approach to the same problem. + +Several patch queue managers have been built on top of git (such as topgit, +stgit, and quilt). They address the same user need. However they also rely on +state managed outside git that needs to be kept in sync. Such state can be +easily damaged when running a git native command that is unaware of the patch +queue. They also typically require an explicit initialization step to be done by +the user which creates workflow problems. + +Mercurial implements a very similar feature in its EvolveExtension. The behavior +of the evolve command itself is very similar, but the storage format for the +change graph differs. In the case of mercurial, each change set can have one or +more obsolescence markers that point to other changesets that they replace. This +is similar to the "Commit Headers" approach considered in the other options +appendix. The approach proposed here stores obsolescence information in a +separate metacommit graph, which makes exchanging of obsolescence information +optional. + +Mercurial's default behavior makes it easy to find and switch between +non-obsolete changesets that aren't currently on any branch. We introduce the +notion of a new ref namespace that enables a similar workflow via a different +mechanism. Mercurial has the notion of changeset phases which isn't present +in git and creates new ways for a changeset to diverge. Git doesn't need +to deal with these issues, but it has to deal with the problems of picking an +upstream branch as a target for rebases and protecting obsolescence information +from GC. We also introduce some additional transformations (see +obsolescence-over-cherry-pick, below) that aren't present in the mercurial +implementation. + +Semi-related work +----------------- +There are other technologies that address different problems but have some +similarities with this proposal. + +Replacements (refs/replace) are superficially similar to obsolescences in that +they describe that one commit should be replaced by another. However, they +differ in both how they are created and how they are intended to be used. +Obsolescences are created automatically by the commands a user runs, and they +describe the user’s intent to perform a future rebase. Obsolete commits still +appear in branches, logs, etc like normal commits (possibly with an extra +decoration that marks them as obsolete). Replacements are typically created +explicitly by the user, they are meant to be kept around for a long time, and +they describe a replacement to be applied at read-time rather than as the input +to a future operation. When a replaced commit is queried, it is typically hidden +and swapped out with its replacement as though the replacement has already +occurred. + +Git-imerge is a project to help make complicated merges easier, particularly +when merging or rebasing long chains of patches. It is not an alternative to +the change graph, but its algorithm of applying smaller incremental merges +could be used as part of the evolve algorithm in the future. + +Overview +======== +We introduce the notion of “meta-commits” which describe how one commit was +created from other commits. A branch of meta-commits is known as a change. +Changes are created and updated automatically whenever a user runs a command +that creates a commit. They are used for locating obsolete commits, providing a +list of a user’s unsubmitted work in progress, and providing a stable name for +each unsubmitted change. + +Users can exchange edit histories by pushing and fetching changes. + +New commands will be introduced for manipulating changes and resolving +divergence between them. Existing commands that create commits will be updated +to modify the meta-commit graph and create changes where necessary. + +Example usage +------------- +# First create three dependent changes +$ echo foo>bar.txt && git add . +$ git commit -m "This is a test" +created change metas/this_is_a_test +$ echo foo2>bar2.txt && git add . +$ git commit -m "This is also a test" +created change metas/this_is_also_a_test +$ echo foo3>bar3.txt && git add . +$ git commit -m "More testing" +created change metas/more_testing + +# List all our changes in progress +$ git change list +metas/this_is_a_test +metas/this_is_also_a_test +* metas/more_testing +metas/some_change_already_merged_upstream + +# Now modify the earliest change, using its stable name +$ git reset --hard metas/this_is_a_test +$ echo morefoo>>bar.txt && git add . && git commit --amend --no-edit + +# Use git-evolve to fix up any dependent changes +$ git evolve +rebasing metas/this_is_also_a_test onto metas/this_is_a_test +rebasing metas/more_testing onto metas/this_is_also_a_test +Done + +# Use git-obslog to view the history of the this_is_a_test change +$ git log --obslog +93f110 metas/this_is_a_test@{0} commit (amend): This is a test +930219 metas/this_is_a_test@{1} commit: This is a test + +# Now create an unrelated change +$ git reset --hard origin/master +$ echo newchange>unrelated.txt && git add . +$ git commit -m "Unrelated change" +created change metas/unrelated_change + +# Fetch the latest code from origin/master and use git-evolve +# to rebase all dependent changes. +$ git fetch origin master +$ git evolve origin/master +deleting metas/some_change_already_merged_upstream +rebasing metas/this_is_a_test onto origin/master +rebasing metas/this_is_also_a_test onto metas/this_is_a_test +rebasing metas/more_testing onto metas/this_is_also_a_test +rebasing metas/unrelated_change onto origin/master +Conflict detected! Resolve it and then use git evolve --continue to resume. + +# Sort out the conflict +$ git mergetool +$ git evolve origin/master +Done + +# Share the full history of edits for the this_is_a_test change +# with a review server +$ git push origin metas/this_is_a_test:refs/for/master +# Share the lastest commit for “Unrelated change”, without history +$ git push origin HEAD:refs/for/master + +Detailed design +=============== +Obsolescence information is stored as a graph of meta-commits. A meta-commit is +a specially-formatted merge commit that describes how one commit was created +from others. + +Meta-commits look like this: + +$ git cat-file -p +tree 4b825dc642cb6eb9a060e54bf8d69288fbee4904 +parent aa7ce55545bf2c14bef48db91af1a74e2347539a +parent d64309ee51d0af12723b6cb027fc9f195b15a5e9 +parent 7e1bbcd3a0fa854a7a9eac9bf1eea6465de98136 +author Stefan Xenos 1540841596 -0700 +committer Stefan Xenos 1540841596 -0700 +parent-type c r o + +This says “commit aa7ce555 makes commit d64309ee obsolete. It was created by +cherry-picking commit 7e1bbcd3”. + +The tree for meta-commits is always the empty tree, but future versions of git +may attach other trees here. For forward-compatibility fsck should ignore such +trees if found on future repository versions. This will allow future versions of +git to add metadata to the meta-commit tree without breaking forwards +compatibility. + +The commit comment for a meta-commit is an auto-generated user-readable string +describing the command that produced the meta commit. These strings are shown +to the user when they view the obslog. + +Parent-type +----------- +The “parent-type” field in the commit header identifies a commit as a +meta-commit and indicates the meaning for each of its parents. It is never +present for normal commits. It contains a space-deliminated list of enum values +whose order matches the order of the parents. Possible parent types are: + +- c: (content) the content parent identifies the commit that this meta-commit is + describing. +- r: (replaced) indicates that this parent is made obsolete by the content + parent. +- o: (origin) indicates that the content parent was generated by cherry-picking + this parent. +- a: (abandoned) used in place of a content parent for abandoned changes. Points + to the final content commit for the change at the time it was abandoned. + +There must be exactly one content or abandoned parent for each meta-commit and +it is always the first parent. The content commit will always be a normal commit +and not a meta-commit. However, future versions of git may create meta-commits +for other meta-commits and the fsck tool must be aware of this for forwards +compatibility. + +A meta-commit can have zero or more replaced parents. An amend operation creates +a single replaced parent. A merge used to resolve divergence (see divergence, +below) will create multiple replaced parents. A meta-commit may have no +replaced parents if it describes a cherry-pick or squash merge that copies one +or more commits but does not replace them. + +A meta-commit can have zero or more origin parents. A cherry-pick creates a +single origin parent. Certain types of squash merge will create multiple origin +parents. Origin parents don't directly cause their origin to become obsolete, +but are used when computing blame or locating a merge base. The section +on obsolescence over cherry-picks describes how the evolve command uses +origin parents. + +A replaced parent or origin parent may be either a normal commit (indicating +the oldest-known version of a change) or another meta-commit (for a change that +has already been modified one or more times). + +The parent-type field needs to go after the committer field since git's rules +for forwards-compatibility require that new fields to be at the end of the +header. Putting a new field in the middle of the header would break fsck. + +The presence of an abandoned parent indicates that the change should be pruned +by the evolve command, and removed from the repository's history. Any follow-up +changes should rebased onto the parent of the pruned commit. The abandoned +parent points to the version of the change that should be restored if the user +attempts to restore the change. + +Changes +------- +A branch of meta-commits describes how a commit was produced and what previous +commits it is based on. It is also an identifier for a thing the user is +currently working on. We refer to such a meta-branch as a change. + +Local changes are stored in the new refs/metas namespace. Remote changes are +stored in the refs/remote//metas namespace. + +The list of changes in refs/metas is more than just a mechanism for the evolve +command to locate obsolete commits. It is also a convenient list of all of a +user’s work in progress and their current state - a list of things they’re +likely to want to come back to. + +Strictly speaking, it is the presence of the branch in the refs/metas namespace +that marks a branch as being a change, not the fact that it points to a +metacommit. Metacommits are only created when a commit is amended or rebased, so +in the case where a change points to a commit that has never been modified, the +change points to that initial commit rather than a metacommit. + +Changes are also stored in the refs/hiddenmetas namespace. Hiddenmetas holds +metadata for historical changes that are not currently in progress by the user. +Commands like filter-branch and other bulk import commands create metadata in +this namespace. + +Note that the changes in hiddenmetas get special treatment in several ways: + +- They are not cleaned up automatically once merged, since it is expected that + they refer to historical changes. +- User commands that modify changes don't append to these changes as they would + to a change in refs/metas. +- They are not displayed when the user lists their local changes. + +Obsolescence +------------ +A commit is considered obsolete if it is reachable from the “replaces” edges +anywhere in the history of a change and it isn’t the head of that change. +Commits may be the content for 0 or more meta-commits. If the same commit +appears in multiple changes, it is not obsolete if it is the head of any of +those changes. + +Note that there is an exception to this rule. The metas namespace takes +precedence over the hiddenmetas namespace for the purpose of obsolescence. That +is, if a change appears in a replaces edge of a change in the metas namespace, +it is obsolete even if it also appears as the head of a change in the +hiddenmetas namespace. + +This special case prevents the hiddenmetas namespace from creating divergence +with the user's work in progress, and allows the user to resolve historical +divergence by creating new changes in the metas namespace. + +Divergence +---------- +From the user’s perspective, two changes are divergent if they both ask for +different replacements to the same commit. More precisely, a target commit is +considered divergent if there is more than one commit at the head of a change in +refs/metas that leads to the target commit via an unbroken chain of “replaces” +parents. + +Much like a merge conflict, divergence is a situation that requires user +intervention to resolve. The evolve command will stop when it encounters +divergence and prompt the user to resolve the problem. Users can solve the +problem in several ways: + +- Discard one of the changes (by deleting its change branch). +- Merge the two changes (producing a single change branch). +- Copy one of the changes (keep both commits, but one of them gets a new + metacommit appended to its history that is connected to its predecessor via an + origin edge rather than a replaces edge. That new change no longer obsoletes + the original.) + +Obsolescence across cherry-picks +-------------------------------- +By default the evolve command will treat cherry-picks and squash merges as being +completely separate from the original. Further amendments to the original commit +will have no effect on the cherry-picked copy. However, this behavior may not be +desirable in all circumstances. + +The evolve command may at some point support an option to look for cases where +the source of a cherry-pick or squash merge has itself been amended, and +automatically apply that same change to the cherry-picked copy. In such cases, +it would traverse origin edges rather than ignoring them, and would treat a +commit with origin edges as being obsolete if any of its origins were obsolete. + +Garbage collection +------------------ +For GC purposes, meta-commits are normal commits. Just as a commit causes its +parents and tree to be retained, a meta-commit also causes its parents to be +retained. + +Change creation +--------------- +Changes are created automatically whenever the user runs a command like “commit” +that has the semantics of creating a new change. They also move forward +automatically even if they’re not checked out. For example, whenever the user +runs a command like “commit --amend” that modifies a commit, all branches in +refs/metas that pointed to the old commit move forward to point to its +replacement instead. This also happens when the user is working from a detached +head. + +This does not mean that every commit has a corresponding change. By default, +changes only exist for recent locally-created commits. Users may explicitly pull +changes from other users or keep their changes around for a long time, but +either behavior requires a user to opt-in. Code review systems like gerrit may +also choose to keep changes around forever. + +Note that the changes in refs/metas serve a dual function as both a way to +identify obsolete changes and as a way for the user to keep track of their work +in progress. If we were only concerned with identifying obsolete changes, it +would be sufficient to create the change branch lazily the first time a commit +is obsoleted. Addressing the second use - of refs/metas as a mechanism for +keeping track of work in progress - is the reason for eagerly creating the +change on first commit. + +Change naming +------------- +When a change is first created, the only requirement for its name is that it +must be unique. Good names would also serve as useful mnemonics and be easy to +type. For example, a short word from the commit message containing no numbers or +special characters and that shows up with low frequency in other commit messages +would make a good choice. + +Different users may prefer different heuristics for their change names. For this +reason a new hook will be introduced to compute change names. Git will invoke +the hook for all newly-created changes and will append a numeric suffix if the +name isn’t unique. The default heuristics are not specified by this proposal and +may change during implementation. + +Change deletion +--------------- +Changes are normally only interesting to a user while a commit is still in +development and under review. Once the commit has submitted wherever it is +going, its change can be discarded. + +The normal way of deleting changes makes this easy to do - changes are deleted +by the evolve command when it detects that the change is present in an upstream +branch. It does this in two ways: if the latest commit in a change either shows +up in the branch history or the change becomes empty after a rebase, it is +considered merged and the change is discarded. In this context, an “upstream +branch” is any branch passed in as the upstream argument of the evolve command. + +In case this sometimes deletes a useful change, such automatic deletions are +recorded in the reflog allowing them to be easily recovered. + +Sharing changes +--------------- +Change histories are shared by pushing or fetching meta-commits and change +branches. This provides users with a lot of control of what to share and +repository implementations with control over what to retain. + +Users that only want to share the content of a commit can do so by pushing the +commit itself as they currently would. Users that want to share an edit history +for the commit can push its change, which would point to a meta-commit rather +than the commit itself if there is any history to share. Note that multiple +changes can refer to the same commits, so it’s possible to construct and push a +different history for the same commit in order to remove sensitive or irrelevant +intermediate states. + +Imagine the user is working on a change “mychange” that is currently the latest +commit on master. They have two ways to share it: + +# User shares just a commit without its history +> git push origin master + +# User shares the full history of the commit to a review system +> git push origin metas/mychange:refs/for/master + +# User fetches a collaborator’s modifications to their change +> git fetch remotename metas/mychange +# Which updates the ref remote/remotename/metas/mychange + +This will cause more intermediate states to be shared with the server than would +have been shared previously. A review system like gerrit would need to keep +track of which states had been explicitly pushed versus other intermediate +states in order to de-emphasize (or hide) the extra intermediate states from the +user interface. + +Merge-base +---------- +Merge-base will be changed to search the meta-commit graph for common ancestors +as well as the commit graph, and will generally prefer results from the +meta-commit graph over the commit graph. Merge-base will consider meta-commits +from all changes, and will traverse both origin and obsolete edges. + +The reason for this is that - when merging two versions of the same commit +together - an earlier version of that same commit will usually be much more +similar than their common parent. This should make the workflow of collaborating +on unsubmitted patches as convenient as the workflow for collaborating in a +topic branch by eliminating repeated merges. + +Configuration +------------- +The core.enableChanges configuration variable enables the creation and update +of change branches. This is enabled by default. + +User interface +-------------- +All git porcelain commands that create commits are classified as having one of +four behaviors: modify, create, copy, or import. These behaviors are discussed +in more detail below. + +Modify commands +--------------- +Modification commands (commit --amend, rebase) will mark the old commit as +obsolete by creating a new meta-commit that references the old one as a +replaced parent. In the event that multiple changes point to the same commit, +this is done independently for every such change. + +More specifically, modifications work like this: + +1. Locate all existing changes for which the old commit is the content for the + head of the change branch. If no such branch exists, create one that points + to the old commit. Changes that include this commit in their history but not + at their head are explicitly not included. +2. For every such change, create a new meta-commit that references the new + commit as its content and references the old head of the change as a + replaced parent. +3. Move the change branch forward to point to the new meta-commit. + +Copy commands +------------- +Copy commands (cherry-pick, merge --squash) create a new meta-commit that +references the old commits as origin parents. Besides the fact that the new +parents are tagged differently, copy commands work the same way as modify +commands. + +Create commands +--------------- +Creation commands (commit, merge) create a new commit and a new change that +points to that commit. The do not create any meta-commits. + +Import commands +--------------- +Import commands (fetch, pull) do not create any new meta-commits or changes +unless that is specifically what they are importing. For example, the fetch +command would update remote/origin/metas/change35 and fetch all referenced +meta-commits if asked to do so directly, but it wouldn’t create any changes or +meta-commits for commits discovered on the master branch when running “git fetch +origin master”. + +Other commands +-------------- +Some commands don’t fit cleanly into one of the above categories. + +Semantically, filter-branch should be treated as a modify command, but doing so +is likely to create a lot of irrelevant clutter in the changes namespace and the +large number of extra change refs may introduce performance problems. We +recommend treating filter-branch as an import command initially, but making it +behave more like a modify command in future follow-up work. One possible +solution may be to treat commits that are part of existing changes as being +modified but to avoid creating changes for other rewritten changes. Another +solution may be to record the modifications as changes in the hiddenmetas +namespace. + +Once the evolve command can handle obsolescence across cherry-picks, such +cherry-picks will result in a hybrid move-and-copy operation. It will create +cherry-picks that replace other cherry-picks, which will have both origin edges +(pointing to the new source commit being picked) and replacement edges (pointing +to the previous cherry-pick being replaced). + +Evolve +------ +The evolve command performs the correct sequence of rebases such that no change +has an obsolete parent. The syntax looks like this: + +git evolve [upstream…] + +It takes an optional list of upstream branches. All changes whose parent shows +up in the history of one of the upstream branches will be rebased onto the +upstream branch before resolving obsolete parents. + +Any change whose latest state is found in an upstream branch (or that ends up +empty after rebase) will be deleted. This is the normal mechanism for deleting +changes. Changes are created automatically on the first commit, and are deleted +automatically when evolve determines that they’ve been merged upstream. + +Orphan commits are commits with obsolete parents. The evolve command then +repeatedly rebases orphan commits with non-orphan parents until there are either +no orphan commits left, or a merge conflict is discovered. It will also +terminate if it detects a divergent parent or a cycle that can't be resolved +using any of the enabled transformations. + +When evolve discovers divergence, it will first check if it can resolve the +divergence automatically using one of its enabled transformations. Supported +transformations are: + +- Check if the user has already merged the divergent changes in a follow-up + change. That is, look for an existing merge in a follow-up change where all + the parents are divergent versions of the same change. Squash that merge with + its parents and use the result as the resolution for the divergence. + +- Attempt to auto-merge all the divergent changes (disabled by default). + +Each of the transformations can be enabled or disabled by command line options. + +Cycles can occur when two changes reference one another as parents. This can +happen when both changes use an obsolete version of the other change as their +parent. Although there are never cycles in the commit graph, users can create +cycles in the change graph by rebasing changes onto obsolete commits. The evolve +command has a transformation that will detect and break cycles by arbitrarily +picking one of the changes to go first. If this generates a merge conflict, +it tries each of the other changes in sequence to see if any ordering merges +cleanly. If no possible ordering merges cleanly, it picks one and terminates +to let the user resolve the merge conflict. + +If the working tree is dirty, evolve will attempt to stash the user's changes +before applying the evolve and then reapply those changes afterward, in much +the same way as rebase --autostash does. + +Checkout +-------- +Running checkout on a change by name has the same effect as checking out a +detached head pointing to the latest commit on that change-branch. There is no +need to ever have HEAD point to a change since changes always move forward when +necessary, no matter what branch the user has checked out + +Meta-commits themselves cannot be checked out by their hash. + +Reset +----- +Resetting a branch to a change by name is the same as resetting to the content +(or abandoned) commit at that change’s head. + +Commit +------ +Commit --amend gets modify semantics and will move existing changes forward. The +normal form of commit gets create semantics and will create a new change. + +$ touch foo && git add . && git commit -m "foo" && git tag A +$ touch bar && git add . && git commit -m "bar" && git tag B +$ touch baz && git add . && git commit -m "baz" && git tag C + +This produces the following commits: +A(tree=[foo]) +B(tree=[foo, bar], parent=A) +C(tree=[foo, bar, baz], parent=B) + +...along with three changes: +metas/foo = A +metas/bar = B +metas/baz = C + +Running commit --amend does the following: +$ git checkout B +$ touch zoom && git add . && git commit --amend -m "baz and zoom" +$ git tag D + +Commits: +A(tree=[foo]) +B(tree=[foo, bar], parent=A) +C(tree=[foo, bar, baz], parent=B) +D(tree=[foo, bar, zoom], parent=A) +Dmeta(content=D, obsolete=B) + +Changes: +metas/foo = A +metas/bar = Dmeta +metas/baz = C + +Merge +----- +Merge gets create, modify, or copy semantics based on what is being merged and +the options being used. + +The --squash version of merge gets copy semantics (it produces a new change that +is marked as a copy of all the original changes that were squashed into it). + +The “modify” version of merge replaces both of the original commits with the +resulting merge commit. This is one of the standard mechanisms for resolving +divergence. The parents of the merge commit are the parents of the two commits +being merged. The resulting commit will not be a merge commit if both of the +original commits had the same parent or if one was the parent of the other. + +The “create” version of merge creates a new change pointing to a merge commit +that has both original commits as parents. The result is what merge produces now +- a new merge commit. However, this version of merge doesn’t directly resolve +divergence. + +To select between these two behaviors, merge gets new “--amend” and “--noamend” +options which select between the “create” and “modify” behaviors respectively, +with noamend being the default. + +For example, imagine we created two divergent changes like this: + +$ touch foo && git add . && git commit -m "foo" && git tag A +$ touch bar && git add . && git commit -m "bar" && git tag B +$ touch baz && git add . && git commit --amend -m "bar and baz" +$ git tag C +$ git checkout B +$ touch bam && git add . && git commit --amend -m "bar and bam" +$ git tag D + +At this point the commit graph looks like this: + +A(tree=[foo]) +B(tree=[bar], parent=A) +C(tree=[bar, baz], parent=A) +D(tree=[bar, bam], parent=A) +Cmeta(content=C, obsoletes=B) +Dmeta(content=D, obsoletes=B) + +There would be three active changes with heads pointing as follows: + +metas/changeA=A +metas/changeB=Cmeta +metas/changeB2=Dmeta + +ChangeB and changeB2 are divergent at this point. Lets consider what happens if +perform each type of merge between changeB and changeB2. + +Merge example: Amend merge +One way to resolve divergent changes is to use an amend merge. Recall that HEAD +is currently pointing to D at this point. + +$ git merge --amend metas/changeB + +Here we’ve asked for an amend merge since we’re trying to resolve divergence +between two versions of the same change. There are no conflicts so we end up +with this: + +E(tree=[bar, baz, bam], parent=A) +Emeta(content=E, obsoletes=[Cmeta, Dmeta]) + +With the following branches: + +metas/changeA=A +metas/changeB=Emeta +metas/changeB2=Emeta + +Notice that the result of the “amend merge” is a replacement for C and D rather +than a new commit with C and D as parents (as a normal merge would have +produced). The parents of the amend merge are the parents of C and D which - in +this case - is just A, so the result is not a merge commit. Also notice that +changeB and changeB2 are now aliases for the same change. + +Merge example: Noamend merge +Consider what would have happened if we’d used a noamend merge instead. Recall +that HEAD was at D and our branches looked like this: + +metas/changeA=A +metas/changeB=Cmeta +metas/changeB2=Dmeta + +$ git merge --noamend metas/changeB + +That would produce the sort of merge we’d normally expect today: + +F(tree=[bar, baz, bam], parent=[C, D]) + +And our changes would look like this: +metas/changeA=A +metas/changeB=Cmeta +metas/changeB2=Dmeta +metas/changeF=F + +In this case, changeB and changeB2 are still divergent and we’ve created a new +change for our merge commit. However, this is just a temporary state. The next +time we run the “evolve” command, it will discover the divergence but also +discover the merge commit F that resolves it. Evolve will suggest converting F +into an amend merge in order to resolve the divergence and will display the +command for doing so. + +Rebase +------ +In general the rebase command is treated as a modify command. When a change is +rebased, the new commit replaces the original. + +Rebase --abort is special. Its intent is to restore git to the state it had +prior to running rebase. It should move back any changes to point to the refs +they had prior to running rebase and delete any new changes that were created as +part of the rebase. To achieve this, rebase will save the state of all changes +in refs/metas prior to running rebase and will restore the entire namespace +after rebase completes (deleting any newly-created changes). Newly-created +metacommits are left in place, but will have no effect until garbage collected +since metacommits are only used if they are reachable from refs/metas. + +Change +------ +The “change” command can be used to list, rename, reset or delete change. It has +a number of subcommands. + +The "list" subcommand lists local changes. If given the -r argument, it lists +remote changes. + +The "rename" subcommand renames a change, given its old and new name. If the old +name is omitted and there is exactly one change pointing to the current HEAD, +that change is renamed. If there are no changes pointing to the current HEAD, +one is created with the given name. + +The "forget" subcommand deletes a change by deleting its ref from the metas/ +namespace. This is the normal way to delete extra aliases for a change if the +change has more than one name. By default, this will refuse to delete the last +alias for a change if there are any other changes that reference this change as +a parent. + +The "update" subcommand adds a new state to a change. It uses the default +algorithm for assigning change names. If the content commit is omitted, HEAD is +used. If given the optional --force argument, it will overwrite any existing +change of the same name. This latter form of "update" can be used to effectively +reset changes. + +The "update" command can accept any number of --origin and --replace arguments. +If any are present, the resulting change branch will point to a metacommit +containing the given origin and replacement edges. + +The "abandon" command deletes a change using obsolescence markers. It marks the +change as being obsolete and having been replaced by its parent. If given no +arguments, it applies to the current commit. Running evolve will cause any +abandoned changes to be removed from the branch. Any child changes will be +reparented on top of the parent of the abandoned change. If the current change +is abandoned, HEAD will move to point to its parent. + +The "restore" command restores a previously-abandoned change. + +The "prune" command deletes all obsolete changes and all changes that are +present in the given branch. Note that such changes can be recovered from the +reflog. + +Combined with the GC protection that is offered, this is intended to facilitate +a workflow that relies on changes instead of branches. Users could choose to +work with no local branches and use changes instead - both for mailing list and +gerrit workflows. + +Log +--- +When a commit is shown in git log that is part of a change, it is decorated with +extra change information. If it is the head of a change, the name of the change +is shown next to the list of branches. If it is obsolete, it is decorated with +the text “obsolete, commits behind ”. + +Log gets a new --obslog argument indicating that the obsolescence graph should +be followed instead of the commit graph. This also changes the default +formatting options to make them more appropriate for viewing different +iterations of the same commit. + +Pull +---- + +Pull gets an --evolve argument that will automatically attempt to run "evolve" +on any affected branches after pulling. + +We also introduce an "evolve" enum value for the branch..rebase config +value. When set, the evolve behavior will happen automatically for that branch +after every pull even if the --evolve argument is not used. + +Next +---- + +The "next" command will reset HEAD to a non-obsolete commit that refers to this +change as its parent. If there is more than one such change, the user will be +prompted. If given the --evolve argument, the next commit will be evolved if +necessary first. + +The "next" command can be thought of as the opposite of +"git reset --hard HEAD^" in that it navigates to a child commit rather than a +parent. + +Prev +---- + +The "prev" command will reset HEAD to the latest version of the parent change. +If the parent change isn't obsolete, this is equivalent to +"git reset --hard HEAD^". If the parent commit is obsolete, it resets to the +latest replacement for the parent commit. + +Other options considered +======================== +We considered several other options for storing the obsolescence graph. This +section describes the other options and why they were rejected. + +Commit header +------------- +Add an “obsoletes” field to the commit header that points backwards from a +commit to the previous commits it obsoletes. + +Pros: +- Very simple +- Easy to traverse from a commit to the previous commits it obsoletes. +Cons: +- Adds a cost to the storage format, even for commits where the change history + is uninteresting. +- Unconditionally prevents the change history from being garbage collected. +- Always causes the change history to be shared when pushing or pulling changes. + +Git notes +--------- +Instead of storing obsolescence information in metacommits, the metacommit +content could go in a new notes namespace - say refs/notes/metacommit. Each note +would contain the list of obsolete and origin parents. An automerger could +be supplied to make it easy to merge the metacommit notes from different remotes. + +Pros: +- Easy to locate all commits obsoleted by a given commit (since there would only + be one metacommit for any given commit). +Cons: +- Wrong GC behavior (obsolete commits wouldn’t automatically be retained by GC) + unless we introduced a special case for these kinds of notes. +- No way to selectively share or pull the metacommits for one specific change. + It would be all-or-nothing, which would be expensive. This could be addressed + by changes to the protocol, but this would be invasive. +- Requires custom auto-merging behavior on fetch. + +Tags +---- +Put the content of the metacommit in a message attached to tag on the +replacement commit. This is very similar to the git notes approach and has the +same pros and cons. + +Simple forward references +------------------------- +Record an edge from an obsolete commit to its replacement in this form: + +refs/obsoletes/ + +pointing to commit as an indication that B is the replacement for the +obsolete commit A. + +Pros: +- Protects from being garbage collected. +- Fast lookup for the evolve operation, without additional search structures + (“what is the replacement for ?” is very fast). + +Cons: +- Can’t represent divergence (which is a P0 requirement). +- Creates lots of refs (which can be inefficient) +- Doesn’t provide a way to fetch only refs for a specific change. +- The obslog command requires a search of all refs. + +Complex forward references +-------------------------- +Record an edge from an obsolete commit to its replacement in this form: + +refs/obsoletes//obs_ + +Pointing to commit as an indication that B is the replacement for obsolete +commit A. + +Pros: +- Permits sharing and fetching refs for only a specific change. +- Supports divergence +- Protects from being garbage collected. + +Cons: +- Creates lots of refs, which is inefficient. +- Doesn’t provide a good lookup structure for lookups in either direction. + +Backward references +------------------- +Record an edge from a replacement commit to the obsolete one in this form: + +refs/obsolescences/ + +Cons: +- Doesn’t provide a way to resolve divergence (which is a P0 requirement). +- Doesn’t protect from being garbage collected (which could be fixed by + combining this with a refs/metas namespace, as in the metacommit variant). + +Obsolescences file +------------------ +Create a custom file (or files) in .git recording obsolescences. + +Pros: +- Can store exactly the information we want with exactly the performance we want + for all operations. For example, there could be a disk-based hashtable + permitting constant time lookups in either direction. + +Cons: +- Handling GC, pushing, and pulling would all require custom solutions. GC + issues could be addressed with a repository format extension. + +Squash points +------------- +We treat changes like topic branches, and use special squash points to mark +places in the commit graph that separate changes. + +We create and update change branches in refs/metas at the same time we +would have in the metacommit proposal. However, rather than pointing to a +metacommit branch they point to normal commits and are treated as “squash +points” - markers for sequences of commits intended to be squashed together on +submission. + +Amends and rebases work differently than they do now. Rather than actually +containing the desired state of a commit, they contain a delta from the previous +version along with a squash point indicating that the preceding changes are +intended to be squashed on submission. Specifically, amends would become new +changes and rebases would become merge commits with the old commit and new +parent as parents. + +When the changes are finally submitted, the squashes are executed, producing the +final version of the commit. + +In addition to the squash points, git would maintain a set of “nosquash” tags +for commits that were used as ancestors of a change that are not meant to be +included in the squash. + +For example, if we have this commit graph: + +A(...) +B(parent=A) +C(parent=B) + +...and we amend B to produce D, we’d get: + +A(...) +B(parent=A) +C(parent=B) +D(parent=B) + +...along with a new change branch indicating D should be squashed with its +parents when submitted: + +metas/changeB = D +metas/changeC = C + +We’d also create a nosquash tag for A indicating that A shouldn’t be included +when changeB is squashed. + +If a user amends the change again, they’d get: + +A(...) +B(parent=A) +C(parent=B) +D(parent=B) +E(parent=D) + +metas/changeB = E +metas/changeC = C + +Pros: +- Good GC behavior. +- Provides a natural way to share changes (they’re just normal branches). +- Merge-base works automatically without special cases. +- Rewriting the obslog would be easy using existing git commands. +- No new data types needed. +Cons: +- No way to connect the squashed version of a change to the original, so no way + to automatically clean up old changes. This also means users lose all benefits + of the evolve command if they prematurely squash their commits. This may occur + if a user thinks a change is ready for submission, squashes it, and then later + discovers an additional change to make. +- Histories would look very cluttered (users would see all previous edits to + their commit in the commit log, and all previous rebases would show up as + merges). Could be quite hard for users to tell what is going on. (Possible + fix: also implement a new smart log feature that displays the log as though + the squashes had occurred). +- Need to change the current behavior of current commands (like amend and + rebase) in ways that will be unexpected to many users. From patchwork Fri Sep 23 18:55:16 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Chris Poucet X-Patchwork-Id: 12986885 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id EC5E8C04A95 for ; Fri, 23 Sep 2022 18:55:48 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231145AbiIWSzr (ORCPT ); Fri, 23 Sep 2022 14:55:47 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:34022 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232342AbiIWSza (ORCPT ); Fri, 23 Sep 2022 14:55:30 -0400 Received: from mail-wr1-x42e.google.com (mail-wr1-x42e.google.com [IPv6:2a00:1450:4864:20::42e]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 4E879F190E for ; Fri, 23 Sep 2022 11:55:29 -0700 (PDT) Received: by mail-wr1-x42e.google.com with SMTP id cc5so1308541wrb.6 for ; Fri, 23 Sep 2022 11:55:29 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=YrdR4Hozyc3DsXTy4w4GIs3Pp3AzvNFwFd5DXY0kiR4=; b=p7H153SFts+vmWSGhjCsJz0RMeBeBIwtXZrwx4wDxPXxud9P5dwvTDcB6VobmT2yvK vIEIjcHSrgYR2pqjfuobuvuqkGm16d0ECflcE2RHQCQUZJso+faAjXcK8oQo2LjZI+CJ 5ExbBMf8HL8MPPWimL2SK3dfnblHuogf1PO87umMGmfnDLm4zlvsyCwcPt0ao5AAnCPM 0PEUgge8EGeDJYkaOLg5Hi81sJ/Vt24dGaCcmrBlxJCheKo5115pBAIuYjUDdFuSBYci 41Y76FuJbiRxnf7POoEgLOrEqp8VNdBO9u9cCwZ06L/hf4d0pI5zhfUncJHBkC+G3rkW PoFg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=YrdR4Hozyc3DsXTy4w4GIs3Pp3AzvNFwFd5DXY0kiR4=; b=WSEcl4rv65SNPp/cO8qyOqBsrlFVuwOUO3DT+wT3t9ZWi4Qagy1LahA+litSrw7DFr 4i2rbt5NcRl4xWbptwhLBEx2Tz1F1HcXZg5LgMXiK9+AJMPDT9SYxoVMETRgyaEB9PKH FcGCKQv4WF1j8C9FfJ2xZKCAAJZ2+0146ut7gt7AE8G5Zi82SGJ6p10JuqpyJNqedHUd W3cgzE85bKeecC4w8AV09tKeHf0znhxQfWUug4MX+yxHdg/ikJJYoJHUCbKEnalthLx7 bMOfcqvuGQFYRDMUXujpQPR1+7ZqLou4jYktbKqmC9sOkzZxHKMRj62N6jyr8K9kJ+cE kPnw== X-Gm-Message-State: ACrzQf12Om9fOQ860T0kf11Ce8MTIStv14TQWg+Tfm4abH43qzsNupgi FjMzK1HnNLhdmtDJuQ9RRV4MBtZp77A= X-Google-Smtp-Source: AMsMyM4DnaFc1Lzw77JuE2iotvMCaZ7WbPyMgFM59b6veKk+7A1lUNNc7HkbmyxsmLKAij7gpcgNzw== X-Received: by 2002:a05:6000:1ac7:b0:22a:906d:3577 with SMTP id i7-20020a0560001ac700b0022a906d3577mr6367845wry.33.1663959327570; Fri, 23 Sep 2022 11:55:27 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id d12-20020a5d6dcc000000b0022af5e36981sm9503341wrz.9.2022.09.23.11.55.27 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:27 -0700 (PDT) Message-Id: <84588312c1d4a62ff6c6211e85b4e58ab0563daa.1663959324.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:16 +0000 Subject: [PATCH 02/10] sha1-array: implement oid_array_readonly_contains Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Chris Poucet Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Chris Poucet From: Chris Poucet Implement a "readonly_contains" function for oid_array that won't sort the array if it is unsorted. This can be used to test containment in the rare situations where the array order matters. The function has intentionally been given a name that is more cumbersome than the "lookup" function, which is what most callers will will want in most situations. Signed-off-by: Chris Poucet --- oid-array.c | 12 ++++++++++++ oid-array.h | 7 +++++++ t/helper/test-oid-array.c | 6 ++++++ t/t0064-oid-array.sh | 22 ++++++++++++++++++++++ 4 files changed, 47 insertions(+) diff --git a/oid-array.c b/oid-array.c index 73ba76e9e9a..1e12651d245 100644 --- a/oid-array.c +++ b/oid-array.c @@ -28,6 +28,18 @@ static const struct object_id *oid_access(size_t index, const void *table) return &array[index]; } +int oid_array_readonly_contains(const struct oid_array *array, + const struct object_id* oid) { + int i; + + if (array->sorted) + return oid_pos(oid, array->oid, array->nr, oid_access) >= 0; + for (i = 0; i < array->nr; i++) + if (oideq(&array->oid[i], oid)) + return 1; + return 0; +} + int oid_array_lookup(struct oid_array *array, const struct object_id *oid) { oid_array_sort(array); diff --git a/oid-array.h b/oid-array.h index f60f9af6741..e056eb61fa2 100644 --- a/oid-array.h +++ b/oid-array.h @@ -58,6 +58,13 @@ struct oid_array { #define OID_ARRAY_INIT { 0 } +/** + * Sees whether an array contains an object ID. Optimized for when the array is + * sorted but does not require the array to be sorted. + */ +int oid_array_readonly_contains(const struct oid_array *array, + const struct object_id* oid); + /** * Add an item to the set. The object ID will be placed at the end of the array * (but note that some operations below may lose this ordering). diff --git a/t/helper/test-oid-array.c b/t/helper/test-oid-array.c index d1324d086a2..0dbfc91ca8d 100644 --- a/t/helper/test-oid-array.c +++ b/t/helper/test-oid-array.c @@ -28,10 +28,16 @@ int cmd__oid_array(int argc, const char **argv) if (get_oid_hex(arg, &oid)) die("not a hexadecimal oid: %s", arg); printf("%d\n", oid_array_lookup(&array, &oid)); + } else if (skip_prefix(line.buf, "readonly_contains ", &arg)) { + if (get_oid_hex(arg, &oid)) + die("not a hexadecimal oid: %s", arg); + printf("%d\n", oid_array_readonly_contains(&array, &oid)); } else if (!strcmp(line.buf, "clear")) oid_array_clear(&array); else if (!strcmp(line.buf, "for_each_unique")) oid_array_for_each_unique(&array, print_oid, NULL); + else if (!strcmp(line.buf, "for_each")) + oid_array_for_each(&array, print_oid, NULL); else die("unknown command: %s", line.buf); } diff --git a/t/t0064-oid-array.sh b/t/t0064-oid-array.sh index 88c89e8f48a..aa677af132d 100755 --- a/t/t0064-oid-array.sh +++ b/t/t0064-oid-array.sh @@ -35,6 +35,28 @@ test_expect_success 'ordered enumeration with duplicate suppression' ' test_cmp expect actual ' +test_expect_success 'readonly_contains finds existing' ' + echo 1 >expect && + echoid "" 88 44 aa 55 >>expect && + { + echoid append 88 44 aa 55 && + echoid readonly_contains 55 && + echo for_each + } | test-tool oid-array >actual && + test_cmp expect actual +' + +test_expect_success 'readonly_contains non-existing query' ' + echo 0 >expect && + echoid "" 88 44 aa 55 >>expect && + { + echoid append 88 44 aa 55 && + echoid readonly_contains 33 && + echo for_each + } | test-tool oid-array >actual && + test_cmp expect actual +' + test_expect_success 'lookup' ' { echoid append 88 44 aa 55 && From patchwork Fri Sep 23 18:55:17 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Chris Poucet X-Patchwork-Id: 12986886 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 750B7C04A95 for ; Fri, 23 Sep 2022 18:55:51 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231931AbiIWSzu (ORCPT ); Fri, 23 Sep 2022 14:55:50 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60940 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232648AbiIWSzb (ORCPT ); Fri, 23 Sep 2022 14:55:31 -0400 Received: from mail-wr1-x429.google.com (mail-wr1-x429.google.com [IPv6:2a00:1450:4864:20::429]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 4BF7A107582 for ; Fri, 23 Sep 2022 11:55:30 -0700 (PDT) Received: by mail-wr1-x429.google.com with SMTP id y5so1337158wrh.3 for ; Fri, 23 Sep 2022 11:55:30 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=tQxBWPWflkWS44dRHvnk1VbOleVWaGQKFc7Rjod28Hw=; b=Id6/s3oWJEW9fF2TU2a7CqfSCY1SVMlXARVIiz8Vlr8nJY1OQsh43pfrW+iHEG1BxE vKNToyEyLr/A7aob302jPnb+fvDBF979sczrQzdaSntZDNihp7E9mUN3WSpKAQRsnCHT YqrNgzvJ5Oj8XdgFqtRxLkxjGUPakDYp+lKGgy1VgTxKCI9svdat9zDs/OTOK/lZ3n5M 4yiYBkT+D1GUQazjMrUSM2s2FRIvIoxfLpuk1bceVT11EkAnk2a6BBBfPEHLHlrfQaE4 XMlJpg3pL9+e2qdaHFid7baWZyi7QVcbwQ9z0kqeiV93o41RcnXkEBy9AxJhDtMJBP/g yDRw== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=tQxBWPWflkWS44dRHvnk1VbOleVWaGQKFc7Rjod28Hw=; b=QZKmGoHiR4CnQObsitModNMI+McXYuAq2tt9Y7yocRRmF/DvlYHunjJ0xy7iTIjYeG SeCip7ECp67N+nvx+2kyJfAwyl2ymAWKrpsLZ3xt1kytXlA5WrLIl2ibXdtYKXOgLObe ohAcNXWPkmcxBeCQAilo96Gpz8Dcf8QQmbYWa+7kxtG0R74wFP4qdNUG/R0pU1qVM5vH jXP3NogKhmm2+BMs//HQjFk4X+OjdgxGilLlACrcJjrn4yUHzhqozW1WF065JZ5TFnta JCAcDJnOkppspFKRvxJtPs2cFBg4ZMt9mIaPNsjhLE0DJZ5jhb1NdK2vfXJHN0VzbffH OSHg== X-Gm-Message-State: ACrzQf11zJVq0rD4RRtc6x0CIABVrf+CYgb/8x6wW+TO2qEY8rChiSiT iaQ+SR4M2QJ3KnccnO6WZQL1AK/mQMw= X-Google-Smtp-Source: AMsMyM67LOBXQzjUnYDpUk5fw8FG+0pC9yQbsq0vqx/ydFEeN1roPNIzVVxbty01OWQGghxiN2NJlw== X-Received: by 2002:a5d:4571:0:b0:22a:bc4c:3c24 with SMTP id a17-20020a5d4571000000b0022abc4c3c24mr6207498wrc.254.1663959328582; Fri, 23 Sep 2022 11:55:28 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id p4-20020a1c5444000000b003a5c7a942edsm3218876wmi.28.2022.09.23.11.55.27 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:28 -0700 (PDT) Message-Id: <54e559967df55ca314e629b65927a88c7f804a98.1663959324.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:17 +0000 Subject: [PATCH 03/10] ref-filter: add the metas namespace to ref-filter Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Chris Poucet Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Chris Poucet From: Chris Poucet The metas namespace will contain refs for changes in progress. Add support for searching this namespace. Signed-off-by: Chris Poucet --- ref-filter.c | 8 ++++++-- ref-filter.h | 4 +++- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/ref-filter.c b/ref-filter.c index fd1cb14b0f1..6a1789c623f 100644 --- a/ref-filter.c +++ b/ref-filter.c @@ -2200,7 +2200,8 @@ static int ref_kind_from_refname(const char *refname) } ref_kind[] = { { "refs/heads/" , FILTER_REFS_BRANCHES }, { "refs/remotes/" , FILTER_REFS_REMOTES }, - { "refs/tags/", FILTER_REFS_TAGS} + { "refs/tags/", FILTER_REFS_TAGS}, + { "refs/metas/", FILTER_REFS_CHANGES } }; if (!strcmp(refname, "HEAD")) @@ -2218,7 +2219,8 @@ static int filter_ref_kind(struct ref_filter *filter, const char *refname) { if (filter->kind == FILTER_REFS_BRANCHES || filter->kind == FILTER_REFS_REMOTES || - filter->kind == FILTER_REFS_TAGS) + filter->kind == FILTER_REFS_TAGS || + filter->kind == FILTER_REFS_CHANGES) return filter->kind; return ref_kind_from_refname(refname); } @@ -2435,6 +2437,8 @@ int filter_refs(struct ref_array *array, struct ref_filter *filter, unsigned int ret = for_each_fullref_in("refs/remotes/", ref_filter_handler, &ref_cbdata); else if (filter->kind == FILTER_REFS_TAGS) ret = for_each_fullref_in("refs/tags/", ref_filter_handler, &ref_cbdata); + else if (filter->kind == FILTER_REFS_CHANGES) + ret = for_each_fullref_in("refs/metas/", ref_filter_handler, &ref_cbdata); else if (filter->kind & FILTER_REFS_ALL) ret = for_each_fullref_in_pattern(filter, ref_filter_handler, &ref_cbdata); if (!ret && (filter->kind & FILTER_REFS_DETACHED_HEAD)) diff --git a/ref-filter.h b/ref-filter.h index aa0eea4ecf5..064fbef8e50 100644 --- a/ref-filter.h +++ b/ref-filter.h @@ -17,8 +17,10 @@ #define FILTER_REFS_BRANCHES 0x0004 #define FILTER_REFS_REMOTES 0x0008 #define FILTER_REFS_OTHERS 0x0010 +#define FILTER_REFS_CHANGES 0x0040 #define FILTER_REFS_ALL (FILTER_REFS_TAGS | FILTER_REFS_BRANCHES | \ - FILTER_REFS_REMOTES | FILTER_REFS_OTHERS) + FILTER_REFS_REMOTES | FILTER_REFS_OTHERS | \ + FILTER_REFS_CHANGES) #define FILTER_REFS_DETACHED_HEAD 0x0020 #define FILTER_REFS_KIND_MASK (FILTER_REFS_ALL | FILTER_REFS_DETACHED_HEAD) From patchwork Fri Sep 23 18:55:18 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Stefan Xenos X-Patchwork-Id: 12986887 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id B33EDC04A95 for ; Fri, 23 Sep 2022 18:55:53 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232696AbiIWSzv (ORCPT ); Fri, 23 Sep 2022 14:55:51 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60050 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232716AbiIWSzd (ORCPT ); Fri, 23 Sep 2022 14:55:33 -0400 Received: from mail-wr1-x432.google.com (mail-wr1-x432.google.com [IPv6:2a00:1450:4864:20::432]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 240A9109502 for ; Fri, 23 Sep 2022 11:55:31 -0700 (PDT) Received: by mail-wr1-x432.google.com with SMTP id s14so1394701wro.0 for ; Fri, 23 Sep 2022 11:55:31 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=wLeJfwB0x5vNYkphAiWjsjsetnGyL3OOlu/8XEZlk/A=; b=AIBI7dBZ1XmjVgVyBNC9wuYudgcZA8jWxYeoWxgWS2zNg0YAIHyAJo28ZUZgQc8WEk ZS/EWIKH5uxS44PZDqYezyNXEJ8SDIBqfLVqyXLmPZZVi+vhWtu4EYdK+F798tjm9EHi ieoojHoK70jyxnFMqbVvCYpuMuT94KNOGgferB8u5O1WBpJXb9dn5SsOM/cHWw+Wl4dB MbRVfgM5/1nRgveZYuIdMd/Pf4X/uy+NFg8ac8NYl6Njt+RurCYjncAlVHgBtbRUPj4X UUAC6hXbOolH+sgqC5pZMT1VksCLA4vZRlWo8CYj2kSIuP/XaR3J0ZkWnvae/RYGpqs/ 9OXQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=wLeJfwB0x5vNYkphAiWjsjsetnGyL3OOlu/8XEZlk/A=; b=vTb1prb79X2c6Nbe3VTHl7pDcp/FuM6l/F2LGuw2oBHcJwLdLrN49w9zOWaAnVB/D3 TdTjN1qHHtBCgWsB1sMIm3iwMf8zX4rCSZV8J/WvLEfSwCEltIoc1tfBOry+N5bVecoK YXy+OT3e4wkYddlALT4SB1y+cmC2OILm4so7/LNdNzyY3lxjxqPHesR5BHHn6yFJHdgf SVTYtVzNKDdtcnEDEZWB3I+0WQmVYfs1P3BWgaifiIjLKd0XCTquhsHKgA5eVlDO1X0d x1SfQII0/GesHa8skGnWryaGrgJ6cyZGxf902Y3NCQMNjGMCFTUUW47sbpsLOp5xN1/3 6GIg== X-Gm-Message-State: ACrzQf2SdE65DE67CHmMHe7+EN2LD54chAtF54AmTeCeAFBTHAOJW0yG fyk+ETqTxtXyBGfHcgBhh0Rgeop054E= X-Google-Smtp-Source: AMsMyM4/1tFU6ONPZZsbuC1t5oWIy6RKChOAlV0nE36eQE8gWp1mWDDCsyDrJl3mmsEDrbZh+zUM0w== X-Received: by 2002:a5d:5a0d:0:b0:228:da3f:738 with SMTP id bq13-20020a5d5a0d000000b00228da3f0738mr6171050wrb.526.1663959329436; Fri, 23 Sep 2022 11:55:29 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id z2-20020a056000110200b00223b8168b15sm7842149wrw.66.2022.09.23.11.55.28 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:29 -0700 (PDT) Message-Id: <2e9a4a9bd819785404e8a5343385f4fb2bc06109.1663959325.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:18 +0000 Subject: [PATCH 04/10] evolve: add support for parsing metacommits Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Stefan Xenos Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Stefan Xenos From: Stefan Xenos This patch adds the get_metacommit_content method, which can classify commits as either metacommits or normal commits, determine whether they are abandoned, and extract the content commit's object id from the metacommit. Signed-off-by: Stefan Xenos Signed-off-by: Chris Poucet --- Makefile | 1 + metacommit-parser.c | 110 ++++++++++++++++++++++++++++++++++++++++++++ metacommit-parser.h | 19 ++++++++ 3 files changed, 130 insertions(+) create mode 100644 metacommit-parser.c create mode 100644 metacommit-parser.h diff --git a/Makefile b/Makefile index cac3452edb9..b2bcc00c289 100644 --- a/Makefile +++ b/Makefile @@ -999,6 +999,7 @@ LIB_OBJS += merge-ort.o LIB_OBJS += merge-ort-wrappers.o LIB_OBJS += merge-recursive.o LIB_OBJS += merge.o +LIB_OBJS += metacommit-parser.o LIB_OBJS += midx.o LIB_OBJS += name-hash.o LIB_OBJS += negotiator/default.o diff --git a/metacommit-parser.c b/metacommit-parser.c new file mode 100644 index 00000000000..70c1428bfc6 --- /dev/null +++ b/metacommit-parser.c @@ -0,0 +1,110 @@ +#include "cache.h" +#include "metacommit-parser.h" +#include "commit.h" + +/* + * Search the commit buffer for a line starting with the given key. Unlike + * find_commit_header, this also searches the commit message body. + */ +static const char *find_key(const char *msg, const char *key, size_t *out_len) +{ + int key_len = strlen(key); + const char *line = msg; + + while (line) { + const char *eol = strchrnul(line, '\n'); + + if (eol - line > key_len && !memcmp(line, key, key_len) && + line[key_len] == ' ') { + *out_len = eol - line - key_len - 1; + return line + key_len + 1; + } + line = *eol ? eol + 1 : NULL; + } + return NULL; +} + +static struct commit *get_commit_by_index(struct commit_list *to_search, int index) +{ + while (to_search && index) { + to_search = to_search->next; + index--; + } + + if (!to_search) + return NULL; + + return to_search->item; +} + +/* + * Writes the index of the content parent to "result". Returns the metacommit + * type. See the METACOMMIT_TYPE_* constants. + */ +static int index_of_content_commit(const char *buffer, int *result) +{ + int index = 0; + int ret = METACOMMIT_TYPE_NONE; + size_t parent_types_size; + const char *parent_types = find_key(buffer, "parent-type", + &parent_types_size); + const char *end; + const char *enum_start = parent_types; + int enum_length = 0; + + if (!parent_types) + return METACOMMIT_TYPE_NONE; + + end = &parent_types[parent_types_size]; + + while (1) { + char next = *parent_types; + if (next == ' ' || parent_types >= end) { + if (enum_length == 1) { + char first_char_in_enum = *enum_start; + if (first_char_in_enum == 'c') { + ret = METACOMMIT_TYPE_NORMAL; + break; + } + if (first_char_in_enum == 'a') { + ret = METACOMMIT_TYPE_ABANDONED; + break; + } + } + if (parent_types >= end) + return METACOMMIT_TYPE_NONE; + enum_start = parent_types + 1; + enum_length = 0; + index++; + } else { + enum_length++; + } + parent_types++; + } + + *result = index; + return ret; +} + +/* + * Writes the content parent's object id to "content". + * Returns the metacommit type. See the METACOMMIT_TYPE_* constants. + */ +int get_metacommit_content(struct commit *commit, struct object_id *content) +{ + const char *buffer = get_commit_buffer(commit, NULL); + int index = 0; + int ret = index_of_content_commit(buffer, &index); + struct commit *content_parent; + + if (ret == METACOMMIT_TYPE_NONE) + return ret; + + content_parent = get_commit_by_index(commit->parents, index); + + if (!content_parent) + return METACOMMIT_TYPE_NONE; + + oidcpy(content, &(content_parent->object.oid)); + return ret; +} diff --git a/metacommit-parser.h b/metacommit-parser.h new file mode 100644 index 00000000000..1c74bd6d699 --- /dev/null +++ b/metacommit-parser.h @@ -0,0 +1,19 @@ +#ifndef METACOMMIT_PARSER_H +#define METACOMMIT_PARSER_H + +#include "commit.h" +#include "hash.h" + +/* Indicates a normal commit (non-metacommit) */ +#define METACOMMIT_TYPE_NONE 0 +/* Indicates a metacommit with normal content (non-abandoned) */ +#define METACOMMIT_TYPE_NORMAL 1 +/* Indicates a metacommit with abandoned content */ +#define METACOMMIT_TYPE_ABANDONED 2 + +struct commit; + +extern int get_metacommit_content( + struct commit *commit, struct object_id *content); + +#endif From patchwork Fri Sep 23 18:55:19 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Stefan Xenos X-Patchwork-Id: 12986889 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 05234C6FA83 for ; Fri, 23 Sep 2022 18:56:00 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S231409AbiIWSz7 (ORCPT ); Fri, 23 Sep 2022 14:55:59 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:33754 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232239AbiIWSzf (ORCPT ); Fri, 23 Sep 2022 14:55:35 -0400 Received: from mail-wr1-x42b.google.com (mail-wr1-x42b.google.com [IPv6:2a00:1450:4864:20::42b]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 2AC61106A3E for ; Fri, 23 Sep 2022 11:55:32 -0700 (PDT) Received: by mail-wr1-x42b.google.com with SMTP id x18so1303458wrm.7 for ; Fri, 23 Sep 2022 11:55:31 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=saeeViDxA8shKM5+tFUSDhs7ZdAizgQQryebkqDrAkQ=; b=XYkrBazqeuH2qVm6OQtrZ9FP/V4wBlyc6Hixmm63PJWj5dfcEJI5/LhQfBC/DLAZLT 9SS0Lp3T9o7Qz3eWkGi5oJ54dt1qEXY225bQPTnHVbfbc1H7wMS3DrfPYb9kXIPZ6qS7 AbDGhSK9hqjnd7OzKX6m53RphqXxlrT4taR8viCpWL2hW1yu+IBFwW6aZk6ZKiM+GC5L QoOF5iTG0MizWPIvwGY3FyO9JFiSJu73wDL9EjVd7KDtaRh6oN8e0UPizEN9qwn8802s qnmR61At1KyZ08dFtY0g79rNbU0t6sTdGWc6O6OJmHS6uxsAkRGVZnjsl87/s/Browsc BHXA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=saeeViDxA8shKM5+tFUSDhs7ZdAizgQQryebkqDrAkQ=; b=mmIjGNj67r2D3SIWOvVjsAP3WegGIJStnqJrxnc8/zN80L7YzWxvB81QDz0BfwT1Mb 89pUYwV76ME57FtjfsnM9XDRu02J90kj7TwDcE3CNli4hD0rs18SXAQTUvbekloa5lZO cVKupDw7gLjldOx1NsxTLKM11cXCxbCBQrvr3D1Twufmmc1Ztpc5ok+p7fBuZ4jBCnCC 5b38sMJmI5/SC6A4Ky0c1lxpYydycKo3lhbRCIS90NYDnNv0B8M2gH6bE4P6QZSnGv9c x1w3/2GD+Aa33i6jmDFfOOC1vKIw/WsfxfaNGF8lw3gwt5U//OTTP2AyCQYAHFc0xOnn NjWg== X-Gm-Message-State: ACrzQf2qn5jWVzR01vSBuDFbsGrEKJSoub51ZwWqD8pyR7NnwqemwPIE CULR1V7hAfTMy38K+ZXpxQzgH/6bmuA= X-Google-Smtp-Source: AMsMyM4YwwWZcrzPJ/fCQQXS04VLX9NsVXmMieMjgEfvMOcZL+6DfXwt7CZCqeL306EROD42xeQRMg== X-Received: by 2002:a05:6000:2a6:b0:22a:fb53:316 with SMTP id l6-20020a05600002a600b0022afb530316mr6269519wry.55.1663959330271; Fri, 23 Sep 2022 11:55:30 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id p4-20020a1c5444000000b003a5c7a942edsm3218940wmi.28.2022.09.23.11.55.29 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:29 -0700 (PDT) Message-Id: <2b3a00a6702eb8fb12e45b833ca74155939588ef.1663959325.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:19 +0000 Subject: [PATCH 05/10] evolve: add the change-table structure Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Stefan Xenos Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Stefan Xenos From: Stefan Xenos A change table stores a list of changes, and supports efficient lookup from a commit hash to the list of changes that reference that commit directly. It can be used to look up content commits or metacommits at the head of a change, but does not support lookup of commits referenced as part of the commit history. Signed-off-by: Stefan Xenos Signed-off-by: Chris Poucet --- Makefile | 1 + change-table.c | 179 +++++++++++++++++++++++++++++++++++++++++++++++++ change-table.h | 132 ++++++++++++++++++++++++++++++++++++ 3 files changed, 312 insertions(+) create mode 100644 change-table.c create mode 100644 change-table.h diff --git a/Makefile b/Makefile index b2bcc00c289..2b847e7e7de 100644 --- a/Makefile +++ b/Makefile @@ -913,6 +913,7 @@ LIB_OBJS += bulk-checkin.o LIB_OBJS += bundle-uri.o LIB_OBJS += bundle.o LIB_OBJS += cache-tree.o +LIB_OBJS += change-table.o LIB_OBJS += cbtree.o LIB_OBJS += chdir-notify.o LIB_OBJS += checkout.o diff --git a/change-table.c b/change-table.c new file mode 100644 index 00000000000..c61ba29f1ed --- /dev/null +++ b/change-table.c @@ -0,0 +1,179 @@ +#include "cache.h" +#include "change-table.h" +#include "commit.h" +#include "ref-filter.h" +#include "metacommit-parser.h" + +void change_table_init(struct change_table *to_initialize) +{ + memset(to_initialize, 0, sizeof(*to_initialize)); + mem_pool_init(&to_initialize->memory_pool, 0); + to_initialize->memory_pool.block_alloc = 4*1024 - sizeof(struct mp_block); + oidmap_init(&to_initialize->oid_to_metadata_index, 0); + string_list_init_dup(&to_initialize->refname_to_change_head); +} + +static void change_list_clear(struct change_list *to_clear) { + string_list_clear(&to_clear->additional_refnames, 0); +} + +static void commit_change_list_entry_clear( + struct commit_change_list_entry *to_clear) { + change_list_clear(&to_clear->changes); +} + +void change_table_clear(struct change_table *to_clear) +{ + struct oidmap_iter iter; + struct commit_change_list_entry *next; + for (next = oidmap_iter_first(&to_clear->oid_to_metadata_index, &iter); + next; + next = oidmap_iter_next(&iter)) { + + commit_change_list_entry_clear(next); + } + + oidmap_free(&to_clear->oid_to_metadata_index, 0); + string_list_clear(&to_clear->refname_to_change_head, 0); + mem_pool_discard(&to_clear->memory_pool, 0); +} + +static void add_head_to_commit(struct change_table *to_modify, + const struct object_id *to_add, const char *refname) +{ + struct commit_change_list_entry *entry; + + /** + * Note: the indices in the map are 1-based. 0 is used to indicate a missing + * element. + */ + entry = oidmap_get(&to_modify->oid_to_metadata_index, to_add); + if (!entry) { + entry = mem_pool_calloc(&to_modify->memory_pool, 1, + sizeof(*entry)); + oidcpy(&entry->entry.oid, to_add); + oidmap_put(&to_modify->oid_to_metadata_index, entry); + string_list_init_nodup(&entry->changes.additional_refnames); + } + + if (!entry->changes.first_refname) + entry->changes.first_refname = refname; + else + string_list_insert(&entry->changes.additional_refnames, refname); +} + +void change_table_add(struct change_table *to_modify, const char *refname, + struct commit *to_add) +{ + struct change_head *new_head; + struct string_list_item *new_item; + int metacommit_type; + + new_head = mem_pool_calloc(&to_modify->memory_pool, 1, + sizeof(*new_head)); + + oidcpy(&new_head->head, &to_add->object.oid); + + metacommit_type = get_metacommit_content(to_add, &new_head->content); + if (metacommit_type == METACOMMIT_TYPE_NONE) + oidcpy(&new_head->content, &to_add->object.oid); + new_head->abandoned = (metacommit_type == METACOMMIT_TYPE_ABANDONED); + new_head->remote = starts_with(refname, "refs/remote/"); + new_head->hidden = starts_with(refname, "refs/hiddenmetas/"); + + new_item = string_list_insert(&to_modify->refname_to_change_head, refname); + new_item->util = new_head; + /* Use pointers to the copy of the string we're retaining locally */ + refname = new_item->string; + + if (!oideq(&new_head->content, &new_head->head)) + add_head_to_commit(to_modify, &new_head->content, refname); + add_head_to_commit(to_modify, &new_head->head, refname); +} + +void change_table_add_all_visible(struct change_table *to_modify, + struct repository* repo) +{ + struct ref_filter filter; + const char *name_patterns[] = {NULL}; + memset(&filter, 0, sizeof(filter)); + filter.kind = FILTER_REFS_CHANGES; + filter.name_patterns = name_patterns; + + change_table_add_matching_filter(to_modify, repo, &filter); +} + +void change_table_add_matching_filter(struct change_table *to_modify, + struct repository* repo, struct ref_filter *filter) +{ + struct ref_array matching_refs; + int i; + + memset(&matching_refs, 0, sizeof(matching_refs)); + filter_refs(&matching_refs, filter, filter->kind); + + /** + * Determine the object id for the latest content commit for each change. + * Fetch the commit at the head of each change ref. If it's a normal commit, + * that's the commit we want. If it's a metacommit, locate its content parent + * and use that. + */ + + for (i = 0; i < matching_refs.nr; i++) { + struct ref_array_item *item = matching_refs.items[i]; + struct commit *commit = item->commit; + + commit = lookup_commit_reference_gently(repo, &item->objectname, 1); + + if (commit) + change_table_add(to_modify, item->refname, commit); + } + + ref_array_clear(&matching_refs); +} + +static int return_true_callback(const char *refname, void *cb_data) +{ + return 1; +} + +int change_table_has_change_referencing(struct change_table *changes, + const struct object_id *referenced_commit_id) +{ + return for_each_change_referencing(changes, referenced_commit_id, + return_true_callback, NULL); +} + +int for_each_change_referencing(struct change_table *table, + const struct object_id *referenced_commit_id, each_change_fn fn, void *cb_data) +{ + const struct change_list *changes; + int i; + int retvalue; + struct commit_change_list_entry *entry; + + entry = oidmap_get(&table->oid_to_metadata_index, + referenced_commit_id); + /* If this commit isn't referenced by any changes, it won't be in the map */ + if (!entry) + return 0; + changes = &entry->changes; + if (!changes->first_refname) + return 0; + retvalue = fn(changes->first_refname, cb_data); + for (i = 0; retvalue == 0 && i < changes->additional_refnames.nr; i++) + retvalue = fn(changes->additional_refnames.items[i].string, cb_data); + return retvalue; +} + +struct change_head* get_change_head(struct change_table *heads, + const char* refname) +{ + struct string_list_item *item = string_list_lookup( + &heads->refname_to_change_head, refname); + + if (!item) + return NULL; + + return (struct change_head *)item->util; +} diff --git a/change-table.h b/change-table.h new file mode 100644 index 00000000000..166b5ed8073 --- /dev/null +++ b/change-table.h @@ -0,0 +1,132 @@ +#ifndef CHANGE_TABLE_H +#define CHANGE_TABLE_H + +#include "oidmap.h" + +struct commit; +struct ref_filter; + +/** + * This struct holds a list of change refs. The first element is stored inline, + * to optimize for small lists. + */ +struct change_list { + /** + * Ref name for the first change in the list, or null if none. + * + * This field is private. Use for_each_change_in to read. + */ + const char* first_refname; + /** + * List of additional change refs. Note that this is empty if the list + * contains 0 or 1 elements. + * + * This field is private. Use for_each_change_in to read. + */ + struct string_list additional_refnames; +}; + +/** + * Holds information about the head of a single change. + */ +struct change_head { + /** + * The location pointed to by the head of the change. May be a commit or a + * metacommit. + */ + struct object_id head; + /** + * The content commit for the latest commit in the change. Always points to a + * real commit, never a metacommit. + */ + struct object_id content; + /** + * Abandoned: indicates that the content commit should be removed from the + * history. + * + * Hidden: indicates that the change is an inactive change from the + * hiddenmetas namespace. Such changes will be hidden from the user by + * default. + * + * Deleted: indicates that the change has been removed from the repository. + * That is the ref was deleted since the time this struct was created. Such + * entries should be ignored. + */ + unsigned int abandoned:1, + hidden:1, + remote:1, + deleted:1; +}; + +/** + * Holds the list of change refs whose content points to a particular content + * commit. + */ +struct commit_change_list_entry { + struct oidmap_entry entry; + struct change_list changes; +}; + +/** + * Holds information about the heads of each change, and permits effecient + * lookup from a commit to the changes that reference it directly. + * + * All fields should be considered private. Use the change_table functions + * to interact with this struct. + */ +struct change_table { + /** + * Memory pool for the objects allocated by the change table. + */ + struct mem_pool memory_pool; + /* Map object_id to commit_change_list_entry structs. */ + struct oidmap oid_to_metadata_index; + /** + * List of ref names. The util value points to a change_head structure + * allocated from memory_pool. + */ + struct string_list refname_to_change_head; +}; + +extern void change_table_init(struct change_table *to_initialize); +extern void change_table_clear(struct change_table *to_clear); + +/* Adds the given change head to the change_table struct */ +extern void change_table_add(struct change_table *to_modify, + const char *refname, struct commit *target); + +/** + * Adds the non-hidden local changes to the given change_table struct. + */ +extern void change_table_add_all_visible(struct change_table *to_modify, + struct repository *repo); + +/* + * Adds all changes matching the given ref filter to the given change_table + * struct. + */ +extern void change_table_add_matching_filter(struct change_table *to_modify, + struct repository* repo, struct ref_filter *filter); + +typedef int each_change_fn(const char *refname, void *cb_data); + +extern int change_table_has_change_referencing(struct change_table *changes, + const struct object_id *referenced_commit_id); + +/** + * Iterates over all changes that reference the given commit. For metacommits, + * this is the list of changes that point directly to that metacommit. + * For normal commits, this is the list of changes that have this commit as + * their latest content. + */ +extern int for_each_change_referencing(struct change_table *heads, + const struct object_id *referenced_commit_id, each_change_fn fn, void *cb_data); + +/** + * Returns the change head for the given refname. Returns NULL if no such change + * exists. + */ +extern struct change_head* get_change_head(struct change_table *heads, + const char* refname); + +#endif From patchwork Fri Sep 23 18:55:20 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Stefan Xenos X-Patchwork-Id: 12986890 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 49EFCC07E9D for ; Fri, 23 Sep 2022 18:56:02 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232270AbiIWS4A (ORCPT ); Fri, 23 Sep 2022 14:56:00 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:33792 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232758AbiIWSzh (ORCPT ); Fri, 23 Sep 2022 14:55:37 -0400 Received: from mail-wm1-x330.google.com (mail-wm1-x330.google.com [IPv6:2a00:1450:4864:20::330]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 257E6109759 for ; Fri, 23 Sep 2022 11:55:33 -0700 (PDT) Received: by mail-wm1-x330.google.com with SMTP id ay36so805925wmb.0 for ; Fri, 23 Sep 2022 11:55:33 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=NeQWl2qdMmtbw/1ImI+AlCvvdPMWT0rOVIiAL2nrI78=; b=PRzqjZ1be7zil7kbhpzL1t/gtQx73wXhTsqCAHkU7kIiUfcMZMsmdjc5mA0OCJ+RjT o07Z2zAlY2Bec+DPVOLGr3bNzccLvaxY4aUdXjCGl/ubNxEyW0uINNV2HwHhJt9+0LJZ msSWz0nr/mCHBUyxaAHOxfkdtIIJ5TLhwDwmmQ3JVWNyY5g/tx0EUKQ46/SQ8A4pa9BF rU+SXMGeBUZ2QAL0EUXl4n660fjUwpKzX14dxZYPNOJ6SsuoTyylzrIJJG20x5DZo0x2 YPt+kT4C9ZiyqTk9IUvmEyAc/z1+4Xa4jBkKAywLjOr2Om+xz2lJIyvxLYIrjUcgUeNb irwA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=NeQWl2qdMmtbw/1ImI+AlCvvdPMWT0rOVIiAL2nrI78=; b=p84ueSJOZsQunS2NuWAjJvDUscz1cr288v2BzgM5OVM6TEaplRN4kcApJfB4c0n90a +9bPK30vJUlhahgzv1qkOauaVJCkPIMXN6gFOHonEvFq8RsiJ0q9MZNcSpjbwo92uQ9H 53fEvDojkWyCujnYoPMzo3dVILzC8UqxjfDOvr3ga+soGafL8ZtErF1FNCB28jmuod6f K5k20u7bMcbA+TvR9N7cuFOekgGHteTBsvL5ALLSfUtB1tIjIrBsw9C9ufo37OVevQcK YaVxXlJ4mMUh4IGkcua/2M1gciUWuzDijvO0FlOTu4iBdSkEAEPt4FSXyiipQO439xHO qbtw== X-Gm-Message-State: ACrzQf2LT7XHDT/OSdHgdsxtQP56HczIKLPs8khg6TS52zfKcxcbBo/X jsKUqB9+18f2vj1J09RCfrUqrkavMbU= X-Google-Smtp-Source: AMsMyM6rnjyw73EQrO6iONVmVg61fZ8zP4xl/lTQbOw9xe9kAFkn7miJ7XLy/3Yqdw0TQW6O6qLcng== X-Received: by 2002:a05:600c:4f45:b0:3b4:9c9a:7077 with SMTP id m5-20020a05600c4f4500b003b49c9a7077mr7064173wmq.109.1663959330983; Fri, 23 Sep 2022 11:55:30 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id p19-20020a05600c05d300b003b3365b38f9sm3068362wmd.10.2022.09.23.11.55.30 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:30 -0700 (PDT) Message-Id: <56c6770997bbdb1b3b87c2c410dd7f158b03f2d6.1663959325.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:20 +0000 Subject: [PATCH 06/10] evolve: add support for writing metacommits Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Stefan Xenos Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Stefan Xenos From: Stefan Xenos metacommit.c supports the creation of metacommits and adds the API needed to create and update changes. Create the "modify_change" function that can be called from modification commands like "rebase" and "git amend" to record obsolescences in the change graph. Create the "record_metacommit" function for recording more complicated commit relationships in the commit graph. Create the "write_metacommit" function for low-level creation of metacommits. Signed-off-by: Stefan Xenos Signed-off-by: Chris Poucet --- Makefile | 1 + metacommit.c | 404 +++++++++++++++++++++++++++++++++++++++++++++++++++ metacommit.h | 58 ++++++++ 3 files changed, 463 insertions(+) create mode 100644 metacommit.c create mode 100644 metacommit.h diff --git a/Makefile b/Makefile index 2b847e7e7de..68082ef94c7 100644 --- a/Makefile +++ b/Makefile @@ -1000,6 +1000,7 @@ LIB_OBJS += merge-ort.o LIB_OBJS += merge-ort-wrappers.o LIB_OBJS += merge-recursive.o LIB_OBJS += merge.o +LIB_OBJS += metacommit.o LIB_OBJS += metacommit-parser.o LIB_OBJS += midx.o LIB_OBJS += name-hash.o diff --git a/metacommit.c b/metacommit.c new file mode 100644 index 00000000000..d2b859a4d3b --- /dev/null +++ b/metacommit.c @@ -0,0 +1,404 @@ +#include "cache.h" +#include "metacommit.h" +#include "commit.h" +#include "change-table.h" +#include "refs.h" + +void init_metacommit_data(struct metacommit_data *state) +{ + memset(state, 0, sizeof(*state)); +} + +void clear_metacommit_data(struct metacommit_data *state) +{ + oid_array_clear(&state->replace); + oid_array_clear(&state->origin); +} + +static void compute_default_change_name(struct commit *initial_commit, + struct strbuf* result) +{ + struct strbuf default_name; + const char *buffer; + const char *subject; + const char *eol; + int len; + strbuf_init(&default_name, 0); + buffer = get_commit_buffer(initial_commit, NULL); + find_commit_subject(buffer, &subject); + eol = strchrnul(subject, '\n'); + for (len = 0;subject < eol && len < 10; ++subject, ++len) { + char next = *subject; + if (isspace(next)) + continue; + + strbuf_addch(&default_name, next); + } + sanitize_refname_component(default_name.buf, result); +} + +/** + * Computes a change name for a change rooted at the given initial commit. Good + * change names should be memorable, unique, and easy to type. They are not + * required to match the commit comment. + */ +static void compute_change_name(struct commit *initial_commit, struct strbuf* result) +{ + struct strbuf default_name; + struct object_id unused; + + strbuf_init(&default_name, 0); + if (initial_commit) + compute_default_change_name(initial_commit, &default_name); + else + strbuf_addstr(&default_name, "change"); + strbuf_addstr(result, "refs/metas/"); + strbuf_addbuf(result, &default_name); + + /* If there is already a change of this name, append a suffix */ + if (!read_ref(result->buf, &unused)) { + int suffix = 2; + int original_length = result->len; + + while (1) { + strbuf_addf(result, "%d", suffix); + if (read_ref(result->buf, &unused)) + break; + strbuf_remove(result, original_length, result->len - original_length); + ++suffix; + } + } + + strbuf_release(&default_name); +} + +struct resolve_metacommit_callback_data +{ + struct change_table* active_changes; + struct string_list *changes; + struct oid_array *heads; +}; + +static int resolve_metacommit_callback(const char *refname, void *cb_data) +{ + struct resolve_metacommit_callback_data *data = (struct resolve_metacommit_callback_data *)cb_data; + struct change_head *chhead; + + chhead = get_change_head(data->active_changes, refname); + + if (data->changes) + string_list_append(data->changes, refname)->util = &(chhead->head); + if (data->heads) + oid_array_append(data->heads, &(chhead->head)); + + return 0; +} + +/** + * Produces the final form of a metacommit based on the current change refs. + */ +static void resolve_metacommit( + struct repository* repo, + struct change_table* active_changes, + const struct metacommit_data *to_resolve, + struct metacommit_data *resolved_output, + struct string_list *to_advance, + int allow_append) +{ + int i; + int len = to_resolve->replace.nr; + struct resolve_metacommit_callback_data cbdata; + int old_change_list_length = to_advance->nr; + struct commit* content; + + oidcpy(&resolved_output->content, &to_resolve->content); + + /* First look for changes that point to any of the replacement edges in the + * metacommit. These will be the changes that get advanced by this + * metacommit. */ + resolved_output->abandoned = to_resolve->abandoned; + cbdata.active_changes = active_changes; + cbdata.changes = to_advance; + cbdata.heads = &(resolved_output->replace); + + if (allow_append) { + for (i = 0; i < len; i++) { + int old_number = resolved_output->replace.nr; + for_each_change_referencing(active_changes, &(to_resolve->replace.oid[i]), + resolve_metacommit_callback, &cbdata); + /* If no changes were found, use the unresolved value. */ + if (old_number == resolved_output->replace.nr) + oid_array_append(&(resolved_output->replace), &(to_resolve->replace.oid[i])); + } + } + + cbdata.changes = NULL; + cbdata.heads = &(resolved_output->origin); + + len = to_resolve->origin.nr; + for (i = 0; i < len; i++) { + int old_number = resolved_output->origin.nr; + for_each_change_referencing(active_changes, &(to_resolve->origin.oid[i]), + resolve_metacommit_callback, &cbdata); + if (old_number == resolved_output->origin.nr) + oid_array_append(&(resolved_output->origin), &(to_resolve->origin.oid[i])); + } + + /* If no changes were advanced by this metacommit, we'll need to create a new + * one. */ + if (to_advance->nr == old_change_list_length) { + struct strbuf change_name; + + strbuf_init(&change_name, 80); + content = lookup_commit_reference_gently(repo, &(to_resolve->content), 1); + + compute_change_name(content, &change_name); + string_list_append(to_advance, change_name.buf); + strbuf_release(&change_name); + } +} + +static void lookup_commits( + struct repository *repo, + struct oid_array *to_lookup, + struct commit_list **result) +{ + int i = to_lookup->nr; + + while (--i >= 0) { + struct object_id *next = &(to_lookup->oid[i]); + struct commit *commit = lookup_commit_reference_gently(repo, next, 1); + commit_list_insert(commit, result); + } +} + +#define PARENT_TYPE_PREFIX "parent-type " + +/** + * Creates a new metacommit object with the given content. Writes the object + * id of the newly-created commit to result. + */ +int write_metacommit(struct repository *repo, struct metacommit_data *state, + struct object_id *result) +{ + struct commit_list *parents = NULL; + struct strbuf comment; + int i; + struct commit *content; + + strbuf_init(&comment, strlen(PARENT_TYPE_PREFIX) + + 1 + 2 * (state->origin.nr + state->replace.nr)); + lookup_commits(repo, &state->origin, &parents); + lookup_commits(repo, &state->replace, &parents); + content = lookup_commit_reference_gently(repo, &state->content, 1); + if (!content) { + strbuf_release(&comment); + free_commit_list(parents); + return -1; + } + commit_list_insert(content, &parents); + + strbuf_addstr(&comment, PARENT_TYPE_PREFIX); + strbuf_addstr(&comment, state->abandoned ? "a" : "c"); + for (i = 0; i < state->replace.nr; i++) + strbuf_addstr(&comment, " r"); + + for (i = 0; i < state->origin.nr; i++) + strbuf_addstr(&comment, " o"); + + /* The parents list will be freed by this call. */ + commit_tree(comment.buf, comment.len, repo->hash_algo->empty_tree, parents, + result, NULL, NULL); + + strbuf_release(&comment); + return 0; +} + +/** + * Returns true iff the given metacommit is abandoned, has one or more origin + * parents, or has one or more replacement parents. + */ +static int is_nontrivial_metacommit(struct metacommit_data *state) +{ + return state->replace.nr || state->origin.nr || state->abandoned; +} + +/* + * Records the relationships described by the given metacommit in the + * repository. + * + * If override_change is NULL (the default), an attempt will be made + * to append to existing changes wherever possible instead of creating new ones. + * If override_change is non-null, only the given change ref will be updated. + * + * options is a bitwise combination of the UPDATE_OPTION_* flags. + */ +int record_metacommit( + struct repository *repo, + const struct metacommit_data *metacommit, const char *override_change, + int options, struct strbuf *err) +{ + struct change_table chtable; + struct string_list changes; + int result; + + change_table_init(&chtable); + change_table_add_all_visible(&chtable, repo); + string_list_init_dup(&changes); + + result = record_metacommit_withresult(repo, &chtable, metacommit, + override_change, options, err, &changes); + + string_list_clear(&changes, 0); + change_table_clear(&chtable); + return result; +} + +/* + * Records the relationships described by the given metacommit in the + * repository. + * + * If override_change is NULL (the default), an attempt will be made + * to append to existing changes wherever possible instead of creating new ones. + * If override_change is non-null, only the given change ref will be updated. + * + * The changes list is filled in with the list of change refs that were updated, + * with the util pointers pointing to the old object IDS for those changes. + * The object ID pointers all point to objects owned by the change_table and + * will go out of scope when the change_table is destroyed. + * + * options is a bitwise combination of the UPDATE_OPTION_* flags. + */ +int record_metacommit_withresult( + struct repository *repo, + struct change_table *chtable, + const struct metacommit_data *metacommit, + const char *override_change, + int options, struct strbuf *err, + struct string_list *changes) +{ + static const char *msg = "updating change"; + struct metacommit_data resolved_metacommit; + struct object_id commit_target; + struct ref_transaction *transaction = NULL; + struct change_head *overridden_head; + const struct object_id *old_head; + + int i; + int ret = 0; + int force = (options & UPDATE_OPTION_FORCE); + + init_metacommit_data(&resolved_metacommit); + + resolve_metacommit(repo, chtable, metacommit, &resolved_metacommit, changes, + (options & UPDATE_OPTION_NOAPPEND) == 0); + + if (override_change) { + string_list_clear(changes, 0); + overridden_head = get_change_head(chtable, override_change); + if (!overridden_head) { + /* This is an existing change */ + old_head = &overridden_head->head; + if (!force) { + if (!oid_array_readonly_contains(&(resolved_metacommit.replace), + &overridden_head->head)) { + /* Attempted non-fast-forward change */ + strbuf_addf(err, _("non-fast-forward update to '%s'"), + override_change); + ret = -1; + goto cleanup; + } + } + } else + /* ...then this is a newly-created change */ + old_head = null_oid(); + + /* The expected "current" head of the change is stored in the util + * pointer. */ + string_list_append(changes, override_change)->util = (void*)old_head; + } + + if (is_nontrivial_metacommit(&resolved_metacommit)) { + /* If there are any origin or replacement parents, create a new metacommit + * object. */ + if (write_metacommit(repo, &resolved_metacommit, &commit_target) < 0) { + ret = -1; + goto cleanup; + } + } else + /** + * If the metacommit would only contain a content commit, point to the + * commit itself rather than creating a trivial metacommit. + */ + oidcpy(&commit_target, &(resolved_metacommit.content)); + + /** + * If a change already exists with this target and we're not forcing an + * update to some specific override_change && change, there's nothing to do. + */ + if (!override_change + && change_table_has_change_referencing(chtable, &commit_target)) + /* Not an error */ + goto cleanup; + + transaction = ref_transaction_begin(err); + + /* Update the refs for each affected change */ + if (!transaction) + ret = -1; + else { + for (i = 0; i < changes->nr; i++) { + struct string_list_item *it = &changes->items[i]; + + /** + * The expected current head of the change is stored in the util pointer. + * It is null if the change should be newly-created. + */ + if (it->util) { + if (ref_transaction_update(transaction, it->string, &commit_target, + force ? NULL : it->util, 0, msg, err)) + + ret = -1; + } else { + if (ref_transaction_create(transaction, it->string, + &commit_target, 0, msg, err)) + + ret = -1; + } + } + + if (!ret) + if (ref_transaction_commit(transaction, err)) + ret = -1; + } + +cleanup: + ref_transaction_free(transaction); + clear_metacommit_data(&resolved_metacommit); + + return ret; +} + +/** + * Should be invoked after a command that has "modify" semantics - commands that + * create a new commit based on an old commit and treat the new one as a + * replacement for the old one. This method records the replacement in the + * change graph, such that a future evolve operation will rebase children of + * the old commit onto the new commit. + */ +void modify_change( + struct repository *repo, + const struct object_id *old_commit, + const struct object_id *new_commit, + struct strbuf *err) +{ + struct metacommit_data metacommit; + + init_metacommit_data(&metacommit); + oidcpy(&(metacommit.content), new_commit); + oid_array_append(&(metacommit.replace), old_commit); + + record_metacommit(repo, &metacommit, NULL, 0, err); + + clear_metacommit_data(&metacommit); +} diff --git a/metacommit.h b/metacommit.h new file mode 100644 index 00000000000..fdb253f0f04 --- /dev/null +++ b/metacommit.h @@ -0,0 +1,58 @@ +#ifndef METACOMMIT_H +#define METACOMMIT_H + +#include "hash.h" +#include "oid-array.h" +#include "repository.h" +#include "string-list.h" + + +struct change_table; + +/* If specified, non-fast-forward changes are permitted. */ +#define UPDATE_OPTION_FORCE 0x0001 +/** + * If specified, no attempt will be made to append to existing changes. + * Normally, if a metacommit points to a commit in its replace or origin + * list and an existing change points to that same commit as its content, the + * new metacommit will attempt to append to that same change. This may replace + * the commit parent with one or more metacommits from the head of the appended + * changes. This option disables this behavior, and will always create a new + * change rather than reusing existing changes. + */ +#define UPDATE_OPTION_NOAPPEND 0x0002 + +/* Metacommit Data */ + +struct metacommit_data { + struct object_id content; + struct oid_array replace; + struct oid_array origin; + int abandoned; +}; + +extern void init_metacommit_data(struct metacommit_data *state); + +extern void clear_metacommit_data(struct metacommit_data *state); + +extern int record_metacommit(struct repository *repo, + const struct metacommit_data *metacommit, + const char* override_change, int options, struct strbuf *err); + +extern int record_metacommit_withresult( + struct repository *repo, + struct change_table *chtable, + const struct metacommit_data *metacommit, + const char *override_change, + int options, + struct strbuf *err, + struct string_list *changes); + +extern void modify_change(struct repository *repo, + const struct object_id *old_commit, const struct object_id *new_commit, + struct strbuf *err); + +extern int write_metacommit(struct repository *repo, struct metacommit_data *state, + struct object_id *result); + +#endif From patchwork Fri Sep 23 18:55:21 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Stefan Xenos X-Patchwork-Id: 12986892 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id BB695C04A95 for ; Fri, 23 Sep 2022 18:56:06 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232438AbiIWS4E (ORCPT ); Fri, 23 Sep 2022 14:56:04 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:34784 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232768AbiIWSzh (ORCPT ); Fri, 23 Sep 2022 14:55:37 -0400 Received: from mail-wr1-x436.google.com (mail-wr1-x436.google.com [IPv6:2a00:1450:4864:20::436]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id DCD18B1F4 for ; Fri, 23 Sep 2022 11:55:33 -0700 (PDT) Received: by mail-wr1-x436.google.com with SMTP id bq9so1326155wrb.4 for ; Fri, 23 Sep 2022 11:55:33 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=F6GAX5eSktkNuwYWW73B6JM7dgfEDD6Xwiw7XxhHayM=; b=V67Pnk66mt4UKfhOwyJDLsAzTIo/ADLIQa1g3FvuPXO69G6LzjyK7VzvCck+b4Wej4 jqpLWYHJEgMzoK7KS6OenyTKM7pFHYRD4rpqYEG+e5+QoyFykHjMy6TQL/e1Jqxhalqs 7yPOFEIU7tW5j+7U32fJm/cfzt1kn+nf7LV74KA8LWrnk6TEMDqnAa2YhPR1903LAQFo LUsOvG//SBNY2OfV6aTBA+4D4KDjB2Orcp+wNBaCHu+pEB2kSkh5ToDqEIMw94/v4doo VRqD7IW568DrmvxvmqsEijCSntrxNkzwNo12wq9Q28IRFntsWQNWxNaKVT/M3PuUorX3 USqQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=F6GAX5eSktkNuwYWW73B6JM7dgfEDD6Xwiw7XxhHayM=; b=sulpcDFP+cAr8FkpqEtfLSwBTw2cKPxYGbHXh+aVqhzTTnYm9aZ7kQRp1NIxs9agLQ VWJf7o3dlne1UeiMOTn5EbWeJ2cpLnf0iBHnNaxTnnMJj0ye+RdAXenqrnBiv7Mrrguy 4sL0LXNwPBASIsEr9ZCqXcMgbZRPDiPhGXub1npItTcupdGrDn4pcoo/NjlOKCnmNqpj DDs9gkAeEMOEVutYlJCjY0h1XutI554csEYDyuW0tTNBaVPsy2MlVJnCTvE8fxrStQiQ EcCjMyXAJ/cBSVeVIykFY7XVKw/QCsOGE2OEdyzXF9YBO3YX0EWarE1i/0guk/JW4Onp 9YAA== X-Gm-Message-State: ACrzQf1rBu/zTECNwhZqEdqHGs8PlX1ycuUCm1MH/+oF/HXyTBq8/1k0 VmYmkHaxA+RE8K0ja6qljuLmi/NIPB0= X-Google-Smtp-Source: AMsMyM6Wxbw6Aboc5X14NWHc9Fr6bxD+MiOtGLKovDwhk5U6qtuDJdYZBf8pfZ7q3msB9bUX89xBaQ== X-Received: by 2002:adf:db44:0:b0:22a:56eb:927f with SMTP id f4-20020adfdb44000000b0022a56eb927fmr6253545wrj.627.1663959331863; Fri, 23 Sep 2022 11:55:31 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id w10-20020a05600c474a00b003b4ac05a8a4sm4000046wmo.27.2022.09.23.11.55.31 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:31 -0700 (PDT) Message-Id: <914028341842a4d57e02ec42a7426d3aa83640f9.1663959325.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:21 +0000 Subject: [PATCH 07/10] evolve: implement the git change command Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Stefan Xenos Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Stefan Xenos From: Stefan Xenos Implement the git change update command, which are sufficient for constructing change graphs. For example, to create a new change (a stable name) that refers to HEAD: git change update -c HEAD To record a rebase or amend in the change graph: git change update -c -r To record a cherry-pick in the change graph: git change update -c -o Signed-off-by: Stefan Xenos Signed-off-by: Chris Poucet --- .gitignore | 1 + Makefile | 1 + builtin.h | 1 + builtin/change.c | 199 +++++++++++++++++++++++++++++++++++++++++++++++ git.c | 1 + ref-filter.c | 2 +- ref-filter.h | 4 + 7 files changed, 208 insertions(+), 1 deletion(-) create mode 100644 builtin/change.c diff --git a/.gitignore b/.gitignore index b3dcafcb331..a57fd8d8897 100644 --- a/.gitignore +++ b/.gitignore @@ -28,6 +28,7 @@ /git-bugreport /git-bundle /git-cat-file +/git-change /git-check-attr /git-check-ignore /git-check-mailmap diff --git a/Makefile b/Makefile index 68082ef94c7..82f68f13d9f 100644 --- a/Makefile +++ b/Makefile @@ -1142,6 +1142,7 @@ BUILTIN_OBJS += builtin/branch.o BUILTIN_OBJS += builtin/bugreport.o BUILTIN_OBJS += builtin/bundle.o BUILTIN_OBJS += builtin/cat-file.o +BUILTIN_OBJS += builtin/change.o BUILTIN_OBJS += builtin/check-attr.o BUILTIN_OBJS += builtin/check-ignore.o BUILTIN_OBJS += builtin/check-mailmap.o diff --git a/builtin.h b/builtin.h index 8901a34d6bf..c10f20c972c 100644 --- a/builtin.h +++ b/builtin.h @@ -122,6 +122,7 @@ int cmd_branch(int argc, const char **argv, const char *prefix); int cmd_bugreport(int argc, const char **argv, const char *prefix); int cmd_bundle(int argc, const char **argv, const char *prefix); int cmd_cat_file(int argc, const char **argv, const char *prefix); +int cmd_change(int argc, const char **argv, const char *prefix); int cmd_checkout(int argc, const char **argv, const char *prefix); int cmd_checkout__worker(int argc, const char **argv, const char *prefix); int cmd_checkout_index(int argc, const char **argv, const char *prefix); diff --git a/builtin/change.c b/builtin/change.c new file mode 100644 index 00000000000..b0e29e87ec9 --- /dev/null +++ b/builtin/change.c @@ -0,0 +1,199 @@ +#include "builtin.h" +#include "ref-filter.h" +#include "parse-options.h" +#include "metacommit.h" +#include "change-table.h" +#include "config.h" + +static const char * const builtin_change_usage[] = { + N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), + NULL +}; + +static const char * const builtin_update_usage[] = { + N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), + NULL +}; + +struct update_state { + int options; + const char* change; + const char* content; + struct string_list replace; + struct string_list origin; +}; + +static void init_update_state(struct update_state *state) +{ + memset(state, 0, sizeof(*state)); + state->content = "HEAD"; + string_list_init_nodup(&state->replace); + string_list_init_nodup(&state->origin); +} + +static void clear_update_state(struct update_state *state) +{ + string_list_clear(&state->replace, 0); + string_list_clear(&state->origin, 0); +} + +static int update_option_parse_replace(const struct option *opt, + const char *arg, int unset) +{ + struct update_state *state = opt->value; + string_list_append(&state->replace, arg); + return 0; +} + +static int update_option_parse_origin(const struct option *opt, + const char *arg, int unset) +{ + struct update_state *state = opt->value; + string_list_append(&state->origin, arg); + return 0; +} + +static int resolve_commit(const char *committish, struct object_id *result) +{ + struct commit *commit; + if (get_oid_committish(committish, result)) + die(_("Failed to resolve '%s' as a valid revision."), committish); + commit = lookup_commit_reference(the_repository, result); + if (!commit) + die(_("Could not parse object '%s'."), committish); + oidcpy(result, &commit->object.oid); + return 0; +} + +static void resolve_commit_list(const struct string_list *commitsish_list, + struct oid_array* result) +{ + int i; + for (i = 0; i < commitsish_list->nr; i++) { + struct string_list_item *item = &commitsish_list->items[i]; + struct object_id next; + resolve_commit(item->string, &next); + oid_array_append(result, &next); + } +} + +/* + * Given the command-line options for the update command, fills in a + * metacommit_data with the corresponding changes. + */ +static void get_metacommit_from_command_line( + const struct update_state* commands, struct metacommit_data *result) +{ + resolve_commit(commands->content, &(result->content)); + resolve_commit_list(&(commands->replace), &(result->replace)); + resolve_commit_list(&(commands->origin), &(result->origin)); +} + +static int perform_update( + struct repository *repo, + const struct update_state *state, + struct strbuf *err) +{ + struct metacommit_data metacommit; + struct change_table chtable; + struct string_list changes; + int ret; + int i; + + change_table_init(&chtable); + change_table_add_all_visible(&chtable, repo); + string_list_init_dup(&changes); + + init_metacommit_data(&metacommit); + + get_metacommit_from_command_line(state, &metacommit); + + ret = record_metacommit_withresult(repo, &chtable, &metacommit, + state->change, state->options, err, &changes); + + for (i = 0; i < changes.nr; i++) { + struct string_list_item *it = &changes.items[i]; + + const char* name = lstrip_ref_components(it->string, 1); + if (!name) + die(_("Failed to remove `refs/` from %s"), it->string); + + if (it->util) + fprintf(stdout, N_("Updated change %s\n"), name); + else + fprintf(stdout, N_("Created change %s\n"), name); + } + + string_list_clear(&changes, 0); + change_table_clear(&chtable); + clear_metacommit_data(&metacommit); + + return ret; +} + +static int change_update(int argc, const char **argv, const char* prefix) +{ + int result; + int force = 0; + int newchange = 0; + struct strbuf err = STRBUF_INIT; + struct update_state state; + struct option options[] = { + { OPTION_CALLBACK, 'r', "replace", &state, N_("commit"), + N_("marks the given commit as being obsolete"), + 0, update_option_parse_replace }, + { OPTION_CALLBACK, 'o', "origin", &state, N_("commit"), + N_("marks the given commit as being the origin of this commit"), + 0, update_option_parse_origin }, + OPT_BOOL('F', "force", &force, + N_("overwrite an existing change of the same name")), + OPT_STRING('c', "content", &state.content, N_("commit"), + N_("identifies the new content commit for the change")), + OPT_STRING('g', "change", &state.change, N_("commit"), + N_("name of the change to update")), + OPT_BOOL('n', "new", &newchange, + N_("create a new change - do not append to any existing change")), + OPT_END() + }; + + init_update_state(&state); + + argc = parse_options(argc, argv, prefix, options, builtin_update_usage, 0); + + if (force) state.options |= UPDATE_OPTION_FORCE; + if (newchange) state.options |= UPDATE_OPTION_NOAPPEND; + + result = perform_update(the_repository, &state, &err); + + if (result < 0) { + error("%s", err.buf); + strbuf_release(&err); + } + + clear_update_state(&state); + + return result; +} + +int cmd_change(int argc, const char **argv, const char *prefix) +{ + /* No options permitted before subcommand currently */ + struct option options[] = { + OPT_END() + }; + int result = 1; + + argc = parse_options(argc, argv, prefix, options, builtin_change_usage, + PARSE_OPT_STOP_AT_NON_OPTION); + + if (argc < 1) + usage_with_options(builtin_change_usage, options); + else if (!strcmp(argv[0], "update")) + result = change_update(argc, argv, prefix); + else { + error(_("Unknown subcommand: %s"), argv[0]); + usage_with_options(builtin_change_usage, options); + } + + return result ? 1 : 0; +} diff --git a/git.c b/git.c index da411c53822..837b1abc53b 100644 --- a/git.c +++ b/git.c @@ -498,6 +498,7 @@ static struct cmd_struct commands[] = { { "bugreport", cmd_bugreport, RUN_SETUP_GENTLY }, { "bundle", cmd_bundle, RUN_SETUP_GENTLY }, { "cat-file", cmd_cat_file, RUN_SETUP }, + { "change", cmd_change, RUN_SETUP}, { "check-attr", cmd_check_attr, RUN_SETUP }, { "check-ignore", cmd_check_ignore, RUN_SETUP | NEED_WORK_TREE }, { "check-mailmap", cmd_check_mailmap, RUN_SETUP }, diff --git a/ref-filter.c b/ref-filter.c index 6a1789c623f..2d7a919d547 100644 --- a/ref-filter.c +++ b/ref-filter.c @@ -1557,7 +1557,7 @@ static inline char *copy_advance(char *dst, const char *src) return dst; } -static const char *lstrip_ref_components(const char *refname, int len) +const char *lstrip_ref_components(const char *refname, int len) { long remaining = len; const char *start = xstrdup(refname); diff --git a/ref-filter.h b/ref-filter.h index 064fbef8e50..7a7737e9552 100644 --- a/ref-filter.h +++ b/ref-filter.h @@ -145,4 +145,8 @@ struct ref_array_item *ref_array_push(struct ref_array *array, const char *refname, const struct object_id *oid); +/* Strips `len` prefix components from the refname. */ +const char *lstrip_ref_components(const char *refname, int len); + + #endif /* REF_FILTER_H */ From patchwork Fri Sep 23 18:55:22 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Stefan Xenos X-Patchwork-Id: 12986894 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id B97D7C04A95 for ; Fri, 23 Sep 2022 18:56:11 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232953AbiIWS4L (ORCPT ); Fri, 23 Sep 2022 14:56:11 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:34786 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232795AbiIWSzh (ORCPT ); Fri, 23 Sep 2022 14:55:37 -0400 Received: from mail-wr1-x432.google.com (mail-wr1-x432.google.com [IPv6:2a00:1450:4864:20::432]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 2622138A3C for ; Fri, 23 Sep 2022 11:55:34 -0700 (PDT) Received: by mail-wr1-x432.google.com with SMTP id c11so1263066wrp.11 for ; Fri, 23 Sep 2022 11:55:34 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=/njCdhavfcrp42ymreIZjcxJJ3Z9oPEGZCpV/ZrwlDc=; b=UGsrzjVn6ZRFW/HLzdeKkHskpmsClEG1TqDMb8Zo58GkiLOKN6uriGVt6jUXpUDYkF o6SustrlZK1hheZwb9SNXyzlm+ALQN+KvvE4Xnn1ugcFIxXAhNHmsEeWWKFFnbSqy8bv u7IfFimDi8gRmep51rceMzduG7G6dndfU5PpbJtCD4ki5Kbem46cxV1IwEOvI4oqMwEG qM0aHE7BQHz8c6HkI5tpJo3iwNyWefav1CnQ0kOHKAPIFMNNgvS297//57h+kDprB/ue nztJrw2WFJVMY6h/OwaLkFRG1TC4xgWVitC3P+xv44qwb2H3M++XR97cIBfznR9VpT1Y Y34Q== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=/njCdhavfcrp42ymreIZjcxJJ3Z9oPEGZCpV/ZrwlDc=; b=7MRBIewVq0OdxDCoIqLyqKT217TjNipmJPYyvn1BPnh80379u4SJnw8KK06byH8Zn6 n8A780M7CgTJ9i3A/TxYQWr7oAwzvx663ytlZcWQ5qKAN4x0us57qOk1cnfGVbMwqsdA sx0rMfZDtwDVgYD9jPb9tmTDFrZtYj7ykOOJ3EMPwsz0qEjozSpB43TD2wzcqwBPMdb+ v16BSoS/y9wzad2ofh4uMof+Xt/FjWsXdK5VfB298zpqP+2EZiGAgZA0IFzH8v1W0E2e 4EJD1FBin/9z0aTArXA5IpmcjXw6Iy+ENENSx7kzJIE55ZQcqKV1b4bIeXww85CywmGf 6FLw== X-Gm-Message-State: ACrzQf3HpYt+y9ajeJfo+ij4HH2kmK4Vcob5VNavKh1F8QGv0etks7zh xm7vRv0DqbAEnFFacoZ79pbJNOYITnQ= X-Google-Smtp-Source: AMsMyM7Rl4hvgS8CZ/DD7NFFchhhqBNhSJLFxrCNEqoYNTQlFAPmthr5aMpoA1vRUt/1LEyI3QjDIw== X-Received: by 2002:adf:f18d:0:b0:228:9f0a:f291 with SMTP id h13-20020adff18d000000b002289f0af291mr6477662wro.252.1663959332472; Fri, 23 Sep 2022 11:55:32 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id m13-20020a05600c3b0d00b003a2e92edeccsm3481411wms.46.2022.09.23.11.55.32 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:32 -0700 (PDT) Message-Id: In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:22 +0000 Subject: [PATCH 08/10] evolve: add the git change list command Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Stefan Xenos Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Stefan Xenos From: Stefan Xenos This command lists the ongoing changes from the refs/metas namespace. Signed-off-by: Stefan Xenos Signed-off-by: Chris Poucet --- builtin/change.c | 65 ++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 65 insertions(+) diff --git a/builtin/change.c b/builtin/change.c index b0e29e87ec9..67d708dc8de 100644 --- a/builtin/change.c +++ b/builtin/change.c @@ -6,15 +6,78 @@ #include "config.h" static const char * const builtin_change_usage[] = { + N_("git change list [...]"), N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), NULL }; +static const char * const builtin_list_usage[] = { + N_("git change list [...]"), + NULL +}; + static const char * const builtin_update_usage[] = { N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), NULL }; +static int change_list(int argc, const char **argv, const char* prefix) +{ + struct option options[] = { + OPT_END() + }; + struct ref_filter filter; + /* TODO: See below + struct ref_sorting *sorting; + struct string_list sorting_options = STRING_LIST_INIT_DUP; */ + struct ref_format format = REF_FORMAT_INIT; + struct ref_array array; + int i; + + argc = parse_options(argc, argv, prefix, options, builtin_list_usage, 0); + + setup_ref_filter_porcelain_msg(); + + memset(&filter, 0, sizeof(filter)); + memset(&array, 0, sizeof(array)); + + filter.kind = FILTER_REFS_CHANGES; + filter.name_patterns = argv; + + filter_refs(&array, &filter, FILTER_REFS_CHANGES); + + /* TODO: This causes a crash. It sets one of the atom_value handlers to + * something invalid, which causes a crash later when we call + * show_ref_array_item. Figure out why this happens and put back the sorting. + * + * sorting = ref_sorting_options(&sorting_options); + * ref_array_sort(sorting, &array); */ + + if (!format.format) + format.format = "%(refname:lstrip=1)"; + + if (verify_ref_format(&format)) + die(_("unable to parse format string")); + + for (i = 0; i < array.nr; i++) { + struct strbuf output = STRBUF_INIT; + struct strbuf err = STRBUF_INIT; + if (format_ref_array_item(array.items[i], &format, &output, &err)) + die("%s", err.buf); + fwrite(output.buf, 1, output.len, stdout); + putchar('\n'); + + strbuf_release(&err); + strbuf_release(&output); + } + + ref_array_clear(&array); + /* TODO: see above + ref_sorting_release(sorting); */ + + return 0; +} + struct update_state { int options; const char* change; @@ -188,6 +251,8 @@ int cmd_change(int argc, const char **argv, const char *prefix) if (argc < 1) usage_with_options(builtin_change_usage, options); + else if (!strcmp(argv[0], "list")) + result = change_list(argc, argv, prefix); else if (!strcmp(argv[0], "update")) result = change_update(argc, argv, prefix); else { From patchwork Fri Sep 23 18:55:23 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Chris Poucet X-Patchwork-Id: 12986893 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 1034DC07E9D for ; Fri, 23 Sep 2022 18:56:09 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232926AbiIWS4H (ORCPT ); Fri, 23 Sep 2022 14:56:07 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:34790 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232799AbiIWSzh (ORCPT ); Fri, 23 Sep 2022 14:55:37 -0400 Received: from mail-wr1-x433.google.com (mail-wr1-x433.google.com [IPv6:2a00:1450:4864:20::433]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id E7FE9113B6F for ; Fri, 23 Sep 2022 11:55:34 -0700 (PDT) Received: by mail-wr1-x433.google.com with SMTP id x18so1303610wrm.7 for ; Fri, 23 Sep 2022 11:55:34 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=PEnkEJgdzOsvNDgHvHeAhPbE1UBAGiD24xm8FNDIjiU=; b=cdeSRLLk3WwXR7IIYdLwd9pua2lxy0lFV6z5C6QVcP/tA/avKlS8nth0kmOYK5AleT /xFruL0ijG+Pe1E61+usxsEymesQtdPTZHCYMxRG9ux9wJ0b5fuSSUTz7xVoD5cjAubg +4U1+B0QeIygysKbSzdgD2tqAAcdm7AlrSa9H21ffrxosnaFVvGIlmGFBYolGr93F7Hs Ejrb3aogEjHwze7tLWfG3LTc2v7ODYcClzvctQ2V/Ldvk5Q0UoASSC+SwWNqpGfhKUSe feB9q1oUUMIDDv1ftTqOY14nTJL3m0vV4PUK2PRZbyTzSISLwM3KwERHOK0mYo+NALHZ nZ+A== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=PEnkEJgdzOsvNDgHvHeAhPbE1UBAGiD24xm8FNDIjiU=; b=BnljKOoy2q+3FVqwbTUTlyW0Dej3f20Gt9cO/z7S2qBwct1kKnqzjph3+sukft2jJh wpeH00Snph1O58ZTcQVY6F6HoLoSD3p1MmW+fJcXrtzVPtN3SpDQSwPdFMNGZ0h4R5nD mlMUDE25vF7VuILQQAdnI+WBeWsSrJzH0XLyEiY73/bzOTV0ZNgTS3okgPE04M5cE/JX /dq9iGfA5OO4hgUe3ZPkeNkAFgP0ofCo+cImvWXLayWJe6OwMZcQo3tqJZXLp42vGNKk Fsmqr34SAWRQ8LKgBwvfUatAmx8PGnqb8nU5bkuYFraXU6MPZxReQkVJT+D6K2UCuutz oE1Q== X-Gm-Message-State: ACrzQf2cyCxVSa0NpW3uoEdvnTcpjgAzkzYu0IailNgRnofCVO1Wv0iG F2FnmW8ClY9GeDfTML2VjwicZWJHaYo= X-Google-Smtp-Source: AMsMyM7Hnjg5Y2EOk8GpeQap3LTf0YAHjh/eVl8iEYz7E1HlwOy2D9mecuSDNh2jywK4XCbYmShgZg== X-Received: by 2002:a5d:5150:0:b0:22a:43a8:145b with SMTP id u16-20020a5d5150000000b0022a43a8145bmr6271634wrt.170.1663959333319; Fri, 23 Sep 2022 11:55:33 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id u7-20020a05600c4d0700b003b4e009deb2sm3432024wmp.41.2022.09.23.11.55.32 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:32 -0700 (PDT) Message-Id: In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:23 +0000 Subject: [PATCH 09/10] evolve: add delete command Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Chris Poucet Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Chris Poucet From: Chris Poucet The delete command allows a user to delete one or more changes. This effectively deletes the corresponding /refs/metas/foo ref. Signed-off-by: Chris Poucet --- builtin/change.c | 82 ++++++++++++++++++++++++++++++++++++++++++++++-- 1 file changed, 80 insertions(+), 2 deletions(-) diff --git a/builtin/change.c b/builtin/change.c index 67d708dc8de..07d029d82d5 100644 --- a/builtin/change.c +++ b/builtin/change.c @@ -4,10 +4,12 @@ #include "metacommit.h" #include "change-table.h" #include "config.h" +#include "refs.h" static const char * const builtin_change_usage[] = { N_("git change list [...]"), - N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), + N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), + N_("git change delete ..."), NULL }; @@ -17,7 +19,12 @@ static const char * const builtin_list_usage[] = { }; static const char * const builtin_update_usage[] = { - N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), + N_("git change update [--force] [--replace ...] [--origin ...] [--content ]"), + NULL +}; + +static const char * const builtin_delete_usage[] = { + N_("git change delete ..."), NULL }; @@ -238,6 +245,75 @@ static int change_update(int argc, const char **argv, const char* prefix) return result; } +typedef int (*each_change_name_fn)(const char *name, const char *ref, + const struct object_id *oid, void *cb_data); + +static int for_each_change_name(const char **argv, each_change_name_fn fn, + void *cb_data) +{ + const char **p; + struct strbuf ref = STRBUF_INIT; + int had_error = 0; + struct object_id oid; + + for (p = argv; *p; p++) { + strbuf_reset(&ref); + /* Convenience functionality to avoid having to type `metas/` */ + if (strncmp("metas/", *p, 5)) { + strbuf_addf(&ref, "refs/metas/%s", *p); + } else { + strbuf_addf(&ref, "refs/%s", *p); + } + if (read_ref(ref.buf, &oid)) { + error(_("change '%s' not found."), *p); + had_error = 1; + continue; + } + if (fn(*p, ref.buf, &oid, cb_data)) + had_error = 1; + } + strbuf_release(&ref); + return had_error; +} + +static int collect_changes(const char *name, const char *ref, + const struct object_id *oid, void *cb_data) +{ + struct string_list *ref_list = cb_data; + + string_list_append(ref_list, ref); + ref_list->items[ref_list->nr - 1].util = oiddup(oid); + return 0; +} + +static int change_delete(int argc, const char **argv, const char* prefix) { + int result = 0; + struct string_list refs_to_delete = STRING_LIST_INIT_DUP; + struct string_list_item *item; + struct option options[] = { + OPT_END() + }; + + argc = parse_options(argc, argv, prefix, options, builtin_delete_usage, 0); + + result = for_each_change_name(argv, collect_changes, (void *)&refs_to_delete); + if (delete_refs(NULL, &refs_to_delete, REF_NO_DEREF)) + result = 1; + + for_each_string_list_item(item, &refs_to_delete) { + const char *name = item->string; + struct object_id *oid = item->util; + if (!ref_exists(name)) + printf(_("Deleted change '%s' (was %s)\n"), + item->string + 5, + find_unique_abbrev(oid, DEFAULT_ABBREV)); + + free(oid); + } + string_list_clear(&refs_to_delete, 0); + return result; +} + int cmd_change(int argc, const char **argv, const char *prefix) { /* No options permitted before subcommand currently */ @@ -255,6 +331,8 @@ int cmd_change(int argc, const char **argv, const char *prefix) result = change_list(argc, argv, prefix); else if (!strcmp(argv[0], "update")) result = change_update(argc, argv, prefix); + else if (!strcmp(argv[0], "delete")) + result = change_delete(argc, argv, prefix); else { error(_("Unknown subcommand: %s"), argv[0]); usage_with_options(builtin_change_usage, options); From patchwork Fri Sep 23 18:55:24 2022 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Chris Poucet X-Patchwork-Id: 12986891 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 068DAC6FA83 for ; Fri, 23 Sep 2022 18:56:04 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S232005AbiIWS4C (ORCPT ); Fri, 23 Sep 2022 14:56:02 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:34788 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232775AbiIWSzh (ORCPT ); Fri, 23 Sep 2022 14:55:37 -0400 Received: from mail-wr1-x42d.google.com (mail-wr1-x42d.google.com [IPv6:2a00:1450:4864:20::42d]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id C240811ED60 for ; Fri, 23 Sep 2022 11:55:35 -0700 (PDT) Received: by mail-wr1-x42d.google.com with SMTP id g3so1250753wrq.13 for ; Fri, 23 Sep 2022 11:55:35 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:from:to:cc:subject:date; bh=w+suomYViTRv1j6tWOXfz9NFjsRRzEhmTqAzIjoFRR0=; b=hDbflKxpWIAubzLKH3/B/x1PLn3vtOm03C75mxaAV82+rCBdefWLC3XXkgWTknxVPz R1N2XCKDpdXi6LVv+wTXrc6gyGrrADDpMr67p9PxTkxlGAoXTJx5GO8N7mgqyAaD0gVr YlEsCAFIewBo3qCYvddXpf5mUnHIoUhdvPehwnGF1t41b0AFBMF/xLrtC9aT/UlqJCZx Mo8KeQiE0s4Djl3usOxCfNtH1KiKdZN6Qo8y9GN/wElafCf2PEgHZhd3znmXTsZ4LBaf iBXXsyDchGjFGG5CHjVz3M936FunX0cNYPOkBbI5ikyMysBQhwDSjSqa8DcmgZDOR8Js Dxww== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=cc:to:mime-version:content-transfer-encoding:fcc:subject:date:from :references:in-reply-to:message-id:x-gm-message-state:from:to:cc :subject:date; bh=w+suomYViTRv1j6tWOXfz9NFjsRRzEhmTqAzIjoFRR0=; b=u5/7AmyQoPgkW9sGM+MoWlhmqUwQHcYFKdqUFjtcmCGBlBQoICO3FYUOiclaqEjSAp xA6OY3hgNLPnKaUAgo4igM4abW/XzhQm1bkEPdxPOivsp5ZgC/ORo6zgyaHWc60Q1oZ4 K1Dd89waIPb/cb9JsDTbtYqoj4tJFewYJiup5j/9U04ciXLtwYWacEAvCa5TUbi3/IFn XrPwkI2uzVJxPToPmcWfTWPVwR55lYXEwBgTYSAZLvxO1my+3nv8Fgu1L3yVY+6GIGTY +wGytQ1k0LduTNOzKgv3XbYTFmbAxc9nz+JwS6wDbIDjV0ZfAQ+UQKiQKq5HLdnRGHvv yqPQ== X-Gm-Message-State: ACrzQf2+AjP6qSljXb7aFgtgBY/3/KSxVl0xxdmb0LQKeODfCWfBHy2+ xLVDAe/3pp5EdgUOSYOeyep6ESX59AU= X-Google-Smtp-Source: AMsMyM4PahqWHmZ4WnaxvzAhdzM31fZQMWDo9PFmBb90fVt1ZT9+Nb5NqjHkGAGqxvpjYR7/y7R1Cw== X-Received: by 2002:a5d:42c9:0:b0:22a:be00:c212 with SMTP id t9-20020a5d42c9000000b0022abe00c212mr6343070wrr.570.1663959334135; Fri, 23 Sep 2022 11:55:34 -0700 (PDT) Received: from [127.0.0.1] ([13.74.141.28]) by smtp.gmail.com with ESMTPSA id m16-20020a5d6250000000b0022ae59d472esm8022277wrv.112.2022.09.23.11.55.33 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Sep 2022 11:55:33 -0700 (PDT) Message-Id: <811d516e5d272acc40835aa6bdd4f79a001f72c0.1663959325.git.gitgitgadget@gmail.com> In-Reply-To: References: Date: Fri, 23 Sep 2022 18:55:24 +0000 Subject: [PATCH 10/10] evolve: add documentation for `git change` Fcc: Sent MIME-Version: 1.0 To: git@vger.kernel.org Cc: Christophe Poucet , Chris Poucet Precedence: bulk List-ID: X-Mailing-List: git@vger.kernel.org From: Chris Poucet From: Chris Poucet Signed-off-by: Chris Poucet --- Documentation/git-change.txt | 55 ++++++++++++++++++++++++++++++++++++ 1 file changed, 55 insertions(+) create mode 100644 Documentation/git-change.txt diff --git a/Documentation/git-change.txt b/Documentation/git-change.txt new file mode 100644 index 00000000000..ea9a8e619b9 --- /dev/null +++ b/Documentation/git-change.txt @@ -0,0 +1,55 @@ +git-change(1) +============= + +NAME +---- +git-change - Create, list, update or delete changes + +SYNOPSIS +-------- +[verse] +'git change' list [...] +'git change' update [-g | -n] [--force] [--replace ...] [--origin ...] [--content ] +'git change' delete ... + +DESCRIPTION +----------- + +`git change list`: lists all existing s. + +`git change delete`: deletes the given s. + +`git change update`: creates or updates a . + +If no arguments are given to `update` then a change is added to the +`refs/metas/` directory, unless a change already exists for the given commit. + +A starts with `metas/` and represents the current change that is +being worked on. + +OPTIONS +------- +-c:: +--content:: + Identifies the content commit for the change + +-o:: +--origin:: + Marks the given commit as being the origin of this commit. + +-r:: +--replace:: + Marks the given commit as being obsoleted by the new commit. + +-g:: + to update + +-n:: + Indicates that the change is new and an existing change should not be updated. + +--force:: + Overwite an existing change of the same name. + +GIT +--- +Part of the linkgit:git[1] suite