summaryrefslogtreecommitdiff
path: root/win32/sendmail.c
diff options
context:
space:
mode:
authorMarkus Fischer <mfischer@php.net>2002-06-05 17:08:38 +0000
committerMarkus Fischer <mfischer@php.net>2002-06-05 17:08:38 +0000
commit6c6ac37be47146952212026974532adc6c6a504a (patch)
tree5c8cc499f6b8764ebf540ea7d69165cbd281529a /win32/sendmail.c
parent3fb8db4ca8456ecc21955d22c52909397341bc7e (diff)
downloadphp-git-6c6ac37be47146952212026974532adc6c6a504a.tar.gz
- Headers are now rewritten to always have \r\n line endings for SMTP.
Also automatically removes superflous line breaks at the start and end of the header. # http://cr.yp.to/docs/smtplf.html is a good read about this topic.
Diffstat (limited to 'win32/sendmail.c')
-rw-r--r--win32/sendmail.c97
1 files changed, 92 insertions, 5 deletions
diff --git a/win32/sendmail.c b/win32/sendmail.c
index 3043e87e28..3ef139e7d4 100644
--- a/win32/sendmail.c
+++ b/win32/sendmail.c
@@ -31,6 +31,10 @@
#include "sendmail.h"
#include "php_ini.h"
+#if HAVE_PCRE || HAVE_BUNDLED_PCRE
+#include "ext/pcre/php_pcre.h"
+#endif
+
/*
extern int _daylight;
extern long _timezone;
@@ -100,10 +104,78 @@ static char *ErrorMessages[] =
{"Bad Mail Host"},
{"Bad Message File"},
{"\"sendmail_from\" not set in php.ini or custom \"From:\" header missing"},
- {"Mailserver rejected our \"sendmail_from\" setting"} /* 20 */
+ {"Mailserver rejected our \"sendmail_from\" setting"}, /* 20 */
+ {"Error while trimming mail header with PCRE, please file a bug report at http://bugs.php.net/"} /* 21 */
};
+/* This pattern converts all single occurences of \n (Unix)
+ * withour a leading \r to \r\n and all occurences of \r (Mac)
+ * without a trailing \n to \r\n
+ * Thx to Nibbler from ircnet/#linuxger
+ */
+#define PHP_WIN32_MAIL_UNIFY_PATTERN "/(\r\n?)|\n/"
+#define PHP_WIN32_MAIL_UNIFY_REPLACE "\r\n"
+
+/* This pattern removes \r\n from the start of the string,
+ * \r\n from the end of the string and also makes sure every line
+ * is only wrapped with a single \r\n (thus reduces multiple
+ * occurences of \r\n between lines to a single \r\n) */
+#define PHP_WIN32_MAIL_RMVDBL_PATTERN "/^\r\n|(\r\n)+$/m"
+#define PHP_WIN32_MAIL_RMVDBL_REPLACE ""
+
+/* This function is meant to unify the headers passed to to mail()
+ * This means, use PCRE to transform single occurences of \n or \r in \r\n
+ * As a second step we also eleminate all \r\n occurences which are:
+ * 1) At the start of the header
+ * 2) At the end of the header
+ * 3) Two or more occurences in the header are removed so only one is left
+ *
+ * Returns NULL on error, or the new char* buffer on success.
+ * You have to take care and efree() the buffer on your own.
+ */
+static char *php_win32_mail_trim_header(char *header TSRMLS_DC)
+{
+#if HAVE_PCRE || HAVE_BUNDLED_PCRE
+
+ char *result, *result2;
+ int result_len;
+ zval *replace;
+
+ if (!header) {
+ return NULL;
+ }
+
+ MAKE_STD_ZVAL(replace);
+ ZVAL_STRING(replace, PHP_WIN32_MAIL_UNIFY_REPLACE, 0);
+
+ result = php_pcre_replace(PHP_WIN32_MAIL_UNIFY_PATTERN, sizeof(PHP_WIN32_MAIL_UNIFY_PATTERN)-1,
+ header, strlen(header),
+ replace,
+ 0,
+ &result_len,
+ -1 TSRMLS_CC);
+ if (NULL == result) {
+ FREE_ZVAL(replace);
+ return NULL;
+ }
+
+ ZVAL_STRING(replace, PHP_WIN32_MAIL_RMVDBL_REPLACE, 0);
+
+ result2 = php_pcre_replace(PHP_WIN32_MAIL_RMVDBL_PATTERN, sizeof(PHP_WIN32_MAIL_RMVDBL_PATTERN)-1,
+ result, result_len,
+ replace,
+ 0,
+ &result_len,
+ -1 TSRMLS_CC);
+ efree(result);
+ FREE_ZVAL(replace);
+ return result2;
+#else
+ /* In case we don't have PCRE support (for whatever reason...) simply do nothing and return the unmodified header */
+ return estrdup(header);
+#endif
+}
/*********************************************************************
// Name: TSendMail
@@ -124,6 +196,7 @@ int TSendMail(char *host, int *error, char **error_message,
int ret;
char *RPath = NULL;
char *headers_lc = NULL; /* headers_lc is only created if we've a header at all */
+ TSRMLS_FETCH();
WinsockStarted = FALSE;
@@ -140,9 +213,17 @@ int TSendMail(char *host, int *error, char **error_message,
if (headers) {
char *pos = NULL;
size_t i;
+
+ /* Use PCRE to trim the header into the right format */
+ if (NULL == (headers = php_win32_mail_trim_header(headers TSRMLS_CC))) {
+ *error = W32_SM_PCRE_ERROR;
+ return FAILURE;
+ }
+
/* Create a lowercased header for all the searches so we're finally case
* insensitive when searching for a pattern. */
if (NULL == (headers_lc = estrdup(headers))) {
+ efree(headers);
*error = OUT_OF_MEMORY;
return FAILURE;
}
@@ -155,7 +236,8 @@ int TSendMail(char *host, int *error, char **error_message,
if (INI_STR("sendmail_from")) {
RPath = estrdup(INI_STR("sendmail_from"));
} else {
- if (headers_lc) {
+ if (headers) {
+ efree(headers);
efree(headers_lc);
}
*error = W32_SM_SENDMAIL_FROM_NOT_SET;
@@ -168,7 +250,8 @@ int TSendMail(char *host, int *error, char **error_message,
if (RPath) {
efree(RPath);
}
- if (headers_lc) {
+ if (headers) {
+ efree(headers);
efree(headers_lc);
}
/* 128 is safe here, the specifier in snprintf isn't longer than that */
@@ -186,7 +269,8 @@ int TSendMail(char *host, int *error, char **error_message,
if (RPath) {
efree(RPath);
}
- if (headers_lc) {
+ if (headers) {
+ efree(headers);
efree(headers_lc);
}
if (ret != SUCCESS) {
@@ -311,10 +395,13 @@ int SendText(char *RPath, char *Subject, char *mailTo, char *data, char *headers
while(token != NULL)
{
snprintf(Buffer, MAIL_BUFFER_SIZE, "RCPT TO:<%s>\r\n", token);
- if ((res = Post(Buffer)) != SUCCESS)
+ if ((res = Post(Buffer)) != SUCCESS) {
+ efree(tempMailTo);
return (res);
+ }
if ((res = Ack(&server_response)) != SUCCESS) {
SMTP_ERROR_RESPONSE(server_response);
+ efree(tempMailTo);
return (res);
}
token = strtok(NULL, ",");