Skip to content

Commit

Permalink
Be more careful when determining whether a remote was configured
Browse files Browse the repository at this point in the history
One of the really nice features of the ~/.gitconfig file is that users
can override defaults by their own preferred settings for all of their
repositories.

One such default that some users like to override is whether the
"origin" remote gets auto-pruned or not. The user would simply call

	git config --global remote.origin.prune true

and from now on all "origin" remotes would be pruned automatically when
fetching into the local repository.

There is just one catch: now Git thinks that the "origin" remote is
configured, even if the repository config has no [remote "origin"]
section at all, as it does not realize that the "prune" setting was
configured globally and that there really is no "origin" remote
configured in this repository.

That is a problem e.g. when renaming a remote to a new name, when Git
may be fooled into thinking that there is already a remote of that new
name.

Let's fix this by paying more attention to *where* the remote settings
came from: if they are configured in the local repository config, we
must not overwrite them. If they were configured elsewhere, we cannot
overwrite them to begin with, as we only write the repository config.

There is only one caller of remote_is_configured() (in `git fetch`) that
may want to take remotes into account even if they were configured
outside the repository config; all other callers essentially try to
prevent the Git command from overwriting settings in the repository
config.

To accommodate that fact, the remote_is_configured() function now
requires a parameter that states whether the caller is interested in all
remotes, or only in those that were configured in the repository config.

Many thanks to Jeff King whose tireless review helped with settling for
nothing less than the current strategy.

This fixes git-for-windows#888

Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
  • Loading branch information
dscho committed Jan 19, 2017
1 parent 4e3c94c commit 1605031
Show file tree
Hide file tree
Showing 5 changed files with 21 additions and 13 deletions.
2 changes: 1 addition & 1 deletion builtin/fetch.c
Original file line number Diff line number Diff line change
Expand Up @@ -1177,7 +1177,7 @@ static int add_remote_or_group(const char *name, struct string_list *list)
git_config(get_remote_group, &g);
if (list->nr == prev_nr) {
struct remote *remote = remote_get(name);
if (!remote_is_configured(remote))
if (!remote_is_configured(remote, 0))
return 0;
string_list_append(list, remote->name);
}
Expand Down
14 changes: 7 additions & 7 deletions builtin/remote.c
Original file line number Diff line number Diff line change
Expand Up @@ -186,7 +186,7 @@ static int add(int argc, const char **argv)
url = argv[1];

remote = remote_get(name);
if (remote_is_configured(remote))
if (remote_is_configured(remote, 1))
die(_("remote %s already exists."), name);

strbuf_addf(&buf2, "refs/heads/test:refs/remotes/%s/test", name);
Expand Down Expand Up @@ -618,14 +618,14 @@ static int mv(int argc, const char **argv)
rename.remote_branches = &remote_branches;

oldremote = remote_get(rename.old);
if (!remote_is_configured(oldremote))
if (!remote_is_configured(oldremote, 1))
die(_("No such remote: %s"), rename.old);

if (!strcmp(rename.old, rename.new) && oldremote->origin != REMOTE_CONFIG)
return migrate_file(oldremote);

newremote = remote_get(rename.new);
if (remote_is_configured(newremote))
if (remote_is_configured(newremote, 1))
die(_("remote %s already exists."), rename.new);

strbuf_addf(&buf, "refs/heads/test:refs/remotes/%s/test", rename.new);
Expand Down Expand Up @@ -753,7 +753,7 @@ static int rm(int argc, const char **argv)
usage_with_options(builtin_remote_rm_usage, options);

remote = remote_get(argv[1]);
if (!remote_is_configured(remote))
if (!remote_is_configured(remote, 1))
die(_("No such remote: %s"), argv[1]);

known_remotes.to_delete = remote;
Expand Down Expand Up @@ -1415,7 +1415,7 @@ static int set_remote_branches(const char *remotename, const char **branches,
strbuf_addf(&key, "remote.%s.fetch", remotename);

remote = remote_get(remotename);
if (!remote_is_configured(remote))
if (!remote_is_configured(remote, 1))
die(_("No such remote '%s'"), remotename);

if (!add_mode && remove_all_fetch_refspecs(remotename, key.buf)) {
Expand Down Expand Up @@ -1469,7 +1469,7 @@ static int get_url(int argc, const char **argv)
remotename = argv[0];

remote = remote_get(remotename);
if (!remote_is_configured(remote))
if (!remote_is_configured(remote, 1))
die(_("No such remote '%s'"), remotename);

url_nr = 0;
Expand Down Expand Up @@ -1537,7 +1537,7 @@ static int set_url(int argc, const char **argv)
oldurl = newurl;

remote = remote_get(remotename);
if (!remote_is_configured(remote))
if (!remote_is_configured(remote, 1))
die(_("No such remote '%s'"), remotename);

if (push_mode) {
Expand Down
12 changes: 10 additions & 2 deletions remote.c
Original file line number Diff line number Diff line change
Expand Up @@ -255,6 +255,7 @@ static void read_remotes_file(struct remote *remote)

if (!f)
return;
remote->configured_in_repo = 1;
remote->origin = REMOTE_REMOTES;
while (strbuf_getline(&buf, f) != EOF) {
const char *v;
Expand Down Expand Up @@ -289,6 +290,7 @@ static void read_branches_file(struct remote *remote)
return;
}

remote->configured_in_repo = 1;
remote->origin = REMOTE_BRANCHES;

/*
Expand Down Expand Up @@ -371,6 +373,8 @@ static int handle_config(const char *key, const char *value, void *cb)
}
remote = make_remote(name, namelen);
remote->origin = REMOTE_CONFIG;
if (current_config_scope() == CONFIG_SCOPE_REPO)
remote->configured_in_repo = 1;
if (!strcmp(subkey, "mirror"))
remote->mirror = git_config_bool(key, value);
else if (!strcmp(subkey, "skipdefaultupdate"))
Expand Down Expand Up @@ -714,9 +718,13 @@ struct remote *pushremote_get(const char *name)
return remote_get_1(name, pushremote_for_branch);
}

int remote_is_configured(struct remote *remote)
int remote_is_configured(struct remote *remote, int in_repo)
{
return remote && remote->origin;
if (!remote)
return 0;
if (in_repo)
return remote->configured_in_repo;
return !!remote->origin;
}

int for_each_remote(each_remote_fn fn, void *priv)
Expand Down
4 changes: 2 additions & 2 deletions remote.h
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ struct remote {
struct hashmap_entry ent; /* must be first */

const char *name;
int origin;
int origin, configured_in_repo;

const char *foreign_vcs;

Expand Down Expand Up @@ -60,7 +60,7 @@ struct remote {

struct remote *remote_get(const char *name);
struct remote *pushremote_get(const char *name);
int remote_is_configured(struct remote *remote);
int remote_is_configured(struct remote *remote, int in_repo);

typedef int each_remote_fn(struct remote *remote, void *priv);
int for_each_remote(each_remote_fn fn, void *priv);
Expand Down
2 changes: 1 addition & 1 deletion t/t5505-remote.sh
Original file line number Diff line number Diff line change
Expand Up @@ -764,7 +764,7 @@ test_expect_success 'rename a remote with name prefix of other remote' '
)
'

test_expect_failure 'rename succeeds with existing remote.<target>.prune' '
test_expect_success 'rename succeeds with existing remote.<target>.prune' '
git clone one four.four &&
test_when_finished git config --global --unset remote.upstream.prune &&
git config --global remote.upstream.prune true &&
Expand Down

0 comments on commit 1605031

Please sign in to comment.