You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
878 lines
25 KiB
878 lines
25 KiB
#define INCL_INETSRV_INCS
|
|
#include "smtpinc.h"
|
|
#include "remoteq.hxx"
|
|
#include "dropdir.hxx"
|
|
|
|
DWORD g_dwDropFileCounter = 0;
|
|
CPool CDropDir::m_Pool(DROPDIR_SIG);
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
VOID
|
|
DropDirWriteCompletion(
|
|
PVOID pvContext,
|
|
DWORD cbWritten,
|
|
DWORD dwCompletionStatus,
|
|
OVERLAPPED * lpo
|
|
)
|
|
{
|
|
DECL_TRACE((LPARAM) 0xC0DEC0DE, "DropDirWriteCompletion");
|
|
BOOL WasProcessed = TRUE;
|
|
CDropDir *pCC = (CDropDir *) pvContext;
|
|
|
|
_ASSERT(pCC);
|
|
|
|
pCC->SetHr( pCC->OnIoWriteCompletion(cbWritten, dwCompletionStatus, lpo) );
|
|
|
|
SAFE_RELEASE( pCC );
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
VOID
|
|
DropDirReadCompletion(
|
|
PFIO_CONTEXT pContext,
|
|
PFH_OVERLAPPED lpo,
|
|
DWORD cbRead,
|
|
DWORD dwCompletionStatus
|
|
)
|
|
{
|
|
DECL_TRACE((LPARAM) 0xC0DEC0DE, "DropDirReadCompletion");
|
|
BOOL WasProcessed = TRUE;
|
|
DROPDIR_READ_OVERLAPPED *p = (DROPDIR_READ_OVERLAPPED *) lpo;
|
|
CDropDir *pCC = (CDropDir*) p->ThisPtr;
|
|
|
|
_ASSERT(pCC);
|
|
|
|
pCC->SetHr( pCC->OnIoReadCompletion(cbRead, dwCompletionStatus, (LPOVERLAPPED)lpo) );
|
|
|
|
SAFE_RELEASE( pCC );
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
CDropDir::CDropDir()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::CDropDir");
|
|
|
|
m_dwSig = DROPDIR_SIG;
|
|
m_cRef = 1;
|
|
m_NumRcpts = 0;
|
|
m_hDrop = INVALID_HANDLE_VALUE;
|
|
m_hMail = NULL;
|
|
m_AdvContext = NULL;
|
|
m_rgRcptIndexList = NULL;
|
|
m_pIMsg = NULL;
|
|
m_pBindInterface = NULL;
|
|
m_pAtqContext = NULL;
|
|
CopyMemory( m_acCrLfDotCrLf, "\r\n.\r\n", 5 );
|
|
ZeroMemory( m_acLastBytes, 5 );
|
|
m_pIMsgRecips = NULL;
|
|
m_hr = S_OK;
|
|
m_cbWriteOffset = 0;
|
|
m_cbReadOffset = 0;
|
|
m_cbMsgWritten = 0;
|
|
m_idxRecips = 0;
|
|
ZeroMemory( &m_WriteOverlapped, sizeof( m_WriteOverlapped ) );
|
|
ZeroMemory( &m_ReadOverlapped, sizeof( m_ReadOverlapped ) );
|
|
m_szDropDir[0] = '\0';
|
|
m_pISMTPConnection = NULL;
|
|
m_ReadState = DROPDIR_READ_NULL;
|
|
m_WriteState = DROPDIR_WRITE_NULL;
|
|
m_pParentInst = NULL;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
CDropDir::~CDropDir()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::~CDropDir");
|
|
|
|
SAFE_RELEASE( m_pIMsgRecips );
|
|
|
|
if(m_pBindInterface)
|
|
{
|
|
m_pBindInterface->ReleaseContext();
|
|
SAFE_RELEASE(m_pBindInterface);
|
|
}
|
|
|
|
m_MsgAck.pIMailMsgProperties = m_pIMsg;
|
|
m_MsgAck.pvMsgContext = (DWORD *) m_AdvContext;
|
|
|
|
if(SUCCEEDED(m_hr))
|
|
{
|
|
m_MsgAck.dwMsgStatus = MESSAGE_STATUS_ALL_DELIVERED;
|
|
}
|
|
else
|
|
{
|
|
m_MsgAck.dwMsgStatus = MESSAGE_STATUS_RETRY_ALL;
|
|
}
|
|
|
|
m_MsgAck.dwStatusCode = 0;
|
|
m_pISMTPConnection->AckMessage(&m_MsgAck);
|
|
SAFE_RELEASE(m_pIMsg);
|
|
|
|
if(SUCCEEDED(m_hr))
|
|
{
|
|
BUMP_COUNTER( m_pParentInst, DirectoryDrops);
|
|
}
|
|
|
|
if(SUCCEEDED(m_hr))
|
|
{
|
|
m_pISMTPConnection->AckConnection(CONNECTION_STATUS_OK);
|
|
}
|
|
else
|
|
{
|
|
SetLastError( 0x0000FFFF & m_hr );
|
|
m_pISMTPConnection->AckConnection(CONNECTION_STATUS_FAILED);
|
|
}
|
|
|
|
PATQ_CONTEXT pAtqContext = (PATQ_CONTEXT)InterlockedExchangePointer( (PVOID *)&m_pAtqContext, NULL);
|
|
if ( pAtqContext != NULL )
|
|
{
|
|
if(m_hDrop != INVALID_HANDLE_VALUE)
|
|
AtqCloseFileHandle( pAtqContext );
|
|
AtqFreeContext( pAtqContext, TRUE );
|
|
}
|
|
|
|
// If we failed to completely write the message, delete the drop file
|
|
if (FAILED(m_hr))
|
|
{
|
|
DeleteFile(m_szDropFile);
|
|
}
|
|
|
|
//
|
|
// Now get the next message on this connection and
|
|
// copy it to drop directory.
|
|
//
|
|
|
|
if (m_szDropDir[0])
|
|
AsyncCopyMailToDropDir( m_pISMTPConnection, m_szDropDir, m_pParentInst );
|
|
|
|
SAFE_RELEASE( m_pISMTPConnection );
|
|
|
|
m_dwSig = DROPDIR_SIG_FREE;
|
|
}
|
|
|
|
|
|
//---[ CDropDir::CopyMailToDropDir ]-------------------------------------------
|
|
//
|
|
//
|
|
// Description:
|
|
//
|
|
// Entry point to the Async writing to the mail file. When this is called
|
|
// it sets up a series of alternating reads and writes (read from the queue
|
|
// and write to the mail file). The (rough description of the) algorithm is
|
|
// as follows:
|
|
//
|
|
// WriteHeaders()
|
|
// Post Async Read from MailFile
|
|
// return
|
|
//
|
|
// Read Completion function()
|
|
// Post Async Write of Data to DropDir
|
|
// return
|
|
//
|
|
// Write Completion function()
|
|
// Post Async Read from MailFile
|
|
// return
|
|
//
|
|
// Called by
|
|
// AsyncCopyMailToDropDir()
|
|
//
|
|
// Parameters:
|
|
//
|
|
// Returns:
|
|
//
|
|
// History:
|
|
// Created by PGOPI
|
|
// 3/25/99 - MikeSwa modified for checkin
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT CDropDir::CopyMailToDropDir(
|
|
ISMTPConnection *pISMTPConnection,
|
|
const char *DropDirectory,
|
|
IMailMsgProperties *pIMsg,
|
|
PVOID AdvContext,
|
|
DWORD NumRcpts,
|
|
DWORD *rgRcptIndexList,
|
|
SMTP_SERVER_INSTANCE *pParentInst)
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::CopyMailToDropDir");
|
|
HRESULT hr = S_OK;
|
|
|
|
m_pISMTPConnection = pISMTPConnection;
|
|
SAFE_ADDREF( m_pISMTPConnection );
|
|
m_pIMsg = pIMsg;
|
|
SAFE_ADDREF( m_pIMsg );
|
|
m_pParentInst = pParentInst;
|
|
m_AdvContext = AdvContext;
|
|
m_NumRcpts = NumRcpts;
|
|
m_rgRcptIndexList = rgRcptIndexList;
|
|
|
|
if (!DropDirectory)
|
|
{
|
|
hr = E_INVALIDARG;
|
|
goto error_exit;
|
|
}
|
|
|
|
//Copy drop directory now... so we have a copy in the destructor
|
|
//even if this message cannot be dropped
|
|
lstrcpy(m_szDropDir, DropDirectory);
|
|
|
|
hr = m_pIMsg->QueryInterface(IID_IMailMsgBind, (void **)&m_pBindInterface);
|
|
|
|
if(FAILED(hr))
|
|
{
|
|
ErrorTrace( (LPARAM)this, "m_pIMsg->QueryInterface for IID_IMailMsgBind return hr = 0x%x", hr );
|
|
goto error_exit;
|
|
}
|
|
|
|
hr = m_pBindInterface->GetBinding(&m_hMail, NULL);
|
|
|
|
if(FAILED(hr))
|
|
{
|
|
ErrorTrace( (LPARAM)this, "GetBinding return hr = 0x%x", hr );
|
|
goto error_exit;
|
|
}
|
|
|
|
hr = m_pIMsg->QueryInterface( IID_IMailMsgRecipients, (PVOID *) &m_pIMsgRecips);
|
|
|
|
if( FAILED( hr ) )
|
|
{
|
|
ErrorTrace( (LPARAM)this, "QueryInterface for IID_IMailMsgRecipients failed hr = 0x%x", hr );
|
|
goto error_exit;
|
|
}
|
|
|
|
if( CheckIfAllRcptsHandled() )
|
|
{
|
|
goto exit;
|
|
}
|
|
|
|
|
|
DebugTrace((LPARAM)this, "Dropping file to: %s", DropDirectory);
|
|
|
|
m_hDrop = CreateDropFile(DropDirectory);
|
|
|
|
if (m_hDrop == INVALID_HANDLE_VALUE)
|
|
{
|
|
hr = HRESULT_FROM_WIN32( GetLastError() );
|
|
ErrorTrace( (LPARAM)this, "Unable to create drop directory (%s) : 0x%x", DropDirectory, hr);
|
|
goto error_exit;
|
|
}
|
|
|
|
if (!AtqAddAsyncHandle( &m_pAtqContext,
|
|
NULL,
|
|
(LPVOID) this,
|
|
DropDirWriteCompletion,
|
|
INFINITE,
|
|
m_hDrop))
|
|
{
|
|
hr = HRESULT_FROM_WIN32( GetLastError() );
|
|
ErrorTrace((LPARAM)this, "Error 0x%x while adding async handle to ATQ", hr);
|
|
goto error_exit;
|
|
}
|
|
|
|
// Output the x-headers
|
|
|
|
if (FAILED( hr = CreateXHeaders() ) )
|
|
{
|
|
ErrorTrace( (LPARAM)this, " Error while creating x-headers hr = 0x%x", hr);
|
|
goto error_exit;
|
|
}
|
|
|
|
exit:
|
|
return( hr );
|
|
|
|
error_exit:
|
|
// If we failed here we must make sure we bubble up the failure so
|
|
// it will be seen in the destructor - if we succeeded it is dangerous
|
|
// to set this since we could overwrite a failure code written in the
|
|
// async path resulting from CreateXHeaders (so we don't do that!)
|
|
SetHr(hr);
|
|
goto exit;
|
|
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
HANDLE CDropDir::CreateDropFile(const char * DropDir)
|
|
{
|
|
HANDLE FileHandle = INVALID_HANDLE_VALUE;
|
|
DWORD dwStrLen;
|
|
FILETIME ftTime;
|
|
DWORD Error = 0;
|
|
|
|
DECL_TRACE((LPARAM)this, "CDropDir::CreateDropFile");
|
|
|
|
dwStrLen = lstrlen(DropDir);
|
|
lstrcpy(m_szDropFile, DropDir);
|
|
|
|
do
|
|
{
|
|
GetSystemTimeAsFileTime(&ftTime);
|
|
wsprintf(&m_szDropFile[dwStrLen],
|
|
"%08x%08x%08x%s",
|
|
ftTime.dwLowDateTime,
|
|
ftTime.dwHighDateTime,
|
|
InterlockedIncrement((PLONG)&g_dwDropFileCounter),
|
|
".eml");
|
|
|
|
FileHandle = CreateFile(m_szDropFile,
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
NULL,
|
|
CREATE_NEW,
|
|
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_SEQUENTIAL_SCAN | FILE_FLAG_OVERLAPPED,
|
|
NULL);
|
|
if (FileHandle != INVALID_HANDLE_VALUE)
|
|
break;
|
|
|
|
if((Error = GetLastError()) != ERROR_FILE_EXISTS)
|
|
{
|
|
FileHandle = INVALID_HANDLE_VALUE;
|
|
ErrorTrace( (LPARAM)this, "CreateFile returns err = %u", Error );
|
|
break;
|
|
}
|
|
|
|
} while( (FileHandle == INVALID_HANDLE_VALUE) && !m_pParentInst->IsShuttingDown());
|
|
|
|
return (FileHandle);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::ReadFile(
|
|
IN LPVOID pBuffer,
|
|
IN DWORD cbSize
|
|
)
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::ReadFile");
|
|
HRESULT hr = S_OK;
|
|
|
|
_ASSERT(pBuffer != NULL);
|
|
|
|
m_ReadOverlapped.Overlapped.Offset = LODWORD( m_cbReadOffset );
|
|
m_ReadOverlapped.Overlapped.OffsetHigh = HIDWORD( m_cbReadOffset );
|
|
m_ReadOverlapped.Overlapped.pfnCompletion = DropDirReadCompletion;
|
|
m_ReadOverlapped.ThisPtr = (PVOID)this;
|
|
|
|
AddRef();
|
|
|
|
BOOL fRet = FIOReadFile( m_hMail,
|
|
pBuffer, // Buffer
|
|
cbSize, // BytesToRead
|
|
(PFH_OVERLAPPED)&m_ReadOverlapped) ;
|
|
|
|
if(!fRet)
|
|
{
|
|
DWORD err = GetLastError();
|
|
if( err != ERROR_IO_PENDING )
|
|
{
|
|
hr = HRESULT_FROM_WIN32( err );
|
|
|
|
ErrorTrace( (LPARAM)this, "FIOReadFile return hr = 0x%x", hr );
|
|
}
|
|
|
|
}
|
|
|
|
if( FAILED( hr ) )
|
|
{
|
|
Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::WriteFile(
|
|
IN LPVOID pBuffer,
|
|
IN DWORD cbSize
|
|
)
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::WriteFile");
|
|
CDropDir *p = this;
|
|
_ASSERT(pBuffer != NULL);
|
|
_ASSERT(cbSize > 0);
|
|
HRESULT hr = S_OK;
|
|
|
|
m_WriteOverlapped.Offset = LODWORD( m_cbWriteOffset );
|
|
m_WriteOverlapped.OffsetHigh = HIDWORD( m_cbWriteOffset );
|
|
|
|
AddRef();
|
|
|
|
BOOL fRet = AtqWriteFile( m_pAtqContext, // Atq context
|
|
pBuffer, // Buffer
|
|
cbSize, // BytesToRead
|
|
(OVERLAPPED *) &m_WriteOverlapped) ;
|
|
|
|
if(!fRet)
|
|
{
|
|
DWORD err = GetLastError();
|
|
if( err != ERROR_IO_PENDING )
|
|
{
|
|
hr = HRESULT_FROM_WIN32( err );
|
|
|
|
ErrorTrace( (LPARAM)this, "AtqWriteFile return hr = 0x%x", hr );
|
|
}
|
|
|
|
}
|
|
|
|
if( FAILED( hr ) )
|
|
{
|
|
Release();
|
|
}
|
|
return( hr );
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::OnIoWriteCompletion( DWORD cbSize, DWORD dwErr, OVERLAPPED *lpo )
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::OnIoWriteCompletion");
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
if( dwErr != NO_ERROR )
|
|
{
|
|
ErrorTrace( (LPARAM)this, "OnIoWriteCompletion got err = %u", dwErr );
|
|
return( HRESULT_FROM_WIN32( dwErr ) );
|
|
}
|
|
|
|
m_cbWriteOffset += cbSize;
|
|
|
|
switch( m_ReadState )
|
|
{
|
|
case DROPDIR_READ_X_SENDER:
|
|
_ASSERT(!"Invalid State READ_X_SENDER");
|
|
hr = E_INVALIDARG;
|
|
break;
|
|
case DROPDIR_READ_X_RECEIVER:
|
|
hr = CreateXRecvHeaders();
|
|
break;
|
|
case DROPDIR_READ_DATA:
|
|
hr = OnCopyMessageWrite( cbSize);
|
|
break;
|
|
default:
|
|
_ASSERT(!"INVALID read STATE");
|
|
hr = E_INVALIDARG;
|
|
ErrorTrace( (LPARAM)this, "Invalid ReadState" );
|
|
break;
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::OnIoReadCompletion( DWORD cbSize, DWORD dwErr, OVERLAPPED *lpo )
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::OnIoReadCompletion");
|
|
HRESULT hr = S_OK;
|
|
|
|
if( ( dwErr != NO_ERROR ) && ( dwErr != ERROR_HANDLE_EOF ) )
|
|
{
|
|
hr = HRESULT_FROM_WIN32( dwErr );
|
|
|
|
ErrorTrace( (LPARAM)this, "OnIOReadCompletion got hr = 0x%x", hr );
|
|
}
|
|
else
|
|
{
|
|
m_cbReadOffset += cbSize;
|
|
hr = OnCopyMessageRead( cbSize, dwErr );
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::CreateXHeaders()
|
|
{
|
|
|
|
DECL_TRACE(((LPARAM) this), "CDropDir::CreateXHeaders");
|
|
HRESULT hr = S_OK;
|
|
|
|
DebugTrace( (LPARAM)this, "Setting m_ReadState = DROPDIR_READ_X_SENDER" );
|
|
|
|
m_ReadState = DROPDIR_READ_X_SENDER;
|
|
|
|
strcpy( m_szBuffer, X_SENDER_HEADER );
|
|
|
|
hr = m_pIMsg->GetStringA( IMMPID_MP_SENDER_ADDRESS_SMTP, MAX_INTERNET_NAME, &m_szBuffer[ sizeof(X_SENDER_HEADER) - 1] );
|
|
|
|
if(SUCCEEDED(hr))
|
|
{
|
|
strcat( m_szBuffer, X_HEADER_EOLN);
|
|
|
|
//
|
|
// this WriteFile will complete async, so set the next read state.
|
|
// Make Sure that no class member variables are acessed after calling
|
|
// WriteFile or ReadFile as they complete async & there is no critical
|
|
// section protecting access.To do this trace through all paths
|
|
// thru which ReadFile and WriteFile are called and make sure no member
|
|
// variables are accessed.
|
|
|
|
m_ReadState = DROPDIR_READ_X_RECEIVER;
|
|
|
|
DebugTrace( (LPARAM)this, "Setting m_ReadState = DROPDIR_READ_X_RECEIVER" );
|
|
//
|
|
|
|
if( FAILED( hr = WriteFile( m_szBuffer, strlen( m_szBuffer ) ) ) )
|
|
{
|
|
ErrorTrace((LPARAM)this, "Error %d writing x-sender line %s", hr, m_szBuffer);
|
|
|
|
return( hr );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
|
|
|
|
hr = HRESULT_FROM_WIN32( ERROR_INVALID_DATA );
|
|
|
|
ErrorTrace((LPARAM)this, "Could not get Sender Address returning hr = 0x%x", hr);
|
|
|
|
return( hr );
|
|
}
|
|
|
|
return( hr );
|
|
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::CreateXRecvHeaders()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::CreateXRecvHeaders");
|
|
HRESULT hr = S_OK;
|
|
DWORD dwRecipientFlags = 0;
|
|
BOOL fPendingWrite = FALSE;
|
|
|
|
strcpy( m_szBuffer, X_RECEIVER_HEADER );
|
|
|
|
while( m_idxRecips < m_NumRcpts )
|
|
{
|
|
hr = m_pIMsgRecips->GetDWORD( m_rgRcptIndexList[m_idxRecips], IMMPID_RP_RECIPIENT_FLAGS,&dwRecipientFlags);
|
|
|
|
if( SUCCEEDED( hr ) )
|
|
{
|
|
if( RP_HANDLED != ( dwRecipientFlags & RP_HANDLED ) )
|
|
{
|
|
hr = m_pIMsgRecips->GetStringA( m_rgRcptIndexList[m_idxRecips],
|
|
IMMPID_RP_ADDRESS_SMTP,
|
|
MAX_INTERNET_NAME,
|
|
&m_szBuffer[ sizeof(X_RECEIVER_HEADER) - 1 ]);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
strcat(m_szBuffer, X_HEADER_EOLN);
|
|
|
|
|
|
|
|
// Make Sure that no class member variables are acessed after calling
|
|
// WriteFile or ReadFile as they complete async & there is no critical
|
|
// section protecting access.To do this trace through all paths
|
|
// thru which ReadFile and WriteFile are called and make sure no member
|
|
// variables are accessed.
|
|
|
|
m_idxRecips++;
|
|
|
|
if(FAILED( hr = WriteFile(m_szBuffer, strlen(m_szBuffer) ) ) )
|
|
|
|
{
|
|
ErrorTrace( (LPARAM)this, "WriteFile return hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
fPendingWrite = TRUE;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
ErrorTrace( (LPARAM)this, "GetStringA for IMMPID_RP_ADDRESS_SMTP returns hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
m_idxRecips++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ErrorTrace( (LPARAM)this, "GetDWORD for IMMPID_RP_RECIPIENT_FLAGS returns hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
}
|
|
|
|
if( !fPendingWrite )
|
|
{
|
|
m_ReadState = DROPDIR_READ_DATA;
|
|
m_WriteState = DROPDIR_WRITE_DATA;
|
|
DebugTrace( (LPARAM)this, "Setting m_WriteState & m_ReadState to WRITE_DATA & READ_DATA respectively" );
|
|
return( CopyMessage() );
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::CopyMessage()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::CopyMessage");
|
|
|
|
// Make Sure that no class member variables are acessed after calling
|
|
// WriteFile or ReadFile as they complete async & there is no critical
|
|
// section protecting access.To do this trace through all paths
|
|
// thru which ReadFile and WriteFile are called and make sure no member
|
|
// variables are accessed.
|
|
|
|
HRESULT hr = ReadFile( m_szBuffer, sizeof( m_szBuffer ) );
|
|
|
|
if( hr == HRESULT_FROM_WIN32( ERROR_HANDLE_EOF ) )
|
|
{
|
|
m_WriteState = DROPDIR_WRITE_CRLF;
|
|
|
|
DebugTrace( (LPARAM)this, "ReadFile returns err = ERROR_HANDLE_EOF, Setting m_WriteState = DROPDIR_WRITE_CRLF" );
|
|
|
|
return( DoLastFileOp() );
|
|
}
|
|
|
|
return( hr );
|
|
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::OnCopyMessageRead( DWORD dwBytesRead, DWORD dwErr )
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::OnCopyMessageRead");
|
|
HRESULT hr = S_OK;
|
|
|
|
if (dwBytesRead )
|
|
{
|
|
if (dwBytesRead > 4)
|
|
{
|
|
CopyMemory(m_acLastBytes, &m_szBuffer[dwBytesRead-5], 5);
|
|
}
|
|
else
|
|
{
|
|
MoveMemory(m_acLastBytes, &m_acLastBytes[dwBytesRead], 5-dwBytesRead);
|
|
CopyMemory(&m_acLastBytes[5-dwBytesRead], m_szBuffer, dwBytesRead);
|
|
}
|
|
|
|
//
|
|
// the write file will complete async & then we'll do the
|
|
// last fileop.
|
|
//
|
|
|
|
if( dwErr == ERROR_HANDLE_EOF )
|
|
{
|
|
DebugTrace( (LPARAM)this, " Interesting case err = ERROR_HANDLE_EOF, Setting m_WriteState = DROPDIR_WRITE_CRLF" );
|
|
|
|
m_WriteState = DROPDIR_WRITE_CRLF;
|
|
}
|
|
|
|
// Make Sure that no class member variables are acessed after calling
|
|
// WriteFile or ReadFile as they complete async & there is no critical
|
|
// section protecting access.To do this trace through all paths
|
|
// thru which ReadFile and WriteFile are called and make sure no member
|
|
// variables are accessed.
|
|
|
|
if ( FAILED( hr = WriteFile(m_szBuffer, dwBytesRead ) ) )
|
|
{
|
|
ErrorTrace( (LPARAM)this, "WriteFile return hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
m_WriteState = DROPDIR_WRITE_CRLF;
|
|
|
|
DebugTrace( (LPARAM)this, "Setting m_WriteState = DROPDIR_WRITE_CRLF" );
|
|
|
|
return( DoLastFileOp() );
|
|
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::OnCopyMessageWrite( DWORD cbWritten )
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::OnCopyMessageWrite");
|
|
HRESULT hr = S_OK;
|
|
|
|
m_cbMsgWritten += cbWritten;
|
|
|
|
switch( m_WriteState )
|
|
{
|
|
case DROPDIR_WRITE_DATA:
|
|
return( CopyMessage() );
|
|
break;
|
|
case DROPDIR_WRITE_CRLF:
|
|
return( DoLastFileOp() );
|
|
break;
|
|
case DROPDIR_WRITE_SETPOS:
|
|
return( AdjustFilePos() );
|
|
break;
|
|
default:
|
|
_ASSERT(!"Invalid WriteState");
|
|
ErrorTrace( (LPARAM)this, "Invalid WriteState" );
|
|
hr = E_INVALIDARG;
|
|
break;
|
|
}
|
|
|
|
return( hr );
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::DoLastFileOp()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::DoLastFileOp");
|
|
HRESULT hr = S_OK;
|
|
DebugTrace( (LPARAM)this, "DoLastFileOp called" );
|
|
|
|
// Now, see if the file ends with a CRLF, if not, add it
|
|
if ((m_cbMsgWritten > 1) && memcmp(&m_acLastBytes[3], &m_acCrLfDotCrLf[3], 2))
|
|
{
|
|
//
|
|
// the write will complete async. set next write state
|
|
//
|
|
|
|
DebugTrace( (LPARAM)this, "Setting m_WriteState = DROPDIR_WRITE_SETPOS" );
|
|
|
|
m_WriteState = DROPDIR_WRITE_SETPOS;
|
|
|
|
// Make Sure that no class member variables are acessed after calling
|
|
// WriteFile or ReadFile as they complete async & there is no critical
|
|
// section protecting access.To do this trace through all paths
|
|
// thru which ReadFile and WriteFile are called and make sure no member
|
|
// variables are accessed.
|
|
|
|
// Add the trailing CRLF
|
|
if(FAILED( hr = WriteFile(m_acCrLfDotCrLf, 2 ) ) )
|
|
{
|
|
ErrorTrace( (LPARAM)this, "WriteFile returns hr = 0x%x", hr );
|
|
return(hr);
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
DebugTrace( (LPARAM)this, "Setting m_WriteState = DROPDIR_WRITE_SETPOS" );
|
|
|
|
m_WriteState = DROPDIR_WRITE_SETPOS;
|
|
|
|
return( AdjustFilePos() );
|
|
}
|
|
|
|
|
|
return( hr );
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::AdjustFilePos()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::AdjustFilePos");
|
|
HRESULT hr = S_OK;
|
|
|
|
DebugTrace( (LPARAM)this, "AdjustFilePos called" );
|
|
|
|
//If file ends with CRLF.CRLF, remove the trailing .CRLF
|
|
//Do not modify the file otherwise
|
|
|
|
DWORD dwLo = LODWORD( m_cbWriteOffset );
|
|
DWORD dwHi = HIDWORD( m_cbWriteOffset );
|
|
|
|
if ((m_cbMsgWritten > 4) && !memcmp(m_acLastBytes, m_acCrLfDotCrLf, 5))
|
|
{
|
|
DebugTrace( (LPARAM)this, "Removing the trailing CRLF-DOT-CRLF" );
|
|
|
|
// Remove the trailing .CRLF
|
|
if ((SetFilePointer(m_hDrop, dwLo-3, (LONG*)&dwHi, FILE_BEGIN) == 0xffffffff) ||
|
|
!SetEndOfFile(m_hDrop))
|
|
{
|
|
hr = HRESULT_FROM_WIN32( GetLastError() );
|
|
ErrorTrace( (LPARAM)this, "SetFilePointer or SetEndofFile return hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
}
|
|
|
|
//We need to flush the context before ACK'ing the message
|
|
if (!FlushFileBuffers(m_hDrop))
|
|
{
|
|
hr = HRESULT_FROM_WIN32( GetLastError() );
|
|
if (SUCCEEDED(hr))
|
|
hr = E_FAIL;
|
|
ErrorTrace( (LPARAM)this, "FlushFileBuffers failed with hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
|
|
hr = SetAllRcptsHandled();
|
|
if( FAILED( hr ) )
|
|
{
|
|
ErrorTrace( (LPARAM)this, "SetAllRcptsHandledreturn hr = 0x%x", hr );
|
|
return( hr );
|
|
}
|
|
|
|
return( hr );
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
BOOL CDropDir::CheckIfAllRcptsHandled()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::CheckIfAllRcptsHandled");
|
|
BOOL fRet = TRUE;
|
|
|
|
for( DWORD i = 0; i < m_NumRcpts; i++ )
|
|
{
|
|
if ( m_rgRcptIndexList[i] != INVALID_RCPT_IDX_VALUE)
|
|
{
|
|
DWORD dwRecipientFlags = 0;
|
|
HRESULT hr = m_pIMsgRecips->GetDWORD(m_rgRcptIndexList[i], IMMPID_RP_RECIPIENT_FLAGS,&dwRecipientFlags);
|
|
if (FAILED(hr))
|
|
{
|
|
fRet = FALSE;
|
|
break;
|
|
}
|
|
|
|
if( RP_HANDLED != ( dwRecipientFlags & RP_HANDLED ) )
|
|
{
|
|
fRet = FALSE;
|
|
break;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return( fRet );
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CDropDir::SetAllRcptsHandled()
|
|
{
|
|
DECL_TRACE((LPARAM) this, "CDropDir::SetAllRcptsHandled");
|
|
HRESULT hr = S_OK;
|
|
|
|
for( DWORD i = 0; i < m_NumRcpts; i++ )
|
|
{
|
|
if (m_rgRcptIndexList[i] != INVALID_RCPT_IDX_VALUE)
|
|
{
|
|
DWORD dwRecipientFlags = 0;
|
|
hr = m_pIMsgRecips->GetDWORD(m_rgRcptIndexList[i], IMMPID_RP_RECIPIENT_FLAGS,&dwRecipientFlags);
|
|
if (FAILED(hr))
|
|
{
|
|
break;
|
|
}
|
|
|
|
if( RP_HANDLED != ( dwRecipientFlags & RP_HANDLED ) )
|
|
{
|
|
dwRecipientFlags |= RP_DELIVERED;
|
|
|
|
|
|
hr = m_pIMsgRecips->PutDWORD(m_rgRcptIndexList[i], IMMPID_RP_RECIPIENT_FLAGS,dwRecipientFlags);
|
|
if (FAILED(hr))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
return( hr );
|
|
}
|