714 lines
22 KiB
714 lines
22 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
GNU General Public License for more details. |
|
|
|
You should have received a copy of the GNU General Public License |
|
along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
/***************************************************************************** |
|
* |
|
* $/Plasma20/Sources/Plasma/NucleusLib/pnMail/pnMail.cpp |
|
* |
|
***/ |
|
|
|
#include "Pch.h" |
|
#pragma hdrstop |
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* Private |
|
* |
|
***/ |
|
|
|
enum EMailStep { |
|
kMailStepConnect, // wait for 220, then send "helo" command |
|
kMailStepAuth, // wait for 250, then send "AUTH PLAIN <base64data>" |
|
kMailStepSender, // wait for 235/250, then send "mail from:" command |
|
kMailStepRecipient, // wait for 250, then send "rcpt to:" command |
|
kMailStepStartLetter, // wait for 250, then send "data" command |
|
kMailStepLetter, // wait for 354, then send letter |
|
kMailStepQuit, // wait for 250, then send "quit" command |
|
kMailStepDisconnect, // wait for 221, then disconnect |
|
}; |
|
|
|
|
|
struct MailTransaction { |
|
LINK(MailTransaction) link; |
|
AsyncSocket sock; |
|
const char ** stepTable; |
|
EMailStep step; |
|
unsigned subStep; |
|
EMailError error; |
|
FMailResult callback; |
|
void * param; |
|
|
|
char * smtp; |
|
char * sender; |
|
char * recipient; |
|
char * replyTo; |
|
char * subject; |
|
char * bodyEncoding; |
|
char * body; |
|
char * auth; |
|
char buffer[1]; |
|
}; |
|
|
|
static bool MailNotifyProc ( |
|
AsyncSocket sock, |
|
EAsyncNotifySocket code, |
|
AsyncNotifySocket * notify, |
|
void ** userState |
|
); |
|
|
|
static void __cdecl Send ( |
|
AsyncSocket sock, |
|
const char str[], |
|
... |
|
); |
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* Private data |
|
* |
|
***/ |
|
|
|
static bool s_shutdown; |
|
static CCritSect s_critsect; |
|
static LISTDECL(MailTransaction, link) s_mail; |
|
|
|
|
|
//=========================================================================== |
|
// Authenticated email sequence |
|
// -> "220 catherine.cyan.com ESMTP\r\n" |
|
// <- "HELO cyan.com\r\n" |
|
// -> "250 catherine.cyan.com\r\n" |
|
// <- "AUTH PLAIN XXXXXXXXX\r\n" |
|
// -> "235 go ahead\r\n" |
|
// <- "MAIL FROM:<UruClient@cyanworlds.com>\r\n" |
|
// -> "250 ok mail from accepted\r\n" |
|
// <- "RCPT TO:<UruCrashReport@cyanworlds.com>\r\n" |
|
// -> "250 ok recipient accepted\r\n" |
|
// <- "DATA\r\n" |
|
// -> "354 ok, go ahead:\r\n" |
|
// -> <message> + "\r\n.\r\n" |
|
// -> "250 message accepted for delivery\r\n" |
|
// <- "QUIT\r\n" |
|
// -> "221 catherine.cyan.com\r\n" |
|
//=========================================================================== |
|
static const char * s_waitStrAuth[] = { |
|
"220 ", // kMailStepConnect |
|
"250 ", // kMailStepAuth |
|
"235 ", // kMailStepSender |
|
"250 ", // kMailStepRecipient |
|
"250 ", // kMailStepStartLetter |
|
"354 ", // kMailStepLetter |
|
"250 ", // kMailStepQuit |
|
"221 ", // kMailStepDisconnect |
|
}; |
|
|
|
//=========================================================================== |
|
// Unauthenticated email seqeunce |
|
// -> "220 catherine.cyan.com ESMTP\r\n" |
|
// <- "HELO cyan.com\r\n" |
|
// -> "250 catherine.cyan.com\r\n" |
|
// <- "MAIL FROM:<UruClient@cyanworlds.com>\r\n" |
|
// -> "250 ok mail from accepted\r\n" |
|
// <- "RCPT TO:<UruCrashReport@cyanworlds.com>\r\n" |
|
// -> "250 ok recipient accepted\r\n" |
|
// <- "DATA\r\n" |
|
// -> "354 ok, go ahead:\r\n" |
|
// -> <message> + "\r\n.\r\n" |
|
// -> "250 message accepted for delivery\r\n" |
|
// <- "QUIT\r\n" |
|
// -> "221 catherine.cyan.com\r\n" |
|
//=========================================================================== |
|
static const char * s_waitStrNoAuth[] = { |
|
"220 ", // kMailStepConnect |
|
nil, // kMailStepAuth |
|
"250 ", // kMailStepSender |
|
"250 ", // kMailStepRecipient |
|
"250 ", // kMailStepStartLetter |
|
"354 ", // kMailStepLetter |
|
"250 ", // kMailStepQuit |
|
"221 ", // kMailStepDisconnect |
|
}; |
|
|
|
|
|
|
|
/***************************************************************************** |
|
* |
|
* Internal functions |
|
* |
|
***/ |
|
|
|
//=========================================================================== |
|
static bool AdvanceStep ( |
|
MailTransaction * transaction, |
|
AsyncSocket sock |
|
) { |
|
switch (transaction->step) { |
|
case kMailStepConnect: { |
|
const char * host = transaction->sender; |
|
const char * separator = StrChr(host, '@'); |
|
if (separator) |
|
host = separator + 1; |
|
Send(sock, "helo ", host, "\r\n", nil); |
|
transaction->step = kMailStepSender; |
|
|
|
if (transaction->auth[0]) |
|
transaction->step = kMailStepAuth; |
|
} |
|
break; |
|
|
|
case kMailStepAuth: |
|
Send(sock, "AUTH PLAIN ", transaction->auth, "\r\n", nil); |
|
transaction->step = kMailStepSender; |
|
break; |
|
|
|
case kMailStepSender: |
|
Send(sock, "mail from:<", transaction->sender, ">\r\n", nil); |
|
transaction->step = kMailStepRecipient; |
|
break; |
|
|
|
case kMailStepRecipient: { |
|
const char * start = transaction->recipient + transaction->subStep; |
|
while (*start == ' ') |
|
++start; |
|
const char * term = StrChr(start, ';'); |
|
if (term) { |
|
char * buffer = ALLOCA(char, term + 1 - start); |
|
StrCopy(buffer, start, term + 1 - start); |
|
Send(sock, "rcpt to:<", buffer, ">\r\n", nil); |
|
transaction->subStep = term + 1 - transaction->recipient; |
|
} |
|
else { |
|
Send(sock, "rcpt to:<", start, ">\r\n", nil); |
|
transaction->step = kMailStepStartLetter; |
|
transaction->subStep = 0; |
|
} |
|
} |
|
break; |
|
|
|
case kMailStepStartLetter: |
|
Send(sock, "data\r\n", nil); |
|
transaction->step = kMailStepLetter; |
|
break; |
|
|
|
case kMailStepLetter: |
|
if (transaction->replyTo[0]) { |
|
Send( |
|
sock, |
|
"Reply-to: ", |
|
transaction->replyTo, |
|
"\r\n", |
|
nil |
|
); |
|
} |
|
if (transaction->bodyEncoding) { |
|
Send( |
|
sock, |
|
"From: ", |
|
transaction->sender, |
|
"\r\nTo: ", |
|
transaction->recipient, |
|
"\r\nSubject: ", |
|
transaction->subject, |
|
"\r\nContent-Type: text/plain; charset=", |
|
transaction->bodyEncoding, |
|
"\r\n\r\n", |
|
transaction->body, |
|
"\r\n.\r\n", |
|
nil |
|
); |
|
} |
|
else { |
|
Send( |
|
sock, |
|
"From: ", |
|
transaction->sender, |
|
"\r\nTo: ", |
|
transaction->recipient, |
|
"\r\nSubject: ", |
|
transaction->subject, |
|
"\r\n\r\n", |
|
transaction->body, |
|
"\r\n.\r\n", |
|
nil |
|
); |
|
} |
|
transaction->step = kMailStepQuit; |
|
break; |
|
|
|
case kMailStepQuit: |
|
Send(sock, "quit\r\n", nil); |
|
transaction->step = kMailStepDisconnect; |
|
break; |
|
|
|
case kMailStepDisconnect: |
|
return false; |
|
|
|
DEFAULT_FATAL(transaction->step); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
//=========================================================================== |
|
static bool NotifyRead ( |
|
AsyncSocket sock, |
|
AsyncNotifySocketRead * read, |
|
MailTransaction * transaction |
|
) { |
|
|
|
// Parse available lines looking for an acknowledgement of last command |
|
const char * compareStr = transaction->stepTable[transaction->step]; |
|
unsigned compareLen = StrLen(compareStr); |
|
unsigned offset = 0; |
|
for (;;) { |
|
const char * source = (const char *)&read->buffer[offset]; |
|
|
|
// Search for an end-of-line marker |
|
const char * eol = StrChr(source, '\n', read->bytes - offset); |
|
if (!eol) |
|
break; |
|
offset += (eol + 1 - source); |
|
|
|
// Search for an acknowledgement |
|
if (!StrCmp(source, compareStr, compareLen)) { |
|
read->bytesProcessed = offset; |
|
return AdvanceStep(transaction, sock); |
|
} |
|
|
|
// An error was received; log it and bail |
|
LogMsg( |
|
kLogError, |
|
"Mail: step %u error '%.*s'", |
|
transaction->step, |
|
eol - source - 1, |
|
source |
|
); |
|
transaction->error = kMailErrServerError; |
|
return false; |
|
} |
|
|
|
// We did not find an acknowledgement, so skip past fully-received lines |
|
// and wait for more data to arrive |
|
read->bytesProcessed = offset; |
|
return true; |
|
|
|
} |
|
|
|
//=========================================================================== |
|
static void DestroyTransaction (MailTransaction * transaction) { |
|
|
|
// Remove transaction from list so that it can |
|
// be safely deleted outside the critical section |
|
s_critsect.Enter(); |
|
{ |
|
s_mail.Unlink(transaction); |
|
} |
|
s_critsect.Leave(); |
|
|
|
// Perform callback if requested |
|
if (transaction->callback) { |
|
transaction->callback( |
|
transaction->param, |
|
transaction->error |
|
); |
|
} |
|
|
|
DEL(transaction); |
|
} |
|
|
|
//=========================================================================== |
|
static void MailLookupProc ( |
|
void * param, |
|
const wchar * , |
|
unsigned addrCount, |
|
const NetAddress addrs[] |
|
) { |
|
|
|
// If no address was found, cancel the transaction |
|
MailTransaction * transaction = (MailTransaction *) param; |
|
if (!addrCount) { |
|
LogMsg(kLogError,"Mail: failed to resolve %s", transaction->smtp); |
|
transaction->error = kMailErrDnsFailed; |
|
DestroyTransaction(transaction); |
|
return; |
|
} |
|
|
|
// Initiate a connection |
|
AsyncCancelId cancelId; |
|
AsyncSocketConnect( |
|
&cancelId, |
|
addrs[0], |
|
MailNotifyProc, |
|
transaction |
|
); |
|
|
|
} |
|
|
|
//=========================================================================== |
|
static bool MailNotifyProc ( |
|
AsyncSocket sock, |
|
EAsyncNotifySocket code, |
|
AsyncNotifySocket * notify, |
|
void ** userState |
|
) { |
|
switch (code) { |
|
case kNotifySocketConnectFailed: { |
|
MailTransaction * transaction = (MailTransaction *)notify->param; |
|
LogMsg(kLogError, "Mail: unable to connect to %s", transaction->smtp); |
|
transaction->error = kMailErrConnectFailed; |
|
DestroyTransaction(transaction); |
|
} |
|
break; |
|
|
|
case kNotifySocketConnectSuccess: { |
|
MailTransaction * transaction = (MailTransaction *) notify->param; |
|
*userState = notify->param; |
|
transaction->sock = sock; |
|
} |
|
break; |
|
|
|
case kNotifySocketRead: { |
|
MailTransaction * transaction = (MailTransaction *) *userState; |
|
return NotifyRead(sock, (AsyncNotifySocketRead *) notify, transaction); |
|
} |
|
break; |
|
|
|
case kNotifySocketDisconnect: { |
|
MailTransaction * transaction = (MailTransaction *) *userState; |
|
if ((transaction->step != kMailStepDisconnect) && !transaction->error) { |
|
transaction->error = kMailErrDisconnected; |
|
LogMsg(kLogError, "Mail: unexpected disconnection from %s", transaction->smtp); |
|
} |
|
DestroyTransaction(transaction); |
|
AsyncSocketDelete(sock); |
|
} |
|
break; |
|
} |
|
|
|
return !s_shutdown; |
|
} |
|
|
|
//=========================================================================== |
|
static void __cdecl Send ( |
|
AsyncSocket sock, |
|
const char str[], |
|
... |
|
) { |
|
// Count bytes |
|
unsigned bytes = 1; |
|
{ |
|
va_list argList; |
|
va_start(argList, str); |
|
for (const char * source = str; source; source = va_arg(argList, const char *)) |
|
bytes += StrLen(source); |
|
va_end(argList); |
|
} |
|
|
|
// Allocate string buffer |
|
char * packed; |
|
const unsigned kStackBufSize = 8 * 1024; |
|
if (bytes > kStackBufSize) |
|
packed = (char *) ALLOC(bytes); |
|
else |
|
packed = (char *) _alloca(bytes); |
|
|
|
// Pack the string |
|
{ |
|
va_list argList; |
|
va_start(argList, str); |
|
char * dest = packed; |
|
for (const char * source = str; source; source = va_arg(argList, const char *)) |
|
dest += StrCopyLen(dest, source, packed + bytes - dest); |
|
va_end(argList); |
|
} |
|
|
|
// Send the string |
|
AsyncSocketSend(sock, packed, bytes - 1); |
|
|
|
// Free the string |
|
if (bytes > kStackBufSize) |
|
FREE(packed); |
|
} |
|
|
|
//=========================================================================== |
|
static void IMail ( |
|
const char * stepTable[], |
|
const char smtp[], |
|
const char sender[], |
|
const char recipient[], |
|
const char replyTo[], |
|
const char subject[], |
|
const char body[], |
|
const char auth[], |
|
EMailEncoding bodyEncoding, |
|
FMailResult callback, |
|
void * param |
|
) { |
|
static const char * utf8 = "\"utf-8\""; |
|
const char * encodingStr; |
|
|
|
if (bodyEncoding == kMailEncodeUtf8) |
|
encodingStr = utf8; |
|
else |
|
encodingStr = ""; |
|
|
|
// Calculate string lengths |
|
unsigned lenSmtp = StrLen(smtp); |
|
unsigned lenSender = StrLen(sender); |
|
unsigned lenRecipient = StrLen(recipient); |
|
unsigned lenReplyTo = StrLen(replyTo); |
|
unsigned lenSubject = StrLen(subject); |
|
unsigned lenBodyEncoding = StrLen(encodingStr); |
|
unsigned lenBody = StrLen(body); |
|
unsigned lenAuth = StrLen(auth); |
|
|
|
// Calculate the required buffer size for all strings |
|
unsigned bytes = ( |
|
lenSmtp + 1 + |
|
lenSender + 1 + |
|
lenRecipient + 1 + |
|
lenReplyTo + 1 + |
|
lenSubject + 1 + |
|
lenBody + 1 + |
|
lenAuth + 1 |
|
) * sizeof(char); |
|
|
|
if (lenBodyEncoding) |
|
bytes += (lenBodyEncoding + 1) * sizeof(char); |
|
|
|
// Create a transaction record |
|
MailTransaction * transaction = new( |
|
ALLOC(offsetof(MailTransaction, buffer) + bytes) |
|
) MailTransaction; |
|
transaction->stepTable = stepTable; |
|
transaction->sock = nil; |
|
transaction->step = kMailStepConnect; |
|
transaction->subStep = 0; |
|
transaction->error = kMailSuccess; |
|
transaction->callback = callback; |
|
transaction->param = param; |
|
|
|
unsigned offset = 0; |
|
transaction->smtp = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->smtp, smtp, lenSmtp + 1) + 1; |
|
transaction->sender = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->sender, sender, lenSender + 1) + 1; |
|
transaction->recipient = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->recipient, recipient, lenRecipient + 1) + 1; |
|
transaction->replyTo = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->replyTo, replyTo, lenReplyTo + 1) + 1; |
|
transaction->subject = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->subject, subject, lenSubject + 1) + 1; |
|
if (lenBodyEncoding) { |
|
transaction->bodyEncoding = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->bodyEncoding, encodingStr, lenBodyEncoding + 1) + 1; |
|
} |
|
else { |
|
transaction->bodyEncoding = nil; |
|
} |
|
transaction->body = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->body, body, lenBody + 1) + 1; |
|
transaction->auth = transaction->buffer + offset; |
|
offset += StrCopyLen(transaction->auth, auth, lenAuth + 1) + 1; |
|
ASSERT(offset == bytes); |
|
|
|
// Start the transaction with a dns lookup |
|
const unsigned kSmtpPort = 25; |
|
wchar smtpName[256]; |
|
StrToUnicode(smtpName, smtp, arrsize(smtpName)); |
|
|
|
// Add transaction to global list |
|
bool shutdown; |
|
s_critsect.Enter(); |
|
{ |
|
shutdown = s_shutdown; |
|
s_mail.Link(transaction); |
|
} |
|
s_critsect.Leave(); |
|
|
|
if (shutdown) { |
|
DestroyTransaction(transaction); |
|
} |
|
else { |
|
NetAddress addr; |
|
if (NetAddressFromString(&addr, smtpName, kSmtpPort)) { |
|
AsyncCancelId cancelId; |
|
AsyncSocketConnect( |
|
&cancelId, |
|
addr, |
|
MailNotifyProc, |
|
transaction |
|
); |
|
} |
|
else { |
|
AsyncCancelId cancelId; |
|
AsyncAddressLookupName( |
|
&cancelId, |
|
MailLookupProc, |
|
smtpName, |
|
kSmtpPort, |
|
transaction |
|
); |
|
} |
|
} |
|
} |
|
|
|
/**************************************************************************** |
|
* |
|
* Exported functions |
|
* |
|
***/ |
|
|
|
//=========================================================================== |
|
void MailEncodePassword ( |
|
const char username[], |
|
const char password[], |
|
ARRAY(char) * emailAuth |
|
) { |
|
// base64_encode("\0#{user}\0#{secret}") |
|
emailAuth->Reserve(512); |
|
emailAuth->Push(0); |
|
emailAuth->Add(username, StrBytes(username)); |
|
emailAuth->Add(password, StrBytes(password)); |
|
unsigned srcChars = emailAuth->Bytes(); |
|
|
|
// Allocate space for encoded data |
|
unsigned dstChars = Base64EncodeSize(srcChars); |
|
char * dstData = emailAuth->New(dstChars); |
|
|
|
// Encode data and move it back to the front of the array |
|
dstChars = Base64Encode( |
|
srcChars, |
|
(const byte *) emailAuth->Ptr(), |
|
dstChars, |
|
dstData |
|
); |
|
emailAuth->Move( |
|
0, |
|
srcChars, |
|
dstChars |
|
); |
|
emailAuth->SetCountFewer(dstChars); |
|
} |
|
|
|
//=========================================================================== |
|
void Mail ( |
|
const char smtp[], |
|
const char sender[], |
|
const char recipient[], // multiple recipients separated by semicolons |
|
const char subject[], |
|
const char body[], |
|
const char username[], |
|
const char password[], |
|
const char replyTo[], |
|
EMailEncoding bodyEncoding, |
|
FMailResult callback, |
|
void * param |
|
) { |
|
s_shutdown = false; |
|
|
|
// Get email authorization |
|
const char * auth; |
|
const char ** stepTable; |
|
ARRAY(char) authBuffer; |
|
if (!password || !*password) { |
|
// No password is specified, use non-authenticated email |
|
auth = ""; |
|
stepTable = s_waitStrNoAuth; |
|
} |
|
else if (!username || !*username) { |
|
// No username specified, user is providing the base64 encoded secret |
|
auth = password; |
|
stepTable = s_waitStrAuth; |
|
} |
|
else { |
|
MailEncodePassword(username, password, &authBuffer); |
|
auth = authBuffer.Ptr(); |
|
stepTable = s_waitStrAuth; |
|
} |
|
|
|
IMail( |
|
stepTable, |
|
smtp, |
|
sender, |
|
recipient, |
|
replyTo ? replyTo : "", |
|
subject, |
|
body, |
|
auth, |
|
bodyEncoding, |
|
callback, |
|
param |
|
); |
|
} |
|
|
|
//=========================================================================== |
|
void MailStop () { |
|
s_critsect.Enter(); |
|
{ |
|
s_shutdown = true; |
|
MailTransaction * transaction = s_mail.Head(); |
|
for (; transaction; transaction = s_mail.Next(transaction)) { |
|
if (transaction->sock) |
|
AsyncSocketDisconnect(transaction->sock, true); |
|
transaction->error = kMailErrClientCanceled; |
|
} |
|
} |
|
s_critsect.Leave(); |
|
|
|
AsyncAddressLookupCancel( |
|
MailLookupProc, |
|
0 // cancel all |
|
); |
|
AsyncSocketConnectCancel( |
|
MailNotifyProc, |
|
0 // cancel all |
|
); |
|
} |
|
|
|
//=========================================================================== |
|
bool MailQueued () { |
|
bool queued; |
|
s_critsect.Enter(); |
|
queued = s_mail.Head() != nil; |
|
s_critsect.Leave(); |
|
return queued; |
|
} |
|
|
|
//============================================================================ |
|
const wchar * MailErrorToString (EMailError error) { |
|
|
|
switch (error) { |
|
case kMailSuccess: return L"kMailSuccess"; |
|
case kMailErrDnsFailed: return L"kMailErrDnsFailed"; |
|
case kMailErrConnectFailed: return L"kMailErrConnectFailed"; |
|
case kMailErrDisconnected: return L"kMailErrDisconnected"; |
|
case kMailErrClientCanceled: return L"kMailErrClientCanceled"; |
|
case kMailErrServerError: return L"kMailErrServerError"; |
|
DEFAULT_FATAL(error); |
|
} |
|
}
|
|
|