* [PATCH] worktree: detect from secondary worktree if main worktree is bare @ 2024-11-15 6:52 Olga Pilipenco via GitGitGadget 2025-01-16 21:35 ` [PATCH v2] " Olga Pilipenco via GitGitGadget 0 siblings, 1 reply; 19+ messages in thread From: Olga Pilipenco via GitGitGadget @ 2024-11-15 6:52 UTC (permalink / raw) To: git; +Cc: Olga Pilipenco, Olga Pilipenco From: Olga Pilipenco <olga.pilipenco@shopify.com> Setup: 1. Have a bare repo with core.bare = true in config.worktree 2. Create a new worktree Behavior: From the secondary worktree the main worktree appears as non-bare. Expected: From the secondary worktree the main worktree should appear as bare. Why current behavior is not good? If the main worktree is detected as not bare it doesn't allow checking out the branch of the main worktree. There are possibly other problems associated with that behavior. Why is it happening? While we're inside the secondary worktree we don't initialize the main worktree's repository with its configuration. How is it fixed? Load actual configs of the main worktree. Also, skip the config loading step if we're already inside the current worktree because in that case we rely on is_bare_repository() to return the correct result. Other solutions considered: Alternatively, instead of incorrectly always using `the_repository` as the main worktree's repository, we can detect and load the actual repository of the main worktree and then use that repository's `is_bare` value extracted from correct configs. However, this approach is a bit riskier and could also affect performance. Since we had the assignment `worktree->repo = the_repository` for a long time already, I decided it's safe to keep it as it is for now; it can be still fixed separately from this change. Real life use case: 1. Have a bare repo 2. Create a worktree from the bare repo 3. In the secondary worktree enable sparse-checkout - this enables extensions.worktreeConfig and keeps core.bare=true setting in config.worktree of the bare worktree 4. The secondary worktree or any other non-bare worktree created won't be able to use branch main (not even once), but it should be able to. Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> --- worktree: detect from secondary worktree if main worktree is bare Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1829%2Folga-mcbfe%2Ffix-bare-repo-detection-with-worktree-config-v1 Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1829/olga-mcbfe/fix-bare-repo-detection-with-worktree-config-v1 Pull-Request: https://github.com/gitgitgadget/git/pull/1829 t/t3200-branch.sh | 14 ++++++++++++++ worktree.c | 38 +++++++++++++++++++++++++++++--------- 2 files changed, 43 insertions(+), 9 deletions(-) diff --git a/t/t3200-branch.sh b/t/t3200-branch.sh index ccfa6a720d0..819228a3344 100755 --- a/t/t3200-branch.sh +++ b/t/t3200-branch.sh @@ -411,6 +411,20 @@ test_expect_success 'bare main worktree has HEAD at branch deleted by secondary git -C secondary branch -D main ' +test_expect_success 'secondary worktree can switch to main if common dir is bare worktree' ' + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && + git init -b main non_bare_repo && + test_commit -C non_bare_repo x && + + git clone --bare non_bare_repo bare_repo && + git -C bare_repo config extensions.worktreeConfig true && + git -C bare_repo config unset core.bare && + git -C bare_repo config --worktree core.bare true && + + git -C bare_repo worktree add ../secondary_worktree && + git -C secondary_worktree checkout main +' + test_expect_success 'git branch --list -v with --abbrev' ' test_when_finished "git branch -D t" && git branch t && diff --git a/worktree.c b/worktree.c index 77ff484d3ec..1bce505bdd7 100644 --- a/worktree.c +++ b/worktree.c @@ -64,6 +64,28 @@ static int is_current_worktree(struct worktree *wt) return is_current; } +static int is_bare_git_dir(const char *git_dir) +{ + int bare = 0; + struct config_set cs = { { 0 } }; + char *config_file; + char *worktree_config_file; + + config_file = xstrfmt("%s/config", git_dir); + worktree_config_file = xstrfmt("%s/config.worktree", git_dir); + + git_configset_init(&cs); + git_configset_add_file(&cs, config_file); + git_configset_add_file(&cs, worktree_config_file); + + git_configset_get_bool(&cs, "core.bare", &bare); + + git_configset_clear(&cs); + free(config_file); + free(worktree_config_file); + return bare; +} + /** * get the main worktree */ @@ -76,18 +98,16 @@ static struct worktree *get_main_worktree(int skip_reading_head) strbuf_strip_suffix(&worktree_path, "/.git"); CALLOC_ARRAY(worktree, 1); + /* + * NEEDSWORK: the_repository is not always main worktree's repository + */ worktree->repo = the_repository; worktree->path = strbuf_detach(&worktree_path, NULL); - /* - * NEEDSWORK: If this function is called from a secondary worktree and - * config.worktree is present, is_bare_repository_cfg will reflect the - * contents of config.worktree, not the contents of the main worktree. - * This means that worktree->is_bare may be set to 0 even if the main - * worktree is configured to be bare. - */ - worktree->is_bare = (is_bare_repository_cfg == 1) || - is_bare_repository(); worktree->is_current = is_current_worktree(worktree); + worktree->is_bare = (is_bare_repository_cfg == 1) || + is_bare_repository() || + (!worktree->is_current && is_bare_git_dir(repo_get_common_dir(the_repository))); + if (!skip_reading_head) add_head_info(worktree); return worktree; base-commit: 25b0f41288718625b18495de23cc066394c09a92 -- gitgitgadget ^ permalink raw reply related [flat|nested] 19+ messages in thread
* [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2024-11-15 6:52 [PATCH] worktree: detect from secondary worktree if main worktree is bare Olga Pilipenco via GitGitGadget @ 2025-01-16 21:35 ` Olga Pilipenco via GitGitGadget 2025-01-19 22:30 ` Eric Sunshine 2025-01-31 18:08 ` [PATCH v3] " Olga Pilipenco via GitGitGadget 0 siblings, 2 replies; 19+ messages in thread From: Olga Pilipenco via GitGitGadget @ 2025-01-16 21:35 UTC (permalink / raw) To: git Cc: Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe, Junio C Hamano, Olga Pilipenco, Olga Pilipenco From: Olga Pilipenco <olga.pilipenco@shopify.com> Setup: 1. Have a bare repo with core.bare = true in config.worktree 2. Create a new worktree Behavior: From the secondary worktree the main worktree appears as non-bare. Expected: From the secondary worktree the main worktree should appear as bare. Why current behavior is not good? If the main worktree is detected as not bare it doesn't allow checking out the branch of the main worktree. There are possibly other problems associated with that behavior. Why is it happening? While we're inside the secondary worktree we don't initialize the main worktree's repository with its configuration. How is it fixed? Load actual configs of the main worktree. Also, skip the config loading step if we're already inside the current worktree because in that case we rely on is_bare_repository() to return the correct result. Other solutions considered: Alternatively, instead of incorrectly always using `the_repository` as the main worktree's repository, we can detect and load the actual repository of the main worktree and then use that repository's `is_bare` value extracted from correct configs. However, this approach is a bit riskier and could also affect performance. Since we had the assignment `worktree->repo = the_repository` for a long time already, I decided it's safe to keep it as it is for now; it can be still fixed separately from this change. Real life use case: 1. Have a bare repo 2. Create a worktree from the bare repo 3. In the secondary worktree enable sparse-checkout - this enables extensions.worktreeConfig and keeps core.bare=true setting in config.worktree of the bare worktree 4. The secondary worktree or any other non-bare worktree created won't be able to use branch main (not even once), but it should be able to. Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> --- worktree: detect from secondary worktree if main worktree is bare Changes since v1: * no code changes * rebased with maint * CC added Existing broken functionality forces our project to use hacks on bare repo that we'd like to avoid. I would really appreciate reviews of this patch to move closer towards fixing the issue. This is my first contribution to git/git, I apologize if I got lost in the instructions, but I tried my best to follow the rules. Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1829%2Folga-mcbfe%2Ffix-bare-repo-detection-with-worktree-config-v2 Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1829/olga-mcbfe/fix-bare-repo-detection-with-worktree-config-v2 Pull-Request: https://github.com/gitgitgadget/git/pull/1829 Range-diff vs v1: 1: 9e7170f07fc = 1: 17f4b24d1da worktree: detect from secondary worktree if main worktree is bare t/t3200-branch.sh | 14 ++++++++++++++ worktree.c | 38 +++++++++++++++++++++++++++++--------- 2 files changed, 43 insertions(+), 9 deletions(-) diff --git a/t/t3200-branch.sh b/t/t3200-branch.sh index a3a21c54cf6..7ca50c9a78d 100755 --- a/t/t3200-branch.sh +++ b/t/t3200-branch.sh @@ -410,6 +410,20 @@ test_expect_success 'bare main worktree has HEAD at branch deleted by secondary git -C secondary branch -D main ' +test_expect_success 'secondary worktree can switch to main if common dir is bare worktree' ' + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && + git init -b main non_bare_repo && + test_commit -C non_bare_repo x && + + git clone --bare non_bare_repo bare_repo && + git -C bare_repo config extensions.worktreeConfig true && + git -C bare_repo config unset core.bare && + git -C bare_repo config --worktree core.bare true && + + git -C bare_repo worktree add ../secondary_worktree && + git -C secondary_worktree checkout main +' + test_expect_success 'git branch --list -v with --abbrev' ' test_when_finished "git branch -D t" && git branch t && diff --git a/worktree.c b/worktree.c index 248bbb39d43..a7726c67747 100644 --- a/worktree.c +++ b/worktree.c @@ -65,6 +65,28 @@ static int is_current_worktree(struct worktree *wt) return is_current; } +static int is_bare_git_dir(const char *git_dir) +{ + int bare = 0; + struct config_set cs = { { 0 } }; + char *config_file; + char *worktree_config_file; + + config_file = xstrfmt("%s/config", git_dir); + worktree_config_file = xstrfmt("%s/config.worktree", git_dir); + + git_configset_init(&cs); + git_configset_add_file(&cs, config_file); + git_configset_add_file(&cs, worktree_config_file); + + git_configset_get_bool(&cs, "core.bare", &bare); + + git_configset_clear(&cs); + free(config_file); + free(worktree_config_file); + return bare; +} + /** * get the main worktree */ @@ -77,18 +99,16 @@ static struct worktree *get_main_worktree(int skip_reading_head) strbuf_strip_suffix(&worktree_path, "/.git"); CALLOC_ARRAY(worktree, 1); + /* + * NEEDSWORK: the_repository is not always main worktree's repository + */ worktree->repo = the_repository; worktree->path = strbuf_detach(&worktree_path, NULL); - /* - * NEEDSWORK: If this function is called from a secondary worktree and - * config.worktree is present, is_bare_repository_cfg will reflect the - * contents of config.worktree, not the contents of the main worktree. - * This means that worktree->is_bare may be set to 0 even if the main - * worktree is configured to be bare. - */ - worktree->is_bare = (is_bare_repository_cfg == 1) || - is_bare_repository(); worktree->is_current = is_current_worktree(worktree); + worktree->is_bare = (is_bare_repository_cfg == 1) || + is_bare_repository() || + (!worktree->is_current && is_bare_git_dir(repo_get_common_dir(the_repository))); + if (!skip_reading_head) add_head_info(worktree); return worktree; base-commit: f93ff170b93a1782659637824b25923245ac9dd1 -- gitgitgadget ^ permalink raw reply related [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-16 21:35 ` [PATCH v2] " Olga Pilipenco via GitGitGadget @ 2025-01-19 22:30 ` Eric Sunshine 2025-01-28 21:44 ` Olga Pilipenco 2025-01-31 18:08 ` [PATCH v3] " Olga Pilipenco via GitGitGadget 1 sibling, 1 reply; 19+ messages in thread From: Eric Sunshine @ 2025-01-19 22:30 UTC (permalink / raw) To: Olga Pilipenco via GitGitGadget Cc: git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano, Olga Pilipenco On Thu, Jan 16, 2025 at 4:35 PM Olga Pilipenco via GitGitGadget <gitgitgadget@gmail.com> wrote: > Setup: > 1. Have a bare repo with core.bare = true in config.worktree > 2. Create a new worktree > > Behavior: > From the secondary worktree the main worktree appears as non-bare. > > Expected: > From the secondary worktree the main worktree should appear as bare. > > Why current behavior is not good? > If the main worktree is detected as not bare it doesn't allow > checking out the branch of the main worktree. There are possibly > other problems associated with that behavior. > > Why is it happening? > While we're inside the secondary worktree we don't initialize the main > worktree's repository with its configuration. Okay, this is clearly a very real problem and explains this comment added by f3534c98e4 (worktree: update is_bare heuristics, 2019-04-19): NEEDSWORK: If this function is called from a secondary worktree and config.worktree is present, is_bare_repository_cfg will reflect the contents of config.worktree, not the contents of the main worktree. This means that worktree->is_bare may be set to 0 even if the main worktree is configured to be bare. (Aside: I recall reading this comment when Jonathan added it but wasn't able to dig into it at the time to really understand it, and never got back around to it. Now, after studying your patch, I understand what it was about. > How is it fixed? > Load actual configs of the main worktree. Also, skip the config loading > step if we're already inside the current worktree because in that case we > rely on is_bare_repository() to return the correct result. I found that I had to dig around a bit to fully understand the problem expressed by this commit message. Perhaps adding a bit more detail would help? Here's my attempt at rewriting the above (also in a way which is more idiomatic to this project): When extensions.worktreeConfig is true and the main worktree is bare -- that is, its config.worktree file contains core.bare=true -- commands run from secondary worktrees incorrectly see the main worktree as not bare. As such, those commands incorrectly think that the repository's default branch (typically "main" or "master") is checked out in the bare repository even though it's not. This makes it impossible, for instance, to checkout or delete the default branch from a secondary worktree, among other shortcomings. This problem occurs because, when extensions.worktreeConfig is true, commands run in secondary worktrees only consult $commondir/config and $commondir/worktrees/<id>/config.worktree, thus they never see the main worktree's core.bare=true setting in $commondir/config.worktree. Fix this problem by consulting the main worktree's config.worktree file when checking whether it is bare. (This extra work is performed only when running from a secondary worktree.) > Other solutions considered: > Alternatively, instead of incorrectly always using > `the_repository` as the main worktree's repository, we can detect > and load the actual repository of the main worktree and then use > that repository's `is_bare` value extracted from correct configs. > However, this approach is a bit riskier and could also affect > performance. Since we had the assignment `worktree->repo = > the_repository` for a long time already, I decided it's safe to > keep it as it is for now; it can be still fixed separately from > this change. I found this paragraph somewhat confusing because it seems to conflate a repository (i.e. the shared object database) with the `struct repository` type, and the configuration which happens to get loaded and stored (as one of *many* members) of the repository structure. I had to read it several times to understand that this was talking about instantiating a separate `struct repository` initialized from the main worktree configuration. I agree that doing so would likely be overkill and could impact performance negatively. I understand that you added this paragraph because SubmittingPatches suggests to do so, but I think it can probably be omitted in this case unless it can be rewritten to be more clear (but even then I doubt it is necessary to keep it). > Real life use case: > 1. Have a bare repo > 2. Create a worktree from the bare repo > 3. In the secondary worktree enable sparse-checkout - this enables > extensions.worktreeConfig and keeps core.bare=true setting in > config.worktree of the bare worktree > 4. The secondary worktree or any other non-bare worktree created > won't be able to use branch main (not even once), but it should be > able to. This is mostly repeating what was said earlier, thus probably isn't adding any value to the commit message. I'd probably drop it. > Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> > --- > Changes since v1: > > * no code changes > * rebased with maint > * CC added Sorry. I've had your v1 sitting in my ever-increasingly-large backlog of patches to look at, but have been extra busy the last many months and never managed to get to it. > Existing broken functionality forces our project to use hacks on bare > repo that we'd like to avoid. I would really appreciate reviews of this > patch to move closer towards fixing the issue. This is my first > contribution to git/git, I apologize if I got lost in the instructions, > but I tried my best to follow the rules. Your submission is fine. Unfortunately, the project has a lack of reviewers but no lack of submitters, so sometimes patches get overlooked or simply buried. > diff --git a/t/t3200-branch.sh b/t/t3200-branch.sh > @@ -410,6 +410,20 @@ test_expect_success 'bare main worktree has HEAD at branch deleted by secondary > +test_expect_success 'secondary worktree can switch to main if common dir is bare worktree' ' The use of "common dir" is a bit confusing. Also, this patch is fixing the more general problem that secondary worktrees think that the bare main worktree has a branch checked out. So, perhaps a better title would be: secondary worktrees recognize core.bare=true in main config.worktree or something? > + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && > + git init -b main non_bare_repo && > + test_commit -C non_bare_repo x && > + > + git clone --bare non_bare_repo bare_repo && > + git -C bare_repo config extensions.worktreeConfig true && > + git -C bare_repo config unset core.bare && > + git -C bare_repo config --worktree core.bare true && > + > + git -C bare_repo worktree add ../secondary_worktree && > + git -C secondary_worktree checkout main > +' Very straightforward and exactly what I expected to see once I understood the problem. > diff --git a/worktree.c b/worktree.c > @@ -65,6 +65,28 @@ static int is_current_worktree(struct worktree *wt) > +static int is_bare_git_dir(const char *git_dir) Nit: I wonder if a name such as is_main_worktree_bare() would clue readers in a bit more? > +{ > + int bare = 0; > + struct config_set cs = { { 0 } }; This is not your fault since this construct is used elsewhere in this file (from which I presume you copied it), but project consensus is that using the notation `{{0}}` to work around a complaint from the Apple compiler (and only the Apple compiler) should be avoided, and that `{0}` is preferred. So, if you reroll, changing this to `{0}` may make other reviewers happy (or you can leave it as is to be consistent with existing precedence in this file; I don't feel strongly about it). > + char *config_file; > + char *worktree_config_file; > + > + config_file = xstrfmt("%s/config", git_dir); > + worktree_config_file = xstrfmt("%s/config.worktree", git_dir); > + > + git_configset_init(&cs); > + git_configset_add_file(&cs, config_file); > + git_configset_add_file(&cs, worktree_config_file); Genuine question: I haven't thought too deeply about it, but do we gain anything by loading $commondir/config here -- which is shared by the main worktree and all secondary worktrees -- considering that it was already loaded and consulted by the earlier is-bare check before this function was even called? > + git_configset_get_bool(&cs, "core.bare", &bare); > + > + git_configset_clear(&cs); > + free(config_file); > + free(worktree_config_file); > + return bare; Everything gets cleaned up correctly. Good. > @@ -77,18 +99,16 @@ static struct worktree *get_main_worktree(int skip_reading_head) > + /* > + * NEEDSWORK: the_repository is not always main worktree's repository > + */ > worktree->repo = the_repository; > worktree->path = strbuf_detach(&worktree_path, NULL); I found this new NEEDSWORK comment rather confusing the first several times I read the patch. It wasn't until I finally realized that the reference to `the_repository` here is the same reference to `the_repository` in the commit message -- which confused me, as well -- that I understood what this was trying to say. The actual problem, of course, is that the _configuration_ stored in `the_repository` is the secondary worktree's configuration, not the main worktree's configuration. Considering that this patch addresses that problem, I'd probably just drop this new comment altogether (unless, perhaps, you rewrite it to talk about the _configuration_ stored in `the_repository`). > - /* > - * NEEDSWORK: If this function is called from a secondary worktree and > - * config.worktree is present, is_bare_repository_cfg will reflect the > - * contents of config.worktree, not the contents of the main worktree. > - * This means that worktree->is_bare may be set to 0 even if the main > - * worktree is configured to be bare. > - */ > - worktree->is_bare = (is_bare_repository_cfg == 1) || > - is_bare_repository(); > worktree->is_current = is_current_worktree(worktree); > + worktree->is_bare = (is_bare_repository_cfg == 1) || > + is_bare_repository() || > + (!worktree->is_current && is_bare_git_dir(repo_get_common_dir(the_repository))); This is performing the expensive check only if the earlier checks left the question unanswered. Good. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-19 22:30 ` Eric Sunshine @ 2025-01-28 21:44 ` Olga Pilipenco 2025-01-29 13:41 ` Eric Sunshine 0 siblings, 1 reply; 19+ messages in thread From: Olga Pilipenco @ 2025-01-28 21:44 UTC (permalink / raw) To: Eric Sunshine Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano Thank you for the review, I totally understand the delay in the review process and appreciate your time spent on this. > On Jan 19, 2025, at 3:30 PM, Eric Sunshine <sunshine@sunshineco.com> wrote: > > On Thu, Jan 16, 2025 at 4:35 PM Olga Pilipenco via GitGitGadget > <gitgitgadget@gmail.com> wrote: >> Setup: >> 1. Have a bare repo with core.bare = true in config.worktree >> 2. Create a new worktree >> >> Behavior: >> From the secondary worktree the main worktree appears as non-bare. >> >> Expected: >> From the secondary worktree the main worktree should appear as bare. >> >> Why current behavior is not good? >> If the main worktree is detected as not bare it doesn't allow >> checking out the branch of the main worktree. There are possibly >> other problems associated with that behavior. >> >> Why is it happening? >> While we're inside the secondary worktree we don't initialize the main >> worktree's repository with its configuration. > > Okay, this is clearly a very real problem and explains this comment > added by f3534c98e4 (worktree: update is_bare heuristics, 2019-04-19): > > NEEDSWORK: If this function is called from a secondary worktree and > config.worktree is present, is_bare_repository_cfg will reflect the > contents of config.worktree, not the contents of the main worktree. > This means that worktree->is_bare may be set to 0 even if the main > worktree is configured to be bare. > > (Aside: I recall reading this comment when Jonathan added it but > wasn't able to dig into it at the time to really understand it, and > never got back around to it. Now, after studying your patch, I > understand what it was about. > >> How is it fixed? >> Load actual configs of the main worktree. Also, skip the config loading >> step if we're already inside the current worktree because in that case we >> rely on is_bare_repository() to return the correct result. > > I found that I had to dig around a bit to fully understand the problem > expressed by this commit message. Perhaps adding a bit more detail > would help? Here's my attempt at rewriting the above (also in a way > which is more idiomatic to this project): > > When extensions.worktreeConfig is true and the main worktree is > bare -- that is, its config.worktree file contains core.bare=true > -- commands run from secondary worktrees incorrectly see the main > worktree as not bare. As such, those commands incorrectly think > that the repository's default branch (typically "main" or > "master") is checked out in the bare repository even though it's > not. This makes it impossible, for instance, to checkout or delete > the default branch from a secondary worktree, among other > shortcomings. > > This problem occurs because, when extensions.worktreeConfig is > true, commands run in secondary worktrees only consult > $commondir/config and $commondir/worktrees/<id>/config.worktree, > thus they never see the main worktree's core.bare=true setting in > $commondir/config.worktree. > > Fix this problem by consulting the main worktree's config.worktree > file when checking whether it is bare. (This extra work is > performed only when running from a secondary worktree.) Wow, your explanation is so much better than mine.Thank you for “translating" it for the world :) I’m still trying to get used to the terminology used in this codebase. I’ll steal your description for sure (if you don’t mind). > >> Other solutions considered: >> Alternatively, instead of incorrectly always using >> `the_repository` as the main worktree's repository, we can detect >> and load the actual repository of the main worktree and then use >> that repository's `is_bare` value extracted from correct configs. >> However, this approach is a bit riskier and could also affect >> performance. Since we had the assignment `worktree->repo = >> the_repository` for a long time already, I decided it's safe to >> keep it as it is for now; it can be still fixed separately from >> this change. > > I found this paragraph somewhat confusing because it seems to conflate > a repository (i.e. the shared object database) with the `struct > repository` type, and the configuration which happens to get loaded > and stored (as one of *many* members) of the repository structure. I > had to read it several times to understand that this was talking about > instantiating a separate `struct repository` initialized from the main > worktree configuration. I agree that doing so would likely be overkill > and could impact performance negatively. I understand that you added > this paragraph because SubmittingPatches suggests to do so, but I > think it can probably be omitted in this case unless it can be > rewritten to be more clear (but even then I doubt it is necessary to > keep it). Trust me, it took me a while to wrap my head around `struct repository` as well. I agree if the explanation is too confusing and doesn’t bring any value, it can be omitted. > >> Real life use case: >> 1. Have a bare repo >> 2. Create a worktree from the bare repo >> 3. In the secondary worktree enable sparse-checkout - this enables >> extensions.worktreeConfig and keeps core.bare=true setting in >> config.worktree of the bare worktree >> 4. The secondary worktree or any other non-bare worktree created >> won't be able to use branch main (not even once), but it should be >> able to. > > This is mostly repeating what was said earlier, thus probably isn't > adding any value to the commit message. I'd probably drop it. I agree, your improved description captures this scenario perfectly. > >> Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> >> --- >> Changes since v1: >> >> * no code changes >> * rebased with maint >> * CC added > > Sorry. I've had your v1 sitting in my ever-increasingly-large backlog > of patches to look at, but have been extra busy the last many months > and never managed to get to it. Totally understand. Thanks again for getting to it eventually. > >> Existing broken functionality forces our project to use hacks on bare >> repo that we'd like to avoid. I would really appreciate reviews of this >> patch to move closer towards fixing the issue. This is my first >> contribution to git/git, I apologize if I got lost in the instructions, >> but I tried my best to follow the rules. > > Your submission is fine. Unfortunately, the project has a lack of > reviewers but no lack of submitters, so sometimes patches get > overlooked or simply buried. > >> diff --git a/t/t3200-branch.sh b/t/t3200-branch.sh >> @@ -410,6 +410,20 @@ test_expect_success 'bare main worktree has HEAD at branch deleted by secondary >> +test_expect_success 'secondary worktree can switch to main if common dir is bare worktree' ' > > The use of "common dir" is a bit confusing. Also, this patch is fixing > the more general problem that secondary worktrees think that the bare > main worktree has a branch checked out. So, perhaps a better title > would be: > > secondary worktrees recognize core.bare=true in main config.worktree > > or something? Sounds good, will update. > >> + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && >> + git init -b main non_bare_repo && >> + test_commit -C non_bare_repo x && >> + >> + git clone --bare non_bare_repo bare_repo && >> + git -C bare_repo config extensions.worktreeConfig true && >> + git -C bare_repo config unset core.bare && >> + git -C bare_repo config --worktree core.bare true && >> + >> + git -C bare_repo worktree add ../secondary_worktree && >> + git -C secondary_worktree checkout main >> +' > > Very straightforward and exactly what I expected to see once I > understood the problem. > >> diff --git a/worktree.c b/worktree.c >> @@ -65,6 +65,28 @@ static int is_current_worktree(struct worktree *wt) >> +static int is_bare_git_dir(const char *git_dir) > > Nit: I wonder if a name such as is_main_worktree_bare() would clue > readers in a bit more? I was about to explain how I wanted this function to be more generic and handle all sorts of bare and non-bare cases - whether it’s the main worktree or not. However, after seeing your comments and after revisiting the code, I realized that generalization doesn’t really provide much benefit here. It is much clearer if we're explicit that the bare check in this case is only performed on the main worktree. I’ll update it in the next version. > >> +{ >> + int bare = 0; >> + struct config_set cs = { { 0 } }; > > This is not your fault since this construct is used elsewhere in this > file (from which I presume you copied it), but project consensus is > that using the notation `{{0}}` to work around a complaint from the > Apple compiler (and only the Apple compiler) should be avoided, and > that `{0}` is preferred. So, if you reroll, changing this to `{0}` may > make other reviewers happy (or you can leave it as is to be consistent > with existing precedence in this file; I don't feel strongly about > it). I’ll fix it, sounds like a good reason. > >> + char *config_file; >> + char *worktree_config_file; >> + >> + config_file = xstrfmt("%s/config", git_dir); >> + worktree_config_file = xstrfmt("%s/config.worktree", git_dir); >> + >> + git_configset_init(&cs); >> + git_configset_add_file(&cs, config_file); >> + git_configset_add_file(&cs, worktree_config_file); > > Genuine question: I haven't thought too deeply about it, but do we > gain anything by loading $commondir/config here -- which is shared by > the main worktree and all secondary worktrees -- considering that it > was already loaded and consulted by the earlier is-bare check before > this function was even called? This function determines if a worktree is bare or not. I want this logic to work even when it’s called from a different context and not rely on other is-bare checks (that are a bit confusing tbh). > >> + git_configset_get_bool(&cs, "core.bare", &bare); >> + >> + git_configset_clear(&cs); >> + free(config_file); >> + free(worktree_config_file); >> + return bare; > > Everything gets cleaned up correctly. Good. > >> @@ -77,18 +99,16 @@ static struct worktree *get_main_worktree(int skip_reading_head) >> + /* >> + * NEEDSWORK: the_repository is not always main worktree's repository >> + */ >> worktree->repo = the_repository; >> worktree->path = strbuf_detach(&worktree_path, NULL); > > I found this new NEEDSWORK comment rather confusing the first several > times I read the patch. It wasn't until I finally realized that the > reference to `the_repository` here is the same reference to > `the_repository` in the commit message -- which confused me, as well > -- that I understood what this was trying to say. The actual problem, > of course, is that the _configuration_ stored in `the_repository` is > the secondary worktree's configuration, not the main worktree's > configuration. Considering that this patch addresses that problem, I'd > probably just drop this new comment altogether (unless, perhaps, you > rewrite it to talk about the _configuration_ stored in > `the_repository`). This `the_repository` structure is soooo confusing, took me a while to figure out what it is! I would feel guilty not mentioning that under some circumstances `the_repository` assigned here could be not actual configuration of the worktree object. I don’t know if that will ever matter or not, but I find this assignment kinda “stinky” and want everyone to know about it. I don’t want to change this assignment in this patch because it didn’t bring any harm so far. I’ll try again to rephrase this comment, just to give a heads up in case someone experiences “weird” behaviour in this area (same way the previous NEEDSWORK comment gave me ideas why my workflow didn’t work and inspired me to try to fix it). > >> - /* >> - * NEEDSWORK: If this function is called from a secondary worktree and >> - * config.worktree is present, is_bare_repository_cfg will reflect the >> - * contents of config.worktree, not the contents of the main worktree. >> - * This means that worktree->is_bare may be set to 0 even if the main >> - * worktree is configured to be bare. >> - */ >> - worktree->is_bare = (is_bare_repository_cfg == 1) || >> - is_bare_repository(); >> worktree->is_current = is_current_worktree(worktree); >> + worktree->is_bare = (is_bare_repository_cfg == 1) || >> + is_bare_repository() || >> + (!worktree->is_current && is_bare_git_dir(repo_get_common_dir(the_repository))); > > This is performing the expensive check only if the earlier checks left > the question unanswered. Good. Thanks for the review. I’ll incorporate the changes in my next version and hopefully it will be good to go :tada: I hope I responded to all the comments, it’s a bit nerve-wrecking to contribute for the first time (so many rules and instructions!) :) ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-28 21:44 ` Olga Pilipenco @ 2025-01-29 13:41 ` Eric Sunshine 2025-01-29 17:08 ` Junio C Hamano [not found] ` <F15C12AB-2238-4553-AFA5-18277B18CE5A@shopify.com> 0 siblings, 2 replies; 19+ messages in thread From: Eric Sunshine @ 2025-01-29 13:41 UTC (permalink / raw) To: Olga Pilipenco Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano On Tue, Jan 28, 2025 at 4:45 PM Olga Pilipenco <olga.pilipenco@shopify.com> wrote: > > On Jan 19, 2025, at 3:30 PM, Eric Sunshine <sunshine@sunshineco.com> wrote: > > On Thu, Jan 16, 2025 at 4:35 PM Olga Pilipenco via GitGitGadget > > <gitgitgadget@gmail.com> wrote: > > I found that I had to dig around a bit to fully understand the problem > > expressed by this commit message. Perhaps adding a bit more detail > > would help? Here's my attempt at rewriting the above (also in a way > > which is more idiomatic to this project): > > > > When extensions.worktreeConfig is true and the main worktree is > > bare -- that is, its config.worktree file contains core.bare=true > > -- commands run from secondary worktrees incorrectly see the main > > worktree as not bare. As such, those commands incorrectly think > > that the repository's default branch (typically "main" or > > "master") is checked out in the bare repository even though it's > > not. This makes it impossible, for instance, to checkout or delete > > the default branch from a secondary worktree, among other > > shortcomings. > > > > This problem occurs because, when extensions.worktreeConfig is > > true, commands run in secondary worktrees only consult > > $commondir/config and $commondir/worktrees/<id>/config.worktree, > > thus they never see the main worktree's core.bare=true setting in > > $commondir/config.worktree. > > > > Fix this problem by consulting the main worktree's config.worktree > > file when checking whether it is bare. (This extra work is > > performed only when running from a secondary worktree.) > > Wow, your explanation is so much better than mine.Thank you for > “translating" it for the world :) I’m still trying to get used to > the terminology used in this codebase. I’ll steal your description > for sure (if you don’t mind). You are more than welcome to use the proposed commit message rewrite. (If you want to acknowledge assistance rendered, a Helped-by: trailer, preceding your Signed-off-by:, is the way to do so. Or not. It's up to you.) > >> diff --git a/worktree.c b/worktree.c > >> @@ -65,6 +65,28 @@ static int is_current_worktree(struct worktree *wt) > >> +static int is_bare_git_dir(const char *git_dir) > > > > Nit: I wonder if a name such as is_main_worktree_bare() would clue > > readers in a bit more? > > I was about to explain how I wanted this function to be more generic > and handle all sorts of bare and non-bare cases - whether it’s the > main worktree or not. However, after seeing your comments and after > revisiting the code, I realized that generalization doesn’t really > provide much benefit here. It is much clearer if we're explicit that > the bare check in this case is only performed on the main > worktree. I’ll update it in the next version. I see. When reviewing, I was wondering why the git-dir was being passed into the function. Your explanation above answers that question. On that note, in addition to renaming the function as suggested, for clarity, I would probably go a bit further and pass in a `struct repository *` rather than passing in the git-dir itself, just to make it clear that the function is checking main-worktree bareness of the repository in question, as opposed to merely checking bareness of any arbitrary directory. (At least, I would find the intention more clear at-a-glance with that additional change applied.) > >> + config_file = xstrfmt("%s/config", git_dir); > >> + worktree_config_file = xstrfmt("%s/config.worktree", git_dir); > >> + > >> + git_configset_init(&cs); > >> + git_configset_add_file(&cs, config_file); > >> + git_configset_add_file(&cs, worktree_config_file); > > > > Genuine question: I haven't thought too deeply about it, but do we > > gain anything by loading $commondir/config here -- which is shared by > > the main worktree and all secondary worktrees -- considering that it > > was already loaded and consulted by the earlier is-bare check before > > this function was even called? > > This function determines if a worktree is bare or not. I want this > logic to work even when it’s called from a different context and not > rely on other is-bare checks (that are a bit confusing tbh). Agreed about the is-bare checks -- and indeed the entire Git startup sequence -- being difficult to digest, however... One reason I asked the question was due to concern that future readers of this code may very well wonder (as I did) why $commondir/config is being loaded when doing so is (apparently) unnecessary in this particular context. The question is especially pertinent given that this is a private helper function with a single caller. A second reason was that, over the years, a good deal of effort has been put into optimizing Git's startup to avoid doing unnecessary work, and this appears to be unnecessary since $commondir/config would already have been consulted by earlier checks before this function gets called (assuming I'm correctly understanding the code-flow). Anyhow, we can probably punt on the question for the moment and leave the code as you wrote it if you feel strongly about it or if you think it is clearer this way for future readers. > >> + /* > >> + * NEEDSWORK: the_repository is not always main worktree's repository > >> + */ > >> worktree->repo = the_repository; > >> worktree->path = strbuf_detach(&worktree_path, NULL); > > > > I found this new NEEDSWORK comment rather confusing the first several > > times I read the patch. It wasn't until I finally realized that the > > reference to `the_repository` here is the same reference to > > `the_repository` in the commit message -- which confused me, as well > > -- that I understood what this was trying to say. The actual problem, > > of course, is that the _configuration_ stored in `the_repository` is > > the secondary worktree's configuration, not the main worktree's > > configuration. Considering that this patch addresses that problem, I'd > > probably just drop this new comment altogether (unless, perhaps, you > > rewrite it to talk about the _configuration_ stored in > > `the_repository`). > > This `the_repository` structure is soooo confusing, took me a while > to figure out what it is! I would feel guilty not mentioning that > under some circumstances `the_repository` assigned here could be not > actual configuration of the worktree object. I don’t know if that > will ever matter or not, but I find this assignment kinda “stinky” > and want everyone to know about it. I don’t want to change this > assignment in this patch because it didn’t bring any harm so far. > I’ll try again to rephrase this comment, just to give a heads up in > case someone experiences “weird” behaviour in this area (same way > the previous NEEDSWORK comment gave me ideas why my workflow didn’t > work and inspired me to try to fix it). Likely, the confusion is an outcome of the natural evolution (mutation) software undergoes. The development of linked worktrees and the concept of a `repository` structure did not necessarily occur concurrently. I suppose one could develop one of two views (if not more) of the `repository`: (1) an in-memory representation of the ".git" directory or bare-repository "object database", including all worktrees hanging off it, or (2) a single worktree's view/representation of the repository, meaning paths, configuration, "index" specific to that worktree. In the present state of the code, the second view is the more accurate one, so the existing `worktree->repo = the_repository` assignment does make sense without any further commentary. My main concern with the NEEDSWORK comment is that it implies that there is a problem with the assignment, even though there isn't. While it may be true that the entire `repository` idea needs to be rethought or clarified or expanded, that's a global issue permeating the entire code-base, not specific to this one spot, which is why it feels inappropriate to have a NEEDSWORK comment here. So, I'm not, in general, opposed to a comment explaining the the `worktree->repo` assignment for future readers if you think that would be valuable, but I am concerned about giving it a "NEEDSWORK" prefix, which feels misleading for this particular piece of code. > Thanks for the review. I’ll incorporate the changes in my next > version and hopefully it will be good to go :tada: I hope I > responded to all the comments, it’s a bit nerve-wrecking to > contribute for the first time (so many rules and instructions!) :) Understood, and it didn't help the nerve situation when your v1 was apparently ignored. Rest assured, though, that your submission was nicely done and fixes a real problem which ought to be addressed. (In fact, I'm surprised it took this long for someone to tackle it. So, thanks.) ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-29 13:41 ` Eric Sunshine @ 2025-01-29 17:08 ` Junio C Hamano [not found] ` <F15C12AB-2238-4553-AFA5-18277B18CE5A@shopify.com> 1 sibling, 0 replies; 19+ messages in thread From: Junio C Hamano @ 2025-01-29 17:08 UTC (permalink / raw) To: Eric Sunshine Cc: Olga Pilipenco, Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe Eric Sunshine <sunshine@sunshineco.com> writes: >> Wow, your explanation is so much better than mine.Thank you for >> “translating" it for the world :) I’m still trying to get used to >> the terminology used in this codebase. I’ll steal your description >> for sure (if you don’t mind). > > You are more than welcome to use the proposed commit message rewrite. > > (If you want to acknowledge assistance rendered, a Helped-by: trailer, > preceding your Signed-off-by:, is the way to do so. Or not. It's up to > you.) > ... > Agreed about the is-bare checks -- and indeed the entire Git startup > sequence -- being difficult to digest, however... > ... Thanks for an easy-to-read review (and thanks OP for working on it, of course). ^ permalink raw reply [flat|nested] 19+ messages in thread
[parent not found: <F15C12AB-2238-4553-AFA5-18277B18CE5A@shopify.com>]
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare [not found] ` <F15C12AB-2238-4553-AFA5-18277B18CE5A@shopify.com> @ 2025-01-30 14:32 ` Eric Sunshine 2025-01-30 14:44 ` Eric Sunshine 0 siblings, 1 reply; 19+ messages in thread From: Eric Sunshine @ 2025-01-30 14:32 UTC (permalink / raw) To: Olga Pilipenco Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano On Thu, Jan 30, 2025 at 2:09 AM Olga Pilipenco <olga.pilipenco@shopify.com> wrot > On Jan 29, 2025, at 6:41 AM, Eric Sunshine <sunshine@sunshineco.com> wrote: > > I see. When reviewing, I was wondering why the git-dir was being > > passed into the function. Your explanation above answers that > > question. On that note, in addition to renaming the function as > > suggested, for clarity, I would probably go a bit further and pass in > > a `struct repository *` rather than passing in the git-dir itself, > > just to make it clear that the function is checking main-worktree > > bareness of the repository in question, as opposed to merely checking > > bareness of any arbitrary directory. (At least, I would find the > > intention more clear at-a-glance with that additional change applied.) > > Indeed, no need to pass git-dir anymore. There is actually no need > to pass `the_repository` because it’s global. I like how this > simplified things and made code clearer. The reason I suggested passing in a `struct repository *` is that the project is slowly moving away from the `the_repository` global, so making this new function accept a `struct repository *` as its sole argument means less work later on. > > One reason I asked the question was due to concern that future readers > > of this code may very well wonder (as I did) why $commondir/config is > > being loaded when doing so is (apparently) unnecessary in this > > particular context. The question is especially pertinent given that > > this is a private helper function with a single caller. A second > > reason was that, over the years, a good deal of effort has been put > > into optimizing Git's startup to avoid doing unnecessary work, and > > this appears to be unnecessary since $commondir/config would already > > have been consulted by earlier checks before this function gets called > > (assuming I'm correctly understanding the code-flow). > > I trust your judgement and knowledge of the code and really like the > reasons presented. I’ll change this function to only check for > worktree config. However I’d like to give it a good name where it’s > clear we only check worktree config. It’s a bit challenging to make > it short-ish and not to include multiple “worktree” words in the > name. Before I submit a new release, maybe I have time to quickly > align on the name. What do you think about this one: > > is_main_worktree_bare_in_worktree_config > > (It will check if bare=true in the main worktree’s worktree.config) > > Naming is harder than the code itself :) It's a historic "accident" that when worktree support was designed, the idea of linking worktrees to a bare repository was not considered. Support for using worktrees with a bare repository was added later. However, by that time, the term "main worktree" was already well established, with the very unfortunate result that even when there is no actual "main worktree" but only a bare repository with "linked worktrees" hanging off it, the repository itself is usually referred to as the "bare main worktree", which is an obvious misnomer; the repository is just a repository (i.e. the object database and other meta-information) and there is no actual main worktree. Given the very real potential for confusion when employing the "bare main worktree" misnomer, I suspect that we won't be able to come up with a good name which easily conveys the function's purpose. Since this is an internal helper (hence, there is slightly less pressure to come up with a perfect name) rather than public API, this might be one of those cases in which it makes more sense to choose a concise name and then explain the function's purpose with a short comment block. Perhaps something like this would be most helpful to future readers of the code: /* * When in a secondary worktree, and when extensions.worktreeConfig * is true, only $commondir/config and $commondir/worktrees/<id>/ * config.worktree are consulted, hence any core.bare=true setting in * $commondir/config.worktree gets overlooked. Thus, check it manually * to determine if the repository is bare. */ static int is_repo_bare(struct repository *r) {...} > Thank you for this thorough explanation. I’ll drop the comment > completely. Less code to read. (To be honest I’m not a big fan of > comments and definitely don’t want to introduce confusing comments > :) Understood. Self-explanatory code is preferred. That said, a comment such as the one proposed above can really help readers not intimately familiar with this otherwise nonobvious behavior, thus may be justified. > Fun fact: my email app kept changing “worktree” to “corktree”. Maybe > another git feature? Your mailer obviously suffers from Gitophobia or is perhaps a nemophilist. (By the way, when replying, please use the normal ">", "> >", "> > >" markers to signify quoted portions of earlier messages in the thread rather than using only indentation. The reason I make this request is that when I replied to your message, my mailer stripped away all indentation from your message, leaving all earlier quoted portions flush with the left margin, which made it very difficult to figure out which quotes came from which authors from which earlier messages, and I ended up having to reinsert the "> >" markers manually to restore structure to my reply.) ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-30 14:32 ` Eric Sunshine @ 2025-01-30 14:44 ` Eric Sunshine 2025-01-31 7:05 ` Olga Pilipenco 0 siblings, 1 reply; 19+ messages in thread From: Eric Sunshine @ 2025-01-30 14:44 UTC (permalink / raw) To: Olga Pilipenco Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano On Thu, Jan 30, 2025 at 9:32 AM Eric Sunshine <sunshine@sunshineco.com> wrote: > (By the way, when replying, please use the normal ">", "> >", "> > >" > markers to signify quoted portions of earlier messages in the thread > rather than using only indentation. The reason I make this request is > that when I replied to your message, my mailer stripped away all > indentation from your message, leaving all earlier quoted portions > flush with the left margin, which made it very difficult to figure out > which quotes came from which authors from which earlier messages, and > I ended up having to reinsert the "> >" markers manually to restore > structure to my reply.) Examining more closely, I see that your message was in fact multipart MIME with an HTML portion which was using <blockquote> and whatnot, which my (plain text) mailer stripped out, hence lost all formatting. If you can configure your mailer to send plain text and use the normal ">" markers, that would be generally helpful on this list. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-30 14:44 ` Eric Sunshine @ 2025-01-31 7:05 ` Olga Pilipenco 2025-01-31 13:28 ` Eric Sunshine 0 siblings, 1 reply; 19+ messages in thread From: Olga Pilipenco @ 2025-01-31 7:05 UTC (permalink / raw) To: Eric Sunshine Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano On Thu, Jan 30, 2025 at 7:44 AM Eric Sunshine <sunshine@sunshineco.com> wrote: > > On Thu, Jan 30, 2025 at 9:32 AM Eric Sunshine <sunshine@sunshineco.com> wrote: > > (By the way, when replying, please use the normal ">", "> >", "> > >" > > markers to signify quoted portions of earlier messages in the thread > > rather than using only indentation. The reason I make this request is > > that when I replied to your message, my mailer stripped away all > > indentation from your message, leaving all earlier quoted portions > > flush with the left margin, which made it very difficult to figure out > > which quotes came from which authors from which earlier messages, and > > I ended up having to reinsert the "> >" markers manually to restore > > structure to my reply.) > > Examining more closely, I see that your message was in fact multipart > MIME with an HTML portion which was using <blockquote> and whatnot, > which my (plain text) mailer stripped out, hence lost all formatting. > If you can configure your mailer to send plain text and use the normal > ">" markers, that would be generally helpful on this list. Ufff, my bad, sorry about that (should be plain text now). I'm having trouble responding to my previous email and keeping that quoting format, so I'll not waste time on trying to fix that (please forgive me, everyone). Just want to let you know that I agree with your response, I'll adjust my patch. Enough talking, let's get the code out! ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v2] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 7:05 ` Olga Pilipenco @ 2025-01-31 13:28 ` Eric Sunshine 0 siblings, 0 replies; 19+ messages in thread From: Eric Sunshine @ 2025-01-31 13:28 UTC (permalink / raw) To: Olga Pilipenco Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Johannes Schindelin, René Scharfe, Junio C Hamano On Fri, Jan 31, 2025 at 2:05 AM Olga Pilipenco <olga.pilipenco@shopify.com> wrote: > On Thu, Jan 30, 2025 at 7:44 AM Eric Sunshine <sunshine@sunshineco.com> wrote: > > Examining more closely, I see that your message was in fact multipart > > MIME with an HTML portion which was using <blockquote> and whatnot, > > which my (plain text) mailer stripped out, hence lost all formatting. > > If you can configure your mailer to send plain text and use the normal > > ">" markers, that would be generally helpful on this list. > > Ufff, my bad, sorry about that (should be plain text now). Yup, looks good. > I'm having trouble responding to my previous email and keeping that > quoting format, > so I'll not waste time on trying to fix that (please forgive me, everyone). > Just want to let you know that I agree with your response, I'll adjust my patch. > Enough talking, let's get the code out! Thanks. ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-01-16 21:35 ` [PATCH v2] " Olga Pilipenco via GitGitGadget 2025-01-19 22:30 ` Eric Sunshine @ 2025-01-31 18:08 ` Olga Pilipenco via GitGitGadget 2025-01-31 19:19 ` Junio C Hamano 2025-02-05 6:30 ` [PATCH v4] " Olga Pilipenco via GitGitGadget 1 sibling, 2 replies; 19+ messages in thread From: Olga Pilipenco via GitGitGadget @ 2025-01-31 18:08 UTC (permalink / raw) To: git Cc: Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe, Junio C Hamano, Olga Pilipenco, Olga Pilipenco From: Olga Pilipenco <olga.pilipenco@shopify.com> When extensions.worktreeConfig is true and the main worktree is bare -- that is, its config.worktree file contains core.bare=true -- commands run from secondary worktrees incorrectly see the main worktree as not bare. As such, those commands incorrectly think that the repository's default branch (typically "main" or "master") is checked out in the bare repository even though it's not. This makes it impossible, for instance, to checkout or delete the default branch from a secondary worktree, among other shortcomings. This problem occurs because, when extensions.worktreeConfig is true, commands run in secondary worktrees only consult $commondir/config and $commondir/worktrees/<id>/config.worktree, thus they never see the main worktree's core.bare=true setting in $commondir/config.worktree. Fix this problem by consulting the main worktree's config.worktree file when checking whether it is bare. (This extra work is performed only when running from a secondary worktree.) Helped-by: Eric Sunshine <sunshine@sunshineco.com> Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> --- worktree: detect from secondary worktree if main worktree is bare Changes since v2, all results of the amazing review: * updated description & comments; * private function is_bare_git_dir is replaced with is_main_worktree_bare. The new implementation only checks if main worktree's worktree.config contains information if main worktree is bare or not. It's assumed that other configs of main worktree are already checked for bareness prior this call. * notation { { 0 } } is replaced with {0} that is preferred by the project. Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1829%2Folga-mcbfe%2Ffix-bare-repo-detection-with-worktree-config-v3 Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1829/olga-mcbfe/fix-bare-repo-detection-with-worktree-config-v3 Pull-Request: https://github.com/gitgitgadget/git/pull/1829 Range-diff vs v2: 1: 17f4b24d1da ! 1: f9207746b38 worktree: detect from secondary worktree if main worktree is bare @@ Metadata ## Commit message ## worktree: detect from secondary worktree if main worktree is bare - Setup: - 1. Have a bare repo with core.bare = true in config.worktree - 2. Create a new worktree + When extensions.worktreeConfig is true and the main worktree is + bare -- that is, its config.worktree file contains core.bare=true + -- commands run from secondary worktrees incorrectly see the main + worktree as not bare. As such, those commands incorrectly think + that the repository's default branch (typically "main" or + "master") is checked out in the bare repository even though it's + not. This makes it impossible, for instance, to checkout or delete + the default branch from a secondary worktree, among other + shortcomings. - Behavior: - From the secondary worktree the main worktree appears as non-bare. + This problem occurs because, when extensions.worktreeConfig is + true, commands run in secondary worktrees only consult + $commondir/config and $commondir/worktrees/<id>/config.worktree, + thus they never see the main worktree's core.bare=true setting in + $commondir/config.worktree. - Expected: - From the secondary worktree the main worktree should appear as bare. - - Why current behavior is not good? - If the main worktree is detected as not bare it doesn't allow - checking out the branch of the main worktree. There are possibly - other problems associated with that behavior. - - Why is it happening? - While we're inside the secondary worktree we don't initialize the main - worktree's repository with its configuration. - - How is it fixed? - Load actual configs of the main worktree. Also, skip the config loading - step if we're already inside the current worktree because in that case we - rely on is_bare_repository() to return the correct result. - - Other solutions considered: - Alternatively, instead of incorrectly always using - `the_repository` as the main worktree's repository, we can detect - and load the actual repository of the main worktree and then use - that repository's `is_bare` value extracted from correct configs. - However, this approach is a bit riskier and could also affect - performance. Since we had the assignment `worktree->repo = - the_repository` for a long time already, I decided it's safe to - keep it as it is for now; it can be still fixed separately from - this change. - - Real life use case: - 1. Have a bare repo - 2. Create a worktree from the bare repo - 3. In the secondary worktree enable sparse-checkout - this enables - extensions.worktreeConfig and keeps core.bare=true setting in - config.worktree of the bare worktree - 4. The secondary worktree or any other non-bare worktree created - won't be able to use branch main (not even once), but it should be - able to. + Fix this problem by consulting the main worktree's config.worktree + file when checking whether it is bare. (This extra work is + performed only when running from a secondary worktree.) + Helped-by: Eric Sunshine <sunshine@sunshineco.com> Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> ## t/t3200-branch.sh ## @@ t/t3200-branch.sh: test_expect_success 'bare main worktree has HEAD at branch de git -C secondary branch -D main ' -+test_expect_success 'secondary worktree can switch to main if common dir is bare worktree' ' ++test_expect_success 'secondary worktrees recognize core.bare=true in main config.worktree' ' + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && + git init -b main non_bare_repo && + test_commit -C non_bare_repo x && @@ worktree.c: static int is_current_worktree(struct worktree *wt) return is_current; } -+static int is_bare_git_dir(const char *git_dir) ++/* ++* When in a secondary worktree, and when extensions.worktreeConfig ++* is true, only $commondir/config and $commondir/worktrees/<id>/ ++* config.worktree are consulted, hence any core.bare=true setting in ++* $commondir/config.worktree gets overlooked. Thus, check it manually ++* to determine if the repository is bare. ++*/ ++static int is_main_worktree_bare(struct repository *repo) +{ + int bare = 0; -+ struct config_set cs = { { 0 } }; -+ char *config_file; -+ char *worktree_config_file; -+ -+ config_file = xstrfmt("%s/config", git_dir); -+ worktree_config_file = xstrfmt("%s/config.worktree", git_dir); ++ struct config_set cs = {0}; ++ char *worktree_config = xstrfmt("%s/config.worktree", repo_get_common_dir(repo)); + + git_configset_init(&cs); -+ git_configset_add_file(&cs, config_file); -+ git_configset_add_file(&cs, worktree_config_file); -+ ++ git_configset_add_file(&cs, worktree_config); + git_configset_get_bool(&cs, "core.bare", &bare); + + git_configset_clear(&cs); -+ free(config_file); -+ free(worktree_config_file); ++ free(worktree_config); + return bare; +} + @@ worktree.c: static int is_current_worktree(struct worktree *wt) * get the main worktree */ @@ worktree.c: static struct worktree *get_main_worktree(int skip_reading_head) - strbuf_strip_suffix(&worktree_path, "/.git"); - CALLOC_ARRAY(worktree, 1); -+ /* -+ * NEEDSWORK: the_repository is not always main worktree's repository -+ */ worktree->repo = the_repository; worktree->path = strbuf_detach(&worktree_path, NULL); - /* @@ worktree.c: static struct worktree *get_main_worktree(int skip_reading_head) worktree->is_current = is_current_worktree(worktree); + worktree->is_bare = (is_bare_repository_cfg == 1) || + is_bare_repository() || -+ (!worktree->is_current && is_bare_git_dir(repo_get_common_dir(the_repository))); ++ (!worktree->is_current && is_main_worktree_bare(the_repository)); + if (!skip_reading_head) add_head_info(worktree); t/t3200-branch.sh | 14 ++++++++++++++ worktree.c | 35 ++++++++++++++++++++++++++--------- 2 files changed, 40 insertions(+), 9 deletions(-) diff --git a/t/t3200-branch.sh b/t/t3200-branch.sh index a3a21c54cf6..f3e720dc10d 100755 --- a/t/t3200-branch.sh +++ b/t/t3200-branch.sh @@ -410,6 +410,20 @@ test_expect_success 'bare main worktree has HEAD at branch deleted by secondary git -C secondary branch -D main ' +test_expect_success 'secondary worktrees recognize core.bare=true in main config.worktree' ' + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && + git init -b main non_bare_repo && + test_commit -C non_bare_repo x && + + git clone --bare non_bare_repo bare_repo && + git -C bare_repo config extensions.worktreeConfig true && + git -C bare_repo config unset core.bare && + git -C bare_repo config --worktree core.bare true && + + git -C bare_repo worktree add ../secondary_worktree && + git -C secondary_worktree checkout main +' + test_expect_success 'git branch --list -v with --abbrev' ' test_when_finished "git branch -D t" && git branch t && diff --git a/worktree.c b/worktree.c index 248bbb39d43..6df4ccf97f7 100644 --- a/worktree.c +++ b/worktree.c @@ -65,6 +65,28 @@ static int is_current_worktree(struct worktree *wt) return is_current; } +/* +* When in a secondary worktree, and when extensions.worktreeConfig +* is true, only $commondir/config and $commondir/worktrees/<id>/ +* config.worktree are consulted, hence any core.bare=true setting in +* $commondir/config.worktree gets overlooked. Thus, check it manually +* to determine if the repository is bare. +*/ +static int is_main_worktree_bare(struct repository *repo) +{ + int bare = 0; + struct config_set cs = {0}; + char *worktree_config = xstrfmt("%s/config.worktree", repo_get_common_dir(repo)); + + git_configset_init(&cs); + git_configset_add_file(&cs, worktree_config); + git_configset_get_bool(&cs, "core.bare", &bare); + + git_configset_clear(&cs); + free(worktree_config); + return bare; +} + /** * get the main worktree */ @@ -79,16 +101,11 @@ static struct worktree *get_main_worktree(int skip_reading_head) CALLOC_ARRAY(worktree, 1); worktree->repo = the_repository; worktree->path = strbuf_detach(&worktree_path, NULL); - /* - * NEEDSWORK: If this function is called from a secondary worktree and - * config.worktree is present, is_bare_repository_cfg will reflect the - * contents of config.worktree, not the contents of the main worktree. - * This means that worktree->is_bare may be set to 0 even if the main - * worktree is configured to be bare. - */ - worktree->is_bare = (is_bare_repository_cfg == 1) || - is_bare_repository(); worktree->is_current = is_current_worktree(worktree); + worktree->is_bare = (is_bare_repository_cfg == 1) || + is_bare_repository() || + (!worktree->is_current && is_main_worktree_bare(the_repository)); + if (!skip_reading_head) add_head_info(worktree); return worktree; base-commit: f93ff170b93a1782659637824b25923245ac9dd1 -- gitgitgadget ^ permalink raw reply related [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 18:08 ` [PATCH v3] " Olga Pilipenco via GitGitGadget @ 2025-01-31 19:19 ` Junio C Hamano 2025-01-31 19:26 ` Junio C Hamano 2025-02-05 6:30 ` [PATCH v4] " Olga Pilipenco via GitGitGadget 1 sibling, 1 reply; 19+ messages in thread From: Junio C Hamano @ 2025-01-31 19:19 UTC (permalink / raw) To: Olga Pilipenco via GitGitGadget Cc: git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe, Olga Pilipenco "Olga Pilipenco via GitGitGadget" <gitgitgadget@gmail.com> writes: > +/* > +* When in a secondary worktree, and when extensions.worktreeConfig > +* is true, only $commondir/config and $commondir/worktrees/<id>/ > +* config.worktree are consulted, hence any core.bare=true setting in > +* $commondir/config.worktree gets overlooked. Thus, check it manually > +* to determine if the repository is bare. > +*/ > +static int is_main_worktree_bare(struct repository *repo) > +{ > + int bare = 0; > + struct config_set cs = {0}; > + char *worktree_config = xstrfmt("%s/config.worktree", repo_get_common_dir(repo)); > + > + git_configset_init(&cs); > + git_configset_add_file(&cs, worktree_config); > + git_configset_get_bool(&cs, "core.bare", &bare); > + > + git_configset_clear(&cs); > + free(worktree_config); > + return bare; > +} That is nicely described. > /** > * get the main worktree > */ > @@ -79,16 +101,11 @@ static struct worktree *get_main_worktree(int skip_reading_head) > CALLOC_ARRAY(worktree, 1); > worktree->repo = the_repository; > worktree->path = strbuf_detach(&worktree_path, NULL); > - /* > - * NEEDSWORK: If this function is called from a secondary worktree and > - * config.worktree is present, is_bare_repository_cfg will reflect the > - * contents of config.worktree, not the contents of the main worktree. > - * This means that worktree->is_bare may be set to 0 even if the main > - * worktree is configured to be bare. > - */ > - worktree->is_bare = (is_bare_repository_cfg == 1) || > - is_bare_repository(); > worktree->is_current = is_current_worktree(worktree); > + worktree->is_bare = (is_bare_repository_cfg == 1) || > + is_bare_repository() || > + (!worktree->is_current && is_main_worktree_bare(the_repository)); Is "this worktree does not have is_current bit set" equivalent to "this worktree is the main one, so is_main_worktree_bare() needs to be consulted"? That linkage between "the is_current bit unset" and "is the main worktree" is not obvious to me. Thanks. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 19:19 ` Junio C Hamano @ 2025-01-31 19:26 ` Junio C Hamano 2025-01-31 20:11 ` Olga Pilipenco 0 siblings, 1 reply; 19+ messages in thread From: Junio C Hamano @ 2025-01-31 19:26 UTC (permalink / raw) To: Olga Pilipenco via GitGitGadget Cc: git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe, Olga Pilipenco Junio C Hamano <gitster@pobox.com> writes: > "Olga Pilipenco via GitGitGadget" <gitgitgadget@gmail.com> writes: > >> +/* >> +* When in a secondary worktree, and when extensions.worktreeConfig >> +* is true, only $commondir/config and $commondir/worktrees/<id>/ >> +* config.worktree are consulted, hence any core.bare=true setting in >> +* $commondir/config.worktree gets overlooked. Thus, check it manually >> +* to determine if the repository is bare. >> +*/ >> +static int is_main_worktree_bare(struct repository *repo) >> +{ >> + int bare = 0; >> + struct config_set cs = {0}; >> + char *worktree_config = xstrfmt("%s/config.worktree", repo_get_common_dir(repo)); >> + >> + git_configset_init(&cs); >> + git_configset_add_file(&cs, worktree_config); >> + git_configset_get_bool(&cs, "core.bare", &bare); >> + >> + git_configset_clear(&cs); >> + free(worktree_config); >> + return bare; >> +} > > That is nicely described. > >> /** >> * get the main worktree >> */ >> @@ -79,16 +101,11 @@ static struct worktree *get_main_worktree(int skip_reading_head) >> CALLOC_ARRAY(worktree, 1); >> worktree->repo = the_repository; >> worktree->path = strbuf_detach(&worktree_path, NULL); >> - /* >> - * NEEDSWORK: If this function is called from a secondary worktree and >> - * config.worktree is present, is_bare_repository_cfg will reflect the >> - * contents of config.worktree, not the contents of the main worktree. >> - * This means that worktree->is_bare may be set to 0 even if the main >> - * worktree is configured to be bare. >> - */ >> - worktree->is_bare = (is_bare_repository_cfg == 1) || >> - is_bare_repository(); >> worktree->is_current = is_current_worktree(worktree); >> + worktree->is_bare = (is_bare_repository_cfg == 1) || >> + is_bare_repository() || >> + (!worktree->is_current && is_main_worktree_bare(the_repository)); > > Is "this worktree does not have is_current bit set" equivalent to > "this worktree is the main one, so is_main_worktree_bare() needs to > be consulted"? That linkage between "the is_current bit unset" and > "is the main worktree" is not obvious to me. Does the thinking behind it go like this? We grabbed the "main" worktree object and stored it in worktree; it is either our current worktree (in which case is_current is true), or it is not (in which case, is_current is false). We know that the old logic failed when asking the "is it bare" question from a secondary worktree. !worktree->is_current tells us that we _are_ asking the question from a secondary worktree, so we need to make the extra call to check config.worktree file as well in that case. Perhaps the logic is clear to those who diagnosed the problem, wrote the patch, and reviewed it, in which case there is no reason to reroll. Perhaps it was just me to whom it was not obvious that the purpose of "is_current" check was not about "are we looking at the main worktree" but was about "if we are not in the main worktree, we need this extra check". Thanks. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 19:26 ` Junio C Hamano @ 2025-01-31 20:11 ` Olga Pilipenco 2025-01-31 20:20 ` Junio C Hamano 0 siblings, 1 reply; 19+ messages in thread From: Olga Pilipenco @ 2025-01-31 20:11 UTC (permalink / raw) To: Junio C Hamano Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe On Fri, Jan 31, 2025 at 12:26 PM Junio C Hamano <gitster@pobox.com> wrote: > > Junio C Hamano <gitster@pobox.com> writes: > > > "Olga Pilipenco via GitGitGadget" <gitgitgadget@gmail.com> writes: > > > >> +/* > >> +* When in a secondary worktree, and when extensions.worktreeConfig > >> +* is true, only $commondir/config and $commondir/worktrees/<id>/ > >> +* config.worktree are consulted, hence any core.bare=true setting in > >> +* $commondir/config.worktree gets overlooked. Thus, check it manually > >> +* to determine if the repository is bare. > >> +*/ > >> +static int is_main_worktree_bare(struct repository *repo) > >> +{ > >> + int bare = 0; > >> + struct config_set cs = {0}; > >> + char *worktree_config = xstrfmt("%s/config.worktree", repo_get_common_dir(repo)); > >> + > >> + git_configset_init(&cs); > >> + git_configset_add_file(&cs, worktree_config); > >> + git_configset_get_bool(&cs, "core.bare", &bare); > >> + > >> + git_configset_clear(&cs); > >> + free(worktree_config); > >> + return bare; > >> +} > > > > That is nicely described. > > > >> /** > >> * get the main worktree > >> */ > >> @@ -79,16 +101,11 @@ static struct worktree *get_main_worktree(int skip_reading_head) > >> CALLOC_ARRAY(worktree, 1); > >> worktree->repo = the_repository; > >> worktree->path = strbuf_detach(&worktree_path, NULL); > >> - /* > >> - * NEEDSWORK: If this function is called from a secondary worktree and > >> - * config.worktree is present, is_bare_repository_cfg will reflect the > >> - * contents of config.worktree, not the contents of the main worktree. > >> - * This means that worktree->is_bare may be set to 0 even if the main > >> - * worktree is configured to be bare. > >> - */ > >> - worktree->is_bare = (is_bare_repository_cfg == 1) || > >> - is_bare_repository(); > >> worktree->is_current = is_current_worktree(worktree); > >> + worktree->is_bare = (is_bare_repository_cfg == 1) || > >> + is_bare_repository() || > >> + (!worktree->is_current && is_main_worktree_bare(the_repository)); > > > > Is "this worktree does not have is_current bit set" equivalent to > > "this worktree is the main one, so is_main_worktree_bare() needs to > > be consulted"? That linkage between "the is_current bit unset" and > > "is the main worktree" is not obvious to me. > > Does the thinking behind it go like this? > > We grabbed the "main" worktree object and stored it in worktree; > it is either our current worktree (in which case is_current is > true), or it is not (in which case, is_current is false). We > know that the old logic failed when asking the "is it bare" > question from a secondary worktree. !worktree->is_current tells > us that we _are_ asking the question from a secondary worktree, > so we need to make the extra call to check config.worktree file > as well in that case. > > Perhaps the logic is clear to those who diagnosed the problem, wrote > the patch, and reviewed it, in which case there is no reason to > reroll. Perhaps it was just me to whom it was not obvious that > the purpose of "is_current" check was not about "are we looking at > the main worktree" but was about "if we are not in the main worktree, > we need this extra check". > > Thanks. You did a great job figuring it out and I agree it's confusing at first, but we tried our best to make it less confusing. `is_current` check is actually not necessary there, but having it there saves extra unnecessary calculations, also describes & fixes the exact scenario that didn't work (not being able to see main worktree as bare from a secondary worktree). Thanks for looking into that. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 20:11 ` Olga Pilipenco @ 2025-01-31 20:20 ` Junio C Hamano 2025-02-04 19:03 ` Olga Pilipenco 0 siblings, 1 reply; 19+ messages in thread From: Junio C Hamano @ 2025-01-31 20:20 UTC (permalink / raw) To: Olga Pilipenco Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe Olga Pilipenco <olga.pilipenco@shopify.com> writes: >> Perhaps the logic is clear to those who diagnosed the problem, wrote >> the patch, and reviewed it, in which case there is no reason to >> reroll. Perhaps it was just me to whom it was not obvious that >> the purpose of "is_current" check was not about "are we looking at >> the main worktree" but was about "if we are not in the main worktree, >> we need this extra check". >> >> Thanks. > > You did a great job figuring it out and I agree it's confusing at > first, but we tried our best to make it less confusing. > `is_current` check is actually not necessary there, but having it there saves > extra unnecessary calculations, also describes & fixes the exact scenario > that didn't work (not being able to see main worktree as bare from a > secondary worktree). If I had to do a great job there, then the code does deserve to be explained a bit better for later developers who wonder why it is written in the way it is, perhaps we a single-liner comment? Thanks. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 20:20 ` Junio C Hamano @ 2025-02-04 19:03 ` Olga Pilipenco 2025-02-04 19:43 ` Junio C Hamano 0 siblings, 1 reply; 19+ messages in thread From: Olga Pilipenco @ 2025-02-04 19:03 UTC (permalink / raw) To: Junio C Hamano Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe On Fri, Jan 31, 2025 at 1:20 PM Junio C Hamano <gitster@pobox.com> wrote: > > Olga Pilipenco <olga.pilipenco@shopify.com> writes: > > >> Perhaps the logic is clear to those who diagnosed the problem, wrote > >> the patch, and reviewed it, in which case there is no reason to > >> reroll. Perhaps it was just me to whom it was not obvious that > >> the purpose of "is_current" check was not about "are we looking at > >> the main worktree" but was about "if we are not in the main worktree, > >> we need this extra check". > >> > >> Thanks. > > > > You did a great job figuring it out and I agree it's confusing at > > first, but we tried our best to make it less confusing. > > `is_current` check is actually not necessary there, but having it there saves > > extra unnecessary calculations, also describes & fixes the exact scenario > > that didn't work (not being able to see main worktree as bare from a > > secondary worktree). > > If I had to do a great job there, then the code does deserve to be > explained a bit better for later developers who wonder why it is > written in the way it is, perhaps we a single-liner comment? I have 2 versions for comment: 1. Since is_main_worktree_bare explains quite well what it does we can have a shorter explanation of `!worktree->is_current` part, something like: /* Additional checks are needed if main worktree is not current (checking from secondary worktree) */ (!worktree->is_current && is_main_worktree_bare(the_repository)); 2. Or a bit longer inline explanation that partially repeats the explanation of is_main_worktree_bare + adds explanation about efficiency: /* * When in a secondary worktree we have to also verify if the main worktree * is bare in $commondir/config.worktree. * This check is unnecessary if we're currently in the main worktree, * as prior checks already consulted all configs of the current worktree. */ (!worktree->is_current && is_main_worktree_bare(the_repository)); Let me know if any of these work. Thanks. > Thanks. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-02-04 19:03 ` Olga Pilipenco @ 2025-02-04 19:43 ` Junio C Hamano 2025-02-04 20:33 ` Olga Pilipenco 0 siblings, 1 reply; 19+ messages in thread From: Junio C Hamano @ 2025-02-04 19:43 UTC (permalink / raw) To: Olga Pilipenco Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe Olga Pilipenco <olga.pilipenco@shopify.com> writes: > I have 2 versions for comment: > > 1. Since is_main_worktree_bare explains quite well what it does we can have > a shorter explanation of `!worktree->is_current` part, something like: > > /* Additional checks are needed if main worktree is not current > (checking from secondary worktree) */ > (!worktree->is_current && is_main_worktree_bare(the_repository)); For somebody who thought about the issue themselves (like me, before writing the message you are responding to), this shorter form would suffice. I'd rephrase it more like so /* When a secondary worktree, an extra check is needed */ for brevity, though. > 2. Or a bit longer inline explanation that partially repeats the > explanation of is_main_worktree_bare > + adds explanation about efficiency: > /* > * When in a secondary worktree we have to also verify if the main worktree > * is bare in $commondir/config.worktree. > * This check is unnecessary if we're currently in the main worktree, > * as prior checks already consulted all configs of the current worktree. > */ > (!worktree->is_current && is_main_worktree_bare(the_repository)); And this more extended version would have helped me by not having to ask Is "this worktree does not have is_current bit set" equivalent to "this worktree is the main one, so is_main_worktree_bare() needs to be consulted"? That linkage between "the is_current bit unset" and "is the main worktree" is not obvious to me. in the first place. In short, both should work, and I personally find that the latter may be a bit more helpful to readers. THanks. ^ permalink raw reply [flat|nested] 19+ messages in thread
* Re: [PATCH v3] worktree: detect from secondary worktree if main worktree is bare 2025-02-04 19:43 ` Junio C Hamano @ 2025-02-04 20:33 ` Olga Pilipenco 0 siblings, 0 replies; 19+ messages in thread From: Olga Pilipenco @ 2025-02-04 20:33 UTC (permalink / raw) To: Junio C Hamano Cc: Olga Pilipenco via GitGitGadget, git, Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe On Tue, Feb 4, 2025 at 12:43 PM Junio C Hamano <gitster@pobox.com> wrote: > > Olga Pilipenco <olga.pilipenco@shopify.com> writes: > > > I have 2 versions for comment: > > > > 1. Since is_main_worktree_bare explains quite well what it does we can have > > a shorter explanation of `!worktree->is_current` part, something like: > > > > /* Additional checks are needed if main worktree is not current > > (checking from secondary worktree) */ > > (!worktree->is_current && is_main_worktree_bare(the_repository)); > > For somebody who thought about the issue themselves (like me, before > writing the message you are responding to), this shorter form would > suffice. I'd rephrase it more like so > > /* When a secondary worktree, an extra check is needed */ > > for brevity, though. > > > > 2. Or a bit longer inline explanation that partially repeats the > > explanation of is_main_worktree_bare > > + adds explanation about efficiency: > > /* > > * When in a secondary worktree we have to also verify if the main worktree > > * is bare in $commondir/config.worktree. > > * This check is unnecessary if we're currently in the main worktree, > > * as prior checks already consulted all configs of the current worktree. > > */ > > (!worktree->is_current && is_main_worktree_bare(the_repository)); > > And this more extended version would have helped me by not having to > ask > > Is "this worktree does not have is_current bit set" equivalent > to "this worktree is the main one, so is_main_worktree_bare() > needs to be consulted"? That linkage between "the is_current > bit unset" and "is the main worktree" is not obvious to me. > > in the first place. > > In short, both should work, and I personally find that the latter > may be a bit more helpful to readers. > > THanks. Perfect, I'll add the latter one then. Thank you! ^ permalink raw reply [flat|nested] 19+ messages in thread
* [PATCH v4] worktree: detect from secondary worktree if main worktree is bare 2025-01-31 18:08 ` [PATCH v3] " Olga Pilipenco via GitGitGadget 2025-01-31 19:19 ` Junio C Hamano @ 2025-02-05 6:30 ` Olga Pilipenco via GitGitGadget 1 sibling, 0 replies; 19+ messages in thread From: Olga Pilipenco via GitGitGadget @ 2025-02-05 6:30 UTC (permalink / raw) To: git Cc: Patrick Steinhardt, Eric Sunshine, Johannes Schindelin, René Scharfe, Junio C Hamano, Olga Pilipenco, Olga Pilipenco From: Olga Pilipenco <olga.pilipenco@shopify.com> When extensions.worktreeConfig is true and the main worktree is bare -- that is, its config.worktree file contains core.bare=true -- commands run from secondary worktrees incorrectly see the main worktree as not bare. As such, those commands incorrectly think that the repository's default branch (typically "main" or "master") is checked out in the bare repository even though it's not. This makes it impossible, for instance, to checkout or delete the default branch from a secondary worktree, among other shortcomings. This problem occurs because, when extensions.worktreeConfig is true, commands run in secondary worktrees only consult $commondir/config and $commondir/worktrees/<id>/config.worktree, thus they never see the main worktree's core.bare=true setting in $commondir/config.worktree. Fix this problem by consulting the main worktree's config.worktree file when checking whether it is bare. (This extra work is performed only when running from a secondary worktree.) Helped-by: Eric Sunshine <sunshine@sunshineco.com> Signed-off-by: Olga Pilipenco <olga.pilipenco@shopify.com> --- worktree: detect from secondary worktree if main worktree is bare Changes since v3: * added a comment to explain why extra check is needed to detect if main worktree bare Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1829%2Folga-mcbfe%2Ffix-bare-repo-detection-with-worktree-config-v4 Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1829/olga-mcbfe/fix-bare-repo-detection-with-worktree-config-v4 Pull-Request: https://github.com/gitgitgadget/git/pull/1829 Range-diff vs v3: 1: f9207746b38 ! 1: a34459b5681 worktree: detect from secondary worktree if main worktree is bare @@ worktree.c: static struct worktree *get_main_worktree(int skip_reading_head) worktree->is_current = is_current_worktree(worktree); + worktree->is_bare = (is_bare_repository_cfg == 1) || + is_bare_repository() || ++ /* ++ * When in a secondary worktree we have to also verify if the main ++ * worktree is bare in $commondir/config.worktree. ++ * This check is unnecessary if we're currently in the main worktree, ++ * as prior checks already consulted all configs of the current worktree. ++ */ + (!worktree->is_current && is_main_worktree_bare(the_repository)); + if (!skip_reading_head) t/t3200-branch.sh | 14 ++++++++++++++ worktree.c | 41 ++++++++++++++++++++++++++++++++--------- 2 files changed, 46 insertions(+), 9 deletions(-) diff --git a/t/t3200-branch.sh b/t/t3200-branch.sh index a3a21c54cf6..f3e720dc10d 100755 --- a/t/t3200-branch.sh +++ b/t/t3200-branch.sh @@ -410,6 +410,20 @@ test_expect_success 'bare main worktree has HEAD at branch deleted by secondary git -C secondary branch -D main ' +test_expect_success 'secondary worktrees recognize core.bare=true in main config.worktree' ' + test_when_finished "rm -rf bare_repo non_bare_repo secondary_worktree" && + git init -b main non_bare_repo && + test_commit -C non_bare_repo x && + + git clone --bare non_bare_repo bare_repo && + git -C bare_repo config extensions.worktreeConfig true && + git -C bare_repo config unset core.bare && + git -C bare_repo config --worktree core.bare true && + + git -C bare_repo worktree add ../secondary_worktree && + git -C secondary_worktree checkout main +' + test_expect_success 'git branch --list -v with --abbrev' ' test_when_finished "git branch -D t" && git branch t && diff --git a/worktree.c b/worktree.c index 248bbb39d43..d4a68c9c235 100644 --- a/worktree.c +++ b/worktree.c @@ -65,6 +65,28 @@ static int is_current_worktree(struct worktree *wt) return is_current; } +/* +* When in a secondary worktree, and when extensions.worktreeConfig +* is true, only $commondir/config and $commondir/worktrees/<id>/ +* config.worktree are consulted, hence any core.bare=true setting in +* $commondir/config.worktree gets overlooked. Thus, check it manually +* to determine if the repository is bare. +*/ +static int is_main_worktree_bare(struct repository *repo) +{ + int bare = 0; + struct config_set cs = {0}; + char *worktree_config = xstrfmt("%s/config.worktree", repo_get_common_dir(repo)); + + git_configset_init(&cs); + git_configset_add_file(&cs, worktree_config); + git_configset_get_bool(&cs, "core.bare", &bare); + + git_configset_clear(&cs); + free(worktree_config); + return bare; +} + /** * get the main worktree */ @@ -79,16 +101,17 @@ static struct worktree *get_main_worktree(int skip_reading_head) CALLOC_ARRAY(worktree, 1); worktree->repo = the_repository; worktree->path = strbuf_detach(&worktree_path, NULL); - /* - * NEEDSWORK: If this function is called from a secondary worktree and - * config.worktree is present, is_bare_repository_cfg will reflect the - * contents of config.worktree, not the contents of the main worktree. - * This means that worktree->is_bare may be set to 0 even if the main - * worktree is configured to be bare. - */ - worktree->is_bare = (is_bare_repository_cfg == 1) || - is_bare_repository(); worktree->is_current = is_current_worktree(worktree); + worktree->is_bare = (is_bare_repository_cfg == 1) || + is_bare_repository() || + /* + * When in a secondary worktree we have to also verify if the main + * worktree is bare in $commondir/config.worktree. + * This check is unnecessary if we're currently in the main worktree, + * as prior checks already consulted all configs of the current worktree. + */ + (!worktree->is_current && is_main_worktree_bare(the_repository)); + if (!skip_reading_head) add_head_info(worktree); return worktree; base-commit: f93ff170b93a1782659637824b25923245ac9dd1 -- gitgitgadget ^ permalink raw reply related [flat|nested] 19+ messages in thread
end of thread, other threads:[~2025-02-05 6:30 UTC | newest] Thread overview: 19+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2024-11-15 6:52 [PATCH] worktree: detect from secondary worktree if main worktree is bare Olga Pilipenco via GitGitGadget 2025-01-16 21:35 ` [PATCH v2] " Olga Pilipenco via GitGitGadget 2025-01-19 22:30 ` Eric Sunshine 2025-01-28 21:44 ` Olga Pilipenco 2025-01-29 13:41 ` Eric Sunshine 2025-01-29 17:08 ` Junio C Hamano [not found] ` <F15C12AB-2238-4553-AFA5-18277B18CE5A@shopify.com> 2025-01-30 14:32 ` Eric Sunshine 2025-01-30 14:44 ` Eric Sunshine 2025-01-31 7:05 ` Olga Pilipenco 2025-01-31 13:28 ` Eric Sunshine 2025-01-31 18:08 ` [PATCH v3] " Olga Pilipenco via GitGitGadget 2025-01-31 19:19 ` Junio C Hamano 2025-01-31 19:26 ` Junio C Hamano 2025-01-31 20:11 ` Olga Pilipenco 2025-01-31 20:20 ` Junio C Hamano 2025-02-04 19:03 ` Olga Pilipenco 2025-02-04 19:43 ` Junio C Hamano 2025-02-04 20:33 ` Olga Pilipenco 2025-02-05 6:30 ` [PATCH v4] " Olga Pilipenco via GitGitGadget
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).