config: avoid using the global variable `store`
It is much easier to reason about, when the config code to set/unset variables or to remove/rename sections does not rely on a global (or file-local) variable. Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de> Signed-off-by: Junio C Hamano <gitster@pobox.com>maint
parent
8032cc4462
commit
fee8572c6d
119
config.c
119
config.c
|
@ -2288,7 +2288,7 @@ void git_die_config(const char *key, const char *err, ...)
|
||||||
* Find all the stuff for git_config_set() below.
|
* Find all the stuff for git_config_set() below.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static struct {
|
struct config_store_data {
|
||||||
int baselen;
|
int baselen;
|
||||||
char *key;
|
char *key;
|
||||||
int do_not_match;
|
int do_not_match;
|
||||||
|
@ -2298,56 +2298,58 @@ static struct {
|
||||||
unsigned int offset_alloc;
|
unsigned int offset_alloc;
|
||||||
enum { START, SECTION_SEEN, SECTION_END_SEEN, KEY_SEEN } state;
|
enum { START, SECTION_SEEN, SECTION_END_SEEN, KEY_SEEN } state;
|
||||||
unsigned int seen;
|
unsigned int seen;
|
||||||
} store;
|
};
|
||||||
|
|
||||||
static int matches(const char *key, const char *value)
|
static int matches(const char *key, const char *value,
|
||||||
|
const struct config_store_data *store)
|
||||||
{
|
{
|
||||||
if (strcmp(key, store.key))
|
if (strcmp(key, store->key))
|
||||||
return 0; /* not ours */
|
return 0; /* not ours */
|
||||||
if (!store.value_regex)
|
if (!store->value_regex)
|
||||||
return 1; /* always matches */
|
return 1; /* always matches */
|
||||||
if (store.value_regex == CONFIG_REGEX_NONE)
|
if (store->value_regex == CONFIG_REGEX_NONE)
|
||||||
return 0; /* never matches */
|
return 0; /* never matches */
|
||||||
|
|
||||||
return store.do_not_match ^
|
return store->do_not_match ^
|
||||||
(value && !regexec(store.value_regex, value, 0, NULL, 0));
|
(value && !regexec(store->value_regex, value, 0, NULL, 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
static int store_aux(const char *key, const char *value, void *cb)
|
static int store_aux(const char *key, const char *value, void *cb)
|
||||||
{
|
{
|
||||||
const char *ep;
|
const char *ep;
|
||||||
size_t section_len;
|
size_t section_len;
|
||||||
|
struct config_store_data *store = cb;
|
||||||
|
|
||||||
switch (store.state) {
|
switch (store->state) {
|
||||||
case KEY_SEEN:
|
case KEY_SEEN:
|
||||||
if (matches(key, value)) {
|
if (matches(key, value, store)) {
|
||||||
if (store.seen == 1 && store.multi_replace == 0) {
|
if (store->seen == 1 && store->multi_replace == 0) {
|
||||||
warning(_("%s has multiple values"), key);
|
warning(_("%s has multiple values"), key);
|
||||||
}
|
}
|
||||||
|
|
||||||
ALLOC_GROW(store.offset, store.seen + 1,
|
ALLOC_GROW(store->offset, store->seen + 1,
|
||||||
store.offset_alloc);
|
store->offset_alloc);
|
||||||
|
|
||||||
store.offset[store.seen] = cf->do_ftell(cf);
|
store->offset[store->seen] = cf->do_ftell(cf);
|
||||||
store.seen++;
|
store->seen++;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case SECTION_SEEN:
|
case SECTION_SEEN:
|
||||||
/*
|
/*
|
||||||
* What we are looking for is in store.key (both
|
* What we are looking for is in store->key (both
|
||||||
* section and var), and its section part is baselen
|
* section and var), and its section part is baselen
|
||||||
* long. We found key (again, both section and var).
|
* long. We found key (again, both section and var).
|
||||||
* We would want to know if this key is in the same
|
* We would want to know if this key is in the same
|
||||||
* section as what we are looking for. We already
|
* section as what we are looking for. We already
|
||||||
* know we are in the same section as what should
|
* know we are in the same section as what should
|
||||||
* hold store.key.
|
* hold store->key.
|
||||||
*/
|
*/
|
||||||
ep = strrchr(key, '.');
|
ep = strrchr(key, '.');
|
||||||
section_len = ep - key;
|
section_len = ep - key;
|
||||||
|
|
||||||
if ((section_len != store.baselen) ||
|
if ((section_len != store->baselen) ||
|
||||||
memcmp(key, store.key, section_len+1)) {
|
memcmp(key, store->key, section_len+1)) {
|
||||||
store.state = SECTION_END_SEEN;
|
store->state = SECTION_END_SEEN;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2355,26 +2357,27 @@ static int store_aux(const char *key, const char *value, void *cb)
|
||||||
* Do not increment matches: this is no match, but we
|
* Do not increment matches: this is no match, but we
|
||||||
* just made sure we are in the desired section.
|
* just made sure we are in the desired section.
|
||||||
*/
|
*/
|
||||||
ALLOC_GROW(store.offset, store.seen + 1,
|
ALLOC_GROW(store->offset, store->seen + 1,
|
||||||
store.offset_alloc);
|
store->offset_alloc);
|
||||||
store.offset[store.seen] = cf->do_ftell(cf);
|
store->offset[store->seen] = cf->do_ftell(cf);
|
||||||
/* fallthru */
|
/* fallthru */
|
||||||
case SECTION_END_SEEN:
|
case SECTION_END_SEEN:
|
||||||
case START:
|
case START:
|
||||||
if (matches(key, value)) {
|
if (matches(key, value, store)) {
|
||||||
ALLOC_GROW(store.offset, store.seen + 1,
|
ALLOC_GROW(store->offset, store->seen + 1,
|
||||||
store.offset_alloc);
|
store->offset_alloc);
|
||||||
store.offset[store.seen] = cf->do_ftell(cf);
|
store->offset[store->seen] = cf->do_ftell(cf);
|
||||||
store.state = KEY_SEEN;
|
store->state = KEY_SEEN;
|
||||||
store.seen++;
|
store->seen++;
|
||||||
} else {
|
} else {
|
||||||
if (strrchr(key, '.') - key == store.baselen &&
|
if (strrchr(key, '.') - key == store->baselen &&
|
||||||
!strncmp(key, store.key, store.baselen)) {
|
!strncmp(key, store->key, store->baselen)) {
|
||||||
store.state = SECTION_SEEN;
|
store->state = SECTION_SEEN;
|
||||||
ALLOC_GROW(store.offset,
|
ALLOC_GROW(store->offset,
|
||||||
store.seen + 1,
|
store->seen + 1,
|
||||||
store.offset_alloc);
|
store->offset_alloc);
|
||||||
store.offset[store.seen] = cf->do_ftell(cf);
|
store->offset[store->seen] =
|
||||||
|
cf->do_ftell(cf);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2389,31 +2392,33 @@ static int write_error(const char *filename)
|
||||||
return 4;
|
return 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct strbuf store_create_section(const char *key)
|
static struct strbuf store_create_section(const char *key,
|
||||||
|
const struct config_store_data *store)
|
||||||
{
|
{
|
||||||
const char *dot;
|
const char *dot;
|
||||||
int i;
|
int i;
|
||||||
struct strbuf sb = STRBUF_INIT;
|
struct strbuf sb = STRBUF_INIT;
|
||||||
|
|
||||||
dot = memchr(key, '.', store.baselen);
|
dot = memchr(key, '.', store->baselen);
|
||||||
if (dot) {
|
if (dot) {
|
||||||
strbuf_addf(&sb, "[%.*s \"", (int)(dot - key), key);
|
strbuf_addf(&sb, "[%.*s \"", (int)(dot - key), key);
|
||||||
for (i = dot - key + 1; i < store.baselen; i++) {
|
for (i = dot - key + 1; i < store->baselen; i++) {
|
||||||
if (key[i] == '"' || key[i] == '\\')
|
if (key[i] == '"' || key[i] == '\\')
|
||||||
strbuf_addch(&sb, '\\');
|
strbuf_addch(&sb, '\\');
|
||||||
strbuf_addch(&sb, key[i]);
|
strbuf_addch(&sb, key[i]);
|
||||||
}
|
}
|
||||||
strbuf_addstr(&sb, "\"]\n");
|
strbuf_addstr(&sb, "\"]\n");
|
||||||
} else {
|
} else {
|
||||||
strbuf_addf(&sb, "[%.*s]\n", store.baselen, key);
|
strbuf_addf(&sb, "[%.*s]\n", store->baselen, key);
|
||||||
}
|
}
|
||||||
|
|
||||||
return sb;
|
return sb;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t write_section(int fd, const char *key)
|
static ssize_t write_section(int fd, const char *key,
|
||||||
|
const struct config_store_data *store)
|
||||||
{
|
{
|
||||||
struct strbuf sb = store_create_section(key);
|
struct strbuf sb = store_create_section(key, store);
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
|
|
||||||
ret = write_in_full(fd, sb.buf, sb.len);
|
ret = write_in_full(fd, sb.buf, sb.len);
|
||||||
|
@ -2422,11 +2427,12 @@ static ssize_t write_section(int fd, const char *key)
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t write_pair(int fd, const char *key, const char *value)
|
static ssize_t write_pair(int fd, const char *key, const char *value,
|
||||||
|
const struct config_store_data *store)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
int length = strlen(key + store.baselen + 1);
|
int length = strlen(key + store->baselen + 1);
|
||||||
const char *quote = "";
|
const char *quote = "";
|
||||||
struct strbuf sb = STRBUF_INIT;
|
struct strbuf sb = STRBUF_INIT;
|
||||||
|
|
||||||
|
@ -2446,7 +2452,7 @@ static ssize_t write_pair(int fd, const char *key, const char *value)
|
||||||
quote = "\"";
|
quote = "\"";
|
||||||
|
|
||||||
strbuf_addf(&sb, "\t%.*s = %s",
|
strbuf_addf(&sb, "\t%.*s = %s",
|
||||||
length, key + store.baselen + 1, quote);
|
length, key + store->baselen + 1, quote);
|
||||||
|
|
||||||
for (i = 0; value[i]; i++)
|
for (i = 0; value[i]; i++)
|
||||||
switch (value[i]) {
|
switch (value[i]) {
|
||||||
|
@ -2556,6 +2562,9 @@ int git_config_set_multivar_in_file_gently(const char *config_filename,
|
||||||
char *filename_buf = NULL;
|
char *filename_buf = NULL;
|
||||||
char *contents = NULL;
|
char *contents = NULL;
|
||||||
size_t contents_sz;
|
size_t contents_sz;
|
||||||
|
struct config_store_data store;
|
||||||
|
|
||||||
|
memset(&store, 0, sizeof(store));
|
||||||
|
|
||||||
/* parse-key returns negative; flip the sign to feed exit(3) */
|
/* parse-key returns negative; flip the sign to feed exit(3) */
|
||||||
ret = 0 - git_config_parse_key(key, &store.key, &store.baselen);
|
ret = 0 - git_config_parse_key(key, &store.key, &store.baselen);
|
||||||
|
@ -2598,8 +2607,8 @@ int git_config_set_multivar_in_file_gently(const char *config_filename,
|
||||||
}
|
}
|
||||||
|
|
||||||
store.key = (char *)key;
|
store.key = (char *)key;
|
||||||
if (write_section(fd, key) < 0 ||
|
if (write_section(fd, key, &store) < 0 ||
|
||||||
write_pair(fd, key, value) < 0)
|
write_pair(fd, key, value, &store) < 0)
|
||||||
goto write_err_out;
|
goto write_err_out;
|
||||||
} else {
|
} else {
|
||||||
struct stat st;
|
struct stat st;
|
||||||
|
@ -2638,7 +2647,7 @@ int git_config_set_multivar_in_file_gently(const char *config_filename,
|
||||||
* As a side effect, we make sure to transform only a valid
|
* As a side effect, we make sure to transform only a valid
|
||||||
* existing config file.
|
* existing config file.
|
||||||
*/
|
*/
|
||||||
if (git_config_from_file(store_aux, config_filename, NULL)) {
|
if (git_config_from_file(store_aux, config_filename, &store)) {
|
||||||
error("invalid config file %s", config_filename);
|
error("invalid config file %s", config_filename);
|
||||||
free(store.key);
|
free(store.key);
|
||||||
if (store.value_regex != NULL &&
|
if (store.value_regex != NULL &&
|
||||||
|
@ -2722,10 +2731,10 @@ int git_config_set_multivar_in_file_gently(const char *config_filename,
|
||||||
/* write the pair (value == NULL means unset) */
|
/* write the pair (value == NULL means unset) */
|
||||||
if (value != NULL) {
|
if (value != NULL) {
|
||||||
if (store.state == START) {
|
if (store.state == START) {
|
||||||
if (write_section(fd, key) < 0)
|
if (write_section(fd, key, &store) < 0)
|
||||||
goto write_err_out;
|
goto write_err_out;
|
||||||
}
|
}
|
||||||
if (write_pair(fd, key, value) < 0)
|
if (write_pair(fd, key, value, &store) < 0)
|
||||||
goto write_err_out;
|
goto write_err_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2849,7 +2858,8 @@ static int section_name_is_ok(const char *name)
|
||||||
|
|
||||||
/* if new_name == NULL, the section is removed instead */
|
/* if new_name == NULL, the section is removed instead */
|
||||||
static int git_config_copy_or_rename_section_in_file(const char *config_filename,
|
static int git_config_copy_or_rename_section_in_file(const char *config_filename,
|
||||||
const char *old_name, const char *new_name, int copy)
|
const char *old_name,
|
||||||
|
const char *new_name, int copy)
|
||||||
{
|
{
|
||||||
int ret = 0, remove = 0;
|
int ret = 0, remove = 0;
|
||||||
char *filename_buf = NULL;
|
char *filename_buf = NULL;
|
||||||
|
@ -2859,6 +2869,9 @@ static int git_config_copy_or_rename_section_in_file(const char *config_filename
|
||||||
FILE *config_file = NULL;
|
FILE *config_file = NULL;
|
||||||
struct stat st;
|
struct stat st;
|
||||||
struct strbuf copystr = STRBUF_INIT;
|
struct strbuf copystr = STRBUF_INIT;
|
||||||
|
struct config_store_data store;
|
||||||
|
|
||||||
|
memset(&store, 0, sizeof(store));
|
||||||
|
|
||||||
if (new_name && !section_name_is_ok(new_name)) {
|
if (new_name && !section_name_is_ok(new_name)) {
|
||||||
ret = error("invalid section name: %s", new_name);
|
ret = error("invalid section name: %s", new_name);
|
||||||
|
@ -2928,7 +2941,7 @@ static int git_config_copy_or_rename_section_in_file(const char *config_filename
|
||||||
}
|
}
|
||||||
store.baselen = strlen(new_name);
|
store.baselen = strlen(new_name);
|
||||||
if (!copy) {
|
if (!copy) {
|
||||||
if (write_section(out_fd, new_name) < 0) {
|
if (write_section(out_fd, new_name, &store) < 0) {
|
||||||
ret = write_error(get_lock_file_path(&lock));
|
ret = write_error(get_lock_file_path(&lock));
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -2949,7 +2962,7 @@ static int git_config_copy_or_rename_section_in_file(const char *config_filename
|
||||||
output[0] = '\t';
|
output[0] = '\t';
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
copystr = store_create_section(new_name);
|
copystr = store_create_section(new_name, &store);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
remove = 0;
|
remove = 0;
|
||||||
|
|
Loading…
Reference in New Issue