Nguyễn Thái Ngọc Duy  <pclo...@gmail.com> writes:

> The main worktree has to be treated specially because well.. it's
> special from the beginning. So HEAD from the main worktree is
> acccessible via the name "main/HEAD" (we can't use
> "worktrees/main/HEAD" because "main" under "worktrees" is not
> reserved).

I do not quite follow.  So with this, both refs/heads/master and
main/refs/heads/master are good names for the master branch (even
though the local branch names are not per worktree), because
in the main worktree, refs/bisect/bad and main/refs/bisect/bad ought
to mean the same thing.

        side note: Or is this only for pseudo-refs
        (i.e. $GIT_DIR/$name where $name consists of all caps or
        underscore and typically ends with HEAD)?  Even if that were
        the case, I do not think it essentially changes the issue
        around disambiguation that much.

The disambiguation rule has always been: if you have a confusingly
named ref, you can spell it out fully to avoid any ambiguity, e.g.
refs/heads/refs/heads/foo can be given to "git rev-parse" and will
mean the tip of the branch whose name is "refs/heads/foo", even when
another branch whose name is "foo" exists.

Would we have a reasonable disambiguation rules that work well with
the main/ and worktrees/* prefixes?  When somebody has main/HEAD branch
and writes "git rev-parse main/HEAD", does it find refs/heads/main/HEAD
or $GIT_DIR/HEAD, if the user is in the main worktree?

It could be simply that the design is underdocumented in this patch
set (in which case I would have appreciated 'RFC' near 'PATCH'), but
I have a feeling that the code came way too early before such design
issues are fleshed out.

> diff --git a/refs.h b/refs.h
> index bd52c1bbae..9b53dbeae8 100644
> --- a/refs.h
> +++ b/refs.h
> @@ -704,9 +704,11 @@ int parse_hide_refs_config(const char *var, const char 
> *value, const char *);
>  int ref_is_hidden(const char *, const char *);
>  
>  enum ref_type {
> -     REF_TYPE_PER_WORKTREE,
> -     REF_TYPE_PSEUDOREF,
> -     REF_TYPE_NORMAL,
> +     REF_TYPE_PER_WORKTREE,    /* refs inside refs/ but not shared       */
> +     REF_TYPE_PSEUDOREF,       /* refs outside refs/ in current worktree */
> +     REF_TYPE_MAIN_PSEUDOREF,  /* pseudo refs from the main worktree     */
> +     REF_TYPE_OTHER_PSEUDOREF, /* pseudo refs from other worktrees       */
> +     REF_TYPE_NORMAL,          /* normal/shared refs inside refs/        */
>  };
>  
>  enum ref_type ref_type(const char *refname);
> diff --git a/refs/files-backend.c b/refs/files-backend.c
> index 416eafa453..bf9ed633b1 100644
> --- a/refs/files-backend.c
> +++ b/refs/files-backend.c
> @@ -149,6 +149,23 @@ static struct files_ref_store *files_downcast(struct 
> ref_store *ref_store,
>       return refs;
>  }
>  
> +static void files_reflog_path_other_worktrees(struct files_ref_store *refs,
> +                                           struct strbuf *sb,
> +                                           const char *refname)
> +{
> +     const char *real_ref;
> +
> +     if (!skip_prefix(refname, "worktrees/", &real_ref))
> +             BUG("refname %s is not a other-worktree ref", refname);
> +     real_ref = strchr(real_ref, '/');
> +     if (!real_ref)
> +             BUG("refname %s is not a other-worktree ref", refname);
> +     real_ref++;
> +
> +     strbuf_addf(sb, "%s/%.*slogs/%s", refs->gitcommondir,
> +                 (int)(real_ref - refname), refname, real_ref);
> +}
> +
>  static void files_reflog_path(struct files_ref_store *refs,
>                             struct strbuf *sb,
>                             const char *refname)
> @@ -158,6 +175,12 @@ static void files_reflog_path(struct files_ref_store 
> *refs,
>       case REF_TYPE_PSEUDOREF:
>               strbuf_addf(sb, "%s/logs/%s", refs->gitdir, refname);
>               break;
> +     case REF_TYPE_OTHER_PSEUDOREF:
> +             return files_reflog_path_other_worktrees(refs, sb, refname);
> +     case REF_TYPE_MAIN_PSEUDOREF:
> +             if (!skip_prefix(refname, "main/", &refname))
> +                     BUG("ref %s is not a main pseudoref", refname);
> +             /* passthru */
>       case REF_TYPE_NORMAL:
>               strbuf_addf(sb, "%s/logs/%s", refs->gitcommondir, refname);
>               break;
> @@ -176,6 +199,11 @@ static void files_ref_path(struct files_ref_store *refs,
>       case REF_TYPE_PSEUDOREF:
>               strbuf_addf(sb, "%s/%s", refs->gitdir, refname);
>               break;
> +     case REF_TYPE_MAIN_PSEUDOREF:
> +             if (!skip_prefix(refname, "main/", &refname))
> +                     BUG("ref %s is not a main pseudoref", refname);
> +             /* passthru */
> +     case REF_TYPE_OTHER_PSEUDOREF:
>       case REF_TYPE_NORMAL:
>               strbuf_addf(sb, "%s/%s", refs->gitcommondir, refname);
>               break;
> diff --git a/t/t1415-worktree-refs.sh b/t/t1415-worktree-refs.sh
> index 0c2d5f89a9..46ca7bfc19 100755
> --- a/t/t1415-worktree-refs.sh
> +++ b/t/t1415-worktree-refs.sh
> @@ -33,4 +33,34 @@ test_expect_success 'refs/local are per-worktree' '
>       ( cd wt2 && test_cmp_rev local/foo wt2 )
>  '
>  
> +test_expect_success 'resolve main/HEAD' '
> +     test_cmp_rev main/HEAD initial &&
> +     ( cd wt1 && test_cmp_rev main/HEAD initial ) &&
> +     ( cd wt2 && test_cmp_rev main/HEAD initial )
> +'
> +
> +test_expect_success 'resolve worktrees/xx/HEAD' '
> +     test_cmp_rev worktrees/wt1/HEAD wt1 &&
> +     ( cd wt1 && test_cmp_rev worktrees/wt1/HEAD wt1 ) &&
> +     ( cd wt2 && test_cmp_rev worktrees/wt1/HEAD wt1 )
> +'
> +
> +test_expect_success 'reflog of main/HEAD' '
> +     git reflog HEAD | sed "s/HEAD/main\/HEAD/" >expected &&
> +     git reflog main/HEAD >actual &&
> +     test_cmp expected actual &&
> +     git -C wt1 reflog main/HEAD >actual.wt1 &&
> +     test_cmp expected actual.wt1
> +'
> +
> +test_expect_success 'reflog of worktrees/xx/HEAD' '
> +     git -C wt2 reflog HEAD | sed "s/HEAD/worktrees\/wt2\/HEAD/" >expected &&
> +     git reflog worktrees/wt2/HEAD >actual &&
> +     test_cmp expected actual &&
> +     git -C wt1 reflog worktrees/wt2/HEAD >actual.wt1 &&
> +     test_cmp expected actual.wt1 &&
> +     git -C wt2 reflog worktrees/wt2/HEAD >actual.wt2 &&
> +     test_cmp expected actual.wt2
> +'
> +
>  test_done

Reply via email to