masqmail
changeset 117:5ec5e6637049
added server-side SMTP SIZE support (patch by Paolo)
``SIZE 0'' (= unlimited) is currently not supported
client-side support was already implemented
author | meillo@marmaro.de |
---|---|
date | Thu, 01 Jul 2010 13:08:53 +0200 (2010-07-01) |
parents | ddc8041fdee1 |
children | 7f1f364c2a29 |
files | man/masqmail.conf.5 src/accept.c src/conf.c src/masqmail.c src/masqmail.h src/smtp_in.c |
diffstat | 6 files changed, 77 insertions(+), 8 deletions(-) [+] |
line diff
1.1 --- a/man/masqmail.conf.5 Wed Jun 30 15:45:34 2010 +0200 1.2 +++ b/man/masqmail.conf.5 Thu Jul 01 13:08:53 2010 +0200 1.3 @@ -1,4 +1,4 @@ 1.4 -.TH masqmail.conf 5 2010-06-30 masqmail-0.2.25 "File Formats" 1.5 +.TH masqmail.conf 5 2010-07-01 masqmail-0.2.25 "File Formats" 1.6 1.7 .SH NAME 1.8 masqmail.conf \- masqmail configuration file 1.9 @@ -482,6 +482,18 @@ 1.10 For example you can feed your mails into a program like hypermail 1.11 for archiving purpose by placing an appropriate pipe command in masqmail.alias 1.12 1.13 +.TP 1.14 +\fBmax_msg_size\fR = \fIbytes\fR 1.15 + 1.16 +This option sets the maximum size in bytes masqmail will accept for delivery. 1.17 +This value is advertised to the SMTP client by the `SIZE' message during SMTP 1.18 +session setup. 1.19 +Clients pretending to send, or actually send, 1.20 +more than \fIbytes\fR will get a 552 error message. 1.21 + 1.22 +Default is 104857600 (= 100MB). 1.23 +(This should be sufficient for most cases.) 1.24 + 1.25 1.26 .SH AUTHOR 1.27
2.1 --- a/src/accept.c Wed Jun 30 15:45:34 2010 +0200 2.2 +++ b/src/accept.c Thu Jul 01 13:08:53 2010 +0200 2.3 @@ -155,6 +155,13 @@ 2.4 line_cnt++; 2.5 } 2.6 } 2.7 + if (data_size > conf.max_msg_size) { 2.8 + DEBUG(4) debugf("accept_message_stream(): " 2.9 + "received %d bytes (conf.max_msg_size=%d)\n", 2.10 + data_size, conf.max_msg_size); 2.11 + return AERR_SIZE; 2.12 + } 2.13 + 2.14 } 2.15 2.16 if (msg->data_list != NULL)
3.1 --- a/src/conf.c Wed Jun 30 15:45:34 2010 +0200 3.2 +++ b/src/conf.c Thu Jul 01 13:08:53 2010 +0200 3.3 @@ -431,6 +431,7 @@ 3.4 conf.do_relay = TRUE; 3.5 conf.alias_local_cmp = strcmp; 3.6 conf.max_defer_time = 86400 * 4; /* 4 days */ 3.7 + conf.max_msg_size = 100*1024*1024; /* in bytes (100MB are probably enough) */ 3.8 3.9 if ((in = fopen(filename, "r")) == NULL) { 3.10 fprintf(stderr, "could not open config file %s: %s\n", filename, strerror(errno)); 3.11 @@ -581,7 +582,11 @@ 3.12 conf.max_defer_time = ival; 3.13 } else if (strcmp(lval, "log_user") == 0) 3.14 conf.log_user = g_strdup(rval); 3.15 - 3.16 + else if(strcmp(lval, "max_msg_size") == 0) { 3.17 + conf.max_msg_size = atol(rval); 3.18 + DEBUG(6) fprintf(stderr,"rval=%s, conf.max_msg_size=%ld\n", 3.19 + rval, conf.max_msg_size); 3.20 + } 3.21 else 3.22 fprintf(stderr, "var '%s' not (yet) known, ignored\n", lval); 3.23 }
4.1 --- a/src/masqmail.c Wed Jun 30 15:45:34 2010 +0200 4.2 +++ b/src/masqmail.c Thu Jul 01 13:08:53 2010 +0200 4.3 @@ -317,6 +317,9 @@ 4.4 case AERR_NORCPT: 4.5 fprintf(stderr, "no recipients.\n"); 4.6 exit(EXIT_FAILURE); 4.7 + case AERR_SIZE: 4.8 + fprintf(stderr, "max message size exceeded.\n"); 4.9 + exit(EXIT_FAILURE); 4.10 default: 4.11 /* should never happen: */ 4.12 fprintf(stderr, "Unknown error (%d)\r\n", err);
5.1 --- a/src/masqmail.h Wed Jun 30 15:45:34 2010 +0200 5.2 +++ b/src/masqmail.h Thu Jul 01 13:08:53 2010 +0200 5.3 @@ -174,6 +174,10 @@ 5.4 5.5 guint remote_port; 5.6 5.7 + /* ANSI C defines unsigned long to be at least 32bit 5.8 + i.e. ca. 4GB max; that should be enough. */ 5.9 + gulong max_msg_size; 5.10 + 5.11 gboolean do_save_envelope_to; 5.12 5.13 gboolean defer_all; 5.14 @@ -332,6 +336,7 @@ 5.15 AERR_SYNTAX, 5.16 AERR_NOSPOOL, 5.17 AERR_NORCPT, 5.18 + AERR_SIZE, /* max msg size exeeded (SMTP SIZE) */ 5.19 AERR_UNKNOWN 5.20 } accept_error; 5.21
6.1 --- a/src/smtp_in.c Wed Jun 30 15:45:34 2010 +0200 6.2 +++ b/src/smtp_in.c Thu Jul 01 13:08:53 2010 +0200 6.3 @@ -55,8 +55,29 @@ 6.4 return SMTP_ERROR; 6.5 } 6.6 6.7 +static gboolean 6.8 +get_size(gchar *line, unsigned long *msize) { 6.9 + gchar *s = NULL; 6.10 + 6.11 + /* hope we need not to handle cases like SiZe= ...*/ 6.12 + s = strstr(line, "SIZE="); 6.13 + if (!s) { 6.14 + /* try it in lowercase too */ 6.15 + if (!(s = strstr(line, "size="))) { 6.16 + return FALSE; 6.17 + } 6.18 + } 6.19 + s += 5; 6.20 + *msize = atol(s); 6.21 + DEBUG(5) debugf("get_size(): line=%s, msize=%ld\n", line, *msize); 6.22 + 6.23 + return TRUE; 6.24 +} 6.25 + 6.26 + 6.27 /* this is a quick hack: we expect the address to be syntactically correct 6.28 - and containing the mailbox only: 6.29 + and containing the mailbox only, though we first check for size in 6.30 + smtp_in(). 6.31 */ 6.32 static gboolean 6.33 get_address(gchar * line, gchar * addr) 6.34 @@ -135,6 +156,7 @@ 6.35 message *msg = NULL; 6.36 smtp_connection *psc; 6.37 int len; 6.38 + unsigned long size, msize; 6.39 6.40 DEBUG(5) debugf("smtp_in entered, remote_host = %s\n", remote_host); 6.41 6.42 @@ -168,7 +190,7 @@ 6.43 6.44 if (psc->prot == PROT_ESMTP) { 6.45 smtp_printf(out, "250-%s Nice to meet you with ESMTP\r\n", conf.host_name); 6.46 - /* not yet: fprintf(out, "250-SIZE\r\n"); */ 6.47 + smtp_printf(out, "250-SIZE %d\r\n", conf.max_msg_size); 6.48 smtp_printf(out, "250-PIPELINING\r\n" "250 HELP\r\n"); 6.49 } else { 6.50 smtp_printf(out, "250 %s pretty old mailer, huh?\r\n", conf.host_name); 6.51 @@ -176,6 +198,15 @@ 6.52 break; 6.53 6.54 case SMTP_MAIL_FROM: 6.55 + if (get_size(buffer, &msize)) { 6.56 + DEBUG(5) debugf("smtp_in(): get_size: msize=%ld, conf.mms=%d\n", 6.57 + msize, conf.max_msg_size); 6.58 + if (msize > conf.max_msg_size) { 6.59 + smtp_printf(out, "552 Message size exceeds fixed limit.\r\n"); 6.60 + break; 6.61 + } 6.62 + } 6.63 + 6.64 { 6.65 gchar buf[MAX_ADDRESS]; 6.66 address *addr; 6.67 @@ -278,12 +309,18 @@ 6.68 6.69 err = accept_message(in, msg, conf.do_save_envelope_to ? ACC_SAVE_ENVELOPE_TO : 0); 6.70 if (err != AERR_OK) { 6.71 - if (err == AERR_TIMEOUT || err == AERR_EOF) { 6.72 + switch (err) { 6.73 + case AERR_TIMEOUT: 6.74 + case AERR_EOF: 6.75 + return; 6.76 + case AERR_SIZE: 6.77 + smtp_printf(out, "552 Error: message too large.\r\n"); 6.78 + return; 6.79 + default: 6.80 + /* should never happen: */ 6.81 + smtp_printf(out, "451 Unknown error\r\n"); 6.82 return; 6.83 } 6.84 - /* should never happen: */ 6.85 - smtp_printf(out, "451 Unknown error\r\n"); 6.86 - return; 6.87 } 6.88 6.89