Merge branch 'jc/am-read-author-file'
Extract a small helper out of the function that reads the authors script file "git am" internally uses. * jc/am-read-author-file: am: refactor read_author_script()
This commit is contained in:
commit
87f5de387c
101
builtin/am.c
101
builtin/am.c
@ -28,6 +28,7 @@
|
|||||||
#include "rerere.h"
|
#include "rerere.h"
|
||||||
#include "prompt.h"
|
#include "prompt.h"
|
||||||
#include "mailinfo.h"
|
#include "mailinfo.h"
|
||||||
|
#include "string-list.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns 1 if the file is empty or does not exist, 0 otherwise.
|
* Returns 1 if the file is empty or does not exist, 0 otherwise.
|
||||||
@ -258,38 +259,29 @@ static int read_state_file(struct strbuf *sb, const struct am_state *state,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reads a KEY=VALUE shell variable assignment from `fp`, returning the VALUE
|
* Take a series of KEY='VALUE' lines where VALUE part is
|
||||||
* as a newly-allocated string. VALUE must be a quoted string, and the KEY must
|
* sq-quoted, and append <KEY, VALUE> at the end of the string list
|
||||||
* match `key`. Returns NULL on failure.
|
|
||||||
*
|
|
||||||
* This is used by read_author_script() to read the GIT_AUTHOR_* variables from
|
|
||||||
* the author-script.
|
|
||||||
*/
|
*/
|
||||||
static char *read_shell_var(FILE *fp, const char *key)
|
static int parse_key_value_squoted(char *buf, struct string_list *list)
|
||||||
{
|
{
|
||||||
struct strbuf sb = STRBUF_INIT;
|
while (*buf) {
|
||||||
const char *str;
|
struct string_list_item *item;
|
||||||
|
char *np;
|
||||||
|
char *cp = strchr(buf, '=');
|
||||||
|
if (!cp)
|
||||||
|
return -1;
|
||||||
|
np = strchrnul(cp, '\n');
|
||||||
|
*cp++ = '\0';
|
||||||
|
item = string_list_append(list, buf);
|
||||||
|
|
||||||
if (strbuf_getline_lf(&sb, fp))
|
buf = np + (*np == '\n');
|
||||||
goto fail;
|
*np = '\0';
|
||||||
|
cp = sq_dequote(cp);
|
||||||
if (!skip_prefix(sb.buf, key, &str))
|
if (!cp)
|
||||||
goto fail;
|
return -1;
|
||||||
|
item->util = xstrdup(cp);
|
||||||
if (!skip_prefix(str, "=", &str))
|
}
|
||||||
goto fail;
|
return 0;
|
||||||
|
|
||||||
strbuf_remove(&sb, 0, str - sb.buf);
|
|
||||||
|
|
||||||
str = sq_dequote(sb.buf);
|
|
||||||
if (!str)
|
|
||||||
goto fail;
|
|
||||||
|
|
||||||
return strbuf_detach(&sb, NULL);
|
|
||||||
|
|
||||||
fail:
|
|
||||||
strbuf_release(&sb);
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -311,44 +303,39 @@ fail:
|
|||||||
static int read_author_script(struct am_state *state)
|
static int read_author_script(struct am_state *state)
|
||||||
{
|
{
|
||||||
const char *filename = am_path(state, "author-script");
|
const char *filename = am_path(state, "author-script");
|
||||||
FILE *fp;
|
struct strbuf buf = STRBUF_INIT;
|
||||||
|
struct string_list kv = STRING_LIST_INIT_DUP;
|
||||||
|
int retval = -1; /* assume failure */
|
||||||
|
int fd;
|
||||||
|
|
||||||
assert(!state->author_name);
|
assert(!state->author_name);
|
||||||
assert(!state->author_email);
|
assert(!state->author_email);
|
||||||
assert(!state->author_date);
|
assert(!state->author_date);
|
||||||
|
|
||||||
fp = fopen(filename, "r");
|
fd = open(filename, O_RDONLY);
|
||||||
if (!fp) {
|
if (fd < 0) {
|
||||||
if (errno == ENOENT)
|
if (errno == ENOENT)
|
||||||
return 0;
|
return 0;
|
||||||
die_errno(_("could not open '%s' for reading"), filename);
|
die_errno(_("could not open '%s' for reading"), filename);
|
||||||
}
|
}
|
||||||
|
strbuf_read(&buf, fd, 0);
|
||||||
|
close(fd);
|
||||||
|
if (parse_key_value_squoted(buf.buf, &kv))
|
||||||
|
goto finish;
|
||||||
|
|
||||||
state->author_name = read_shell_var(fp, "GIT_AUTHOR_NAME");
|
if (kv.nr != 3 ||
|
||||||
if (!state->author_name) {
|
strcmp(kv.items[0].string, "GIT_AUTHOR_NAME") ||
|
||||||
fclose(fp);
|
strcmp(kv.items[1].string, "GIT_AUTHOR_EMAIL") ||
|
||||||
return -1;
|
strcmp(kv.items[2].string, "GIT_AUTHOR_DATE"))
|
||||||
}
|
goto finish;
|
||||||
|
state->author_name = kv.items[0].util;
|
||||||
state->author_email = read_shell_var(fp, "GIT_AUTHOR_EMAIL");
|
state->author_email = kv.items[1].util;
|
||||||
if (!state->author_email) {
|
state->author_date = kv.items[2].util;
|
||||||
fclose(fp);
|
retval = 0;
|
||||||
return -1;
|
finish:
|
||||||
}
|
string_list_clear(&kv, !!retval);
|
||||||
|
strbuf_release(&buf);
|
||||||
state->author_date = read_shell_var(fp, "GIT_AUTHOR_DATE");
|
return retval;
|
||||||
if (!state->author_date) {
|
|
||||||
fclose(fp);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fgetc(fp) != EOF) {
|
|
||||||
fclose(fp);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
fclose(fp);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Loading…
Reference in New Issue
Block a user