723 lines
15 KiB
C
723 lines
15 KiB
C
#include "readmsg-common.h"
|
|
|
|
#include <sys/stat.h>
|
|
#ifndef _MSC_VER
|
|
# include <sys/mman.h>
|
|
# include <unistd.h>
|
|
#endif
|
|
#include <fcntl.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
/* returns TRUE is given MIME part is a text part */
|
|
|
|
int etpan_mime_is_text(struct mailmime * build_info)
|
|
{
|
|
if (build_info->mm_type == MAILMIME_SINGLE) {
|
|
if (build_info->mm_content_type != NULL) {
|
|
if (build_info->mm_content_type->ct_type->tp_type ==
|
|
MAILMIME_TYPE_DISCRETE_TYPE) {
|
|
if (build_info->mm_content_type->ct_type->tp_data.tp_discrete_type->dt_type ==
|
|
MAILMIME_DISCRETE_TYPE_TEXT)
|
|
return 1;
|
|
}
|
|
}
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* display content type */
|
|
|
|
int show_part_info(FILE * f,
|
|
struct mailmime_single_fields * mime_fields,
|
|
struct mailmime_content * content)
|
|
{
|
|
char * description;
|
|
char * filename;
|
|
int col;
|
|
int r;
|
|
|
|
description = mime_fields->fld_description;
|
|
filename = mime_fields->fld_disposition_filename;
|
|
|
|
col = 0;
|
|
|
|
r = fprintf(f, " [ Part ");
|
|
if (r < 0)
|
|
goto err;
|
|
|
|
if (content != NULL) {
|
|
r = mailmime_content_type_write(f, &col, content);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
}
|
|
|
|
if (filename != NULL) {
|
|
r = fprintf(f, " (%s)", filename);
|
|
if (r < 0)
|
|
goto err;
|
|
}
|
|
|
|
if (description != NULL) {
|
|
r = fprintf(f, " : %s", description);
|
|
if (r < 0)
|
|
goto err;
|
|
}
|
|
|
|
r = fprintf(f, " ]\n\n");
|
|
if (r < 0)
|
|
goto err;
|
|
|
|
return NO_ERROR;
|
|
|
|
err:
|
|
return ERROR_FILE;
|
|
}
|
|
|
|
/*
|
|
fetch the data of the mailmime_data structure whether it is a file
|
|
or a string.
|
|
|
|
data must be freed with mmap_string_unref()
|
|
*/
|
|
|
|
#if 0
|
|
static int fetch_data(struct mailmime_data * data,
|
|
char ** result, size_t * result_len)
|
|
{
|
|
int fd;
|
|
int r;
|
|
char * text;
|
|
struct stat buf;
|
|
int res;
|
|
MMAPString * mmapstr;
|
|
|
|
switch (data->dt_type) {
|
|
case MAILMIME_DATA_TEXT:
|
|
mmapstr = mmap_string_new_len(data->dt_data.dt_text.dt_data,
|
|
data->dt_data.dt_text.dt_length);
|
|
if (mmapstr == NULL) {
|
|
res = ERROR_MEMORY;
|
|
goto err;
|
|
}
|
|
|
|
* result = mmapstr->str;
|
|
* result_len = mmapstr->len;
|
|
|
|
return NO_ERROR;
|
|
|
|
case MAILMIME_DATA_FILE:
|
|
fd = open(data->dt_data.dt_filename, O_RDONLY);
|
|
if (fd < 0) {
|
|
res = ERROR_FILE;
|
|
goto err;
|
|
}
|
|
|
|
r = fstat(fd, &buf);
|
|
if (r < 0) {
|
|
res = ERROR_FILE;
|
|
goto close;
|
|
}
|
|
|
|
if (buf.st_size != 0) {
|
|
text = mmap(NULL, buf.st_size, PROT_READ, MAP_SHARED, fd, 0);
|
|
if (text == (char *)MAP_FAILED) {
|
|
res = ERROR_FILE;
|
|
goto close;
|
|
}
|
|
|
|
mmapstr = mmap_string_new_len(text, buf.st_size);
|
|
if (mmapstr == NULL) {
|
|
res = r;
|
|
goto unmap;
|
|
}
|
|
|
|
munmap(text, buf.st_size);
|
|
}
|
|
else {
|
|
mmapstr = mmap_string_new("");
|
|
if (mmapstr == NULL) {
|
|
res = r;
|
|
goto close;
|
|
}
|
|
}
|
|
|
|
close(fd);
|
|
|
|
* result = mmapstr->str;
|
|
* result_len = mmapstr->len;
|
|
|
|
return NO_ERROR;
|
|
|
|
default:
|
|
return ERROR_INVAL;
|
|
}
|
|
|
|
unmap:
|
|
munmap(text, buf.st_size);
|
|
close:
|
|
close(fd);
|
|
err:
|
|
return res;
|
|
}
|
|
#endif
|
|
|
|
/* fetch message and decode if it is base64 or quoted-printable */
|
|
|
|
int etpan_fetch_message(mailmessage * msg_info,
|
|
struct mailmime * mime_part,
|
|
struct mailmime_single_fields * fields,
|
|
char ** result, size_t * result_len)
|
|
{
|
|
char * data;
|
|
size_t len;
|
|
int r;
|
|
int encoding;
|
|
char * decoded;
|
|
size_t decoded_len;
|
|
size_t cur_token;
|
|
int res;
|
|
int encoded;
|
|
|
|
encoded = 0;
|
|
|
|
r = mailmessage_fetch_section(msg_info,
|
|
mime_part, &data, &len);
|
|
if (r != MAIL_NO_ERROR) {
|
|
res = ERROR_FETCH;
|
|
goto err;
|
|
}
|
|
|
|
encoded = 1;
|
|
|
|
/* decode message */
|
|
|
|
if (encoded) {
|
|
if (fields->fld_encoding != NULL)
|
|
encoding = fields->fld_encoding->enc_type;
|
|
else
|
|
encoding = MAILMIME_MECHANISM_8BIT;
|
|
}
|
|
else {
|
|
encoding = MAILMIME_MECHANISM_8BIT;
|
|
}
|
|
|
|
cur_token = 0;
|
|
r = mailmime_part_parse(data, len, &cur_token,
|
|
encoding, &decoded, &decoded_len);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
res = ERROR_FETCH;
|
|
goto free;
|
|
}
|
|
|
|
mailmessage_fetch_result_free(msg_info, data);
|
|
|
|
* result = decoded;
|
|
* result_len = decoded_len;
|
|
|
|
return NO_ERROR;
|
|
|
|
free:
|
|
mailmessage_fetch_result_free(msg_info, data);
|
|
err:
|
|
return res;
|
|
}
|
|
|
|
|
|
/* fetch fields */
|
|
|
|
struct mailimf_fields * fetch_fields(mailmessage * msg_info,
|
|
struct mailmime * mime)
|
|
{
|
|
char * data;
|
|
size_t len;
|
|
int r;
|
|
size_t cur_token;
|
|
struct mailimf_fields * fields;
|
|
|
|
r = mailmessage_fetch_section_header(msg_info, mime, &data, &len);
|
|
if (r != MAIL_NO_ERROR)
|
|
return NULL;
|
|
|
|
cur_token = 0;
|
|
r = mailimf_fields_parse(data, len, &cur_token, &fields);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
mailmessage_fetch_result_free(msg_info, data);
|
|
return NULL;
|
|
}
|
|
|
|
mailmessage_fetch_result_free(msg_info, data);
|
|
|
|
return fields;
|
|
}
|
|
|
|
|
|
|
|
#define MAX_MAIL_COL 72
|
|
|
|
/* write decoded mailbox */
|
|
|
|
static int
|
|
etpan_mailbox_write(FILE * f, int * col,
|
|
struct mailimf_mailbox * mb)
|
|
{
|
|
int r;
|
|
|
|
if (* col > 1) {
|
|
|
|
if (* col + strlen(mb->mb_addr_spec) >= MAX_MAIL_COL) {
|
|
r = mailimf_string_write(f, col, "\r\n ", 3);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
* col = 1;
|
|
}
|
|
}
|
|
|
|
if (mb->mb_display_name) {
|
|
char * decoded_from;
|
|
size_t cur_token;
|
|
|
|
cur_token = 0;
|
|
r = mailmime_encoded_phrase_parse(DEST_CHARSET,
|
|
mb->mb_display_name, strlen(mb->mb_display_name),
|
|
&cur_token, DEST_CHARSET,
|
|
&decoded_from);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
decoded_from = strdup(mb->mb_display_name);
|
|
if (decoded_from == NULL)
|
|
return ERROR_MEMORY;
|
|
}
|
|
|
|
r = mailimf_quoted_string_write(f, col, decoded_from,
|
|
strlen(decoded_from));
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
free(decoded_from);
|
|
return ERROR_FILE;
|
|
}
|
|
|
|
if (* col > 1) {
|
|
|
|
if (* col + strlen(decoded_from) + 3 >= MAX_MAIL_COL) {
|
|
r = mailimf_string_write(f, col, "\r\n ", 3);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
free(decoded_from);
|
|
return r;
|
|
}
|
|
* col = 1;
|
|
}
|
|
}
|
|
|
|
free(decoded_from);
|
|
|
|
r = mailimf_string_write(f, col, " <", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
|
|
r = mailimf_string_write(f, col,
|
|
mb->mb_addr_spec, strlen(mb->mb_addr_spec));
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
|
|
r = mailimf_string_write(f, col, ">", 1);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
}
|
|
else {
|
|
r = mailimf_string_write(f, col,
|
|
mb->mb_addr_spec, strlen(mb->mb_addr_spec));
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
}
|
|
|
|
|
|
return NO_ERROR;
|
|
|
|
}
|
|
|
|
/* write decoded mailbox list */
|
|
|
|
int
|
|
etpan_mailbox_list_write(FILE * f, int * col,
|
|
struct mailimf_mailbox_list * mb_list)
|
|
{
|
|
clistiter * cur;
|
|
int r;
|
|
int first;
|
|
|
|
first = 1;
|
|
|
|
for(cur = clist_begin(mb_list->mb_list) ; cur != NULL ;
|
|
cur = clist_next(cur)) {
|
|
struct mailimf_mailbox * mb;
|
|
|
|
mb = cur->data;
|
|
|
|
if (!first) {
|
|
r = mailimf_string_write(f, col, ", ", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
}
|
|
else {
|
|
first = 0;
|
|
}
|
|
|
|
r = etpan_mailbox_write(f, col, mb);
|
|
if (r != NO_ERROR)
|
|
return r;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
/* write decoded group */
|
|
|
|
static int
|
|
etpan_group_write(FILE * f, int * col,
|
|
struct mailimf_group * group)
|
|
{
|
|
int r;
|
|
|
|
r = mailimf_string_write(f, col, group->grp_display_name,
|
|
strlen(group->grp_display_name));
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
|
|
r = mailimf_string_write(f, col, ": ", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
|
|
if (group->grp_mb_list != NULL) {
|
|
r = etpan_mailbox_list_write(f, col, group->grp_mb_list);
|
|
if (r != NO_ERROR)
|
|
return r;
|
|
}
|
|
|
|
r = mailimf_string_write(f, col, ";", 1);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
/* write decoded address */
|
|
|
|
int
|
|
etpan_address_write(FILE * f, int * col,
|
|
struct mailimf_address * addr)
|
|
{
|
|
int r;
|
|
|
|
switch(addr->ad_type) {
|
|
case MAILIMF_ADDRESS_MAILBOX:
|
|
r = etpan_mailbox_write(f, col, addr->ad_data.ad_mailbox);
|
|
if (r != NO_ERROR)
|
|
return r;
|
|
|
|
break;
|
|
|
|
case MAILIMF_ADDRESS_GROUP:
|
|
r = etpan_group_write(f, col, addr->ad_data.ad_group);
|
|
if (r != NO_ERROR)
|
|
return r;
|
|
|
|
break;
|
|
}
|
|
|
|
return MAILIMF_NO_ERROR;
|
|
}
|
|
|
|
/* write decoded address list */
|
|
|
|
int
|
|
etpan_address_list_write(FILE * f, int * col,
|
|
struct mailimf_address_list * addr_list)
|
|
{
|
|
clistiter * cur;
|
|
int r;
|
|
int first;
|
|
|
|
first = 1;
|
|
|
|
for(cur = clist_begin(addr_list->ad_list) ; cur != NULL ;
|
|
cur = clist_next(cur)) {
|
|
struct mailimf_address * addr;
|
|
|
|
addr = clist_content(cur);
|
|
|
|
if (!first) {
|
|
r = mailimf_string_write(f, col, ", ", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
return ERROR_FILE;
|
|
}
|
|
else {
|
|
first = 0;
|
|
}
|
|
|
|
r = etpan_address_write(f, col, addr);
|
|
if (r != NO_ERROR)
|
|
return r;
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
/* write decoded subject */
|
|
|
|
static int etpan_subject_write(FILE * f, int * col,
|
|
char * subject)
|
|
{
|
|
int r;
|
|
char * decoded_subject;
|
|
size_t cur_token;
|
|
|
|
r = mailimf_string_write(f, col, "Subject: ", 9);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
return ERROR_FILE;
|
|
}
|
|
|
|
cur_token = 0;
|
|
r = mailmime_encoded_phrase_parse(DEST_CHARSET,
|
|
subject, strlen(subject),
|
|
&cur_token, DEST_CHARSET,
|
|
&decoded_subject);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
decoded_subject = strdup(subject);
|
|
if (decoded_subject == NULL)
|
|
return ERROR_MEMORY;
|
|
}
|
|
|
|
r = mailimf_string_write(f, col, decoded_subject, strlen(decoded_subject));
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
free(decoded_subject);
|
|
return ERROR_FILE;
|
|
}
|
|
|
|
free(decoded_subject);
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR) {
|
|
return ERROR_FILE;
|
|
}
|
|
* col = 0;
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
/* write decoded fields */
|
|
|
|
int fields_write(FILE * f, int * col,
|
|
struct mailimf_fields * fields)
|
|
{
|
|
clistiter * cur;
|
|
int r;
|
|
|
|
for(cur = clist_begin(fields->fld_list) ; cur != NULL ;
|
|
cur = clist_next(cur)) {
|
|
struct mailimf_field * field;
|
|
|
|
field = clist_content(cur);
|
|
|
|
switch (field->fld_type) {
|
|
case MAILIMF_FIELD_FROM:
|
|
r = mailimf_string_write(f, col, "From: ", 6);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_mailbox_list_write(f, col,
|
|
field->fld_data.fld_from->frm_mb_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
|
|
case MAILIMF_FIELD_REPLY_TO:
|
|
r = mailimf_string_write(f, col, "Reply-To: ", 10);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_reply_to->rt_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
|
|
case MAILIMF_FIELD_TO:
|
|
r = mailimf_string_write(f, col, "To: ", 4);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_to->to_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
|
|
case MAILIMF_FIELD_CC:
|
|
r = mailimf_string_write(f, col, "Cc: ", 4);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_cc->cc_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
|
|
case MAILIMF_FIELD_BCC:
|
|
r = mailimf_string_write(f, col, "Bcc: ", 10);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
if (field->fld_data.fld_bcc->bcc_addr_list != NULL) {
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_bcc->bcc_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
}
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
|
|
case MAILIMF_FIELD_SUBJECT:
|
|
r = etpan_subject_write(f, col, field->fld_data.fld_subject->sbj_value);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
break;
|
|
|
|
case MAILIMF_FIELD_RESENT_FROM:
|
|
r = mailimf_string_write(f, col, "Resent-From: ", 13);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_mailbox_list_write(f, col,
|
|
field->fld_data.fld_resent_from->frm_mb_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
break;
|
|
|
|
case MAILIMF_FIELD_RESENT_TO:
|
|
r = mailimf_string_write(f, col, "Resent-To: ", 11);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_resent_to->to_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
case MAILIMF_FIELD_RESENT_CC:
|
|
r = mailimf_string_write(f, col, "Resent-Cc: ", 11);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_resent_cc->cc_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
case MAILIMF_FIELD_RESENT_BCC:
|
|
r = mailimf_string_write(f, col, "Resent-Bcc: ", 12);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
|
|
if (field->fld_data.fld_resent_bcc->bcc_addr_list != NULL) {
|
|
r = etpan_address_list_write(f, col,
|
|
field->fld_data.fld_resent_bcc->bcc_addr_list);
|
|
if (r != NO_ERROR)
|
|
goto err;
|
|
}
|
|
|
|
r = mailimf_string_write(f, col, "\r\n", 2);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
* col = 0;
|
|
|
|
break;
|
|
|
|
case MAILIMF_FIELD_ORIG_DATE:
|
|
case MAILIMF_FIELD_RESENT_DATE:
|
|
r = mailimf_field_write(f, col, field);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
break;
|
|
|
|
case MAILIMF_FIELD_OPTIONAL_FIELD:
|
|
if ((strcasecmp(field->fld_data.fld_optional_field->fld_name,
|
|
"X-Mailer") == 0)
|
|
|| (strncasecmp(field->fld_data.fld_optional_field->fld_name,
|
|
"Resent-", 7) == 0)
|
|
|| (strcasecmp(field->fld_data.fld_optional_field->fld_name,
|
|
"Newsgroups") == 0)
|
|
|| (strcasecmp(field->fld_data.fld_optional_field->fld_name,
|
|
"Followup-To") == 0)
|
|
|| (strcasecmp(field->fld_data.fld_optional_field->fld_name,
|
|
"User-Agent") == 0)) {
|
|
r = mailimf_field_write(f, col, field);
|
|
if (r != MAILIMF_NO_ERROR)
|
|
goto err;
|
|
}
|
|
break;
|
|
|
|
case MAILIMF_FIELD_MESSAGE_ID:
|
|
case MAILIMF_FIELD_SENDER:
|
|
case MAILIMF_FIELD_IN_REPLY_TO:
|
|
case MAILIMF_FIELD_REFERENCES:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return NO_ERROR;
|
|
|
|
err:
|
|
return ERROR_FILE;
|
|
}
|