wt-status: tolerate dangling marks
When a user checks out the upstream branch of HEAD, the upstream branch not being a local branch, and then runs "git status", like this: git clone $URL client cd client git checkout @{u} git status no status is printed, but instead an error message: fatal: HEAD does not point to a branch (This error message when running "git branch" persists even after checking out other things - it only stops after checking out a branch.) This is because "git status" reads the reflog when determining the "HEAD detached" message, and thus attempts to DWIM "@{u}", but that doesn't work because HEAD no longer points to a branch. Therefore, when calculating the status of a worktree, tolerate dangling marks. This is done by adding an additional parameter to dwim_ref() and repo_dwim_ref(). Signed-off-by: Jonathan Tan <jonathantanmy@google.com> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
ec06b05568
commit
f24c30e0b6
@ -397,10 +397,10 @@ static void parse_treeish_arg(const char **argv,
|
||||
const char *colon = strchrnul(name, ':');
|
||||
int refnamelen = colon - name;
|
||||
|
||||
if (!dwim_ref(name, refnamelen, &oid, &ref))
|
||||
if (!dwim_ref(name, refnamelen, &oid, &ref, 0))
|
||||
die(_("no such ref: %.*s"), refnamelen, name);
|
||||
} else {
|
||||
dwim_ref(name, strlen(name), &oid, &ref);
|
||||
dwim_ref(name, strlen(name), &oid, &ref, 0);
|
||||
}
|
||||
|
||||
if (get_oid(name, &oid))
|
||||
|
2
branch.c
2
branch.c
@ -281,7 +281,7 @@ void create_branch(struct repository *r,
|
||||
die(_("Not a valid object name: '%s'."), start_name);
|
||||
}
|
||||
|
||||
switch (dwim_ref(start_name, strlen(start_name), &oid, &real_ref)) {
|
||||
switch (dwim_ref(start_name, strlen(start_name), &oid, &real_ref, 0)) {
|
||||
case 0:
|
||||
/* Not branching from any existing branch */
|
||||
if (explicit_tracking)
|
||||
|
@ -650,7 +650,7 @@ static void setup_branch_path(struct branch_info *branch)
|
||||
* If this is a ref, resolve it; otherwise, look up the OID for our
|
||||
* expression. Failure here is okay.
|
||||
*/
|
||||
if (!dwim_ref(branch->name, strlen(branch->name), &branch->oid, &branch->refname))
|
||||
if (!dwim_ref(branch->name, strlen(branch->name), &branch->oid, &branch->refname, 0))
|
||||
repo_get_oid_committish(the_repository, branch->name, &branch->oid);
|
||||
|
||||
strbuf_branchname(&buf, branch->name, INTERPRET_BRANCH_LOCAL);
|
||||
@ -1349,7 +1349,7 @@ static void die_expecting_a_branch(const struct branch_info *branch_info)
|
||||
struct object_id oid;
|
||||
char *to_free;
|
||||
|
||||
if (dwim_ref(branch_info->name, strlen(branch_info->name), &oid, &to_free) == 1) {
|
||||
if (dwim_ref(branch_info->name, strlen(branch_info->name), &oid, &to_free, 0) == 1) {
|
||||
const char *ref = to_free;
|
||||
|
||||
if (skip_prefix(ref, "refs/tags/", &ref))
|
||||
|
@ -943,7 +943,7 @@ static void get_tags_and_duplicates(struct rev_cmdline_info *info)
|
||||
if (e->flags & UNINTERESTING)
|
||||
continue;
|
||||
|
||||
if (dwim_ref(e->name, strlen(e->name), &oid, &full_name) != 1)
|
||||
if (dwim_ref(e->name, strlen(e->name), &oid, &full_name, 0) != 1)
|
||||
continue;
|
||||
|
||||
if (refspecs.nr) {
|
||||
|
@ -1062,7 +1062,7 @@ static char *find_branch_name(struct rev_info *rev)
|
||||
return NULL;
|
||||
ref = rev->cmdline.rev[positive].name;
|
||||
tip_oid = &rev->cmdline.rev[positive].item->oid;
|
||||
if (dwim_ref(ref, strlen(ref), &branch_oid, &full_ref) &&
|
||||
if (dwim_ref(ref, strlen(ref), &branch_oid, &full_ref, 0) &&
|
||||
skip_prefix(full_ref, "refs/heads/", &v) &&
|
||||
oideq(tip_oid, &branch_oid))
|
||||
branch = xstrdup(v);
|
||||
|
@ -501,7 +501,7 @@ static void merge_name(const char *remote, struct strbuf *msg)
|
||||
if (!remote_head)
|
||||
die(_("'%s' does not point to a commit"), remote);
|
||||
|
||||
if (dwim_ref(remote, strlen(remote), &branch_head, &found_ref) > 0) {
|
||||
if (dwim_ref(remote, strlen(remote), &branch_head, &found_ref, 0) > 0) {
|
||||
if (starts_with(found_ref, "refs/heads/")) {
|
||||
strbuf_addf(msg, "%s\t\tbranch '%s' of .\n",
|
||||
oid_to_hex(&branch_head), remote);
|
||||
|
@ -423,7 +423,7 @@ int cmd_reset(int argc, const char **argv, const char *prefix)
|
||||
char *ref = NULL;
|
||||
int err;
|
||||
|
||||
dwim_ref(rev, strlen(rev), &dummy, &ref);
|
||||
dwim_ref(rev, strlen(rev), &dummy, &ref, 0);
|
||||
if (ref && !starts_with(ref, "refs/"))
|
||||
ref = NULL;
|
||||
|
||||
|
@ -136,7 +136,7 @@ static void show_rev(int type, const struct object_id *oid, const char *name)
|
||||
struct object_id discard;
|
||||
char *full;
|
||||
|
||||
switch (dwim_ref(name, strlen(name), &discard, &full)) {
|
||||
switch (dwim_ref(name, strlen(name), &discard, &full, 0)) {
|
||||
case 0:
|
||||
/*
|
||||
* Not found -- not a ref. We could
|
||||
|
@ -741,7 +741,7 @@ int cmd_show_branch(int ac, const char **av, const char *prefix)
|
||||
die(Q_("only %d entry can be shown at one time.",
|
||||
"only %d entries can be shown at one time.",
|
||||
MAX_REVS), MAX_REVS);
|
||||
if (!dwim_ref(*av, strlen(*av), &oid, &ref))
|
||||
if (!dwim_ref(*av, strlen(*av), &oid, &ref, 0))
|
||||
die(_("no such ref %s"), *av);
|
||||
|
||||
/* Has the base been specified? */
|
||||
|
@ -185,7 +185,7 @@ static int get_stash_info(struct stash_info *info, int argc, const char **argv)
|
||||
end_of_rev = strchrnul(revision, '@');
|
||||
strbuf_add(&symbolic, revision, end_of_rev - revision);
|
||||
|
||||
ret = dwim_ref(symbolic.buf, symbolic.len, &dummy, &expanded_ref);
|
||||
ret = dwim_ref(symbolic.buf, symbolic.len, &dummy, &expanded_ref, 0);
|
||||
strbuf_release(&symbolic);
|
||||
switch (ret) {
|
||||
case 0: /* Not found, but valid ref */
|
||||
|
2
bundle.c
2
bundle.c
@ -377,7 +377,7 @@ static int write_bundle_refs(int bundle_fd, struct rev_info *revs)
|
||||
|
||||
if (e->item->flags & UNINTERESTING)
|
||||
continue;
|
||||
if (dwim_ref(e->name, strlen(e->name), &oid, &ref) != 1)
|
||||
if (dwim_ref(e->name, strlen(e->name), &oid, &ref, 0) != 1)
|
||||
goto skip_write_ref;
|
||||
if (read_ref_full(e->name, RESOLVE_REF_READING, &oid, &flag))
|
||||
flag = 0;
|
||||
|
7
cache.h
7
cache.h
@ -1567,6 +1567,13 @@ struct interpret_branch_name_options {
|
||||
* allowed, even ones to refs outside of those namespaces.
|
||||
*/
|
||||
unsigned allowed;
|
||||
|
||||
/*
|
||||
* If ^{upstream} or ^{push} (or equivalent) is requested, and the
|
||||
* branch in question does not have such a reference, return -1 instead
|
||||
* of die()-ing.
|
||||
*/
|
||||
unsigned nonfatal_dangling_mark : 1;
|
||||
};
|
||||
int repo_interpret_branch_name(struct repository *r,
|
||||
const char *str, int len,
|
||||
|
2
commit.c
2
commit.c
@ -921,7 +921,7 @@ struct commit *get_fork_point(const char *refname, struct commit *commit)
|
||||
struct commit *ret = NULL;
|
||||
char *full_refname;
|
||||
|
||||
switch (dwim_ref(refname, strlen(refname), &oid, &full_refname)) {
|
||||
switch (dwim_ref(refname, strlen(refname), &oid, &full_refname, 0)) {
|
||||
case 0:
|
||||
die("No such ref: '%s'", refname);
|
||||
case 1:
|
||||
|
14
refs.c
14
refs.c
@ -598,10 +598,13 @@ const char *git_default_branch_name(void)
|
||||
* to name a branch.
|
||||
*/
|
||||
static char *substitute_branch_name(struct repository *r,
|
||||
const char **string, int *len)
|
||||
const char **string, int *len,
|
||||
int nonfatal_dangling_mark)
|
||||
{
|
||||
struct strbuf buf = STRBUF_INIT;
|
||||
struct interpret_branch_name_options options = { 0 } ;
|
||||
struct interpret_branch_name_options options = {
|
||||
.nonfatal_dangling_mark = nonfatal_dangling_mark
|
||||
};
|
||||
int ret = repo_interpret_branch_name(r, *string, *len, &buf, &options);
|
||||
|
||||
if (ret == *len) {
|
||||
@ -615,9 +618,10 @@ static char *substitute_branch_name(struct repository *r,
|
||||
}
|
||||
|
||||
int repo_dwim_ref(struct repository *r, const char *str, int len,
|
||||
struct object_id *oid, char **ref)
|
||||
struct object_id *oid, char **ref, int nonfatal_dangling_mark)
|
||||
{
|
||||
char *last_branch = substitute_branch_name(r, &str, &len);
|
||||
char *last_branch = substitute_branch_name(r, &str, &len,
|
||||
nonfatal_dangling_mark);
|
||||
int refs_found = expand_ref(r, str, len, oid, ref);
|
||||
free(last_branch);
|
||||
return refs_found;
|
||||
@ -661,7 +665,7 @@ int repo_dwim_log(struct repository *r, const char *str, int len,
|
||||
struct object_id *oid, char **log)
|
||||
{
|
||||
struct ref_store *refs = get_main_ref_store(r);
|
||||
char *last_branch = substitute_branch_name(r, &str, &len);
|
||||
char *last_branch = substitute_branch_name(r, &str, &len, 0);
|
||||
const char **p;
|
||||
int logs_found = 0;
|
||||
struct strbuf path = STRBUF_INIT;
|
||||
|
8
refs.h
8
refs.h
@ -151,12 +151,14 @@ struct argv_array;
|
||||
void expand_ref_prefix(struct argv_array *prefixes, const char *prefix);
|
||||
|
||||
int expand_ref(struct repository *r, const char *str, int len, struct object_id *oid, char **ref);
|
||||
int repo_dwim_ref(struct repository *r, const char *str, int len, struct object_id *oid, char **ref);
|
||||
int repo_dwim_ref(struct repository *r, const char *str, int len,
|
||||
struct object_id *oid, char **ref, int nonfatal_dangling_mark);
|
||||
int repo_dwim_log(struct repository *r, const char *str, int len, struct object_id *oid, char **ref);
|
||||
static inline int dwim_ref(const char *str, int len, struct object_id *oid,
|
||||
char **ref)
|
||||
char **ref, int nonfatal_dangling_mark)
|
||||
{
|
||||
return repo_dwim_ref(the_repository, str, len, oid, ref);
|
||||
return repo_dwim_ref(the_repository, str, len, oid, ref,
|
||||
nonfatal_dangling_mark);
|
||||
}
|
||||
int dwim_log(const char *str, int len, struct object_id *oid, char **ref);
|
||||
|
||||
|
2
remote.c
2
remote.c
@ -1558,7 +1558,7 @@ static void set_merge(struct branch *ret)
|
||||
strcmp(ret->remote_name, "."))
|
||||
continue;
|
||||
if (dwim_ref(ret->merge_name[i], strlen(ret->merge_name[i]),
|
||||
&oid, &ref) == 1)
|
||||
&oid, &ref, 0) == 1)
|
||||
ret->merge[i]->dst = ref;
|
||||
else
|
||||
ret->merge[i]->dst = xstrdup(ret->merge_name[i]);
|
||||
|
16
sha1-name.c
16
sha1-name.c
@ -809,7 +809,7 @@ static int get_oid_basic(struct repository *r, const char *str, int len,
|
||||
|
||||
if (len == r->hash_algo->hexsz && !get_oid_hex(str, oid)) {
|
||||
if (warn_ambiguous_refs && warn_on_object_refname_ambiguity) {
|
||||
refs_found = repo_dwim_ref(r, str, len, &tmp_oid, &real_ref);
|
||||
refs_found = repo_dwim_ref(r, str, len, &tmp_oid, &real_ref, 0);
|
||||
if (refs_found > 0) {
|
||||
warning(warn_msg, len, str);
|
||||
if (advice_object_name_warning)
|
||||
@ -860,11 +860,11 @@ static int get_oid_basic(struct repository *r, const char *str, int len,
|
||||
|
||||
if (!len && reflog_len)
|
||||
/* allow "@{...}" to mean the current branch reflog */
|
||||
refs_found = repo_dwim_ref(r, "HEAD", 4, oid, &real_ref);
|
||||
refs_found = repo_dwim_ref(r, "HEAD", 4, oid, &real_ref, 0);
|
||||
else if (reflog_len)
|
||||
refs_found = repo_dwim_log(r, str, len, oid, &real_ref);
|
||||
else
|
||||
refs_found = repo_dwim_ref(r, str, len, oid, &real_ref);
|
||||
refs_found = repo_dwim_ref(r, str, len, oid, &real_ref, 0);
|
||||
|
||||
if (!refs_found)
|
||||
return -1;
|
||||
@ -1496,8 +1496,14 @@ static int interpret_branch_mark(struct repository *r,
|
||||
branch = branch_get(NULL);
|
||||
|
||||
value = get_data(branch, &err);
|
||||
if (!value)
|
||||
die("%s", err.buf);
|
||||
if (!value) {
|
||||
if (options->nonfatal_dangling_mark) {
|
||||
strbuf_release(&err);
|
||||
return -1;
|
||||
} else {
|
||||
die("%s", err.buf);
|
||||
}
|
||||
}
|
||||
|
||||
if (!branch_interpret_allowed(value, options->allowed))
|
||||
return -1;
|
||||
|
@ -846,6 +846,18 @@ test_expect_success 'status refreshes the index' '
|
||||
test_cmp expect output
|
||||
'
|
||||
|
||||
test_expect_success 'status shows detached HEAD properly after checking out non-local upstream branch' '
|
||||
test_when_finished rm -rf upstream downstream actual &&
|
||||
|
||||
test_create_repo upstream &&
|
||||
test_commit -C upstream foo &&
|
||||
|
||||
git clone upstream downstream &&
|
||||
git -C downstream checkout @{u} &&
|
||||
git -C downstream status >actual &&
|
||||
test_i18ngrep "HEAD detached at [0-9a-f]\\+" actual
|
||||
'
|
||||
|
||||
test_expect_success 'setup status submodule summary' '
|
||||
test_create_repo sm && (
|
||||
cd sm &&
|
||||
|
@ -1574,7 +1574,7 @@ static void wt_status_get_detached_from(struct repository *r,
|
||||
return;
|
||||
}
|
||||
|
||||
if (dwim_ref(cb.buf.buf, cb.buf.len, &oid, &ref) == 1 &&
|
||||
if (dwim_ref(cb.buf.buf, cb.buf.len, &oid, &ref, 1) == 1 &&
|
||||
/* sha1 is a commit? match without further lookup */
|
||||
(oideq(&cb.noid, &oid) ||
|
||||
/* perhaps sha1 is a tag, try to dereference to a commit */
|
||||
|
Loading…
Reference in New Issue
Block a user