Merge branch 'jk/mailinfo-cleanup'
Code clean-up. * jk/mailinfo-cleanup: mailinfo: factor out some repeated header handling mailinfo: be more liberal with header whitespace mailinfo: simplify parsing of header values mailinfo: treat header values as C strings
This commit is contained in:
commit
d880c3de23
62
mailinfo.c
62
mailinfo.c
@ -346,11 +346,17 @@ static const char *header[MAX_HDR_PARSED] = {
|
|||||||
"From","Subject","Date",
|
"From","Subject","Date",
|
||||||
};
|
};
|
||||||
|
|
||||||
static inline int cmp_header(const struct strbuf *line, const char *hdr)
|
static inline int skip_header(const struct strbuf *line, const char *hdr,
|
||||||
|
const char **outval)
|
||||||
{
|
{
|
||||||
int len = strlen(hdr);
|
const char *val;
|
||||||
return !strncasecmp(line->buf, hdr, len) && line->len > len &&
|
if (!skip_iprefix(line->buf, hdr, &val) ||
|
||||||
line->buf[len] == ':' && isspace(line->buf[len + 1]);
|
*val++ != ':')
|
||||||
|
return 0;
|
||||||
|
while (isspace(*val))
|
||||||
|
val++;
|
||||||
|
*outval = val;
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int is_format_patch_separator(const char *line, int len)
|
static int is_format_patch_separator(const char *line, int len)
|
||||||
@ -543,22 +549,36 @@ release_return:
|
|||||||
mi->input_error = -1;
|
mi->input_error = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Returns true if "line" contains a header matching "hdr", in which case "val"
|
||||||
|
* will contain the value of the header with any RFC2047 B and Q encoding
|
||||||
|
* unwrapped, and optionally normalize the meta information to utf8.
|
||||||
|
*/
|
||||||
|
static int parse_header(const struct strbuf *line,
|
||||||
|
const char *hdr,
|
||||||
|
struct mailinfo *mi,
|
||||||
|
struct strbuf *val)
|
||||||
|
{
|
||||||
|
const char *val_str;
|
||||||
|
|
||||||
|
if (!skip_header(line, hdr, &val_str))
|
||||||
|
return 0;
|
||||||
|
strbuf_addstr(val, val_str);
|
||||||
|
decode_header(mi, val);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
static int check_header(struct mailinfo *mi,
|
static int check_header(struct mailinfo *mi,
|
||||||
const struct strbuf *line,
|
const struct strbuf *line,
|
||||||
struct strbuf *hdr_data[], int overwrite)
|
struct strbuf *hdr_data[], int overwrite)
|
||||||
{
|
{
|
||||||
int i, ret = 0, len;
|
int i, ret = 0;
|
||||||
struct strbuf sb = STRBUF_INIT;
|
struct strbuf sb = STRBUF_INIT;
|
||||||
|
|
||||||
/* search for the interesting parts */
|
/* search for the interesting parts */
|
||||||
for (i = 0; header[i]; i++) {
|
for (i = 0; header[i]; i++) {
|
||||||
int len = strlen(header[i]);
|
if ((!hdr_data[i] || overwrite) &&
|
||||||
if ((!hdr_data[i] || overwrite) && cmp_header(line, header[i])) {
|
parse_header(line, header[i], mi, &sb)) {
|
||||||
/* Unwrap inline B and Q encoding, and optionally
|
|
||||||
* normalize the meta information to utf8.
|
|
||||||
*/
|
|
||||||
strbuf_add(&sb, line->buf + len + 2, line->len - len - 2);
|
|
||||||
decode_header(mi, &sb);
|
|
||||||
handle_header(&hdr_data[i], &sb);
|
handle_header(&hdr_data[i], &sb);
|
||||||
ret = 1;
|
ret = 1;
|
||||||
goto check_header_out;
|
goto check_header_out;
|
||||||
@ -566,26 +586,17 @@ static int check_header(struct mailinfo *mi,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* Content stuff */
|
/* Content stuff */
|
||||||
if (cmp_header(line, "Content-Type")) {
|
if (parse_header(line, "Content-Type", mi, &sb)) {
|
||||||
len = strlen("Content-Type: ");
|
|
||||||
strbuf_add(&sb, line->buf + len, line->len - len);
|
|
||||||
decode_header(mi, &sb);
|
|
||||||
handle_content_type(mi, &sb);
|
handle_content_type(mi, &sb);
|
||||||
ret = 1;
|
ret = 1;
|
||||||
goto check_header_out;
|
goto check_header_out;
|
||||||
}
|
}
|
||||||
if (cmp_header(line, "Content-Transfer-Encoding")) {
|
if (parse_header(line, "Content-Transfer-Encoding", mi, &sb)) {
|
||||||
len = strlen("Content-Transfer-Encoding: ");
|
|
||||||
strbuf_add(&sb, line->buf + len, line->len - len);
|
|
||||||
decode_header(mi, &sb);
|
|
||||||
handle_content_transfer_encoding(mi, &sb);
|
handle_content_transfer_encoding(mi, &sb);
|
||||||
ret = 1;
|
ret = 1;
|
||||||
goto check_header_out;
|
goto check_header_out;
|
||||||
}
|
}
|
||||||
if (cmp_header(line, "Message-Id")) {
|
if (parse_header(line, "Message-Id", mi, &sb)) {
|
||||||
len = strlen("Message-Id: ");
|
|
||||||
strbuf_add(&sb, line->buf + len, line->len - len);
|
|
||||||
decode_header(mi, &sb);
|
|
||||||
if (mi->add_message_id)
|
if (mi->add_message_id)
|
||||||
mi->message_id = strbuf_detach(&sb, NULL);
|
mi->message_id = strbuf_detach(&sb, NULL);
|
||||||
ret = 1;
|
ret = 1;
|
||||||
@ -606,8 +617,9 @@ static int is_inbody_header(const struct mailinfo *mi,
|
|||||||
const struct strbuf *line)
|
const struct strbuf *line)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
const char *val;
|
||||||
for (i = 0; header[i]; i++)
|
for (i = 0; header[i]; i++)
|
||||||
if (!mi->s_hdr_data[i] && cmp_header(line, header[i]))
|
if (!mi->s_hdr_data[i] && skip_header(line, header[i], &val))
|
||||||
return 1;
|
return 1;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -213,4 +213,19 @@ test_expect_failure 'mailinfo -b separated double [PATCH]' '
|
|||||||
test z"$subj" = z"Subject: [other] message"
|
test z"$subj" = z"Subject: [other] message"
|
||||||
'
|
'
|
||||||
|
|
||||||
|
test_expect_success 'mailinfo handles unusual header whitespace' '
|
||||||
|
git mailinfo /dev/null /dev/null >actual <<-\EOF &&
|
||||||
|
From:Real Name <user@example.com>
|
||||||
|
Subject: extra spaces
|
||||||
|
EOF
|
||||||
|
|
||||||
|
cat >expect <<-\EOF &&
|
||||||
|
Author: Real Name
|
||||||
|
Email: user@example.com
|
||||||
|
Subject: extra spaces
|
||||||
|
|
||||||
|
EOF
|
||||||
|
test_cmp expect actual
|
||||||
|
'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
Loading…
Reference in New Issue
Block a user