|
|
|
#include "git-compat-util.h"
|
|
|
|
#include "cache.h"
|
|
|
|
#include "config.h"
|
|
|
|
#include "branch.h"
|
|
|
|
#include "refs.h"
|
|
|
|
#include "remote.h"
|
|
|
|
#include "commit.h"
|
|
|
|
#include "worktree.h"
|
|
|
|
|
|
|
|
struct tracking {
|
|
|
|
struct refspec spec;
|
|
|
|
char *src;
|
|
|
|
const char *remote;
|
|
|
|
int matches;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int find_tracked_branch(struct remote *remote, void *priv)
|
|
|
|
{
|
|
|
|
struct tracking *tracking = priv;
|
|
|
|
|
|
|
|
if (!remote_find_tracking(remote, &tracking->spec)) {
|
|
|
|
if (++tracking->matches == 1) {
|
|
|
|
tracking->src = tracking->spec.src;
|
|
|
|
tracking->remote = remote->name;
|
|
|
|
} else {
|
|
|
|
free(tracking->spec.src);
|
|
|
|
if (tracking->src) {
|
|
|
|
FREE_AND_NULL(tracking->src);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tracking->spec.src = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int should_setup_rebase(const char *origin)
|
|
|
|
{
|
|
|
|
switch (autorebase) {
|
|
|
|
case AUTOREBASE_NEVER:
|
|
|
|
return 0;
|
|
|
|
case AUTOREBASE_LOCAL:
|
|
|
|
return origin == NULL;
|
|
|
|
case AUTOREBASE_REMOTE:
|
|
|
|
return origin != NULL;
|
|
|
|
case AUTOREBASE_ALWAYS:
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char tracking_advice[] =
|
|
|
|
N_("\n"
|
|
|
|
"After fixing the error cause you may try to fix up\n"
|
|
|
|
"the remote tracking information by invoking\n"
|
|
|
|
"\"git branch --set-upstream-to=%s%s%s\".");
|
|
|
|
|
|
|
|
int install_branch_config(int flag, const char *local, const char *origin, const char *remote)
|
|
|
|
{
|
refactor skip_prefix to return a boolean
The skip_prefix() function returns a pointer to the content
past the prefix, or NULL if the prefix was not found. While
this is nice and simple, in practice it makes it hard to use
for two reasons:
1. When you want to conditionally skip or keep the string
as-is, you have to introduce a temporary variable.
For example:
tmp = skip_prefix(buf, "foo");
if (tmp)
buf = tmp;
2. It is verbose to check the outcome in a conditional, as
you need extra parentheses to silence compiler
warnings. For example:
if ((cp = skip_prefix(buf, "foo"))
/* do something with cp */
Both of these make it harder to use for long if-chains, and
we tend to use starts_with() instead. However, the first line
of "do something" is often to then skip forward in buf past
the prefix, either using a magic constant or with an extra
strlen(3) (which is generally computed at compile time, but
means we are repeating ourselves).
This patch refactors skip_prefix() to return a simple boolean,
and to provide the pointer value as an out-parameter. If the
prefix is not found, the out-parameter is untouched. This
lets you write:
if (skip_prefix(arg, "foo ", &arg))
do_foo(arg);
else if (skip_prefix(arg, "bar ", &arg))
do_bar(arg);
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
11 years ago
|
|
|
const char *shortname = NULL;
|
|
|
|
struct strbuf key = STRBUF_INIT;
|
|
|
|
int rebasing = should_setup_rebase(origin);
|
|
|
|
|
refactor skip_prefix to return a boolean
The skip_prefix() function returns a pointer to the content
past the prefix, or NULL if the prefix was not found. While
this is nice and simple, in practice it makes it hard to use
for two reasons:
1. When you want to conditionally skip or keep the string
as-is, you have to introduce a temporary variable.
For example:
tmp = skip_prefix(buf, "foo");
if (tmp)
buf = tmp;
2. It is verbose to check the outcome in a conditional, as
you need extra parentheses to silence compiler
warnings. For example:
if ((cp = skip_prefix(buf, "foo"))
/* do something with cp */
Both of these make it harder to use for long if-chains, and
we tend to use starts_with() instead. However, the first line
of "do something" is often to then skip forward in buf past
the prefix, either using a magic constant or with an extra
strlen(3) (which is generally computed at compile time, but
means we are repeating ourselves).
This patch refactors skip_prefix() to return a simple boolean,
and to provide the pointer value as an out-parameter. If the
prefix is not found, the out-parameter is untouched. This
lets you write:
if (skip_prefix(arg, "foo ", &arg))
do_foo(arg);
else if (skip_prefix(arg, "bar ", &arg))
do_bar(arg);
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
11 years ago
|
|
|
if (skip_prefix(remote, "refs/heads/", &shortname)
|
|
|
|
&& !strcmp(local, shortname)
|
|
|
|
&& !origin) {
|
|
|
|
warning(_("Not setting branch %s as its own upstream."),
|
|
|
|
local);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
strbuf_addf(&key, "branch.%s.remote", local);
|
|
|
|
if (git_config_set_gently(key.buf, origin ? origin : ".") < 0)
|
|
|
|
goto out_err;
|
|
|
|
|
|
|
|
strbuf_reset(&key);
|
|
|
|
strbuf_addf(&key, "branch.%s.merge", local);
|
|
|
|
if (git_config_set_gently(key.buf, remote) < 0)
|
|
|
|
goto out_err;
|
|
|
|
|
|
|
|
if (rebasing) {
|
|
|
|
strbuf_reset(&key);
|
|
|
|
strbuf_addf(&key, "branch.%s.rebase", local);
|
|
|
|
if (git_config_set_gently(key.buf, "true") < 0)
|
|
|
|
goto out_err;
|
|
|
|
}
|
|
|
|
strbuf_release(&key);
|
|
|
|
|
|
|
|
if (flag & BRANCH_CONFIG_VERBOSE) {
|
|
|
|
if (shortname) {
|
|
|
|
if (origin)
|
|
|
|
printf_ln(rebasing ?
|
|
|
|
_("Branch '%s' set up to track remote branch '%s' from '%s' by rebasing.") :
|
|
|
|
_("Branch '%s' set up to track remote branch '%s' from '%s'."),
|
|
|
|
local, shortname, origin);
|
|
|
|
else
|
|
|
|
printf_ln(rebasing ?
|
|
|
|
_("Branch '%s' set up to track local branch '%s' by rebasing.") :
|
|
|
|
_("Branch '%s' set up to track local branch '%s'."),
|
|
|
|
local, shortname);
|
|
|
|
} else {
|
|
|
|
if (origin)
|
|
|
|
printf_ln(rebasing ?
|
|
|
|
_("Branch '%s' set up to track remote ref '%s' by rebasing.") :
|
|
|
|
_("Branch '%s' set up to track remote ref '%s'."),
|
|
|
|
local, remote);
|
|
|
|
else
|
|
|
|
printf_ln(rebasing ?
|
|
|
|
_("Branch '%s' set up to track local ref '%s' by rebasing.") :
|
|
|
|
_("Branch '%s' set up to track local ref '%s'."),
|
|
|
|
local, remote);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
out_err:
|
|
|
|
strbuf_release(&key);
|
|
|
|
error(_("Unable to write upstream branch configuration"));
|
|
|
|
|
|
|
|
advise(_(tracking_advice),
|
|
|
|
origin ? origin : "",
|
|
|
|
origin ? "/" : "",
|
|
|
|
shortname ? shortname : remote);
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is called when new_ref is branched off of orig_ref, and tries
|
|
|
|
* to infer the settings for branch.<new_ref>.{remote,merge} from the
|
|
|
|
* config.
|
|
|
|
*/
|
|
|
|
static void setup_tracking(const char *new_ref, const char *orig_ref,
|
|
|
|
enum branch_track track, int quiet)
|
|
|
|
{
|
|
|
|
struct tracking tracking;
|
|
|
|
int config_flags = quiet ? 0 : BRANCH_CONFIG_VERBOSE;
|
|
|
|
|
|
|
|
memset(&tracking, 0, sizeof(tracking));
|
|
|
|
tracking.spec.dst = (char *)orig_ref;
|
|
|
|
if (for_each_remote(find_tracked_branch, &tracking))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (!tracking.matches)
|
|
|
|
switch (track) {
|
|
|
|
case BRANCH_TRACK_ALWAYS:
|
|
|
|
case BRANCH_TRACK_EXPLICIT:
|
|
|
|
case BRANCH_TRACK_OVERRIDE:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tracking.matches > 1)
|
|
|
|
die(_("Not tracking: ambiguous information for ref %s"),
|
|
|
|
orig_ref);
|
|
|
|
|
|
|
|
if (install_branch_config(config_flags, new_ref, tracking.remote,
|
|
|
|
tracking.src ? tracking.src : orig_ref) < 0)
|
|
|
|
exit(-1);
|
|
|
|
|
|
|
|
free(tracking.src);
|
|
|
|
}
|
|
|
|
|
|
|
|
int read_branch_desc(struct strbuf *buf, const char *branch_name)
|
|
|
|
{
|
|
|
|
char *v = NULL;
|
|
|
|
struct strbuf name = STRBUF_INIT;
|
|
|
|
strbuf_addf(&name, "branch.%s.description", branch_name);
|
|
|
|
if (git_config_get_string(name.buf, &v)) {
|
|
|
|
strbuf_release(&name);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
strbuf_addstr(buf, v);
|
|
|
|
free(v);
|
|
|
|
strbuf_release(&name);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if 'name' can be a valid name for a branch; die otherwise.
|
|
|
|
* Return 1 if the named branch already exists; return 0 otherwise.
|
|
|
|
* Fill ref with the full refname for the branch.
|
|
|
|
*/
|
|
|
|
int validate_branchname(const char *name, struct strbuf *ref)
|
|
|
|
{
|
|
|
|
if (strbuf_check_branch_ref(ref, name))
|
|
|
|
die(_("'%s' is not a valid branch name."), name);
|
|
|
|
|
|
|
|
return ref_exists(ref->buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check if a branch 'name' can be created as a new branch; die otherwise.
|
|
|
|
* 'force' can be used when it is OK for the named branch already exists.
|
|
|
|
* Return 1 if the named branch already exists; return 0 otherwise.
|
|
|
|
* Fill ref with the full refname for the branch.
|
|
|
|
*/
|
|
|
|
int validate_new_branchname(const char *name, struct strbuf *ref, int force)
|
|
|
|
{
|
|
|
|
const char *head;
|
|
|
|
|
|
|
|
if (!validate_branchname(name, ref))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!force)
|
|
|
|
die(_("A branch named '%s' already exists."),
|
|
|
|
ref->buf + strlen("refs/heads/"));
|
|
|
|
|
|
|
|
head = resolve_ref_unsafe("HEAD", 0, NULL, NULL);
|
|
|
|
if (!is_bare_repository() && head && !strcmp(head, ref->buf))
|
|
|
|
die(_("Cannot force update the current branch."));
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
branch.c: Validate tracking branches with refspecs instead of refs/remotes/*
The current code for validating tracking branches (e.g. the argument to
the -t/--track option) hardcodes refs/heads/* and refs/remotes/* as the
potential locations for tracking branches. This works with the refspecs
created by "git clone" or "git remote add", but is suboptimal in other
cases:
- If "refs/remotes/foo/bar" exists without any association to a remote
(i.e. there is no remote named "foo", or no remote with a refspec
that matches "refs/remotes/foo/bar"), then it is impossible to set up
a valid upstream config that tracks it. Currently, the code defaults
to using "refs/remotes/foo/bar" from repo "." as the upstream, which
works, but is probably not what the user had in mind when running
"git branch baz --track foo/bar".
- If the user has tweaked the fetch refspec for a remote to put its
remote-tracking branches outside of refs/remotes/*, e.g. by running
git config remote.foo.fetch "+refs/heads/*:refs/foo_stuff/*"
then the current code will refuse to use its remote-tracking branches
as --track arguments, since they do not match refs/remotes/*.
This patch removes the "refs/remotes/*" requirement for upstream branches,
and replaces it with explicit checking of the refspecs for each remote to
determine whether a given --track argument is a valid remote-tracking
branch. This solves both of the above problems, since the matching refspec
guarantees that there is a both a remote name and a remote branch name
that can be used for the upstream config.
However, this means that refs located within refs/remotes/* without a
corresponding remote/refspec will no longer be usable as upstreams.
The few existing tests which depended on this behavioral quirk has
already been fixed in the preceding patches.
This patch fixes the last remaining test failure in t2024-checkout-dwim.
Signed-off-by: Johan Herland <johan@herland.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
12 years ago
|
|
|
static int check_tracking_branch(struct remote *remote, void *cb_data)
|
|
|
|
{
|
|
|
|
char *tracking_branch = cb_data;
|
|
|
|
struct refspec query;
|
|
|
|
memset(&query, 0, sizeof(struct refspec));
|
|
|
|
query.dst = tracking_branch;
|
|
|
|
return !remote_find_tracking(remote, &query);
|
branch.c: Validate tracking branches with refspecs instead of refs/remotes/*
The current code for validating tracking branches (e.g. the argument to
the -t/--track option) hardcodes refs/heads/* and refs/remotes/* as the
potential locations for tracking branches. This works with the refspecs
created by "git clone" or "git remote add", but is suboptimal in other
cases:
- If "refs/remotes/foo/bar" exists without any association to a remote
(i.e. there is no remote named "foo", or no remote with a refspec
that matches "refs/remotes/foo/bar"), then it is impossible to set up
a valid upstream config that tracks it. Currently, the code defaults
to using "refs/remotes/foo/bar" from repo "." as the upstream, which
works, but is probably not what the user had in mind when running
"git branch baz --track foo/bar".
- If the user has tweaked the fetch refspec for a remote to put its
remote-tracking branches outside of refs/remotes/*, e.g. by running
git config remote.foo.fetch "+refs/heads/*:refs/foo_stuff/*"
then the current code will refuse to use its remote-tracking branches
as --track arguments, since they do not match refs/remotes/*.
This patch removes the "refs/remotes/*" requirement for upstream branches,
and replaces it with explicit checking of the refspecs for each remote to
determine whether a given --track argument is a valid remote-tracking
branch. This solves both of the above problems, since the matching refspec
guarantees that there is a both a remote name and a remote branch name
that can be used for the upstream config.
However, this means that refs located within refs/remotes/* without a
corresponding remote/refspec will no longer be usable as upstreams.
The few existing tests which depended on this behavioral quirk has
already been fixed in the preceding patches.
This patch fixes the last remaining test failure in t2024-checkout-dwim.
Signed-off-by: Johan Herland <johan@herland.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
12 years ago
|
|
|
}
|
|
|
|
|
|
|
|
static int validate_remote_tracking_branch(char *ref)
|
|
|
|
{
|
|
|
|
return !for_each_remote(check_tracking_branch, ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char upstream_not_branch[] =
|
|
|
|
N_("Cannot setup tracking information; starting point '%s' is not a branch.");
|
|
|
|
static const char upstream_missing[] =
|
|
|
|
N_("the requested upstream branch '%s' does not exist");
|
|
|
|
static const char upstream_advice[] =
|
|
|
|
N_("\n"
|
|
|
|
"If you are planning on basing your work on an upstream\n"
|
|
|
|
"branch that already exists at the remote, you may need to\n"
|
|
|
|
"run \"git fetch\" to retrieve it.\n"
|
|
|
|
"\n"
|
|
|
|
"If you are planning to push out a new local branch that\n"
|
|
|
|
"will track its remote counterpart, you may want to use\n"
|
|
|
|
"\"git push -u\" to set the upstream config as you push.");
|
|
|
|
|
|
|
|
void create_branch(const char *name, const char *start_name,
|
|
|
|
int force, int clobber_head_ok, int reflog,
|
|
|
|
int quiet, enum branch_track track)
|
|
|
|
{
|
|
|
|
struct commit *commit;
|
|
|
|
struct object_id oid;
|
|
|
|
char *real_ref;
|
|
|
|
struct strbuf ref = STRBUF_INIT;
|
|
|
|
int forcing = 0;
|
|
|
|
int dont_change_ref = 0;
|
|
|
|
int explicit_tracking = 0;
|
|
|
|
|
|
|
|
if (track == BRANCH_TRACK_EXPLICIT || track == BRANCH_TRACK_OVERRIDE)
|
|
|
|
explicit_tracking = 1;
|
|
|
|
|
|
|
|
if ((track == BRANCH_TRACK_OVERRIDE || clobber_head_ok)
|
|
|
|
? validate_branchname(name, &ref)
|
|
|
|
: validate_new_branchname(name, &ref, force)) {
|
|
|
|
if (!force)
|
|
|
|
dont_change_ref = 1;
|
|
|
|
else
|
|
|
|
forcing = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
real_ref = NULL;
|
|
|
|
if (get_oid(start_name, &oid)) {
|
|
|
|
if (explicit_tracking) {
|
|
|
|
if (advice_set_upstream_failure) {
|
|
|
|
error(_(upstream_missing), start_name);
|
|
|
|
advise(_(upstream_advice));
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
die(_(upstream_missing), start_name);
|
|
|
|
}
|
|
|
|
die(_("Not a valid object name: '%s'."), start_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (dwim_ref(start_name, strlen(start_name), &oid, &real_ref)) {
|
|
|
|
case 0:
|
|
|
|
/* Not branching from any existing branch */
|
|
|
|
if (explicit_tracking)
|
|
|
|
die(_(upstream_not_branch), start_name);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
/* Unique completion -- good, only if it is a real branch */
|
|
|
|
if (!starts_with(real_ref, "refs/heads/") &&
|
branch.c: Validate tracking branches with refspecs instead of refs/remotes/*
The current code for validating tracking branches (e.g. the argument to
the -t/--track option) hardcodes refs/heads/* and refs/remotes/* as the
potential locations for tracking branches. This works with the refspecs
created by "git clone" or "git remote add", but is suboptimal in other
cases:
- If "refs/remotes/foo/bar" exists without any association to a remote
(i.e. there is no remote named "foo", or no remote with a refspec
that matches "refs/remotes/foo/bar"), then it is impossible to set up
a valid upstream config that tracks it. Currently, the code defaults
to using "refs/remotes/foo/bar" from repo "." as the upstream, which
works, but is probably not what the user had in mind when running
"git branch baz --track foo/bar".
- If the user has tweaked the fetch refspec for a remote to put its
remote-tracking branches outside of refs/remotes/*, e.g. by running
git config remote.foo.fetch "+refs/heads/*:refs/foo_stuff/*"
then the current code will refuse to use its remote-tracking branches
as --track arguments, since they do not match refs/remotes/*.
This patch removes the "refs/remotes/*" requirement for upstream branches,
and replaces it with explicit checking of the refspecs for each remote to
determine whether a given --track argument is a valid remote-tracking
branch. This solves both of the above problems, since the matching refspec
guarantees that there is a both a remote name and a remote branch name
that can be used for the upstream config.
However, this means that refs located within refs/remotes/* without a
corresponding remote/refspec will no longer be usable as upstreams.
The few existing tests which depended on this behavioral quirk has
already been fixed in the preceding patches.
This patch fixes the last remaining test failure in t2024-checkout-dwim.
Signed-off-by: Johan Herland <johan@herland.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
12 years ago
|
|
|
validate_remote_tracking_branch(real_ref)) {
|
|
|
|
if (explicit_tracking)
|
|
|
|
die(_(upstream_not_branch), start_name);
|
|
|
|
else
|
|
|
|
real_ref = NULL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
die(_("Ambiguous object name: '%s'."), start_name);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
Convert lookup_commit* to struct object_id
Convert lookup_commit, lookup_commit_or_die,
lookup_commit_reference, and lookup_commit_reference_gently to take
struct object_id arguments.
Introduce a temporary in parse_object buffer in order to convert this
function. This is required since in order to convert parse_object and
parse_object_buffer, lookup_commit_reference_gently and
lookup_commit_or_die would need to be converted. Not introducing a
temporary would therefore require that lookup_commit_or_die take a
struct object_id *, but lookup_commit would take unsigned char *,
leaving a confusing and hard-to-use interface.
parse_object_buffer will lose this temporary in a later patch.
This commit was created with manual changes to commit.c, commit.h, and
object.c, plus the following semantic patch:
@@
expression E1, E2;
@@
- lookup_commit_reference_gently(E1.hash, E2)
+ lookup_commit_reference_gently(&E1, E2)
@@
expression E1, E2;
@@
- lookup_commit_reference_gently(E1->hash, E2)
+ lookup_commit_reference_gently(E1, E2)
@@
expression E1;
@@
- lookup_commit_reference(E1.hash)
+ lookup_commit_reference(&E1)
@@
expression E1;
@@
- lookup_commit_reference(E1->hash)
+ lookup_commit_reference(E1)
@@
expression E1;
@@
- lookup_commit(E1.hash)
+ lookup_commit(&E1)
@@
expression E1;
@@
- lookup_commit(E1->hash)
+ lookup_commit(E1)
@@
expression E1, E2;
@@
- lookup_commit_or_die(E1.hash, E2)
+ lookup_commit_or_die(&E1, E2)
@@
expression E1, E2;
@@
- lookup_commit_or_die(E1->hash, E2)
+ lookup_commit_or_die(E1, E2)
Signed-off-by: brian m. carlson <sandals@crustytoothpaste.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
8 years ago
|
|
|
if ((commit = lookup_commit_reference(&oid)) == NULL)
|
|
|
|
die(_("Not a valid branch point: '%s'."), start_name);
|
|
|
|
oidcpy(&oid, &commit->object.oid);
|
|
|
|
|
|
|
|
if (reflog)
|
|
|
|
log_all_ref_updates = LOG_REFS_NORMAL;
|
|
|
|
|
|
|
|
if (!dont_change_ref) {
|
|
|
|
struct ref_transaction *transaction;
|
|
|
|
struct strbuf err = STRBUF_INIT;
|
|
|
|
char *msg;
|
|
|
|
|
|
|
|
if (forcing)
|
|
|
|
msg = xstrfmt("branch: Reset to %s", start_name);
|
|
|
|
else
|
|
|
|
msg = xstrfmt("branch: Created from %s", start_name);
|
|
|
|
|
|
|
|
transaction = ref_transaction_begin(&err);
|
|
|
|
if (!transaction ||
|
|
|
|
ref_transaction_update(transaction, ref.buf,
|
|
|
|
&oid, forcing ? NULL : &null_oid,
|
|
|
|
0, msg, &err) ||
|
|
|
|
ref_transaction_commit(transaction, &err))
|
|
|
|
die("%s", err.buf);
|
|
|
|
ref_transaction_free(transaction);
|
|
|
|
strbuf_release(&err);
|
|
|
|
free(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (real_ref && track)
|
|
|
|
setup_tracking(ref.buf + 11, real_ref, track, quiet);
|
|
|
|
|
|
|
|
strbuf_release(&ref);
|
|
|
|
free(real_ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
void remove_branch_state(void)
|
|
|
|
{
|
memoize common git-path "constant" files
One of the most common uses of git_path() is to pass a
constant, like git_path("MERGE_MSG"). This has two
drawbacks:
1. The return value is a static buffer, and the lifetime
is dependent on other calls to git_path, etc.
2. There's no compile-time checking of the pathname. This
is OK for a one-off (after all, we have to spell it
correctly at least once), but many of these constant
strings appear throughout the code.
This patch introduces a series of functions to "memoize"
these strings, which are essentially globals for the
lifetime of the program. We compute the value once, take
ownership of the buffer, and return the cached value for
subsequent calls. cache.h provides a helper macro for
defining these functions as one-liners, and defines a few
common ones for global use.
Using a macro is a little bit gross, but it does nicely
document the purpose of the functions. If we need to touch
them all later (e.g., because we learned how to change the
git_dir variable at runtime, and need to invalidate all of
the stored values), it will be much easier to have the
complete list.
Note that the shared-global functions have separate, manual
declarations. We could do something clever with the macros
(e.g., expand it to a declaration in some places, and a
declaration _and_ a definition in path.c). But there aren't
that many, and it's probably better to stay away from
too-magical macros.
Likewise, if we abandon the C preprocessor in favor of
generating these with a script, we could get much fancier.
E.g., normalizing "FOO/BAR-BAZ" into "git_path_foo_bar_baz".
But the small amount of saved typing is probably not worth
the resulting confusion to readers who want to grep for the
function's definition.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
10 years ago
|
|
|
unlink(git_path_cherry_pick_head());
|
|
|
|
unlink(git_path_revert_head());
|
|
|
|
unlink(git_path_merge_head());
|
|
|
|
unlink(git_path_merge_rr());
|
|
|
|
unlink(git_path_merge_msg());
|
|
|
|
unlink(git_path_merge_mode());
|
|
|
|
unlink(git_path_squash_msg());
|
|
|
|
}
|
|
|
|
|
|
|
|
void die_if_checked_out(const char *branch, int ignore_current_worktree)
|
|
|
|
{
|
|
|
|
const struct worktree *wt;
|
|
|
|
|
|
|
|
wt = find_shared_symref("HEAD", branch);
|
|
|
|
if (!wt || (ignore_current_worktree && wt->is_current))
|
|
|
|
return;
|
|
|
|
skip_prefix(branch, "refs/heads/", &branch);
|
|
|
|
die(_("'%s' is already checked out at '%s'"),
|
|
|
|
branch, wt->path);
|
|
|
|
}
|
|
|
|
|
|
|
|
int replace_each_worktree_head_symref(const char *oldref, const char *newref,
|
|
|
|
const char *logmsg)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
struct worktree **worktrees = get_worktrees(0);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; worktrees[i]; i++) {
|
|
|
|
struct ref_store *refs;
|
|
|
|
|
|
|
|
if (worktrees[i]->is_detached)
|
|
|
|
continue;
|
branch: fix branch renaming not updating HEADs correctly
There are two bugs that sort of work together and cause
problems. Let's start with one in replace_each_worktree_head_symref.
Before fa099d2322 (worktree.c: kill parse_ref() in favor of
refs_resolve_ref_unsafe() - 2017-04-24), this code looks like this:
if (strcmp(oldref, worktrees[i]->head_ref))
continue;
set_worktree_head_symref(...);
After fa099d2322, it is possible that head_ref can be NULL. However,
the updated code takes the wrong exit. In the error case (NULL
head_ref), we should "continue;" to the next worktree. The updated
code makes us _skip_ "continue;" and update HEAD anyway.
The NULL head_ref is triggered by the second bug in add_head_info (in
the same commit). With the flag RESOLVE_REF_READING, resolve_ref_unsafe()
will abort if it cannot resolve the target ref. For orphan checkouts,
HEAD always points to an unborned branch, resolving target ref will
always fail. Now we have NULL head_ref. Now we always update HEAD.
Correct the logic in replace_ function so that we don't accidentally
update HEAD on error. As it turns out, correcting the logic bug above
breaks branch renaming completely, thanks to the second bug.
"git branch -[Mm]" does two steps (on a normal checkout, no orphan!):
- rename the branch on disk (e.g. refs/heads/abc to refs/heads/def)
- update HEAD if it points to the branch being renamed.
At the second step, since the branch pointed to by HEAD (e.g. "abc") no
longer exists on disk, we run into a temporary orphan checkout situation
that has been just corrected to _not_ update HEAD. But we need to update
HEAD since it's not actually an orphan checkout. We need to update HEAD
to move out of that orphan state.
Correct add_head_info(), remove RESOLVE_REF_READING flag. With the flag
gone, we should always return good "head_ref" in orphan checkouts (either
temporary or permanent). With good head_ref, things start to work again.
Noticed-by: Nish Aravamudan <nish.aravamudan@canonical.com>
Signed-off-by: Nguyễn Thái Ngọc Duy <pclouds@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
7 years ago
|
|
|
if (!worktrees[i]->head_ref)
|
|
|
|
continue;
|
|
|
|
if (strcmp(oldref, worktrees[i]->head_ref))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
refs = get_worktree_ref_store(worktrees[i]);
|
|
|
|
if (refs_create_symref(refs, "HEAD", newref, logmsg))
|
|
|
|
ret = error(_("HEAD of working tree %s is not updated"),
|
|
|
|
worktrees[i]->path);
|
|
|
|
}
|
|
|
|
|
|
|
|
free_worktrees(worktrees);
|
|
|
|
return ret;
|
|
|
|
}
|