diff options
Diffstat (limited to 'win32/sendmail.c')
| -rw-r--r-- | win32/sendmail.c | 511 | 
1 files changed, 511 insertions, 0 deletions
| diff --git a/win32/sendmail.c b/win32/sendmail.c new file mode 100644 index 0000000000..7adf4d7bf3 --- /dev/null +++ b/win32/sendmail.c @@ -0,0 +1,511 @@ + +/*  + *    PHP Sendmail for Windows. + * + *  This file is rewriten specificly for PHPFI.  Some functionality + *  has been removed (MIME and file attachments).  This code was  + *  modified from code based on code writen by Jarle Aase. + * + *  This class is based on the original code by Jarle Aase, see bellow: + *  wSendmail.cpp  It has been striped of some functionality to match + *  the requirements of phpfi. + * + *  Very simple SMTP Send-mail program for sending command-line level + *  emails and CGI-BIN form response for the Windows platform. + * + *  The complete wSendmail package with source code can be downloaded + *  from http://virtual.icr.com.au:80/jgaa/cgi-bin.htm + * + */ + +#ifdef THREAD_SAFE +#include "tls.h" +#endif +#include "php.h"				/*php specific */ +#include <stdio.h> +#include <stdlib.h> +#include <winsock.h> +#include "time.h" +#include <string.h> +#include <malloc.h> +#include <memory.h> +#include <winbase.h> +#include "sendmail.h" + + +/* +   extern int _daylight; +   extern long _timezone; + */ +/*enum +   { +   DO_CONNECT = WM_USER +1 +   }; + */ + +static char *days[] = +{"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"}; +static char *months[] = +{"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}; + +#ifndef THREAD_SAFE +char Buffer[MAIL_BUFFER_SIZE]; + +// socket related data +SOCKET sc; +WSADATA Data; +struct hostent *adr; +SOCKADDR_IN sock_in; +int WinsockStarted; +// values set by the constructor +char *AppName; +char MailHost[HOST_NAME_LEN]; +char LocalHost[HOST_NAME_LEN]; +#endif +char seps[] = " ,\t\n"; +char *php_mailer = "PHP 3.0 WIN32"; + +char *get_header(char *h, char *headers); + +// Error messages +static char *ErrorMessages[] = +{ +	{"Success"}, +	{"Bad arguments from form"}, +	{"Unable to open temporary mailfile for read"}, +	{"Failed to Start Sockets"}, +	{"Failed to Resolve Host"}, +	{"Failed to obtain socket handle"}, +	{"Failed to Connect"}, +	{"Failed to Send"}, +	{"Failed to Receive"}, +	{"Server Error"}, +	{"Failed to resolve the host IP name"}, +	{"Out of memory"}, +	{"Unknown error"}, +	{"Bad Message Contents"}, +	{"Bad Message Subject"}, +	{"Bad Message destination"}, +	{"Bad Message Return Path"}, +	{"Bad Mail Host"}, +	{"Bad Message File"}, +	{"PHP Internal error: php3.ini sendmail from variable not set!"} +}; + + + +//******************************************************************** +// Name:  TSendMail +// Input:   1) host:    Name of the mail host where the SMTP server resides +//                      max accepted length of name = 256 +//          2) appname: Name of the application to use in the X-mailer +//                      field of the message. if NULL is given the application +//                      name is used as given by the GetCommandLine() function +//                      max accespted length of name = 100 +// Output:  1) error:   Returns the error code if something went wrong or +//                      SUCCESS otherwise. +// +//  See SendText() for additional args! +//******************************************************************** +int TSendMail(char *host, int *error, +			  char *headers, char *Subject, char *mailTo, char *data) +{ +	int ret; +	char *RPath = NULL; +	TLS_VARS; + +	GLOBAL(WinsockStarted) = FALSE; + +	if (host == NULL) { +		*error = BAD_MAIL_HOST; +		return BAD_MAIL_HOST; +	} else if (strlen(host) >= HOST_NAME_LEN) { +		*error = BAD_MAIL_HOST; +		return BAD_MAIL_HOST; +	} else { +		strcpy(GLOBAL(MailHost), host); +	} + +	if (php3_ini.sendmail_from){ +		RPath = estrdup(php3_ini.sendmail_from); +		} else { +			return 19; +	} + +	// attempt to connect with mail host +	*error = MailConnect(); +	if (*error != 0) { +		if(RPath)efree(RPath); +		return *error; +	} else { +		ret = SendText(RPath, Subject, mailTo, data, headers); +		TSMClose(); +		if (ret != SUCCESS) { +			*error = ret; +		} +		if(RPath)efree(RPath); +		return ret; +	} +} + +//******************************************************************** +// Name:  TSendMail::~TSendMail +// Input: +// Output: +// Description: DESTRUCTOR +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +void TSMClose() +{ +	TLS_VARS; + +	Post("QUIT\n"); +	Ack(); +	// to guarantee that the cleanup is not made twice and +	// compomise the rest of the application if sockets are used +	// elesewhere +} + + +//******************************************************************** +// Name:  char *GetSMErrorText +// Input:   Error index returned by the menber functions +// Output:  pointer to a string containing the error description +// Description: +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +char *GetSMErrorText(int index) +{ + +	if ((index > MAX_ERROR_INDEX) || (index < MIN_ERROR_INDEX)) +		return (ErrorMessages[UNKNOWN_ERROR]); +	else +		return (ErrorMessages[index]); +} + + +//******************************************************************** +// Name:  TSendText +// Input:       1) RPath:   return path of the message +//                                  Is used to fill the "Return-Path" and the +//                                  "X-Sender" fields of the message. +//                  2) Subject: Subject field of the message. If NULL is given +//                                  the subject is set to "No Subject" +//                  3) mailTo:  Destination address +//                  4) data:        Null terminated string containing the data to be send. +// Output:      Error code or SUCCESS +// Description: +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +int SendText(char *RPath, char *Subject, char *mailTo, char *data, char *headers) +{ + +	int res, i; +	char *p; +	TLS_VARS; + +	// check for NULL parameters +	if (data == NULL) +		return (BAD_MSG_CONTENTS); +	if (mailTo == NULL) +		return (BAD_MSG_DESTINATION); +	if (RPath == NULL) +		return (BAD_MSG_RPATH); + +	// simple checks for the mailto address +	// have ampersand ? +	if (strchr(mailTo, '@') == NULL) +		return (BAD_MSG_DESTINATION); + +	sprintf(GLOBAL(Buffer), "HELO %s\n", GLOBAL(LocalHost)); + +	// in the beggining of the dialog +	// attempt reconnect if the first Post fail +	if ((res = Post(GLOBAL(Buffer))) != SUCCESS) { +		MailConnect(); +		if ((res = Post(GLOBAL(Buffer))) != SUCCESS) +			return (res); +	} +	if ((res = Ack()) != SUCCESS) +		return (res); + +	sprintf(GLOBAL(Buffer), "MAIL FROM:<%s>\n", RPath); +	if ((res = Post(GLOBAL(Buffer))) != SUCCESS) +		return (res); +	if ((res = Ack()) != SUCCESS) +		return (res); + + +	sprintf(GLOBAL(Buffer), "RCPT TO:<%s>\n", mailTo); +	if ((res = Post(GLOBAL(Buffer))) != SUCCESS) +		return (res); +	if ((res = Ack()) != SUCCESS) +		return (res); + +	if ((res = Post("DATA\n")) != SUCCESS) +		return (res); +	if ((res = Ack()) != SUCCESS) +		return (res); + + +	// send message header +	if (Subject == NULL) +		res = PostHeader(RPath, "No Subject", mailTo, headers); +	else +		res = PostHeader(RPath, Subject, mailTo, headers); +	if (res != SUCCESS) +		return (res); + + +	// send message contents in 1024 chunks +	if (strlen(data) <= 1024) { +		if ((res = Post(data)) != SUCCESS) +			return (res); +	} else { +		p = data; +		while (1) { +			if (*p == '\0') +				break; +			if (strlen(p) >= 1024) +				i = 1024; +			else +				i = strlen(p); + +			// put next chunk in buffer +			strncpy(GLOBAL(Buffer), p, i); +			GLOBAL(Buffer)[i] = '\0'; +			p += i; + +			// send chunk +			if ((res = Post(GLOBAL(Buffer))) != SUCCESS) +				return (res); +		} +	} + +	//send termination dot +	if ((res = Post("\r\n.\r\n")) != SUCCESS) +		return (res); +	if ((res = Ack()) != SUCCESS) +		return (res); + +	return (SUCCESS); +} + + + +//******************************************************************** +// Name:  PostHeader +// Input:       1) return path +//                  2) Subject +//                  3) destination address +//                  4) DoMime flag +// Output:      Error code or Success +// Description: +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +int PostHeader(char *RPath, char *Subject, char *mailTo, char *xheaders) +{ + +	// Print message header according to RFC 822 +	// Return-path, Received, Date, From, Subject, Sender, To, cc + +	time_t tNow = time(NULL); +	struct tm *tm = localtime(&tNow); +	int zoneh = abs(_timezone); +	int zonem, res; +	char *p; +	TLS_VARS; + +	p = GLOBAL(Buffer); +	zoneh /= (60 * 60); +	zonem = (abs(_timezone) / 60) - (zoneh * 60); + +	p += sprintf(p, "Date: %s, %02d %s %04d %02d:%02d:%02d %s%02d%02d\r\n", +				 days[tm->tm_wday], +				 tm->tm_mday, +				 months[tm->tm_mon], +				 tm->tm_year + 1900, +				 tm->tm_hour, +				 tm->tm_min, +				 tm->tm_sec, +				 (_timezone > 0) ? "+" : (_timezone < 0) ? "-" : "", +				 zoneh, +				 zonem); + +	if(xheaders && strnicmp("From:",xheaders,5)){ +		p += sprintf(p, "From: %s\r\n", RPath); +	} +	p += sprintf(p, "Subject: %s\r\n", Subject); +	p += sprintf(p, "To: %s\r\n", mailTo); +	if(xheaders){ +		p += sprintf(p, "%s\r\n", xheaders); +	} + +	if ((res = Post(GLOBAL(Buffer))) != SUCCESS) +		return (res); + +	if ((res = Post("\r\n")) != SUCCESS) +		return (res); + +	return (SUCCESS); +} + + + +//******************************************************************** +// Name:  MailConnect +// Input:   None +// Output:  None +// Description: Connect to the mail host and receive the welcome message. +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +int MailConnect() +{ + +	int res; +	TLS_VARS; + + +	// Create Socket +	if ((GLOBAL(sc) = socket(PF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET) +		return (FAILED_TO_OBTAIN_SOCKET_HANDLE); + +	// Get our own host name +	if (gethostname(GLOBAL(LocalHost), HOST_NAME_LEN)) +		return (FAILED_TO_GET_HOSTNAME); + +	// Resolve the servers IP +	//if (!isdigit(GLOBAL(MailHost)[0])||!gethostbyname(GLOBAL(MailHost))) +	//{ +	//	return (FAILED_TO_RESOLVE_HOST); +	//} + +	// Connect to server +	GLOBAL(sock_in).sin_family = AF_INET; +	GLOBAL(sock_in).sin_port = htons(25); +	GLOBAL(sock_in).sin_addr.S_un.S_addr = GetAddr(GLOBAL(MailHost)); + +	if (connect(GLOBAL(sc), (LPSOCKADDR) & GLOBAL(sock_in), sizeof(GLOBAL(sock_in)))) +		return (FAILED_TO_CONNECT); + +	// receive Server welcome message +	res = Ack(); +	return (res); +} + + + + + + +//******************************************************************** +// Name:  Post +// Input: +// Output: +// Description: +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +int Post(LPCSTR msg) +{ +	int len = strlen(msg); +	int slen; +	int index = 0; +	TLS_VARS; + +	while (len > 0) { +		if ((slen = send(GLOBAL(sc), msg + index, len, 0)) < 1) +			return (FAILED_TO_SEND); +		len -= slen; +		index += slen; +	} +	return (SUCCESS); +} + + + +//******************************************************************** +// Name:  Ack +// Input: +// Output: +// Description: +// Get the response from the server. We only want to know if the +// last command was successful. +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +int Ack() +{ +	static char *buf; +	int rlen; +	int Index = 0; +	int Received = 0; +	TLS_VARS; + +	if (!buf) +		if ((buf = (char *) malloc(1024 * 4)) == NULL) +			return (OUT_OF_MEMORY); + +  again: + +	if ((rlen = recv(GLOBAL(sc), buf + Index, ((1024 * 4) - 1) - Received, 0)) < 1) +		return (FAILED_TO_RECEIVE); + +	Received += rlen; +	buf[Received] = 0; +	//err_msg   fprintf(stderr,"Received: (%d bytes) %s", rlen, buf + Index); + +	// Check for newline +	Index += rlen; +	if ((buf[Received - 2] != '\r') || (buf[Received - 1] != '\n')) +		// err_msg          fprintf(stderr,"Incomplete server message. Awaiting CRLF\n"); +		goto again;				// Incomplete data. Line must be terminated by CRLF + +	if (buf[0] > '3') +		return (SMTP_SERVER_ERROR); + +	return (SUCCESS); +} + + +//******************************************************************** +// Name:  unsigned long GetAddr (LPSTR szHost) +// Input: +// Output: +// Description: Given a string, it will return an IP address. +//   - first it tries to convert the string directly +//   - if that fails, it tries o resolve it as a hostname +// +// WARNING: gethostbyname() is a blocking function +// Author/Date:  jcar 20/9/96 +// History: +//******************************************************************** +unsigned long GetAddr(LPSTR szHost) +{ +	LPHOSTENT lpstHost; +	u_long lAddr = INADDR_ANY; + +	/* check that we have a string */ +	if (*szHost) { + +		/* check for a dotted-IP address string */ +		lAddr = inet_addr(szHost); + +		/* If not an address, then try to resolve it as a hostname */ +		if ((lAddr == INADDR_NONE) && (strcmp(szHost, "255.255.255.255"))) { + +			lpstHost = gethostbyname(szHost); +			if (lpstHost) {		/* success */ +				lAddr = *((u_long FAR *) (lpstHost->h_addr)); +			} else { +				lAddr = INADDR_ANY;		/* failure */ +			} +		} +	} +	return (lAddr); +}								/* end GetAddr() */ | 
