commit: Let the callback of for_each_mergetag return on error

This is yet another patch to be filed under the keyword "libification".

There is one subtle change in behavior here, where a `git log` that has
been asked to show the mergetags would now stop reporting the mergetags
upon the first failure, whereas previously, it would have continued to the
next mergetag, if any.

In practice, that change should not matter, as it is 1) uncommon to
perform octopus merges using multiple tags as merge heads, and 2) when the
user asks to be shown those tags, they really should be there.

Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Johannes Schindelin 2018-04-25 11:54:04 +02:00 committed by Junio C Hamano
parent c5aa6db64f
commit fef461ea5d
4 changed files with 18 additions and 15 deletions

View File

@ -345,7 +345,7 @@ struct check_mergetag_data {
const char **argv; const char **argv;
}; };
static void check_one_mergetag(struct commit *commit, static int check_one_mergetag(struct commit *commit,
struct commit_extra_header *extra, struct commit_extra_header *extra,
void *data) void *data)
{ {
@ -368,20 +368,20 @@ static void check_one_mergetag(struct commit *commit,
if (get_oid(mergetag_data->argv[i], &oid) < 0) if (get_oid(mergetag_data->argv[i], &oid) < 0)
die(_("Not a valid object name: '%s'"), mergetag_data->argv[i]); die(_("Not a valid object name: '%s'"), mergetag_data->argv[i]);
if (!oidcmp(&tag->tagged->oid, &oid)) if (!oidcmp(&tag->tagged->oid, &oid))
return; /* found */ return 0; /* found */
} }
die(_("original commit '%s' contains mergetag '%s' that is discarded; " die(_("original commit '%s' contains mergetag '%s' that is discarded; "
"use --edit instead of --graft"), ref, oid_to_hex(&tag_oid)); "use --edit instead of --graft"), ref, oid_to_hex(&tag_oid));
} }
static void check_mergetags(struct commit *commit, int argc, const char **argv) static int check_mergetags(struct commit *commit, int argc, const char **argv)
{ {
struct check_mergetag_data mergetag_data; struct check_mergetag_data mergetag_data;
mergetag_data.argc = argc; mergetag_data.argc = argc;
mergetag_data.argv = argv; mergetag_data.argv = argv;
for_each_mergetag(check_one_mergetag, commit, &mergetag_data); return for_each_mergetag(check_one_mergetag, commit, &mergetag_data);
} }
static int create_graft(int argc, const char **argv, int force) static int create_graft(int argc, const char **argv, int force)

View File

@ -1288,17 +1288,19 @@ struct commit_extra_header *read_commit_extra_headers(struct commit *commit,
return extra; return extra;
} }
void for_each_mergetag(each_mergetag_fn fn, struct commit *commit, void *data) int for_each_mergetag(each_mergetag_fn fn, struct commit *commit, void *data)
{ {
struct commit_extra_header *extra, *to_free; struct commit_extra_header *extra, *to_free;
int res = 0;
to_free = read_commit_extra_headers(commit, NULL); to_free = read_commit_extra_headers(commit, NULL);
for (extra = to_free; extra; extra = extra->next) { for (extra = to_free; !res && extra; extra = extra->next) {
if (strcmp(extra->key, "mergetag")) if (strcmp(extra->key, "mergetag"))
continue; /* not a merge tag */ continue; /* not a merge tag */
fn(commit, extra, data); res = fn(commit, extra, data);
} }
free_commit_extra_headers(to_free); free_commit_extra_headers(to_free);
return res;
} }
static inline int standard_header_field(const char *field, size_t len) static inline int standard_header_field(const char *field, size_t len)

View File

@ -291,10 +291,10 @@ extern const char *find_commit_header(const char *msg, const char *key,
/* Find the end of the log message, the right place for a new trailer. */ /* Find the end of the log message, the right place for a new trailer. */
extern int ignore_non_trailer(const char *buf, size_t len); extern int ignore_non_trailer(const char *buf, size_t len);
typedef void (*each_mergetag_fn)(struct commit *commit, struct commit_extra_header *extra, typedef int (*each_mergetag_fn)(struct commit *commit, struct commit_extra_header *extra,
void *cb_data); void *cb_data);
extern void for_each_mergetag(each_mergetag_fn fn, struct commit *commit, void *data); extern int for_each_mergetag(each_mergetag_fn fn, struct commit *commit, void *data);
struct merge_remote_desc { struct merge_remote_desc {
struct object *obj; /* the named object, could be a tag */ struct object *obj; /* the named object, could be a tag */

View File

@ -488,7 +488,7 @@ static int is_common_merge(const struct commit *commit)
&& !commit->parents->next->next); && !commit->parents->next->next);
} }
static void show_one_mergetag(struct commit *commit, static int show_one_mergetag(struct commit *commit,
struct commit_extra_header *extra, struct commit_extra_header *extra,
void *data) void *data)
{ {
@ -502,7 +502,7 @@ static void show_one_mergetag(struct commit *commit,
hash_object_file(extra->value, extra->len, type_name(OBJ_TAG), &oid); hash_object_file(extra->value, extra->len, type_name(OBJ_TAG), &oid);
tag = lookup_tag(&oid); tag = lookup_tag(&oid);
if (!tag) if (!tag)
return; /* error message already given */ return -1; /* error message already given */
strbuf_init(&verify_message, 256); strbuf_init(&verify_message, 256);
if (parse_tag_buffer(tag, extra->value, extra->len)) if (parse_tag_buffer(tag, extra->value, extra->len))
@ -536,11 +536,12 @@ static void show_one_mergetag(struct commit *commit,
show_sig_lines(opt, status, verify_message.buf); show_sig_lines(opt, status, verify_message.buf);
strbuf_release(&verify_message); strbuf_release(&verify_message);
return 0;
} }
static void show_mergetag(struct rev_info *opt, struct commit *commit) static int show_mergetag(struct rev_info *opt, struct commit *commit)
{ {
for_each_mergetag(show_one_mergetag, commit, opt); return for_each_mergetag(show_one_mergetag, commit, opt);
} }
void show_log(struct rev_info *opt) void show_log(struct rev_info *opt)