On Thu, May 24 2018, Ævar Arnfjörð Bjarmason wrote:

>  struct tracking_name_data {
>       /* const */ char *src_ref;
>       char *dst_ref;
>       struct object_id *dst_oid;
>       int unique;
> +     const char *implicit_remote;
> +     char *implicit_dst_ref;
>  };

There's a bug here that I'll fix in a v3. We need to have a implicit_*
variant for dst_oid as well. Currently this will be buggy and check out
origin/<branch>, but then check the index out to the tree of whatever
the last <someremote>/<branch> we iterated over was.

Easiy fix and I already have it locally, I just want to improve some of
the testing. I missed it because in my tests I'd just re-add the same
remote again, so the trees were the same.

>  static int check_tracking_name(struct remote *remote, void *cb_data)
> @@ -20,6 +23,8 @@ static int check_tracking_name(struct remote *remote, void 
> *cb_data)
>               free(query.dst);
>               return 0;
>       }
> +     if (cb->implicit_remote && !strcmp(remote->name, cb->implicit_remote))
> +             cb->implicit_dst_ref = xstrdup(query.dst);
>       if (cb->dst_ref) {
>               free(query.dst);
>               cb->unique = 0;
> @@ -31,13 +36,21 @@ static int check_tracking_name(struct remote *remote, 
> void *cb_data)
>
>  const char *unique_tracking_name(const char *name, struct object_id *oid)
>  {
> -     struct tracking_name_data cb_data = { NULL, NULL, NULL, 1 };
> +     const char *implicit_remote = NULL;
> +     struct tracking_name_data cb_data = { NULL, NULL, NULL, 1, NULL, NULL };
> +     if (!git_config_get_string_const("checkout.implicitremote", 
> &implicit_remote))
> +             cb_data.implicit_remote = implicit_remote;
>       cb_data.src_ref = xstrfmt("refs/heads/%s", name);
>       cb_data.dst_oid = oid;
>       for_each_remote(check_tracking_name, &cb_data);
>       free(cb_data.src_ref);
> -     if (cb_data.unique)
> +     free((char *)implicit_remote);
> +     if (cb_data.unique) {
> +             free(cb_data.implicit_dst_ref);
>               return cb_data.dst_ref;
> +     }
>       free(cb_data.dst_ref);
> +     if (cb_data.implicit_dst_ref)
> +             return cb_data.implicit_dst_ref;
>       return NULL;
>  }
> diff --git a/t/t2024-checkout-dwim.sh b/t/t2024-checkout-dwim.sh
> index 3e5ac81bd2..da6bd74bbc 100755
> --- a/t/t2024-checkout-dwim.sh
> +++ b/t/t2024-checkout-dwim.sh
> @@ -68,6 +68,16 @@ test_expect_success 'checkout of branch from multiple 
> remotes fails #1' '
>       test_branch master
>  '
>
> +test_expect_success 'checkout of branch from multiple remotes succeeds with 
> checkout.implicitRemote #1' '
> +     git checkout -B master &&
> +     test_might_fail git branch -D foo &&
> +
> +     git -c checkout.implicitRemote=repo_a checkout foo &&
> +     test_branch foo &&
> +     test_cmp_rev remotes/repo_a/foo HEAD &&
> +     test_branch_upstream foo repo_a foo
> +'
> +
>  test_expect_success 'checkout of branch from a single remote succeeds #1' '
>       git checkout -B master &&
>       test_might_fail git branch -D bar &&
> diff --git a/t/t2025-worktree-add.sh b/t/t2025-worktree-add.sh
> index 2240498924..271a6413f0 100755
> --- a/t/t2025-worktree-add.sh
> +++ b/t/t2025-worktree-add.sh
> @@ -402,6 +402,24 @@ test_expect_success '"add" <path> <branch> dwims' '
>       )
>  '
>
> +test_expect_success '"add" <path> <branch> dwims with 
> checkout.implicitRemote' '
> +     test_when_finished rm -rf repo_upstream repo_dwim foo &&
> +     setup_remote_repo repo_upstream repo_dwim &&
> +     git init repo_dwim &&
> +     (
> +             cd repo_dwim &&
> +             git remote add repo_upstream2 ../repo_upstream &&
> +             git fetch repo_upstream2 &&
> +             test_must_fail git worktree add ../foo foo &&
> +             git -c checkout.implicitRemote=repo_upstream worktree add 
> ../foo foo
> +     ) &&
> +     (
> +             cd foo &&
> +             test_branch_upstream foo repo_upstream foo &&
> +             test_cmp_rev refs/remotes/repo_upstream/foo refs/heads/foo
> +     )
> +'
> +
>  test_expect_success 'git worktree add does not match remote' '
>       test_when_finished rm -rf repo_a repo_b foo &&
>       setup_remote_repo repo_a repo_b &&

Reply via email to