/*++ Copyright (c) 1996 Microsoft Corporation Module Name: smtpinst.cxx Abstract: This module defines the SMTP_SERVER_INSTANCE class Author: Johnson Apacible (JohnsonA) June-04-1996 Revision History: David Howell (dhowell) May-1997 Added Etrn Logic Nimish Khanolkar (NimishK) Jan - 1998 - modified for CAPI store ce --*/ #define INCL_INETSRV_INCS #include "smtpinc.h" #include #include "iiscnfg.h" #include #include #include #include #include #include "smtpcli.hxx" #include "dirnot.hxx" #include // SEO Header files #define _ATL_NO_DEBUG_CRT #define _ATL_STATIC_REGISTRY 1 #define _ASSERTE _ASSERT #define _WINDLL #include "atlbase.h" extern CComModule _Module; #include "atlcom.h" #undef _WINDLL #include "smtpsvr.h" #include "seo.h" #include "seolib.h" #include "smtpdisp.h" #include "seodisp.h" #include "seo_i.c" #include "tran_evntlog.h" extern "C" { int strcasecmp(char *s1, char *s2); int strncasecmp(char *s1, char *s2, int n); } #define STORE_DRIVER_PROGID L"Exchange.NtfsDrv" #define SMTPSERVER_PROGID L"SMTPServer.SMTPServer.1" #if 0 extern VOID ServerEventCompletion( PVOID pvContext, DWORD cbWritten, DWORD dwCompletionStatus, OVERLAPPED * lpo ); #endif //extern "C++" { //BOOL g_IsShuttingDown = FALSE; //} DWORD BuildInitialQueueProc(void *lpThis); VOID ProcessInitialQueueObjects(PVOID pvContext, DWORD cbWritten, DWORD dwCompletionStatus, OVERLAPPED *lpo); DWORD EnumAllDomains(void *ptr); // // Constants // // // Globals // #define MAX_CONNECTION_OBJECTS 5000 PFN_SF_NOTIFY g_pSslKeysNotify = NULL; extern STORE_CHANGE_NOTIFIER *g_pCAPIStoreChangeNotifier; SmtpMappingSupportFunction( PVOID pvInstance, PVOID pData, DWORD dwPropId); SERVICE_MAPPING_CONTEXT g_SmtpSMC = { SmtpMappingSupportFunction}; // // Prototypes // BOOL SetSslKeysNotify( PFN_SF_NOTIFY pFn ); /************************************************************ * Symbolic Constants ************************************************************/ static TCHAR szServicePath[] = TEXT("System\\CurrentControlSet\\Services\\"); static TCHAR szParametersKey[] = TEXT("\\Parameters"); static TCHAR szParamPath[] = TEXT("System\\CurrentControlSet\\Services\\SmtpSvc\\Parameters"); static WCHAR szParamPathW[] = L"System\\CurrentControlSet\\Services\\SmtpSvc\\Parameters"; static TCHAR szTcpipPath[] = TEXT("System\\CurrentControlSet\\Services\\Tcpip\\Parameters"); static TCHAR szTcpipTransient[] = TEXT("Transient"); static TCHAR szMaxSmtpErrors[] = TEXT("MaxErrors"); static WCHAR szMaxSmtpErrorsW[] = L"MaxErrors"; static TCHAR szMaxRemoteTimeOut[] = TEXT("MaxRemoteTimeOut"); static WCHAR szMaxRemoteTimeOutW[] = L"MaxRemoteTimeOut"; static TCHAR szMaxMsgSize[] = TEXT("MaxMsgSize"); static WCHAR szMaxMsgSizeW[] = L"MaxMsgSize"; static TCHAR szMaxMsgSizeBeforeClose[] = TEXT("MaxMsgSizeBeforeClose"); static WCHAR szMaxMsgSizeBeforeCloseW[] = L"MaxMsgSizeBeforeClose"; static TCHAR szMaxRcpts[] = TEXT("MaxRcpts"); static WCHAR szMaxRcptsW[] = L"MaxRcpts"; static TCHAR szEnableReverseLookup[] = TEXT("EnableReverseLookup"); static WCHAR szEnableReverseLookupW[] = L"EnableReverseLookup"; static TCHAR szDomains[] = TEXT("Domains"); static WCHAR szDomainsW[] = L"Domains"; static TCHAR szNameResolution[] = TEXT("NameResolution"); static WCHAR szNameResolutionW[] = L"NameResolution"; static TCHAR szSmartHostType[] = TEXT("SmartHostUseType"); static WCHAR szSmartHostTypeW[] = L"SmartHostUseType"; static TCHAR szRetryAttempts[] = TEXT("MaxRetryAttempts"); static WCHAR szRetryAttemptsW[] = L"MaxRetryAttempts"; static TCHAR szRetryMinutes[] = TEXT("MaxRetryMinutes"); static WCHAR szRetryMinutesW[] = L"MaxRetryMinutes"; static TCHAR szShouldPipelineOut[] = TEXT("PipelineOutput"); static WCHAR szShouldPipelineOutW[] = L"PipelineOutput"; static TCHAR szShouldPipelineIn[] = TEXT("PipelineInput"); static WCHAR szShouldPipelineInW[] = L"PipelineInput"; static TCHAR szMaxHopCount[] = TEXT("MaxHopCount"); static WCHAR szMaxHopCountW[] = L"MaxHopCount"; static TCHAR szMaxOutConnections[] = TEXT("MaxOutConnections"); static WCHAR szMaxOutConnectionsW[] = L"MaxOutConnections"; static TCHAR szSendBadToAdmin[] = TEXT("SendBadToAdmin"); static WCHAR szSendBadToAdminW[] = L"SendBadToAdmin"; static TCHAR szSendNDRToAdmin[] = TEXT("SentNDRToAdmin"); static WCHAR szSendNDRToAdminW[] = L"SentNDRToAdmin"; static TCHAR szRoutingSources[] = TEXT("RoutingSources"); static WCHAR szRoutingSourcesW[] = L"RoutingSources"; static TCHAR szRoutingThreads[] = TEXT("RoutingThreads"); static WCHAR szRoutingThreadsW[] = L"RoutingThreads"; static TCHAR szDirBuffers[] = TEXT("MaxDirectoryBuffers"); static WCHAR szDirBuffersW[] = L"MaxDirectoryBuffers"; static TCHAR szDirBuffersSize[] = TEXT("DirectoryBuffSize"); static WCHAR szDirBuffersSizeW[] = L"DirectoryBufferSize"; static TCHAR szDirPendingIos[] = TEXT("NumDirPendingIos"); static WCHAR szDirPendingIosW[] = L"NumDirPendingIos"; static TCHAR szBadMailDir[] = TEXT("BadMailDir"); static WCHAR szBadMailDirW[] = L"BadMailDir"; static TCHAR szMailQueueDir[] = TEXT("MailQueueDir"); static WCHAR szMailQueueDirW[] = L"MailQueueDir"; static TCHAR szShouldDeliver[] = TEXT("ShouldDeliver"); static WCHAR szShouldDeliverW[] = L"ShouldDeliver"; static TCHAR szShouldDelete[] = TEXT("ShouldDelete"); static WCHAR szShouldDeleteW[] = L"ShouldDelete"; static TCHAR szDeleteDir[] = TEXT("DeleteDir"); static WCHAR szDeleteDirW[] = L"DeleteDir"; static TCHAR szMaxAddrObjects[] = TEXT("MaxAddressObjects"); static WCHAR szMaxAddrObjectsW[] = L"MaxAddressObjects"; static TCHAR szMaxMailObjects[] = TEXT("MaxMailObjects"); static WCHAR szMaxMailObjectsW[] = L"MaxMailObjects"; static TCHAR szRoutingDll[] = TEXT("RoutingDll"); static WCHAR szRoutingDllW[] = L"RoutingDll"; static TCHAR szUseFileLinks[] = TEXT("UseFileLinks"); static WCHAR szUseFileLinksW[] = L"UseFileLinks"; static TCHAR szMsgBatchLimit[] = TEXT("BatchMsgLimit"); static WCHAR szMsgBatchLimitW[] = L"BatchMsgLimit"; static TCHAR szMailPickupDir[] = TEXT("MailPickupDir"); static WCHAR szMailPickupDirW[] = L"MailPickupDir"; static TCHAR szMailDropDir[] = TEXT("MailDropDir"); static WCHAR szMailDropDirW[] = L"MailDropDir"; static TCHAR szShouldPickupMail[] = TEXT("ShouldPickupMail"); static WCHAR szShouldPickupMailW[] = L"ShouldPickupMail"; static TCHAR szCommandLogMask[] = TEXT("CommandLogMask"); static WCHAR szCommandLogMaskW[] = L"CommandLogMask"; static TCHAR szShowEightBitMime[] = TEXT("ShowEightBitMime"); static WCHAR szShowEightBitMimeW[] = L"ShowEightBitMime"; static TCHAR szShowBinaryMime[] = TEXT("ShowBinaryMime"); static WCHAR szShowBinaryMimeW[] = L"ShowBinaryMime"; static TCHAR szShowChunking[] = TEXT("ShowChunking"); static WCHAR szShowChunkingW[] = L"ShowChunking"; static TCHAR szFlushMailFiles[] = TEXT("FlushMailFiles"); static WCHAR szFlushMailFilesW[] = L"FlushMailFiles"; static TCHAR szVirtualRoot[] = TEXT("Virtual Roots"); static WCHAR szVirtualRootW[] = L"Virtual Roots"; static TCHAR szRRetryAttempts[] = TEXT("MaxRemoteRetryAttempts"); static WCHAR szRRetryAttemptsW[] = L"MaxRemoteRetryAttempts"; static TCHAR szRRetryMinutes[] = TEXT("MaxRemoteRetryMinutes"); static WCHAR szRRetryMinutesW[] = L"MaxRemoteRetryMinutes"; static TCHAR szShareRetryMinutes[] = TEXT("MaxShareRetryMinutes"); static WCHAR szShareRetryMinutesW[] = L"MaxShareRetryMinutes"; // Always use ANSI for Internet compatibility, even if UNICODE is defined static WCHAR szDefaultDomainW[] = L"DefaultDomain"; static TCHAR szDefaultDomain[] = TEXT("DefaultDomain"); static WCHAR szConnectResponseW[] = L"ConnectResponse"; static TCHAR szConnectResponse[] = TEXT("ConnectResponse"); static WCHAR szSmartHostNameW[] = L"SmartHost"; static TCHAR szSmartHostName[] = TEXT("SmartHost"); // // Added by keithlau on 7/12/96 // #define SMTP_EVENTLOG_MAX_ITEMS 10 #define SMTP_INIT_ABSORT 0x00000001 #define SMTP_INIT_CSLOCK 0x00000002 #define SMTP_INIT_OUTLOCK 0x00000004 #define SMTP_INIT_GENLOCK 0x00000008 const LPSTR pszPackagesDefault = "NTLM\0GSSAPI\0"; const DWORD ccbPackagesDefault = sizeof( "NTLM\0GSSAPI\0" ); extern AQ_INITIALIZE_EX_FUNCTION g_pfnInitializeAQ; extern AQ_DEINITIALIZE_EX_FUNCTION g_pfnDeinitializeAQ; extern HRESULT CallInstanceInitStoreDriver(DWORD InstanceId, IAdvQueue *pIAq, char * UserName, char * DomainName, char * Password, char * DnToUse); extern void CallInstanceDeInitStoreDriver(DWORD InstanceId); //extern HRESULT HrGetGatewayDN(IN OUT DWORD *pcbBuffer, // IN LPSTR szBuffer); inline BOOL ConvertToMultisz(LPSTR szMulti, DWORD *pdwCount, LPSTR szAuthPack) { CHAR *pcStart = szAuthPack, *pc; DWORD dw = 0; pc = pcStart; if (*pc == '\0' || *pc == ',') return FALSE; *pdwCount = 0; while (TRUE) { if (*pc == '\0') { strcpy(&szMulti[dw], pcStart); (*pdwCount)++; dw += lstrlen(pcStart); szMulti[dw + 1] = '\0'; return TRUE; } else if (*pc == ',') { *pc = '\0'; lstrcpy(&szMulti[dw], pcStart); (*pdwCount)++; dw += lstrlen(pcStart); dw++; *pc = ','; pcStart = ++pc; } else { pc++; } } } // // Quick and dirty range check using inlines (KeithLau 7/28/96) // static inline BOOL pValidateRange(DWORD dwValue, DWORD dwLower, DWORD dwUpper) { // Inclusive if ((dwValue >= dwLower) && (dwValue <= dwUpper)) return (TRUE); SetLastError(ERROR_INVALID_PARAMETER); return (FALSE); } // // Quick and dirty string validation // static inline BOOL pValidateStringPtr(LPWSTR lpwszString, DWORD dwMaxLength) { if (IsBadStringPtr((LPCTSTR)lpwszString, dwMaxLength)) return (FALSE); while (dwMaxLength--) if (*lpwszString++ == 0) return (TRUE); return (FALSE); } static inline BOOL ConvertFromUnicode(LPWSTR pwsz, char * psz, DWORD cchMax) /*++ Converts a given string into unicode string Returns FALSE on failure. Use GetLastError() for details. --*/ { DWORD cch; cch = lstrlenW(pwsz) + 1; if (cchMax < cch) { SetLastError(ERROR_NOT_ENOUGH_MEMORY); return FALSE; } int iRet; iRet = WideCharToMultiByte(CP_ACP, 0, pwsz, cch, psz, cchMax, NULL, NULL); return (iRet != 0); } // ConvertToUnicode() // // Enumeration stuff // DWORD WINAPI EnumBuildQProc(LPVOID pvContext) { SMTP_SERVER_INSTANCE *pInst = (SMTP_SERVER_INSTANCE *)pvContext; _ASSERT(pInst); pInst->TriggerStoreServerEvent(SMTP_STOREDRV_ENUMMESS_EVENT); return (NO_ERROR); } DWORD InitializeInstances( PSMTP_IIS_SERVICE pService ) /*++ Routine Description: Reads the instances from the metabase Arguments: pService - Server instances added to. Return Value: Win32 --*/ { DWORD i; DWORD cInstances = 0; MB mb( (IMDCOM*) pService->QueryMDObject() ); CHAR szKeyName[MAX_PATH+1]; DWORD err = NO_ERROR; BUFFER buff; BOOL fMigrateRoots = FALSE; // // Open the metabase for write to get an atomic snapshot // ReOpen: if ( !mb.Open( "/LM/SMTPSVC/", METADATA_PERMISSION_READ | METADATA_PERMISSION_WRITE )) { DBGPRINTF(( DBG_CONTEXT, "InitializeInstances: Cannot open path %s, error %lu\n", "/LM/SMTPSVC/", GetLastError() )); // // If the web service key isn't here, just create it // if ( !mb.Open(METADATA_MASTER_ROOT_HANDLE, METADATA_PERMISSION_READ | METADATA_PERMISSION_WRITE ) || !mb.AddObject( "/LM/SMTPSVC/" )) { err = GetLastError(); return err; } DBGPRINTF(( DBG_CONTEXT, "/LM/SMTPSvc not found, auto-created\n" )); mb.Close(); goto ReOpen; } //_VERIFY(mb.SetString("", MD_NTAUTHENTICATION_PROVIDERS, IIS_MD_UT_SERVER, "NTLM,LOGIN")); DWORD dwVersion; if (!mb.GetDword("", MD_SMTP_SERVICE_VERSION, IIS_MD_UT_SERVER, &dwVersion)) { _VERIFY(mb.SetDword("", MD_SMTP_SERVICE_VERSION, IIS_MD_UT_SERVER, g_ProductType)); } //_VERIFY(mb.SetDword("", MD_SMTP_AUTHORIZATION, IIS_MD_UT_SERVER, 7)); //_VERIFY(mb.SetDword("", MD_AUTHORIZATION, IIS_MD_UT_SERVER, 7)); // // Loop through instance keys and build a list. We don't keep the // metabase open because the instance instantiation code will need // to write to the metabase // TryAgain: i = 0; while ( mb.EnumObjects( "", szKeyName, i++ )) { DWORD dwInstance; // // Get the instance id // DBGPRINTF((DBG_CONTEXT,"instance key %s\n",szKeyName)); dwInstance = atoi( szKeyName ); if ( dwInstance == 0 ) { continue; } if ( buff.QuerySize() < (cInstances + 1) * sizeof(DWORD) ) { if ( !buff.Resize( (cInstances + 10) * sizeof(DWORD)) ) { return GetLastError(); } } ((DWORD *) buff.QueryPtr())[cInstances++] = dwInstance; } if ( cInstances == 0 ) { DBGPRINTF(( DBG_CONTEXT, "No defined instances\n" )); if ( !mb.AddObject( "1" )) { DBGPRINTF(( DBG_CONTEXT, "Unable to create first instance, error %d\n", GetLastError() )); return GetLastError(); } fMigrateRoots = TRUE; // Force reg->metabase migration of virtual directories goto TryAgain; } DBG_REQUIRE( mb.Close() ); mb.Save(); for ( i = 0; i < cInstances; i++ ) { DWORD dwInstance = ((DWORD *)buff.QueryPtr())[i]; if ( !g_pInetSvc->AddInstanceInfo( dwInstance, fMigrateRoots ) ) { err = GetLastError(); DBGPRINTF((DBG_CONTEXT, "InitializeInstances: cannot create instance %lu, error %lu\n", dwInstance, err)); //break; } } return err; } // InitializeInstances //+--------------------------------------------------------------- // // Function: StopInstance // // Synopsis: Callback from IIS_SERVICE iterator // // Arguments: void // // Returns: TRUE is success, else FALSE // // History: HowardCu Created 23 May 1995 // //---------------------------------------------------------------- BOOL StopSmtpInstances( PVOID pvContext, PVOID pvContext1, PIIS_SERVER_INSTANCE pInstance ) { PSMTP_SERVER_INSTANCE pSmtpInstance = (PSMTP_SERVER_INSTANCE)pInstance; if ( !pSmtpInstance->Stop() ) { } return TRUE; } /*++ Routine Description: Shutdown each instance and terminate all global cpools Arguments: pService - Server instances added to. Return Value: Win32 --*/ VOID TerminateInstances( PSMTP_IIS_SERVICE pService) { PFN_INSTANCE_ENUM pfnInstanceEnum = NULL; TraceFunctEnter("TerminateInstances"); // // Iterate over all instances // StopInstance callback calls SMTP_SERVER_INSTANCE::Stop() // pfnInstanceEnum = (PFN_INSTANCE_ENUM)&StopSmtpInstances; if ( !pService->EnumServiceInstances( NULL, NULL, pfnInstanceEnum ) ) { ErrorTrace(0,"Error enumerating instances"); } TraceFunctLeave(); } SMTP_SERVER_INSTANCE::SMTP_SERVER_INSTANCE( IN PSMTP_IIS_SERVICE pService, IN DWORD dwInstanceId, IN USHORT Port, IN LPCSTR lpszRegParamKey, IN LPWSTR lpwszAnonPasswordSecretName, IN LPWSTR lpwszVirtualRootsSecretName, IN BOOL fMigrateRoots ) : IIS_SERVER_INSTANCE(pService, dwInstanceId, Port, lpszRegParamKey, lpwszAnonPasswordSecretName, lpwszVirtualRootsSecretName, fMigrateRoots) //m_signature (SMTP_SERVER_INSTANCE_SIGNATURE) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE" ); m_pSmtpStats = NULL; m_IsShuttingDown = FALSE; m_QIsShuttingDown = FALSE; m_RetryQIsShuttingDown = FALSE; m_IsFileSystemNtfs = TRUE; m_fShouldStartAcceptingConnections = TRUE; m_SmtpInitializeStatus = 0; m_cCurrentConnections = 0; m_cMaxCurrentConnections = 0; m_cCurrentOutConnections = 0; m_dwNextInboundClientId = 1; m_dwNextOutboundClientId = 1; m_dwStopHint = 2; m_cProcessClientThreads = 0; m_cNumConnInObjsAlloced = 0; m_cNumConnOutObjsAlloced = 0; m_cNumCBufferObjsAlloced = 0; m_cNumAsyncObjsAlloced = 0; m_cNumAsyncDnsObjsAlloced = 0; m_cchConnectResponse = 0; m_cCurrentRoutingThreads = 0; m_cMaxOutConnectionsPerDomain = 0; m_cMaxOutConnections = 0; m_cMaxHopCount = 0; m_RemoteSmtpPort = 0; m_cMaxRcpts = 0; m_fMasquerade = FALSE; m_fIgnoreTime = FALSE; m_fStartRetry = FALSE; m_fRequiresSSL = FALSE; m_fRequires128Bits = FALSE; m_fRequiresCertVerifySubject= FALSE; m_fRequiresCertVerifyIssuer = FALSE; m_pSSLInfo = NULL; m_fLimitRemoteConnections = TRUE; m_RDNSOptions = 0; m_fAllowVerify = FALSE; m_InstBooted = FALSE; m_fStoreDrvStartEventCalled = FALSE; m_fStoreDrvPrepShutDownEventCalled = FALSE; m_fScheduledConnection = FALSE; m_fIsRoutingTable = TRUE; m_fHaveRegisteredPrincipalNames = FALSE; m_fHelloNoValidate = FALSE; m_fMailNoValidate = FALSE; m_fRcptNoValidate = FALSE; m_fEtrnNoValidate = FALSE; m_fPickupNoValidate = FALSE; m_pProviderPackagesInfo = NULL; fInitializedAQ = FALSE; fInitializedStoreDriver = FALSE; // Raid 174038 m_fDisablePickupDotStuff = FALSE; m_szMasqueradeName [0] = '\0'; m_szMailQueueDir[0] = '\0'; m_szMailPickupDir[0] = '\0'; m_szMailDropDir[0] = '\0'; m_szBadMailDir[0] = '\0'; m_szMyDomain[0] = '\0'; m_szDefaultDomain[0] = '\0'; m_szFQDomainName[0] = '\0'; m_szSmartHostName[0] = '\0'; m_szPostDnsSmarthost[0] = '\0'; m_AdminName[0] = '\0'; m_BadMailName[0] = '\0'; m_DefaultRemoteUserName[0] = '\0'; m_DefaultRemotePassword[0] = '\0'; SmtpDir = NULL; DirPickupThreadHandle = NULL; StopHandle = NULL; m_hEnumDomainThreadHandle = NULL; m_pChangeObject = NULL; m_IAQ = NULL; m_ICM = NULL; m_pvAQInstanceContext = NULL; m_pIAdvQueueConfig = NULL; m_RemoteQ = NULL; m_ComSmtpServer = NULL; m_pSmtpInfo = NULL; m_dwEventlogLevel = LOGEVENT_LEVEL_MEDIUM; m_dwDnsFlags = 0; m_dwDeniedIpAction = 0; InitializeCriticalSection( &m_critBoot ) ; m_fInitAsyncCS = FALSE; m_cServerList = 0; ZeroMemory(m_rgpServerList, sizeof(m_rgpServerList)); m_signature = SMTP_INSTANCE_SIGNATURE; TraceFunctLeaveEx((LPARAM)this); return; } // SMTP_SERVER_INSTANCE::SMTP_SERVER_INSTANCE SMTP_SERVER_INSTANCE::~SMTP_SERVER_INSTANCE( VOID ) { ULONG i = 0; TraceFunctEnterEx((LPARAM)this, "~SMTP_SERVER_INSTANCE" ); EnterCriticalSection( &m_critBoot ); if ( m_InstBooted && !m_fShutdownCalled) { InitiateShutDown(); } else ErrorTrace((LPARAM) this, "Shutdown for instance %d was already called", QueryInstanceId()); for(i = 0; i < m_cServerList; i++) { if(m_rgpServerList[i]) delete m_rgpServerList[i]; } LeaveCriticalSection( &m_critBoot ); DeleteCriticalSection( &m_critBoot ) ; TraceFunctLeaveEx((LPARAM) this); } //--------------------------------------------------------------------------------------- // Description: // Called by InitiateStartup() to initialize members of SMTP_SERVER_INSTANCE when // a VSI is started up. InitiateStartup() is called by IIS to start a VSI. // Arguments: // None. // Returns: // None. //--------------------------------------------------------------------------------------- void SMTP_SERVER_INSTANCE::InitializeClassVariables(void) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE:InitializeClassVariables" ); m_pSmtpStats = NULL; m_IsShuttingDown = FALSE; m_QIsShuttingDown = FALSE; m_RetryQIsShuttingDown = FALSE; m_fShouldStartAcceptingConnections = TRUE; m_SmtpInitializeStatus = 0; m_cCurrentConnections = 0; m_cMaxCurrentConnections = 0; m_cCurrentOutConnections = 0; m_dwNextInboundClientId = 1; m_dwNextOutboundClientId = 1; m_dwStopHint = 2; m_cProcessClientThreads = 0; m_cNumConnInObjsAlloced = 0; m_cNumConnOutObjsAlloced = 0; m_cNumCBufferObjsAlloced = 0; m_cNumAsyncObjsAlloced = 0; m_cNumAsyncDnsObjsAlloced = 0; m_cchConnectResponse = 0; m_fMasquerade = FALSE; m_fIgnoreTime = FALSE; m_fStartRetry = FALSE; m_fLimitRemoteConnections = TRUE; m_fShutdownCalled = FALSE; m_fSendNDRToAdmin = FALSE; m_fSendBadToAdmin = FALSE; m_fRequiresSASL = FALSE; m_szDefaultLogonDomain[0] = '\0'; m_fStoreDrvStartEventCalled = FALSE; m_fStoreDrvPrepShutDownEventCalled = FALSE; //directory pickup stuff SmtpDir = NULL; DirPickupThreadHandle = NULL; StopHandle = NULL; m_hEnumDomainThreadHandle = NULL; // Raid 174038 m_fDisablePickupDotStuff = FALSE; m_QStopEvent = NULL; m_hEnumBuildQ = NULL; InitializeListHead( &m_ConnectionsList); InitializeListHead( &m_OutConnectionsList); InitializeListHead( &m_leVRoots); InitializeListHead( &m_AsynConnectList); InitializeListHead( &m_AsyncDnsList); InitializeCriticalSection( &m_csLock); m_SmtpInitializeStatus |= SMTP_INIT_CSLOCK; _ASSERT(!m_fInitAsyncCS && "Doubly initialized critsec"); if(!m_fInitAsyncCS) { InitializeCriticalSection( &m_csAsyncDns ); InitializeCriticalSection( &m_csAsyncConnect ) ; m_fInitAsyncCS = TRUE; } m_szConnectResponse[0] = '\0'; // Set the logging handle m_InstancePropertyBag.SetLogging(&m_Logging); TraceFunctLeaveEx((LPARAM) this); } BOOL SMTP_SERVER_INSTANCE::AllocNewMessage(SMTP_ALLOC_PARAMS * Params) { TraceFunctEnterEx((LPARAM)this, "AllocNewMessage"); HRESULT hr = S_OK; BOOL fRet = TRUE; Params->m_InstanceId = QueryInstanceId(); Params->m_EventSmtpServer = (PVOID *) m_ComSmtpServer; Params->m_pNotify = NULL; hr = TriggerServerEvent(SMTP_STOREDRV_ALLOC_EVENT, (PVOID) Params); if (FAILED(hr)) { fRet = FALSE; } TraceFunctLeaveEx((LPARAM)this); return fRet; } HRESULT SMTP_SERVER_INSTANCE::SinkReadMetabaseDword(DWORD MetabaseId, DWORD * dwValue) { DWORD tmp = 0; HRESULT hr = S_FALSE; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); if (mb.Open(QueryMDPath())) { if ( dwValue && mb.GetDword("", MetabaseId, IIS_MD_UT_SERVER, &tmp)) { *dwValue = tmp; hr = S_OK; } } return hr; } HRESULT SMTP_SERVER_INSTANCE::SinkReadMetabaseString(DWORD MetabaseId, char * Buffer, DWORD * BufferSize, BOOL fSecure) { STR TempString; HRESULT hr = S_OK; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); DWORD MetaOptions = METADATA_INHERIT; TempString.Reset(); if (fSecure) { MetaOptions |= METADATA_SECURE; } if (mb.Open(QueryMDPath())) { if (!mb.GetStr("", MetabaseId, IIS_MD_UT_SERVER, &TempString, MetaOptions) || TempString.IsEmpty()) { hr = S_FALSE; } else { lstrcpyn(Buffer,TempString.QueryStr(), MAX_PATH); } } return hr; } //---[ SMTP_SERVER_INSTANCE::SinkReadMetabaseData ----------------------------- // // // Description: // Reads arbitraty binary Metabase data (like an ACL) // Parameters: // IN MetabaseId ID to read data from // IN OUT pBuffer Buffer to read data into // IN OUT pcbBufferSize Size of buffer/valid data in buffer // Returns: // S_OK on success // HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER) if buffer size is not // not large enough for requested data. // History: // 6/7/99 - MikeSwa Created // //----------------------------------------------------------------------------- HRESULT SMTP_SERVER_INSTANCE::SinkReadMetabaseData(DWORD MetabaseId, BYTE *pBuffer, DWORD *pcbBufferSize) { HRESULT hr = S_OK; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); if (!pcbBufferSize) return E_INVALIDARG; if (mb.Open(QueryMDPath())) { if (!mb.GetData("", MetabaseId, IIS_MD_UT_SERVER, BINARY_METADATA, pBuffer, pcbBufferSize, METADATA_INHERIT)) { hr = HRESULT_FROM_WIN32(GetLastError()); } } else { hr = HRESULT_FROM_WIN32(GetLastError()); } return hr; } HRESULT SMTP_SERVER_INSTANCE::TriggerLocalDelivery(IMailMsgProperties *pMsg, DWORD dwRecipientCount, DWORD * pdwRecipIndexes, IMailMsgNotify *pNotify) { HRESULT hr = S_FALSE; SMTP_ALLOC_PARAMS AllocParams; AllocParams.IMsgPtr = (PVOID) pMsg; AllocParams.m_InstanceId = QueryInstanceId(); AllocParams.m_RecipientCount = dwRecipientCount; AllocParams.pdwRecipIndexes = pdwRecipIndexes; AllocParams.m_EventSmtpServer = (PVOID *) m_ComSmtpServer; AllocParams.hr = S_OK; AllocParams.m_pNotify = (PVOID) pNotify; _ASSERT(pMsg != NULL); _ASSERT(pdwRecipIndexes != NULL); _ASSERT(dwRecipientCount != 0); ADD_COUNTER (this, NumRcptsRecvdLocal, dwRecipientCount); ADD_COUNTER (this, NumRcptsRecvd, dwRecipientCount); hr = TriggerServerEvent(SMTP_STOREDRV_DELIVERY_EVENT, (PVOID) &AllocParams); // // jstamerj 1998/08/04 17:45:15: // If a sink returned a special error, return that to the caller of TriggerLocalDelivery // if (SUCCEEDED(hr) && FAILED(AllocParams.hr)) hr = AllocParams.hr; return hr; } HRESULT SMTP_SERVER_INSTANCE::TriggerDirectoryDrop(IMailMsgProperties *pMsg, DWORD dwRecipientCount, DWORD * pdwRecipIndexes, LPCSTR DropDirectory) { HRESULT hr = S_FALSE; SMTP_ALLOC_PARAMS AllocParams; AllocParams.IMsgPtr = (PVOID) pMsg; AllocParams.m_InstanceId = QueryInstanceId(); AllocParams.m_RecipientCount = dwRecipientCount; AllocParams.pdwRecipIndexes = pdwRecipIndexes; AllocParams.m_EventSmtpServer = (PVOID *) m_ComSmtpServer; AllocParams.m_DropDirectory = DropDirectory; AllocParams.m_pNotify = NULL; _ASSERT(pMsg != NULL); _ASSERT(pdwRecipIndexes != NULL); _ASSERT(DropDirectory != NULL); _ASSERT(dwRecipientCount != 0); if (DropDirectory != NULL) { // hr = TriggerServerEvent(SMTP_MAIL_DROP_EVENT, (PVOID) &AllocParams); } else { hr = S_FALSE; } return hr; } HRESULT SMTP_SERVER_INSTANCE::TriggerStoreServerEvent(DWORD EventType) { HRESULT hr = S_FALSE; SMTP_ALLOC_PARAMS AllocParams; AllocParams.m_InstanceId = QueryInstanceId(); AllocParams.m_EventSmtpServer = (PVOID *) m_ComSmtpServer; AllocParams.m_dwStartupType = SMTP_INIT_VSERVER_STARTUP; AllocParams.m_pNotify = NULL; hr = TriggerServerEvent(EventType, (PVOID) &AllocParams); return hr; } ///////////////////////////////////////////////////////////////// HRESULT SMTP_SERVER_INSTANCE::TriggerDnsResolverEvent( LPSTR HostName, LPSTR pszFQDN, DWORD dwVirtualServerId, DNS_SERVER_INFO **ppDnsServerInfo, IDnsResolverRecord **ppIDnsResolverRecord ) { HRESULT hr = S_FALSE; EVENTPARAMS_DNSRESOLVERRECORD params; params.pszHostName = HostName; params.pszFQDN = pszFQDN; params.dwVirtualServerId = dwVirtualServerId; params.ppIDnsResolverRecord = ppIDnsResolverRecord; params.ppDnsServerInfo = ppDnsServerInfo; hr = TriggerServerEvent( SMTP_DNSRESOLVERRECORDSINK_EVENT, (PVOID) ¶ms); return hr; } ///////////////////////////////////////////////////////////////// HRESULT SMTP_SERVER_INSTANCE::TriggerMaxMsgSizeEvent( IUnknown *pIUnknown, IMailMsgProperties *pIMailMsg, BOOL *pfShouldImposeLimit ) { HRESULT hr = S_FALSE; EVENTPARAMS_MAXMSGSIZE params; params.pIUnknown = pIUnknown; params.pIMailMsg = pIMailMsg; params.pfShouldImposeLimit = pfShouldImposeLimit; hr = TriggerServerEvent( SMTP_MAXMSGSIZE_EVENT, (PVOID)¶ms ); return( hr ); } ///////////////////////////////////////////////////////////////// void SMTP_SERVER_INSTANCE::WriteLog( LPMSG_TRACK_INFO pMsgTrackInfo, IMailMsgProperties *pMsgProps, LPEVENT_LOG_INFO pEventLogInfo, LPSTR pszProtocolLog ) { // // do the message tracking stuff // if( pMsgTrackInfo || pMsgProps ) { EVENTPARAMS_MSGTRACKLOG msgTrackLogParams; msgTrackLogParams.pIServer = GetInstancePropertyBag(); msgTrackLogParams.pIMailMsgProperties = pMsgProps; msgTrackLogParams.pMsgTrackInfo = pMsgTrackInfo; TriggerServerEvent( SMTP_MSGTRACKLOG_EVENT, (PVOID)&msgTrackLogParams ); } // // do the event log stuff // if( pEventLogInfo ) { SmtpLogEventEx( pEventLogInfo->dwEventId, pEventLogInfo->pszEventLogMsg, pEventLogInfo->dwErrorCode ); } // // do protocol logging stuff // if( pszProtocolLog ) { INETLOG_INFORMATION translog; ZeroMemory( &translog, sizeof( translog ) ); translog.pszOperation = "SMTPSVC_LOG"; translog.cbOperation = strlen ("SMTPSVC_LOG"); translog.pszHTTPHeader = pszProtocolLog; translog.cbHTTPHeaderSize = strlen( pszProtocolLog ); m_Logging.LogInformation( &translog ); } } ///////////////////////////////////////////////////////////////// HRESULT SMTP_SERVER_INSTANCE::TriggerLogEvent( IN DWORD idMessage, IN WORD idCategory, IN WORD cSubstrings, IN LPCSTR *rgszSubstrings, IN WORD wType, IN DWORD errCode, IN WORD iDebugLevel, IN LPCSTR szKey, IN DWORD dwOptions, IN DWORD iMessageString, IN HMODULE hModule) { HRESULT hr = S_OK; EVENTPARAMS_LOG LogParms; SMTP_LOG_EVENT_INFO LogEventInfo; // Construct the log event info LogEventInfo.idMessage = idMessage; LogEventInfo.idCategory = idCategory; LogEventInfo.cSubstrings = cSubstrings; LogEventInfo.rgszSubstrings = rgszSubstrings; LogEventInfo.wType = wType; LogEventInfo.errCode = errCode; LogEventInfo.iDebugLevel = iDebugLevel; LogEventInfo.szKey = szKey; LogEventInfo.dwOptions = dwOptions; LogEventInfo.iMessageString = iMessageString; LogEventInfo.hModule = hModule; // Construct the event context LogParms.pSmtpEventLogInfo = &LogEventInfo; LogParms.pDefaultEventLogHandler = &g_EventLog; LogParms.iSelectedDebugLevel = m_dwEventlogLevel; TriggerServerEvent( SMTP_LOG_EVENT, (PVOID)&LogParms ); return hr; } ///////////////////////////////////////////////////////////////// HRESULT SMTP_SERVER_INSTANCE::ResetLogEvent( IN DWORD idMessage, IN LPCSTR szKey) { HRESULT hr = S_OK; hr = g_EventLog.ResetEvent(idMessage, szKey); return hr; } ///////////////////////////////////////////////////////////////// HRESULT SMTP_SERVER_INSTANCE::HrTriggerGetAuxDomainInfoFlagsEvent( IN LPCSTR pszDomainName, OUT DWORD *pdwDomainInfoFlags ) { HRESULT hr = S_OK; EVENTPARAMS_GET_AUX_DOMAIN_INFO_FLAGS EventParams; DWORD dwStartTicks, dwStopTicks; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE:HrTriggerGetAuxDomainInfoFlagsEvent" ); _ASSERT(pdwDomainInfoFlags); // Before we trigger the event, let's pre-set pdwDomainInfoFlags // to be "DOMAIN_INFO_INVALID" so we get the right result if there // were no sinks *pdwDomainInfoFlags = DOMAIN_INFO_INVALID; // Construct the event params EventParams.pIServer = GetInstancePropertyBag(); EventParams.pszDomainName = pszDomainName; EventParams.pdwDomainInfoFlags = pdwDomainInfoFlags; DebugTrace((LPARAM)this, "Triggering event SMTP_GET_AUX_DOMAIN_INFO_FLAGS_EVENT ..."); // Count ticks for calling this event dwStartTicks = GetTickCount(); hr = TriggerServerEvent( SMTP_GET_AUX_DOMAIN_INFO_FLAGS_EVENT, (PVOID)&EventParams ); if(FAILED(hr)) { goto Exit; } dwStopTicks = GetTickCount(); DebugTrace((LPARAM)this, "Event SMTP_GET_AUX_DOMAIN_INFO_FLAGS_EVENT took %d ms.", dwStopTicks - dwStartTicks); if (*EventParams.pdwDomainInfoFlags & DOMAIN_INFO_INVALID) { // domain info not found ... the caller can interpret the flags DebugTrace((LPARAM)this, "Event SMTP_GET_AUX_DOMAIN_INFO_FLAGS_EVENT did NOT return data"); hr = S_OK; } else { DebugTrace((LPARAM)this, "Event SMTP_GET_AUX_DOMAIN_INFO_FLAGS_EVENT returned flags : %d", *EventParams.pdwDomainInfoFlags); hr = S_OK; } Exit: TraceFunctLeaveEx((LPARAM) this); return hr; } ///////////////////////////////////////////////////////////////// HRESULT SMTP_SERVER_INSTANCE::TriggerServerEvent(DWORD dwEventID, PVOID pvContext) { return m_CSMTPSeoMgr.HrTriggerServerEvent(dwEventID, pvContext); } ///////////////////////////////////////////////////////////////// //-------------------------------------------------------------------------------------- // Description: // Performs per-startup initialization of SMTP_SERVER_INSTANCE. All initialization // in here is de-inited by InitiateShutDown(). IIS calls this function each time // a VSI is started up. // Arguments: // None. // Returns: // TRUE on success, FALSE otherwise. // Notes: // InitiateStartup() and InitiateShutdown() could be called multiple times for the // same SMTP_SERVER_INSTANCE by IIS. If this fails, IIS will shutdown the VSI by // calling InitiateShutdown(). //-------------------------------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::InitiateStartup(void) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE:InitiateStartup" ); DWORD error = NO_ERROR, err = NO_ERROR; CSMTPServer * Ptr = NULL; HRESULT hr = S_OK; EnterCriticalSection( &m_critBoot ); InitializeClassVariables(); Ptr = new CSMTPServer(); if (Ptr != NULL) { Ptr->Init(this); hr = Ptr->QueryInterface(IID_ISMTPServer, (void **) &m_ComSmtpServer); if (FAILED(hr) || !m_ComSmtpServer) { err = GetLastError(); ErrorTrace((LPARAM) this, "QueryInterface for m_ComSmtpServer failed - %x", hr); goto error_exit; } } else { ErrorTrace((LPARAM) this, "new CSMTPServer() failed"); err = ERROR_NOT_ENOUGH_MEMORY; goto error_exit; } m_QStopEvent = CreateEvent(NULL, TRUE, FALSE, NULL); if (m_QStopEvent == NULL) { err = GetLastError(); ErrorTrace((LPARAM) this, "Creating stop event failed - %d", err); goto error_exit; } // // Create statistics object before initializing the queues // m_pSmtpStats = new SMTP_SERVER_STATISTICS(this); if ( m_pSmtpStats == NULL ) { err = ERROR_NOT_ENOUGH_MEMORY; ErrorTrace((LPARAM) this, "new SMTP_SERVER_STATISTICS(this) failed - %d", err); SetLastError( ERROR_NOT_ENOUGH_MEMORY ); goto error_exit; } // // initialize the info list - used to collect performance info (a bit of a backdoor // to get the information back // if (!(m_pSmtpInfo = new SMTP_INSTANCE_LIST_ENTRY)) { ErrorTrace((LPARAM) this, "new SMTP_INSTANCE_LIST_ENTRY failed"); SetLastError(ERROR_NOT_ENOUGH_MEMORY); goto error_exit; } m_pSmtpInfo->dwInstanceId = QueryInstanceId(); m_pSmtpInfo->pSmtpServerStatsObj = m_pSmtpStats; ((PSMTP_IIS_SERVICE) g_pInetSvc)->AcquireServiceExclusiveLock(); InsertTailList((((PSMTP_IIS_SERVICE) g_pInetSvc)->GetInfoList()), &(m_pSmtpInfo->ListEntry)); ((PSMTP_IIS_SERVICE) g_pInetSvc)->ReleaseServiceExclusiveLock(); ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); // // shinjuku initialization // hr = m_CSMTPSeoMgr.HrInit(QueryInstanceId()); if (FAILED(hr)) { char szInst[10]; _itoa((int)QueryInstanceId(), szInst, 10); ErrorTrace((LPARAM)this, "Error: Failed to initialize SEO for instance %u", QueryInstanceId()); SmtpLogEventEx(SEO_INIT_FAILED_INSTANCE, szInst, hr); goto error_exit; } if (!InitFromRegistry()) { err= GetLastError(); ErrorTrace((LPARAM) this, "init from registry failed. err = %d", err); if (err == NO_ERROR) err = ERROR_INVALID_PARAMETER; SetLastError(err); goto error_exit; } ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); TriggerStoreServerEvent(SMTP_STOREDRV_STARTUP_EVENT); m_fStoreDrvStartEventCalled = TRUE; // Enumerate files ... DWORD dwThreadId; m_hEnumBuildQ = CreateThread( NULL, 0, EnumBuildQProc, (LPVOID)this, 0, &dwThreadId); if (m_hEnumBuildQ == NULL) { err = GetLastError(); ErrorTrace((LPARAM) this, "Creating startup enumeration thread failed - %d", err); goto error_exit; } // Initialize the mail pickup stuff, only if enabled if (ShouldPickupMail() && !InitDirectoryNotification()) { err = GetLastError(); ErrorTrace((LPARAM) this, "InitDirectoryNotification() failed. err: %u", error); if (err == NO_ERROR) err = ERROR_INVALID_PARAMETER; SetLastError(err); goto error_exit; } LeaveCriticalSection( &m_critBoot ); TraceFunctLeaveEx((LPARAM) this); return TRUE; error_exit: LeaveCriticalSection( &m_critBoot ); TraceFunctLeaveEx((LPARAM) this); return FALSE; } PSMTP_INSTANCE_LIST_ENTRY SMTP_SERVER_INSTANCE::GetSmtpInstanceInfo(void) { AQPerfCounters AqPerf; HRESULT hr = S_FALSE; if (m_pIAdvQueueConfig != NULL) { AqPerf.cbVersion = sizeof(AQPerfCounters); hr = m_pIAdvQueueConfig->GetPerfCounters( &AqPerf, (m_pSmtpInfo) ? &(m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->CatPerfBlock) : NULL); } if(!FAILED(hr)) { if(m_pSmtpInfo) { m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->RemoteQueueLength = AqPerf.cCurrentQueueMsgInstances; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->NumMsgsDelivered = AqPerf.cMsgsDeliveredLocal; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->LocalQueueLength = AqPerf.cCurrentMsgsPendingLocalDelivery; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->RemoteRetryQueueLength = AqPerf.cCurrentMsgsPendingRemoteRetry; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->NumSendRetries = AqPerf.cTotalMsgRemoteSendRetries; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->NumNDRGenerated = AqPerf.cNDRsGenerated; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->RetryQueueLength = AqPerf.cCurrentMsgsPendingLocalRetry; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->NumDeliveryRetries = AqPerf.cTotalMsgLocalRetries; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->ETRNMessages = AqPerf.cTotalMsgsTURNETRN; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->CatQueueLength = AqPerf.cCurrentMsgsPendingCat; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsBadmailNoRecipients = AqPerf.cTotalMsgsBadmailNoRecipients; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsBadmailHopCountExceeded = AqPerf.cTotalMsgsBadmailHopCountExceeded; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsBadmailFailureGeneral = AqPerf.cTotalMsgsBadmailFailureGeneral; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsBadmailBadPickupFile = AqPerf.cTotalMsgsBadmailBadPickupFile; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsBadmailEvent = AqPerf.cTotalMsgsBadmailEvent; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsBadmailNdrOfDsn = AqPerf.cTotalMsgsBadmailNdrOfDsn; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsPendingRouting = AqPerf.cCurrentMsgsPendingRouting; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsPendingUnreachableLink = AqPerf.cCurrentMsgsPendingUnreachableLink; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->SubmittedMessages = AqPerf.cTotalMsgsSubmitted; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->DSNFailures = AqPerf.cTotalDSNFailures; m_pSmtpInfo->pSmtpServerStatsObj->QueryStatsMember()->MsgsInLocalDelivery = AqPerf.cCurrentMsgsInLocalDelivery; } } return m_pSmtpInfo; } //---[ SMTP_SERVER_INSTANCE::HrGetDomainInfoFlags ]---------------------------- // // // Description: // Gets domain info flags for specified domain if present // Parameters: // IN szDomainName Name of domain to check for // OUT pdwDomainInfoFlags DomainInfo flags for this domain // Returns: // S_OK on success // E_INVALIDARG if szDomainName or pdwDomainInfoFlags is NULL // E_FAIL if other failure // History: // 10/6/2000 - dbraun, Created. // //----------------------------------------------------------------------------- HRESULT SMTP_SERVER_INSTANCE::HrGetDomainInfoFlags( IN LPSTR szDomainName, OUT DWORD *pdwDomainInfoFlags) { HRESULT hr = S_OK; IAdvQueueDomainType * pIAdvQueueDomainType = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE:HrGetDomainInfoFlags" ); // Make sure we have a valid IAdvQueue interface if (!m_IAQ) { ErrorTrace((LPARAM) this, "Member m_IAQ is not valid"); _ASSERT(m_IAQ); hr = E_FAIL; goto exit; } // Get the IAdvQueueDomainType interface hr = m_IAQ->QueryInterface(IID_IAdvQueueDomainType, (void **) &pIAdvQueueDomainType); if (FAILED(hr) || !pIAdvQueueDomainType) { ErrorTrace((LPARAM) this, "QueryInterface for IID_IAdvQueueDomainType failed - %x", hr); hr = E_FAIL; goto exit; } // Got the interface, now check for the domain hr = pIAdvQueueDomainType->GetDomainInfoFlags(szDomainName, pdwDomainInfoFlags); exit: if (pIAdvQueueDomainType) pIAdvQueueDomainType->Release(); TraceFunctLeaveEx((LPARAM) this); return hr; } BOOL SMTP_SERVER_INSTANCE::Stop(void) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE:Stop" ); //set the global termination flag m_IsShuttingDown = TRUE; EnterCriticalSection( &m_critBoot ); if (m_ICM) { m_ICM->ReleaseWaitingThreads(); } if (!m_InstBooted) { LeaveCriticalSection( &m_critBoot ); TraceFunctLeaveEx((LPARAM) this); return TRUE; } if (m_fShutdownCalled) { LeaveCriticalSection( &m_critBoot ); TraceFunctLeaveEx((LPARAM) this); return TRUE; } if (m_QStopEvent) { SetEvent(m_QStopEvent); } //disconnect all inbound connections DisconnectAllConnections(); DisconnectAllAsyncDnsConnections(); //disconnect all outbound connections DisconnectAllOutboundConnections(); DisconnectAllAsyncConnections(); //we need to disconnect all outbound connections again here, //incase one slipped past the remote queue shutdown code DisconnectAllOutboundConnections(); m_rfAccessCheck.Reset( (IMDCOM*)m_Service->QueryMDObject() ); LeaveCriticalSection( &m_critBoot ); TraceFunctLeaveEx((LPARAM) this); return TRUE; } //----------------------------------------------------------------------------------- // Description: // Function called by IIS to deinitialize SMTP_SERVER_INSTANCE when the VSI is // stopping. Note that this function can be called multiple times for the same // SMTP_SERVER_INSTANCE. A matching call to InitializeClassVariables() and // InitiateStartup() do is made by IIS (to start SMTP_SERVER_INSTANCE). // Arguments: // None. // Returns: // None. //----------------------------------------------------------------------------------- void SMTP_SERVER_INSTANCE::InitiateShutDown(void) { char IntBuffer [20]; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE:InitiateShutDown" ); EnterCriticalSection( &m_critBoot ); Stop(); // Wait for the enum buildq thread to die ... if (m_hEnumBuildQ) { DWORD dwWait; do { dwWait = WaitForSingleObject(m_hEnumBuildQ, 1000); if (dwWait == WAIT_TIMEOUT) StopHint(); } while (dwWait != WAIT_OBJECT_0); CloseHandle(m_hEnumBuildQ); m_hEnumBuildQ = NULL; } // Wait for the enum domains thread to die ... if (m_hEnumDomainThreadHandle) { DWORD dwWait; do { dwWait = WaitForSingleObject(m_hEnumDomainThreadHandle, 1000); if (dwWait == WAIT_TIMEOUT) StopHint(); } while (dwWait != WAIT_OBJECT_0); CloseHandle(m_hEnumDomainThreadHandle); m_hEnumDomainThreadHandle = NULL; } //make sure directory notification threads are shut down if (ShouldPickupMail()) DestroyDirectoryNotification(); //Prepare for shutdown *must* be called before StopQDrivers... otherwise //we may have stray threads calling into submit messages... which could //AV. if(m_fStoreDrvStartEventCalled && !m_fStoreDrvPrepShutDownEventCalled) { TriggerStoreServerEvent(SMTP_STOREDRV_PREPSHUTDOWN_EVENT); m_fStoreDrvPrepShutDownEventCalled = TRUE; } // After this NOBODY can submit to the queue, the pointers are GONE StopQDrivers(); if( m_fStoreDrvStartEventCalled ) { TriggerStoreServerEvent(SMTP_STOREDRV_SHUTDOWN_EVENT); } //now that all the queues have stopped receiving data, //flush each queue, by removing all data and then delete //the queues. if (m_RemoteQ != NULL) { delete m_RemoteQ; m_RemoteQ = NULL; } DebugTrace((LPARAM)this, "removing and delete the Info List"); if (m_pSmtpInfo != NULL) { ((PSMTP_IIS_SERVICE)g_pInetSvc)->AcquireServiceExclusiveLock(); RemoveEntryList(&(m_pSmtpInfo->ListEntry)); ((PSMTP_IIS_SERVICE)g_pInetSvc)->ReleaseServiceExclusiveLock(); delete (m_pSmtpInfo); m_pSmtpInfo = NULL; } DebugTrace((LPARAM)this, "deleting statistics obj"); // // delete statistics object // if ( m_pSmtpStats != NULL ) { delete m_pSmtpStats; m_pSmtpStats = NULL; } FreeVRootList(&m_leVRoots); DebugTrace((LPARAM)this, "deleting crit sects"); //Delete the critical section objects if (m_SmtpInitializeStatus & SMTP_INIT_CSLOCK) { DeleteCriticalSection( &m_csLock); m_SmtpInitializeStatus &= ~SMTP_INIT_CSLOCK; } if (m_pProviderPackagesInfo != NULL ) { m_pProviderPackagesInfo->Release(); m_pProviderPackagesInfo = NULL; } //free the SSL info object if ( m_pSSLInfo != NULL ) { //If the refcount is not zero - we have a problem DWORD dwCount = IIS_SSL_INFO::Release( m_pSSLInfo ); m_pSSLInfo = NULL; } ResetRelayIpSecList(); // // shutdown shinjuku // this causes SEO to drop all loaded objects // m_CSMTPSeoMgr.Deinit(); if (m_ComSmtpServer != NULL) { m_ComSmtpServer->Release(); m_ComSmtpServer = NULL; } if (m_QStopEvent != NULL) { CloseHandle(m_QStopEvent); m_QStopEvent = NULL; } // // here we see if we are getting shutdown because we are being // deleted. if so then we'll remove all of our bindings from // the shinjuku event binding database // MB mb( (IMDCOM*) g_pInetSvc->QueryMDObject() ); StopHint(); if (mb.Open(QueryMDPath())) { // our metabase path still exists, so we aren't being deleted mb.Close(); } else { StopHint(); // our metabase path is gone, delete the shinjuku binding // database HRESULT hr = UnregisterPlatSEOInstance(QueryInstanceId()); if (FAILED(hr)) { char szInst[10]; _itoa((int)QueryInstanceId(), szInst, 10); ErrorTrace(0, "UnregisterSEOInstance(%lu) failed with %x", QueryInstanceId(), hr); SmtpLogEventEx(SEO_DELETE_INSTANCE_FAILED, szInst, hr); } } _ASSERT(m_fInitAsyncCS && "Deleting un-initialized critsec"); if(m_fInitAsyncCS) { DeleteCriticalSection( &m_csAsyncDns); DeleteCriticalSection( &m_csAsyncConnect); m_fInitAsyncCS = FALSE; } m_fShutdownCalled = TRUE; LeaveCriticalSection( &m_critBoot ); TraceFunctLeaveEx((LPARAM) this); } DWORD SMTP_SERVER_INSTANCE::PauseInstance() { DWORD err = NO_ERROR ; char IntBuffer [20]; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::PauseInstance"); err = IIS_SERVER_INSTANCE::PauseInstance() ; if (err == NO_ERROR) { _itoa(QueryInstanceId(), IntBuffer, 10); SmtpLogEventEx(SMTP_EVENT_SERVICE_INSTANCE_PAUSED, IntBuffer, 0); } TraceFunctLeaveEx((LPARAM)this); return err ; } /*++ Routine Description: Sets instance to RUNNING Arguments: NewState - Receives the new state. Return Value: DWORD - 0 if successful, !0 otherwise. --*/ DWORD SMTP_SERVER_INSTANCE::StartInstance(void) { DWORD RetCode = ERROR_INVALID_SERVICE_CONTROL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::StartInstance(void)" ); _ASSERT(QueryServerState( ) == MD_SERVER_STATE_STOPPED); //initiate our startup first if ( InitiateStartup() ) { RetCode = IIS_SERVER_INSTANCE::StartInstance(); if (RetCode != ERROR_SUCCESS) { // shutdown the instance InitiateShutDown(); DebugTrace((LPARAM)this, "StartInstance() failed, err= %d", RetCode); } else { m_InstBooted = TRUE; } } else { //shutdown everything InitiateShutDown(); } TraceFunctLeaveEx((LPARAM)this); return RetCode; } DWORD SMTP_SERVER_INSTANCE::StopInstance(void) { DWORD RetCode = NO_ERROR; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::StopInstance(void)" ); //call the IIS stuff first. RetCode = IIS_SERVER_INSTANCE::StopInstance(); if (RetCode == NO_ERROR) { //shutdown everything InitiateShutDown(); } else { DebugTrace((LPARAM)this, "StopInstance() failed, err= %d", RetCode); } TraceFunctLeaveEx((LPARAM)this); return RetCode; } BOOL SMTP_SERVER_INSTANCE::InitQueues(void) { DWORD error = NO_ERROR; BOOL fReturn = FALSE; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::InitQueues" ); m_RemoteQ = PERSIST_QUEUE::CreateQueue(REMOTEQ, this); if (m_RemoteQ != NULL) { fReturn = TRUE; } else { error = GetLastError(); ErrorTrace((LPARAM)this, "new failed for PERSIST_QUEUE::CreateQueue(REMOTEQ). err: %u", error); if (error == NO_ERROR) SetLastError (ERROR_NOT_ENOUGH_MEMORY); } return fReturn; } void SMTP_SERVER_INSTANCE::FreeVRootList(PLIST_ENTRY pleHead) { PLIST_ENTRY pEntry; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::FreeVRootList" ); while (!IsListEmpty(pleHead)) { pEntry = RemoveHeadList(pleHead); TCP_FREE(CONTAINING_RECORD(pEntry, SMTP_VROOT_ENTRY, list)); } TraceFunctLeaveEx((LPARAM)this); } BOOL GetVroots( PVOID pvContext, MB * pmb, VIRTUAL_ROOT * pvr ) /*++ Routine Description: Virtual directory enumerater callback that allocates and builds the virtual directory structure list Arguments: pvContext is a pointer to the midl allocated memory Return: TRUE if success, otherwise FALSE --*/ { //LPINET_INFO_VIRTUAL_ROOT_LIST pvrl = (LPINET_INFO_VIRTUAL_ROOT_LIST) pv //DWORD i = pvrl->cEntries; //LPINET_INFO_VIRTUAL_ROOT_ENTRY pvre = &pvrl->aVirtRootEntry[i]; //_ASSERT( pvr->pszMetaPath[0] == '/' && // pvr->pszMetaPath[1] == '/' ); return TRUE; } BOOL SMTP_SERVER_INSTANCE::FindBestVRoot(LPSTR szVRoot) { PLIST_ENTRY pEntry; HANDLE hToken; DWORD cbRoot; char szRoot[MAX_PATH + 1]; DWORD dwErr; DWORD dwBytes; DWORD dwSectors; DWORD dwFree; DWORD dwTotal; DWORD dwRatio; DWORD dwRatioKeep; PLIST_ENTRY pKeep = NULL; LPSTR szT; DWORD cSlash; DWORD dwAccessMask = 0; SMTP_VROOT_ENTRY *pVrEntry = NULL; TraceFunctEnterEx((LPARAM)this, "SMTPCONFIG::FindBestVRoot"); if (IsListEmpty(&m_leVRoots)) { ErrorTrace((LPARAM)this, "Vroots list empty"); return FALSE; } for (pEntry = m_leVRoots.Flink ; pEntry != &m_leVRoots ; pEntry = pEntry->Flink) { cbRoot = sizeof(szRoot); pVrEntry = (SMTP_VROOT_ENTRY *) CONTAINING_RECORD(pEntry, SMTP_VROOT_ENTRY, list); if (!QueryVrootTable()->LookupVirtualRoot(pVrEntry->szVRoot, szRoot, &cbRoot, &dwAccessMask, NULL, NULL, &hToken, NULL)) { dwErr = GetLastError(); ErrorTrace(NULL, "ResolveVirtualRoot failed for %s, %u", CONTAINING_RECORD(pEntry, SMTP_VROOT_ENTRY, list)->szVRoot, dwErr); } else { cSlash = 0; if (szRoot[0] == '\\' && szRoot[1] == '\\') { // UNC Name DebugTrace((LPARAM)this, "Found UNC path %s", szRoot); szT = szRoot; while (*szT) { if (*szT == '\\') { cSlash++; if (cSlash == 4) { *(szT + 1) = '\0'; } } szT++; } if (cSlash != 4) { lstrcat(szRoot, "\\"); } } else { DebugTrace((LPARAM)this, "Found normal directory: %s", szRoot); szRoot[3] = '\0'; } DebugTrace((LPARAM)this, "Getting free disk ratio on %s", szRoot); if (hToken == 0 || ImpersonateLoggedOnUser(hToken)) { if (GetDiskFreeSpace(szRoot, &dwSectors, &dwBytes, &dwFree, &dwTotal)) { dwSectors *= dwBytes; dwRatio = MulDiv(dwSectors, dwTotal, dwFree); if (pKeep == NULL) { dwRatioKeep = dwRatio; pKeep = pEntry; } else { if (dwRatioKeep > dwRatio) { dwRatioKeep = dwRatio; pKeep = pEntry; } } } if (hToken != 0) _VERIFY(RevertToSelf()); } } } if (pKeep != NULL) lstrcpy(szVRoot, CONTAINING_RECORD(pKeep, SMTP_VROOT_ENTRY, list)->szVRoot); else lstrcpy(szVRoot, pVrEntry->szVRoot); TraceFunctLeaveEx((LPARAM)this); return TRUE; } //+--------------------------------------------------------------- // // Function: SMTPCONFIG::WriteRegParams // // Synopsis: Writes parameters from a config info structure // into the registry // // Arguments: SMTP_CONFIG_INFO *: pointer to config information // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::WriteRegParams(SMTP_CONFIG_INFO *pconfig) { //TraceFunctLeaveEx((LPARAM)this); return TRUE; } DWORD ReadMetabaseDword (MB& mb, DWORD Key, DWORD DefaultValue) { DWORD tmp = 0; if ( !mb.GetDword("", Key, IIS_MD_UT_SERVER, &tmp)) { //mb.SetDword("", Key, IIS_MD_UT_SERVER, DefaultValue); tmp = DefaultValue; } return tmp; } BOOL CProviderPackagesInfo::SetProviderPackages() { TraceFunctEnter( "CProviderPackages::SetProviderPackages" ); LPSTR psz; DWORD i; PAUTH_BLOCK pBlock = NULL; if ( m_ProviderNames == NULL || m_cProviderPackages == 0) { ErrorTrace((LPARAM)this, "Invalid Parameters: 0x%08X, %d", m_ProviderNames, m_cProviderPackages ); return FALSE; } pBlock = (PAUTH_BLOCK)LocalAlloc(0, m_cProviderPackages * sizeof(AUTH_BLOCK)); if (pBlock == NULL) { ErrorTrace( 0, "AUTH_BLOCK LocalAlloc failed: %d", GetLastError() ); return FALSE; } // // start at 1 since 0 indicates the Invalid protocol // for ( i=0, psz = (LPSTR)m_ProviderNames; i< m_cProviderPackages; i++ ) { // // this would be the place to check whether the package was valid // DebugTrace( 0, "Protocol: %s", psz); pBlock[i].Name = psz; psz += lstrlen(psz) + 1; } m_ProviderPackages = pBlock; return TRUE; } // SetAuthPackageNames /******************************************************************* NAME: GetDefaultDomainName SYNOPSIS: Fills in the given array with the name of the default domain to use for logon validation. ENTRY: pszDomainName - Pointer to a buffer that will receive the default domain name. cchDomainName - The size (in charactesr) of the domain name buffer. RETURNS: APIERR - 0 if successful, !0 if not. HISTORY: KeithMo 05-Dec-1994 Created. ********************************************************************/ APIERR GetDefaultDomainName( STR * pstrDomainName ) { OBJECT_ATTRIBUTES ObjectAttributes; NTSTATUS NtStatus; INT Result; APIERR err = 0; LSA_HANDLE LsaPolicyHandle = NULL; PPOLICY_ACCOUNT_DOMAIN_INFO DomainInfo = NULL; TraceFunctEnter("GetDefaultDomainName"); // // Open a handle to the local machine's LSA policy object. // InitializeObjectAttributes( &ObjectAttributes, // object attributes NULL, // name 0L, // attributes NULL, // root directory NULL ); // security descriptor NtStatus = LsaOpenPolicy( NULL, // system name &ObjectAttributes, // object attributes POLICY_EXECUTE, // access mask &LsaPolicyHandle ); // policy handle if ( !NT_SUCCESS( NtStatus ) ) { DebugTrace(0,"cannot open lsa policy, error %08lX\n",NtStatus ); err = LsaNtStatusToWinError( NtStatus ); goto Cleanup; } // // Query the domain information from the policy object. // NtStatus = LsaQueryInformationPolicy( LsaPolicyHandle, PolicyAccountDomainInformation, (PVOID *)&DomainInfo ); if ( !NT_SUCCESS( NtStatus ) ) { DebugTrace(0,"cannot query lsa policy info, error %08lX\n",NtStatus ); err = LsaNtStatusToWinError( NtStatus ); goto Cleanup; } // // Compute the required length of the ANSI name. // Result = WideCharToMultiByte( CP_ACP, 0, // dwFlags (LPCWSTR)DomainInfo->DomainName.Buffer, DomainInfo->DomainName.Length /sizeof(WCHAR), NULL, // lpMultiByteStr 0, // cchMultiByte NULL, // lpDefaultChar NULL // lpUsedDefaultChar ); if ( Result <= 0 ) { err = GetLastError(); goto Cleanup; } // // Resize the output string as appropriate, including room for the // terminating '\0'. // if ( !pstrDomainName->Resize( (UINT)Result + 1 ) ) { err = GetLastError(); goto Cleanup; } // // Convert the name from UNICODE to ANSI. // Result = WideCharToMultiByte( CP_ACP, 0, // flags (LPCWSTR)DomainInfo->DomainName.Buffer, DomainInfo->DomainName.Length /sizeof(WCHAR), pstrDomainName->QueryStr(), pstrDomainName->QuerySize() - 1, // for '\0' NULL, NULL ); if ( Result <= 0 ) { err = GetLastError(); DebugTrace(0,"cannot convert domain name to ANSI, error %d\n",err ); goto Cleanup; } // // Ensure the ANSI string is zero terminated. // _ASSERT( (DWORD)Result < pstrDomainName->QuerySize() ); pstrDomainName->QueryStr()[Result] = '\0'; // // Success! // _ASSERT( err == 0 ); DebugTrace(0,"GetDefaultDomainName: default domain = %s\n",pstrDomainName->QueryStr() ); Cleanup: if ( DomainInfo != NULL ) { LsaFreeMemory( (PVOID)DomainInfo ); } if ( LsaPolicyHandle != NULL ) { LsaClose( LsaPolicyHandle ); } return err; } // GetDefaultDomainName() //+--------------------------------------------------------------- // // Function: SMTPCONFIG::ReadRegParams // // Synopsis: Reads parameters from the registry into the config // class member variables and IServer (m_InstancePropertyBag) // // Arguments: FIELD_CONTROL: Bit-field defining what params to // read. // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // // Note: BOOL fInit argument removed by KeithLau on 7/15/96 // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::ReadRegParams( FIELD_CONTROL fc, BOOL fRebuild, BOOL fShowEvents) { BOOL fRet = TRUE; DWORD dwErr = NO_ERROR; DWORD dwAttr; DWORD dwLen; DWORD dwTempVar; DWORD dwAqueueWait = 0; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); char ScratchBuffer [400]; STR TempString; const CHAR * apszSubStrings[4]; CHAR pchAddr1[32] = ""; AQConfigInfo AQConfig; HRESULT hr; char szDomainName[MAX_PATH + 1] = {0}; TraceFunctEnterEx((LPARAM)this, "SMTPCONFIG::SMTP_SERVER_INSTANCE"); SetLastError(NO_ERROR); m_fDefaultInRt = FALSE; ZeroMemory(&AQConfig, sizeof(AQConfig)); // Make sure we've got a valid field control input _ASSERT(fc != 0 && !(fc & ~(FC_SMTP_INFO_ALL))); m_GenLock.ExclusiveLock(); _itoa(QueryInstanceId(), pchAddr1, 10); // // Read metabase data. // if ( !mb.Open( QueryMDPath(), METADATA_PERMISSION_READ | METADATA_PERMISSION_WRITE) ) { m_GenLock.ExclusiveUnlock(); return FALSE; } if (IsFieldSet(fc, FC_SMTP_INFO_ROUTING)) { GetCatInfo(mb, AQConfig); } ZeroMemory(&AQConfig, sizeof(AQConfig)); if (IsFieldSet(fc, FC_SMTP_INFO_REVERSE_LOOKUP)) { dwTempVar = 0; dwTempVar = ReadMetabaseDword(mb, MD_REVERSE_NAME_LOOKUP, 0); // If the value is changing, generate an information NT event... if (dwTempVar != m_RDNSOptions) { apszSubStrings[0] = pchAddr1; if (fShowEvents) { if (dwTempVar) SmtpLogEvent(SMTP_EVENT_SET_REVERSE_LOOKUP_ENABLED, 1, apszSubStrings, 0); else SmtpLogEvent(SMTP_EVENT_SET_REVERSE_LOOKUP_DISABLED, 1, apszSubStrings, 0); } m_RDNSOptions = dwTempVar; StateTrace((LPARAM)this, "m_fEnableReverseLookup = %u", m_RDNSOptions); } } //I am isung this for all commands if (IsFieldSet(fc, FC_SMTP_INFO_INBOUND_SUPPORT_OPTIONS)) { m_InboundCmdOptions = ReadMetabaseDword(mb, MD_INBOUND_COMMAND_SUPPORT_OPTIONS, SMTP_DEFAULT_CMD_SUPPORT); m_OutboundCmdOptions = ReadMetabaseDword(mb, MD_OUTBOUND_COMMAND_SUPPORT_OPTIONS, SMTP_DEFAULT_OUTBOUND_SUPPORT); m_fAddNoHdrs = !!ReadMetabaseDword(mb, MD_ADD_NOHEADERS, FALSE); } if (IsFieldSet(fc, FC_SMTP_INFO_SSL_PERM)) { BOOL fRequiresCertVerifySubject; BOOL fRequiresCertVerifyIssuer; fRequiresCertVerifyIssuer = ReadMetabaseDword (mb, MD_SMTP_SSL_REQUIRE_TRUSTED_CA, FALSE); fRequiresCertVerifySubject = ReadMetabaseDword (mb, MD_SMTP_SSL_CERT_HOSTNAME_VALIDATION, FALSE); m_fRequiresCertVerifySubject = fRequiresCertVerifySubject; m_fRequiresCertVerifyIssuer = fRequiresCertVerifyIssuer; BOOL fRequiresSSL, fRequires128Bits; dwTempVar = ReadMetabaseDword(mb, MD_SSL_ACCESS_PERM, FALSE); fRequiresSSL = IsFieldSet(dwTempVar, MD_ACCESS_SSL); fRequires128Bits = IsFieldSet(dwTempVar, MD_ACCESS_SSL128); apszSubStrings[0] = pchAddr1; // If the value is changing, generate an information NT event... if (fShowEvents && (fRequiresSSL != m_fRequiresSSL)) { SmtpLogEvent( fRequiresSSL ? SMTP_EVENT_REQUIRE_SSL_INBOUND_ENABLE : SMTP_EVENT_REQUIRE_SSL_INBOUND_DISABLE, 1, apszSubStrings, 0); } m_fRequiresSSL = fRequiresSSL; if (fShowEvents && (fRequires128Bits != m_fRequires128Bits)) { SmtpLogEvent( fRequires128Bits ? SMTP_EVENT_REQUIRE_128BIT_SSL_INBOUND_ENABLE : SMTP_EVENT_REQUIRE_128BIT_SSL_INBOUND_DISABLE, 1, apszSubStrings, 0); } m_fRequires128Bits = fRequires128Bits; StateTrace((LPARAM)this, "m_fRequiresSSL = %s, m_fRequires128Bits = %s", m_fRequiresSSL ? "TRUE" : "FALSE", m_fRequires128Bits ? "TRUE" : "FALSE"); } if (IsFieldSet(fc, FC_SMTP_INFO_COMMON_PARAMS)) { DWORD dwDomainValidationFlags = 0; DWORD dwNameResolution = 0; m_dwDnsFlags = ReadMetabaseDword(mb, MD_SMTP_USE_TCP_DNS, 0); m_dwNameResolution = ReadMetabaseDword(mb, MD_NAME_RESOLUTION_TYPE, 1); StateTrace((LPARAM)this, "m_dwNameResolution = %u", m_dwNameResolution); m_cMaxBatchLimit = ReadMetabaseDword(mb, MD_BATCH_MSG_LIMIT, 20); AQConfig.cMinMessagesPerConnection = m_cMaxBatchLimit; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MIN_MSG; AQConfig.dwConnectionWaitMilliseconds = ReadMetabaseDword(mb, MD_SMTP_AQUEUE_WAIT, 60000); //AQConfig.dwConnectionWaitMilliseconds = INFINITE; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_CON_WAIT; StateTrace((LPARAM)this, "m_cMaxBatchLimit = %u", m_cMaxBatchLimit); m_fRelayForAuthUsers = !!ReadMetabaseDword(mb, MD_SMTP_RELAY_FOR_AUTH_USERS, TRUE); m_fIsRelayEnabled = !!ReadMetabaseDword(mb, MD_SMTP_DISABLE_RELAY, TRUE); m_fHelloNoDomain = !!ReadMetabaseDword(mb, MD_SMTP_HELO_NODOMAIN, TRUE); m_fMailFromNoHello = !!ReadMetabaseDword(mb, MD_SMTP_MAIL_NO_HELO, FALSE); //Per spec these two are related if (m_fMailFromNoHello) m_fHelloNoDomain = TRUE; m_fNagleIn = !!ReadMetabaseDword(mb, MD_SMTP_INBOUND_NAGLE, FALSE); m_fNagleOut = !!ReadMetabaseDword(mb, MD_SMTP_OUTBOUND_NAGLE, FALSE); dwDomainValidationFlags = !!ReadMetabaseDword(mb, MD_DOMAIN_VALIDATION_FLAGS, 0); m_fHelloNoValidate = dwDomainValidationFlags & SMTP_NOVALIDATE_EHLO; m_fMailNoValidate = dwDomainValidationFlags & SMTP_NOVALIDATE_MAIL; m_fRcptNoValidate = dwDomainValidationFlags & SMTP_NOVALIDATE_RCPT; m_fEtrnNoValidate = dwDomainValidationFlags & SMTP_NOVALIDATE_ETRN; m_fPickupNoValidate = dwDomainValidationFlags & SMTP_NOVALIDATE_PKUP; m_dwDeniedIpAction = ReadMetabaseDword(mb, MD_SMTP_DENIED_IP_ACTION, SMTPDENIEDIP_DROPCONN); TempString.Reset(); if(mb.GetStr("", MD_SMTP_POST_DNS_SMARTHOST, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT, "")) { lstrcpyn(m_szPostDnsSmarthost, TempString.QueryStr(), sizeof(m_szPostDnsSmarthost)); DebugTrace((LPARAM)this, "Post DNS smarthost = %s", m_szPostDnsSmarthost); } else { m_szPostDnsSmarthost[0] = '\0'; DebugTrace((LPARAM)this, "No post DNS smarthost configured"); } ReadRouteDomainIpSecList(mb); // Raid 174038 m_fDisablePickupDotStuff = !!ReadMetabaseDword(mb, MD_SMTP_DISABLE_PICKUP_DOT_STUFF, FALSE); StateTrace((LPARAM)this, "m_fDisablePickupDotStuff = %u", m_fDisablePickupDotStuff); m_dwEventlogLevel = ReadMetabaseDword(mb, MD_SMTP_EVENTLOG_LEVEL, LOGEVENT_LEVEL_MEDIUM); } if (IsFieldSet(fc, FC_SMTP_INFO_MAX_OUT_CONN_PER_DOMAIN)) { dwTempVar = ReadMetabaseDword(mb, MD_MAX_OUT_CONN_PER_DOMAIN, 20); if (fShowEvents && (dwTempVar != m_cMaxOutConnectionsPerDomain)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_CONN_PER_DOMAIN, 2, apszSubStrings, 0); } m_cMaxOutConnectionsPerDomain = dwTempVar; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MAX_LINK; AQConfig.cMaxLinkConnections = dwTempVar; StateTrace((LPARAM)this, "m_cMaxOutConnectionsPerDomain = %u", m_cMaxOutConnectionsPerDomain); } if (IsFieldSet(fc, FC_SMTP_INFO_AUTHORIZATION)) { m_dwAuth = ReadMetabaseDword(mb, MD_AUTHORIZATION, DEFAULT_AUTHENTICATION); DebugTrace((LPARAM)this, "m_dwAuth=%u", m_dwAuth); } // // update the following data members: // m_ProviderPackages // m_ProviderNames // m_cProviderPackages CProviderPackagesInfo *pProviderPackagesInfo = NULL; if (IsFieldSet(fc, FC_SMTP_INFO_NTAUTHENTICATION_PROVIDERS) && (pProviderPackagesInfo = new CProviderPackagesInfo())) { CHAR szAuthPack[MAX_PATH + 1]; DWORD dwLen; TempString.Reset(); if (mb.GetStr("", MD_NTAUTHENTICATION_PROVIDERS, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT, "")) { lstrcpyn(szAuthPack, TempString.QueryStr(), MAX_PATH); dwLen = lstrlen(szAuthPack); DebugTrace((LPARAM)this, "Authentication packages=%s", szAuthPack); ULONG cProviderPackages = 0; if(ConvertToMultisz( pProviderPackagesInfo->GetProviderNames(), &cProviderPackages, szAuthPack)) { pProviderPackagesInfo->SetProviderPackagesCount(cProviderPackages); } else { CopyMemory(pProviderPackagesInfo->GetProviderNames(), pszPackagesDefault, ccbPackagesDefault); pProviderPackagesInfo->SetProviderPackagesCount(2); } } else { DebugTrace((LPARAM)this, "Use default authentication packages=%s", pszPackagesDefault); CopyMemory(pProviderPackagesInfo->GetProviderNames(), pszPackagesDefault, ccbPackagesDefault); pProviderPackagesInfo->SetProviderPackagesCount(1); } // set the AUTH_BLOCK info if (!pProviderPackagesInfo->SetProviderPackages()) { ErrorTrace((LPARAM)this, "Unable to allocate AUTH_BLOCK"); pProviderPackagesInfo->Release(); } else { if(m_pProviderPackagesInfo) m_pProviderPackagesInfo->Release(); m_pProviderPackagesInfo = pProviderPackagesInfo; } } if (IsFieldSet(fc, FC_SMTP_INFO_SASL_LOGON_DOMAIN)) { TempString.Reset(); if (mb.GetStr("", MD_SASL_LOGON_DOMAIN, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT, "")) { lstrcpyn(m_szDefaultLogonDomain, TempString.QueryStr(), MAX_SERVER_NAME_LEN); DebugTrace((LPARAM)this, "SASL logon domain =%s", m_szDefaultLogonDomain); } else { m_szDefaultLogonDomain[0] = '\0'; DebugTrace((LPARAM)this, "no SASL default logon domain was found"); } } if (IsFieldSet(fc, FC_SMTP_CLEARTEXT_AUTH_PROVIDER)) { TempString.Reset(); m_cbCleartextAuthPackage = sizeof(m_szCleartextAuthPackage); if (mb.GetStr("", MD_SMTP_CLEARTEXT_AUTH_PROVIDER, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT, "")) { lstrcpy(m_szCleartextAuthPackage, TempString.QueryStr()); m_cbCleartextAuthPackage = lstrlen(m_szCleartextAuthPackage) + 1; StateTrace((LPARAM)this, "Cleartext authentication provider: <%s>, length %u", m_szCleartextAuthPackage, m_cbCleartextAuthPackage); } else { m_szCleartextAuthPackage[0] = '\0'; m_cbCleartextAuthPackage = 0; StateTrace((LPARAM)this, "No default cleartext authentication provider specified, using CleartextLogon"); } TempString.Reset(); if (mb.GetStr("", MD_MD_SERVER_SS_AUTH_MAPPING, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT, "")) { lstrcpyn(m_szMembershipBroker, TempString.QueryStr(), MAX_PATH); StateTrace((LPARAM)this, "Membership Broker name is set to %s", m_szMembershipBroker); } else { m_szMembershipBroker[0] = '\0'; StateTrace((LPARAM)this, "No Membership Broker name configured"); } } if (IsFieldSet(fc, FC_SMTP_INFO_MAX_HOP_COUNT)) { dwTempVar = ReadMetabaseDword(mb, MD_HOP_COUNT, 15); StateTrace((LPARAM)this, "m_cMaxHopCount = %u", dwTempVar); if (fShowEvents && (dwTempVar != m_cMaxHopCount)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_HOPCOUNT, 2, apszSubStrings, 0); } m_cMaxHopCount = dwTempVar; } if (IsFieldSet(fc, FC_SMTP_INFO_REMOTE_PORT)) { dwTempVar = ReadMetabaseDword(mb, MD_REMOTE_SMTP_PORT, 25); if (fShowEvents && (dwTempVar != m_RemoteSmtpPort)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_REMOTE_PORT, 2, apszSubStrings, 0); } m_RemoteSmtpPort = dwTempVar; StateTrace((LPARAM)this, "m_RemoteSmtpPort = %u", m_RemoteSmtpPort); } if (IsFieldSet(fc, FC_SMTP_INFO_MAX_ERRORS)) { dwTempVar = ReadMetabaseDword(mb, MD_MAX_SMTP_ERRORS, 10); if (dwTempVar == 0) dwTempVar = 10; if (fShowEvents && (dwTempVar != m_cMaxErrors)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_ERRORS, 2, apszSubStrings, 0); } m_cMaxErrors = dwTempVar; StateTrace((LPARAM)this, "m_cMaxErrors = %u", m_cMaxErrors); dwTempVar = ReadMetabaseDword(mb, MD_MAX_SMTP_AUTHLOGON_ERRORS, 4); if (dwTempVar == 0) dwTempVar = 4; if (fShowEvents && (dwTempVar != m_dwMaxLogonFailures)) { //_itoa(dwTempVar, ScratchBuffer, 10); // apszSubStrings[0] = pchAddr1; // apszSubStrings[1] = ScratchBuffer; //SmtpLogEvent(SMTP_EVENT_SET_MAX_ERRORS, 2, apszSubStrings, 0); } m_dwMaxLogonFailures = dwTempVar; StateTrace((LPARAM)this, "m_dwMaxLogonFailures = %u", m_dwMaxLogonFailures); } if (IsFieldSet(fc, FC_SMTP_INFO_MAX_SIZE)) { dwTempVar = ReadMetabaseDword(mb, MD_MAX_MSG_SIZE, 2 * 1024); if (fShowEvents && (dwTempVar != m_cbMaxMsgSize)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_SIZE_ACCEPTED, 2, apszSubStrings, 0); } m_cbMaxMsgSize = dwTempVar; StateTrace((LPARAM)this, "m_cbMaxMsgSize = %u", m_cbMaxMsgSize); dwTempVar = ReadMetabaseDword(mb, MD_MAX_MSG_SIZE_B4_CLOSE, 10 * 1024); if (fShowEvents && (dwTempVar != m_cbMaxMsgSizeBeforeClose)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_SIZE_BEFORE_CLOSE, 2, apszSubStrings, 0); } m_cbMaxMsgSizeBeforeClose = dwTempVar; StateTrace((LPARAM)this, "m_cbMaxMsgSizeBeforeClose = %u", m_cbMaxMsgSizeBeforeClose); } if (IsFieldSet(fc, FC_SMTP_INFO_REMOTE_TIMEOUT)) { dwTempVar = ReadMetabaseDword(mb, MD_REMOTE_TIMEOUT, 600); if (fShowEvents && (dwTempVar != m_cMaxRemoteTimeOut)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_REMOTE_TIMEOUT, 2, apszSubStrings, 0); } m_cMaxRemoteTimeOut = dwTempVar; StateTrace((LPARAM)this, "m_cMaxRemoteTimeOut = %u", m_cMaxRemoteTimeOut); m_ConnectTimeout = ReadMetabaseDword(mb, MD_SMTP_CONNECT_TIMEOUT, 150); if (m_ConnectTimeout == 0) m_ConnectTimeout = 1; // Calculate timeout in milliseconds from timeout in seconds. m_ConnectTimeout = m_ConnectTimeout * 60 * 1000; m_MailFromTimeout = ReadMetabaseDword(mb, MD_SMTP_MAILFROM_TIMEOUT, 600); m_RcptToTimeout = ReadMetabaseDword(mb, MD_SMTP_RCPTTO_TIMEOUT, 600); m_DataTimeout = ReadMetabaseDword(mb, MD_SMTP_DATA_TIMEOUT, 600); m_AuthTimeout = ReadMetabaseDword(mb, MD_SMTP_AUTH_TIMEOUT, 600); m_SaslTimeout = ReadMetabaseDword(mb, MD_SMTP_SASL_TIMEOUT, 600); m_HeloTimeout = ReadMetabaseDword(mb, MD_SMTP_HELO_TIMEOUT, 600); m_BdatTimeout = ReadMetabaseDword(mb, MD_SMTP_BDAT_TIMEOUT, 600); m_TurnTimeout = ReadMetabaseDword(mb, MD_SMTP_TURN_TIMEOUT, 600); m_RSetTimeout = ReadMetabaseDword(mb, MD_SMTP_RSET_TIMEOUT, 600); m_QuitTimeout = ReadMetabaseDword(mb, MD_SMTP_RSET_TIMEOUT, 600); } if (IsFieldSet(fc, FC_SMTP_INFO_MAX_OUTBOUND_CONN)) { dwTempVar = ReadMetabaseDword(mb, MD_MAX_OUTBOUND_CONNECTION, 100); if (fShowEvents && (dwTempVar != m_cMaxOutConnections)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_OUTBOUND_CONNECTIONS, 2, apszSubStrings, 0); } m_cMaxOutConnections = dwTempVar; AQConfig.cMaxConnections = dwTempVar; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MAX_CON; StateTrace((LPARAM)this, "m_cMaxOutConnections = %u", m_cMaxOutConnections); } if (IsFieldSet(fc, FC_SMTP_INFO_MAX_RECIPS)) { dwTempVar = ReadMetabaseDword(mb, MD_MAX_RECIPIENTS, 100); if (fShowEvents && (dwTempVar != m_cMaxRcpts)) { _itoa(dwTempVar, ScratchBuffer, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = ScratchBuffer; SmtpLogEvent(SMTP_EVENT_SET_MAX_RECIPIENTS, 2, apszSubStrings, 0); } m_cMaxRcpts = dwTempVar; StateTrace((LPARAM)this, "m_cMaxRcpts = %u", m_cMaxRcpts); } if (IsFieldSet(fc, FC_SMTP_INFO_ETRN_SUBDOMAINS)) { m_fAllowEtrnSubDomains = !!ReadMetabaseDword(mb, MD_ETRN_SUBDOMAINS, TRUE); StateTrace((LPARAM)this, "m_fAllowEtrnSubDomains = %u", m_fAllowEtrnSubDomains); } if (IsFieldSet(fc, FC_SMTP_INFO_RETRY)) { DWORD dwTempVar2 = 0; char ScratchBuffer2[50]; dwTempVar = ReadMetabaseDword(mb, MD_LOCAL_RETRY_ATTEMPTS, 48); dwTempVar2 = ReadMetabaseDword(mb, MD_LOCAL_RETRY_MINUTES, 60); if (fShowEvents && ((m_cRetryAttempts != dwTempVar) ||(m_cRetryMinutes != dwTempVar2))) { _itoa(dwTempVar, ScratchBuffer, 10); _itoa(dwTempVar2, ScratchBuffer2, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = "Local"; apszSubStrings[2] = ScratchBuffer; apszSubStrings[3] = ScratchBuffer2; SmtpLogEvent(SMTP_EVENT_SET_RETRY_PARAMETERS, 4, apszSubStrings, 0); } m_cRetryAttempts = dwTempVar; m_cRetryMinutes = dwTempVar2; StateTrace((LPARAM)this, "m_cRetryAttempts = %u", m_cRetryAttempts); StateTrace((LPARAM)this, "m_cRetryMinutes = %u", m_cRetryMinutes); dwTempVar = ReadMetabaseDword(mb, MD_REMOTE_RETRY_ATTEMPTS, 48); dwTempVar2 = ReadMetabaseDword(mb, MD_REMOTE_RETRY_MINUTES, 60); if ( fShowEvents && ((m_cRemoteRetryAttempts != dwTempVar) ||(m_cRemoteRetryMinutes != dwTempVar2))) { _itoa(dwTempVar, ScratchBuffer, 10); _itoa(dwTempVar2, ScratchBuffer2, 10); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = "Remote"; apszSubStrings[2] = ScratchBuffer; apszSubStrings[3] = ScratchBuffer2; SmtpLogEvent(SMTP_EVENT_SET_RETRY_PARAMETERS, 4, apszSubStrings, 0); } m_cRemoteRetryAttempts = dwTempVar; m_cRemoteRetryMinutes = dwTempVar2; StateTrace((LPARAM)this, "m_cRetryAttempts = %u", m_cRemoteRetryAttempts); StateTrace((LPARAM)this, "m_cRetryMinutes = %u", m_cRemoteRetryMinutes); TempString.Reset(); m_cbProgressiveRetryMinutes = sizeof(m_szProgressiveRetryMinutes); if (mb.GetStr("", MD_SMTP_REMOTE_PROGRESSIVE_RETRY_MINUTES, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT, "")) { lstrcpy(m_szProgressiveRetryMinutes, TempString.QueryStr()); m_cbProgressiveRetryMinutes = lstrlen(m_szProgressiveRetryMinutes) + 1; StateTrace((LPARAM)this, "Progressive retry minutes: <%s>, length %u", m_szProgressiveRetryMinutes, m_cbProgressiveRetryMinutes); } //Parse out the string to get at most 4 seperate minute values //the values will be part of a comma delimited string int cRetryMin[4]; DWORD i = 0; char *szTemp; char *Ptr = NULL; szTemp = m_szProgressiveRetryMinutes; //Parse out the 4 integers for (;i<4;) { Ptr = strchr(szTemp,','); if (Ptr) *Ptr = '\0'; cRetryMin[i] = 0; cRetryMin[i] = atoi(szTemp); if (cRetryMin[i] < 1) break; i++; if (Ptr) { *Ptr = ','; szTemp = ++Ptr; Ptr = NULL; } else break; } //check if could parse nothing if (i==0) { //dwRetryMin[i] = 60; //defualt retry is 60 minutes //NK** for now look at the old value - this way we can set something //using MMC cRetryMin[i] = m_cRemoteRetryMinutes; i++; } //If not all four retry intervals were specified //Make the use the last specifed interval in place of unspecified intervals while (i<4) { cRetryMin[i] = cRetryMin[i - 1]; i++; } AQConfig.dwFirstRetrySeconds = cRetryMin[0] * 60; AQConfig.dwSecondRetrySeconds = cRetryMin[1] * 60; AQConfig.dwThirdRetrySeconds = cRetryMin[2] * 60; AQConfig.dwFourthRetrySeconds = cRetryMin[3] * 60; AQConfig.dwConnectionRetryMilliseconds = (m_cRemoteRetryMinutes*60*1000); AQConfig.dwRetryThreshold = ReadMetabaseDword(mb,MD_SMTP_REMOTE_RETRY_THRESHOLD,3); if (!AQConfig.dwRetryThreshold) AQConfig.dwRetryThreshold = 3; StateTrace((LPARAM)this, "RetryThreshold = %u", AQConfig.dwRetryThreshold); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_CON_RETRY; // // If DSN intervals are not set, then derive them from the retry intervals // 10/6/98 - MikeSwa // AQConfig.dwDelayExpireMinutes = ReadMetabaseDword(mb, MD_SMTP_EXPIRE_REMOTE_DELAY_MIN, m_cRemoteRetryMinutes); AQConfig.dwNDRExpireMinutes = ReadMetabaseDword(mb, MD_SMTP_EXPIRE_REMOTE_NDR_MIN, m_cRemoteRetryMinutes * m_cRemoteRetryAttempts); AQConfig.dwAQConfigInfoFlags |= (AQ_CONFIG_INFO_EXPIRE_DELAY | AQ_CONFIG_INFO_EXPIRE_NDR); AQConfig.dwLocalDelayExpireMinutes = ReadMetabaseDword(mb, MD_SMTP_EXPIRE_LOCAL_DELAY_MIN, m_cRetryMinutes); AQConfig.dwLocalNDRExpireMinutes = ReadMetabaseDword(mb, MD_SMTP_EXPIRE_LOCAL_NDR_MIN, m_cRetryMinutes * m_cRetryAttempts); AQConfig.dwAQConfigInfoFlags |= (AQ_CONFIG_INFO_LOCAL_EXPIRE_DELAY | AQ_CONFIG_INFO_LOCAL_EXPIRE_NDR); } if (IsFieldSet(fc, FC_SMTP_INFO_PIPELINE)) { dwTempVar = !!ReadMetabaseDword(mb, MD_SHOULD_PIPELINE_OUT, TRUE); if (fShowEvents && ((dwTempVar && !m_fShouldPipelineOut) || (!dwTempVar && m_fShouldPipelineOut))) { apszSubStrings[0] = pchAddr1; if (dwTempVar) SmtpLogEvent(SMTP_EVENT_SET_PIPELINE_OUT_ENABLED, 1, apszSubStrings, 0); else SmtpLogEvent(SMTP_EVENT_SET_PIPELINE_OUT_DISABLED, 1, apszSubStrings, 0); } m_fShouldPipelineOut = dwTempVar; StateTrace((LPARAM)this, "m_fShouldPipelineOut = %u", m_fShouldPipelineOut); dwTempVar = !!ReadMetabaseDword(mb, MD_SHOULD_PIPELINE_IN, TRUE); if (fShowEvents && ((dwTempVar && !m_fShouldPipelineIn)|| (!dwTempVar && m_fShouldPipelineIn))) { apszSubStrings[0] = pchAddr1; if (dwTempVar) SmtpLogEvent(SMTP_EVENT_SET_PIPELINE_IN_ENABLED, 1, apszSubStrings, 0); else SmtpLogEvent(SMTP_EVENT_SET_PIPELINE_IN_DISABLED, 1, apszSubStrings, 0); } m_fShouldPipelineIn = dwTempVar; StateTrace((LPARAM)this, "m_fShouldPipelineIn = %u", m_fShouldPipelineIn); } if (IsFieldSet(fc, FC_SMTP_INFO_SMART_HOST)) { dwTempVar = ReadMetabaseDword(mb, MD_SMARTHOST_TYPE, 0); StateTrace((LPARAM)this, "m_fSmartHostType = %u", dwTempVar); if (dwTempVar != smarthostNone) { TempString.Reset(); if (! mb.GetStr("", MD_SMARTHOST_NAME, IIS_MD_UT_SERVER, &TempString) || TempString.IsEmpty()) { // Don't have a smart host, so turn off smart host and start. // Log it though, because the settings conflicted and so somebody // probably messed up... dwTempVar = m_fSmartHostType = smarthostNone; m_szSmartHostName[0] = '\0'; ErrorTrace((LPARAM)this, "Unable to read smart host name, error %u", dwErr); apszSubStrings[0] = pchAddr1; SmtpLogEvent(SMTP_EVENT_INVALID_SMART_HOST, 1, apszSubStrings, 0); } else { if (fShowEvents && lstrcmpi(m_szSmartHostName,TempString.QueryStr())) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = TempString.QueryStr(); SmtpLogEvent(SMTP_EVENT_SET_SMART_HOST_NAME, 2, apszSubStrings, 0); } lstrcpyn (m_szSmartHostName,TempString.QueryStr(), MAX_PATH); } StateTrace((LPARAM)this, "m_szSmartHost = %s", m_szSmartHostName); } // If the value is changing, generate an information NT event... if (fShowEvents && (dwTempVar != m_fSmartHostType)) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szSmartHostName; switch (dwTempVar) { case smarthostNone: { SmtpLogEvent(SMTP_EVENT_SET_SMART_HOST_TYPE_NONE, 1, apszSubStrings, 0); break; } case smarthostAfterFail: { SmtpLogEvent(SMTP_EVENT_SET_SMART_HOST_TYPE_AFTER_FAIL, 2, apszSubStrings, 0); break; } case smarthostAlways: { SmtpLogEvent(SMTP_EVENT_SET_SMART_HOST_TYPE_ALWAYS, 2, apszSubStrings, 0); break; } }//end switch }//endf if m_fSmartHostType = dwTempVar; } if (IsFieldSet(fc, FC_SMTP_INFO_MASQUERADE)) { m_fMasquerade = !!ReadMetabaseDword(mb, MD_DO_MASQUERADE, FALSE); StateTrace((LPARAM)this, "m_fSmartHostType = %u", m_fMasquerade); if (m_fMasquerade) { TempString.Reset(); if (! mb.GetStr("", MD_MASQUERADE_NAME, IIS_MD_UT_SERVER, &TempString) || TempString.IsEmpty()) { // Don't have a masquerade, so turn off masquerading and start. // Log it though, because the settings conflicted and so somebody // probably messed up... m_fMasquerade = 0; m_szMasqueradeName[0] = '\0'; ErrorTrace((LPARAM)this, "Unable to read masquerade name, error %u", dwErr); apszSubStrings[0] = pchAddr1; SmtpLogEvent(SMTP_EVENT_INVALID_MASQUERADE, 1, apszSubStrings, 0); } else { if (fShowEvents && lstrcmpi(m_szMasqueradeName,TempString.QueryStr())) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = TempString.QueryStr(); SmtpLogEvent(SMTP_EVENT_SET_MASQUERADE_NAME, 2, apszSubStrings, 0); } lstrcpyn (m_szMasqueradeName,TempString.QueryStr(), AB_MAX_DOMAIN); } StateTrace((LPARAM)this, "m_szMasqueradeName = %s", m_szMasqueradeName); } } if (IsFieldSet(fc, FC_SMTP_INFO_DEFAULT_DOMAIN)) { m_fDefaultDomainExists = TRUE; TempString.Reset(); if (! mb.GetStr("", MD_DEFAULT_DOMAIN_VALUE, IIS_MD_UT_SERVER, &TempString)) { // // we set this value on system start up this is a problem... it should be set. // lstrcpyn(m_szDefaultDomain, ((SMTP_IIS_SERVICE *) g_pInetSvc)->QueryTcpipName() , MAX_PATH); ErrorTrace((LPARAM) this, "Error reading Default Domain from Metabase. Using TcpipValue Name: %s", m_szDefaultDomain); } else if (TempString.IsEmpty()) { // // blow away the value, this will invoke the routine again, and it will be updated // with the higher level default. Just in case, put the Instance value in there as a // placeholder... just in case. // lstrcpyn(m_szDefaultDomain, ((SMTP_IIS_SERVICE *) g_pInetSvc)->QueryTcpipName() , MAX_PATH); ErrorTrace((LPARAM) this, "Default Domain was blank string, removing Instance value."); if (! mb.DeleteData("", MD_DEFAULT_DOMAIN_VALUE, IIS_MD_UT_SERVER, STRING_METADATA)) { ErrorTrace((LPARAM) this, "Error deleting Default Domain from Metabase. Using TcpipValue Name: %s", m_szDefaultDomain); } } else { lstrcpyn(m_szDefaultDomain,TempString.QueryStr(), sizeof(m_szDefaultDomain)); if (fShowEvents) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szDefaultDomain; // If the value is changing, generate an information NT event... SmtpLogEvent(SMTP_EVENT_SET_DEFAULT_DOMAIN, 2, apszSubStrings, 0); } } // // Inform aqueue of the default domain // AQConfig.szDefaultLocalDomain = m_szDefaultDomain; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_DEFAULT_DOMAIN; ErrorTrace((LPARAM)this , "Default domain is %s", m_szDefaultDomain); } if (IsFieldSet(fc, FC_SMTP_INFO_FQDN)) { TempString.Reset(); if (! mb.GetStr("", MD_FQDN_VALUE, IIS_MD_UT_SERVER, &TempString)) { // // We set this value on system start up - see main.cxx, // InitializeService // This is a problem... it should be set. // lstrcpyn(m_szFQDomainName, ((SMTP_IIS_SERVICE *) g_pInetSvc)->QueryTcpipName(), MAX_PATH); ErrorTrace((LPARAM) this, "Error reading FQDN value from Metabase. Using TcpipValue Name: %s", m_szFQDomainName); } else if (TempString.IsEmpty()) { // // blow away the value, this will invoke the routine again, and it will be updated // with the higher level default. Just in case, put the Instance value in there as a // placeholder... just in case. // lstrcpyn(m_szFQDomainName, ((SMTP_IIS_SERVICE *) g_pInetSvc)->QueryTcpipName() , MAX_PATH); ErrorTrace((LPARAM) this, "FQDN Value was blank string, removing Instance value."); if (! mb.DeleteData("", MD_FQDN_VALUE, IIS_MD_UT_SERVER, STRING_METADATA)) { ErrorTrace((LPARAM) this, "Error reading FQDN value from Metabase. Using TcpipValue Name: %s", m_szFQDomainName); } } else { lstrcpyn(m_szFQDomainName,TempString.QueryStr(), sizeof(m_szFQDomainName)); // // Register the SPNs for this virtual server. Currently, we ignore // errors here. // if (!RegisterServicePrincipalNames(FALSE)) { ErrorTrace((LPARAM) this, "Unable to register Kerberos SPNs %d, will try later", GetLastError()); } } //VerifyFQDNWithBindings(); AQConfig.szServerFQDN = m_szFQDomainName; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_SERVER_FQDN; ErrorTrace((LPARAM)this , "Fully qualified domain name is %s", m_szFQDomainName); } if (IsFieldSet(fc, FC_SMTP_INFO_SEND_TO_ADMIN)) { char * DomainOffset = NULL; DWORD NameSize = 0; TempString.Reset(); if (! mb.GetStr("", MD_SEND_NDR_TO, IIS_MD_UT_SERVER, &TempString) || TempString.IsEmpty()) { m_fSendNDRToAdmin = FALSE; m_AdminName[0] = '\0'; ErrorTrace((LPARAM)this, "Unable to read admin name, error %u", dwErr); } else { lstrcpyn (m_AdminName,TempString.QueryStr(), MAX_INTERNET_NAME); DomainOffset = strchr(m_AdminName, '@'); if (CAddr::ValidateCleanEmailName(m_AdminName, DomainOffset)) { m_fSendNDRToAdmin = TRUE; if (DomainOffset == NULL) { lstrcat(m_AdminName, "@"); lstrcat(m_AdminName, m_szDefaultDomain); } } else { ErrorTrace((LPARAM)this, "Unable to read admin name, error %u", dwErr); m_AdminName[0] = '\0'; m_fSendNDRToAdmin = FALSE; } } StateTrace((LPARAM)this, "m_AdminName = %s", m_AdminName); if (fShowEvents && m_fSendNDRToAdmin) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_AdminName; SmtpLogEvent(SMTP_EVENT_SET_SEND_NDR_TO_ADMIN_ENABLED, 2, apszSubStrings, 0); } else if (fShowEvents && !m_fSendNDRToAdmin) { apszSubStrings[0] = pchAddr1; SmtpLogEvent(SMTP_EVENT_SET_SEND_NDR_TO_ADMIN_DISABLED, 1, apszSubStrings, 0); } DomainOffset = NULL; NameSize = 0; TempString.Reset(); if (! mb.GetStr("", MD_SEND_BAD_TO, IIS_MD_UT_SERVER, &TempString) || TempString.IsEmpty()) { m_fSendBadToAdmin = FALSE; m_BadMailName[0] = '\0'; ErrorTrace((LPARAM)this, "Unable to read badmail email name, error %u", dwErr); } else { lstrcpyn(m_BadMailName,TempString.QueryStr(), MAX_INTERNET_NAME); DomainOffset = strchr(m_BadMailName, '@'); if (CAddr::ValidateCleanEmailName(m_BadMailName, DomainOffset)) { m_fSendBadToAdmin = TRUE; if (DomainOffset == NULL) { lstrcat(m_BadMailName, "@"); lstrcat(m_BadMailName, m_szDefaultDomain); } } else { m_fSendBadToAdmin = FALSE; m_BadMailName[0] = '\0'; ErrorTrace((LPARAM)this, "Unable to read badmail email name, error %u", dwErr); } } StateTrace((LPARAM)this, "m_BadMailName = %s", m_BadMailName); if (fShowEvents && m_fSendBadToAdmin) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_BadMailName; SmtpLogEvent(SMTP_EVENT_SET_SEND_BAD_TO_ADMIN_ENABLED, 2, apszSubStrings, 0); } else if (fShowEvents && !m_fSendBadToAdmin) { apszSubStrings[0] = pchAddr1; SmtpLogEvent(SMTP_EVENT_SET_SEND_BAD_TO_ADMIN_DISABLED, 1, apszSubStrings, 0); } } if (IsFieldSet(fc, FC_SMTP_INFO_DEFAULT_DROP_DIR)) { TempString.Reset(); if (! mb.GetStr("", MD_MAIL_DROP_DIR, IIS_MD_UT_SERVER, &TempString, 0) || TempString.IsEmpty()) { // We had a problem reading the metabase. // This is a very bad thing. m_szMailDropDir[0] = '\0'; m_cchMailDropDir = 0; apszSubStrings[0] = pchAddr1; apszSubStrings[1] = szMailDropDir; // // When Exchange is installed, drop-dir is "" which is a valid configuration // so we refrain from logging an event. // if(!TempString.IsEmpty()) SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, 0); } else { lstrcpyn(m_szMailDropDir, TempString.QueryStr(), MAX_PATH); if (fShowEvents) { apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailDropDir; SmtpLogEvent(SMTP_EVENT_SET_MAIL_DROP_DIR, 2, apszSubStrings, dwErr); } // We found a path in the reg, so see if we can use it... dwAttr = GetFileAttributes(m_szMailDropDir); if (dwAttr == 0xFFFFFFFF) { // The path doesn't exist yet, so we'll try to create it. if (!CreateLayerDirectory(m_szMailDropDir) && (dwErr = GetLastError()) != ERROR_ALREADY_EXISTS) { ErrorTrace((LPARAM)this, "Unable to create mail drop directory (%s)", m_szMailQueueDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailDropDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, dwErr); m_szMailDropDir[0] = '\0'; m_cchMailDropDir = 0; } } else { if (!(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) { // The registry points to a file, so we're outta luck. // The directory doesn't exist, and we can't create it. // Because this is the queue directory, we're not going // to be able to start. Log it so it can be fixed, but // set fRet to FALSE so that we shut back down. ErrorTrace((LPARAM)this, "Mail drop directory (%s) already exists as a file", m_szMailDropDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailDropDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, ERROR_ALREADY_EXISTS); m_szMailDropDir[0] = '\0'; } } if (m_szMailDropDir[0] != '\0') { // Looks like everything is going to be OK, so we add a backslash (if // necessary) because it makes things easier later. m_cchMailDropDir = lstrlen(m_szMailDropDir); if (m_cchMailDropDir > 0 && m_szMailDropDir[m_cchMailDropDir - 1] != '\\') { lstrcat(m_szMailDropDir, "\\"); m_cchMailDropDir++; } } } } // // Added by keithlau on 7/8/96 // if (IsFieldSet(fc, FC_SMTP_INFO_BAD_MAIL_DIR)) { TempString.Reset(); if (! mb.GetStr("", MD_BAD_MAIL_DIR, IIS_MD_UT_SERVER, &TempString, 0) || TempString.IsEmpty()) { // We had a problem reading the registry, so // set the flags back to not save bad mail and // log it and hop out. // This is NOT fatal. We'll still start up and run, // but we're not going to save bad mail, and we wanted // to, so at least we'll log the event. m_szBadMailDir[0] = '\0'; ErrorTrace((LPARAM)this, "Unable to read bad mail dir value, error = %u", dwErr); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = szBadMailDir; SmtpLogEvent(SMTP_EVENT_CANNOT_READ_SVC_REGKEY, 2, apszSubStrings, 0); } else { lstrcpyn (m_szBadMailDir,TempString.QueryStr(), MAX_PATH); } if (m_szBadMailDir[0] != '\0') { // We found a path in the registry, so see if it already exists. dwAttr = GetFileAttributes(m_szBadMailDir); if (dwAttr == 0xFFFFFFFF) { // It doesn't exist at all, so try to create it. if (!CreateLayerDirectory(m_szBadMailDir) && (dwErr = GetLastError()) != ERROR_ALREADY_EXISTS) { // We couldn't create it, so we can't save bad mail. // Log it because somebody messed up the registry. // This is NOT fatal. We'll still start up and run, // but we're not going to save bad mail, and we wanted // to, so at least we'll log the event. ErrorTrace((LPARAM)this, "Unable to create bad mail directory (%s)", m_szBadMailDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szBadMailDir; SmtpLogEvent(SMTP_EVENT_INVALID_BAD_MAIL_DIR, 2, apszSubStrings, 0); m_szBadMailDir[0] = '\0'; } } else { if (!(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) { // The registry points to a file, so we're outta luck. // The directory doesn't exist, and we can't create it. // This is NOT fatal. We'll still start up and run, // but we're not going to save bad mail, and we wanted // to, so at least we'll log the event. ErrorTrace((LPARAM)this, "Bad mail directory (%s) already exists", m_szBadMailDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szBadMailDir; SmtpLogEvent(SMTP_EVENT_INVALID_BAD_MAIL_DIR, 2, apszSubStrings, 0); m_szBadMailDir[0] = '\0'; } } StateTrace((LPARAM)this, "Save bad mail is enabled and m_szBadMailDir = %s", m_szBadMailDir); } else { // We at least leave a trace saying that save bad mail is disabled StateTrace((LPARAM)this, "Save bad mail is disabled"); } if (m_szBadMailDir[0] != '\0') { // Finally, if everything has gone OK, then add a trailing backslash so // that things are easier later. dwLen = lstrlen(m_szBadMailDir); if (dwLen > 0 && m_szBadMailDir[dwLen-1] != '\\') lstrcat(m_szBadMailDir, "\\"); } } if (IsFieldSet(fc, FC_SMTP_INFO_DOMAIN_ROUTING)) { dwTempVar = ReadMetabaseDword(mb, MD_ROUTE_ACTION, 0); // // It is not possible to configure the default routeaction // as drop. This will cause all mail to be NDR'd (the config // is pushed to AQ with no drop directory. // if (dwTempVar & SMTP_DROP) { ErrorTrace((LPARAM) this, "Default RouteAction cannot be SMTP_DROP... Action was 0x%X now 0x%X", dwTempVar, dwTempVar & ~SMTP_DROP); dwTempVar &= ~SMTP_DROP; } m_DefaultRouteAction = dwTempVar; TempString.Reset(); if (!mb.GetStr("", MD_ROUTE_USER_NAME, IIS_MD_UT_SERVER, &TempString, 0) || TempString.IsEmpty()) { m_DefaultRemoteUserName[0] = '\0'; m_DefaultRemotePassword[0] = '\0'; m_DefaultRouteAction &= ~SMTP_AUTH_NTLM; m_DefaultRouteAction &= ~SMTP_AUTH_CLEARTEXT; m_DefaultRouteAction &= ~SMTP_AUTH_KERBEROS; m_DefaultRouteAction &= ~SMTP_SASL; } else { lstrcpyn (m_DefaultRemoteUserName,TempString.QueryStr(), MAX_INTERNET_NAME); } TempString.Reset(); if (!mb.GetStr("", MD_ROUTE_PASSWORD, IIS_MD_UT_SERVER, &TempString, METADATA_SECURE) || TempString.IsEmpty()) { m_DefaultRemotePassword[0] = '\0'; } else { lstrcpyn (m_DefaultRemotePassword,TempString.QueryStr(), MAX_PATH); } // Get change domain name. if (m_pChangeObject) { char *psz = strstr((const char *) m_pChangeObject->pszMDPath, "/Domain/"); if (!psz) { // Changing default config or even more, rebuild fRebuild = TRUE; } else { // Found a normal domain change, copy the name strncpy(szDomainName, psz+8, strlen(psz)-8); szDomainName[strlen(psz)-8-1]='\0'; } if (!strcmp(szDomainName, "*")) { // Changing '*' domain, rebuild fRebuild = TRUE; } } fRet = GetRouteDomains(mb, szDomainName, fRebuild); } AQConfig.cbVersion = sizeof(AQConfigInfo); //Send badmail information AQConfig.szBadMailDir = m_szBadMailDir; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_BADMAIL_DIR; //Get DSN Options AQConfig.dwDSNOptions = ReadMetabaseDword(mb, MD_SMTP_DSN_OPTIONS, DSN_OPTIONS_DEFAULT); AQConfig.dwDSNLanguageID = ReadMetabaseDword(mb, MD_SMTP_DSN_LANGUAGE_ID, 0); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_USE_DSN_OPTIONS | AQ_CONFIG_INFO_USE_DSN_LANGUAGE; if (m_fSendNDRToAdmin) AQConfig.szSendCopyOfNDRToAddress = m_AdminName; else AQConfig.szSendCopyOfNDRToAddress = NULL; AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_SEND_DSN_TO; m_pIAdvQueueConfig->SetConfigInfo(&AQConfig); // // jstamerj 1998/11/17 16:34:48: // Set the well known proeprties of ISession // hr = HrSetWellKnownIServerProps(); if(FAILED(hr)) { // // Just like everything else in this fucntion, don't treat // this as fatal; just log it // ErrorTrace((LPARAM)this, "HrSetWellKnownIServerProps failed hr %08lx", hr); } m_GenLock.ExclusiveUnlock(); if (!fRet && (GetLastError() == NO_ERROR)) SetLastError(ERROR_PATH_NOT_FOUND); TraceFunctLeaveEx((LPARAM)this); return fRet; } //+--------------------------------------------------------------- // // Function: DeleteDomainEntry // // Synopsis: Delete a domain from cached tables. // // Arguments: DomainName: name of the domain to delete. // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::DeleteDomainEntry(const char *DomainName) { TraceFunctEnterEx((LPARAM)this, "DeleteDomainEntry"); DebugTrace((LPARAM)this , "Delete Domain [%s] ", DomainName); // Remove this domain from our SMTPSVC tables m_TurnAccessList.RemoveFromTable((const char *) DomainName); TraceFunctLeaveEx((LPARAM)this); return TRUE; } //+--------------------------------------------------------------- // // Function: RemoveRegParams // // Synopsis: Remove metabase info from cached tables. // We only handle domain removal at this moment. // // Arguments: DomainName: name of the domain to delete. // If DomainName is NULL, we get it from // m_pChangeObject. // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::RemoveRegParams(const char *DomainName) { TraceFunctEnterEx((LPARAM)this, "RemoveRegParams"); char szDomainName [AB_MAX_DOMAIN + 1] = {0}; BOOL fRet = TRUE; m_GenLock.ExclusiveLock(); if (!m_pChangeObject) goto Exit; if (DomainName) { strcpy(szDomainName, DomainName); } else { // At this stage, we only handle Domain removale. char *psz = strstr((const char *) m_pChangeObject->pszMDPath, "/Domain/"); if (psz) { // We find it. strncpy(szDomainName, psz+8, strlen(psz)-8); szDomainName[strlen(psz)-8-1]='\0'; } } if (szDomainName[0] == '\0') goto Exit; { HRESULT hr = S_OK; char szRoutePath [MAX_PATH + 1]; char szActionType [MAX_PATH + 1]; char szUserName [MAX_INTERNET_NAME + 1]; char szEtrnDomain [AB_MAX_DOMAIN + 1]; char szPassword [MAX_PATH + 1]; char szTargetName [MAX_PATH + 1]; DomainInfo DefaultDomainInfo; DomainInfo StarDomainInfo; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); if( !mb.Open( QueryMDPath(), METADATA_PERMISSION_READ | METADATA_PERMISSION_WRITE) ) { fRet = FALSE; goto Exit; } if (!strcmp(szDomainName, "*")) { // Deleting the '*' domain, must rebuild fRet = GetRouteDomains(mb, szDomainName, TRUE); goto Exit; } // Delete the domain from the table in SMTPSVC DeleteDomainEntry((const char *) szDomainName); // Since we have no way to tell AQ to delete this domain apart from // doing a full rebuild, we are going to push the defaults to AQ // to overwrite the custom settings this domain had before // Load up the "default config" : We get this by loading the server // default config and then trying to overwrite it with the '*' // domain config (if present) // REVIEW : Ideally this code would be in DeleteDomainEntry but putting it // there hurts us now because we have to call DeleteDomainEntry from // SetRouteDomainParameters to clean the tables in SMTPSVC. If we can get // rid of those tables, the layout of these functions shoule be reconsidered. wsprintf(szRoutePath, ""); SetRouteDomainParameters(mb, szDomainName, szRoutePath, szActionType, szUserName, szEtrnDomain, szPassword, szTargetName, &DefaultDomainInfo); if (AlwaysUseSmartHost() && m_szSmartHostName[0]) { DefaultDomainInfo.dwDomainInfoFlags |= DOMAIN_INFO_REMOTE_SMARTHOST; DefaultDomainInfo.szSmartHostDomainName = m_szSmartHostName; DefaultDomainInfo.cbSmartHostDomainNameLength = lstrlen(m_szSmartHostName); } //check if we should always use SSL if (GetDefaultRouteAction() & SMTP_SSL) { DefaultDomainInfo.dwDomainInfoFlags |= DOMAIN_INFO_USE_SSL; } // Try to overwrite this with the "*" domain if it exists wsprintf(szRoutePath, "/Domain/*"); SetRouteDomainParameters(mb, szDomainName, szRoutePath, szActionType, szUserName, szEtrnDomain, szPassword, szTargetName, &StarDomainInfo); // If we found the star domain - dwDomainInfoFlags is zero if // the domain does not exist in the metabase if (StarDomainInfo.dwDomainInfoFlags) { hr = m_pIAdvQueueConfig->SetDomainInfo(&StarDomainInfo); } // else push the default domain info else { hr = m_pIAdvQueueConfig->SetDomainInfo(&DefaultDomainInfo); } } Exit: m_GenLock.ExclusiveUnlock(); TraceFunctLeaveEx((LPARAM)this); return fRet; } BOOL SMTP_SERVER_INSTANCE::GetCatInfo(MB& mb, AQConfigInfo& AQConfig) { STR TempString; char Password[MAX_PATH]; char BindType[MAX_PATH]; char SchemaType[MAX_PATH]; char Domain[MAX_PATH]; char UserName[MAX_PATH]; char Host[MAX_PATH]; char NamingContext[MAX_PATH]; char DsType[MAX_PATH]; Password[0] = '\0'; BindType[0] = '\0'; SchemaType[0] = '\0'; Domain[0] = '\0'; UserName[0] = '\0'; Host[0] = '\0'; NamingContext[0] = '\0'; DsType [0] = '\0'; if( mb.GetDword("RoutingSources", MD_SMTP_DS_USE_CAT, IIS_MD_UT_SERVER, &AQConfig.dwMsgCatEnable)) { AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_ENABLE; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutDWORD( PE_ISERVID_DW_CATENABLE, AQConfig.dwMsgCatEnable))); } if( mb.GetDword("RoutingSources", MD_SMTP_DS_FLAGS, IIS_MD_UT_SERVER, &AQConfig.dwMsgCatFlags)) { AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_FLAGS; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutDWORD( PE_ISERVID_DW_CATFLAGS, AQConfig.dwMsgCatFlags))); } if( mb.GetDword("RoutingSources", MD_SMTP_DS_PORT, IIS_MD_UT_SERVER, &AQConfig.dwMsgCatPort)) { AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_PORT; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutDWORD( PE_ISERVID_DW_CATPORT, AQConfig.dwMsgCatPort))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_ACCOUNT, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(UserName, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_USER; AQConfig.szMsgCatUser = UserName; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATUSER, TempString.QueryStr()))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_SCHEMA_TYPE, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(SchemaType, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_SCHEMATYPE; AQConfig.szMsgCatSchemaType = SchemaType; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATSCHEMA, TempString.QueryStr()))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_BIND_TYPE, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(BindType, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_BINDTYPE; AQConfig.szMsgCatBindType = BindType; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATBINDTYPE, TempString.QueryStr()))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_PASSWORD, IIS_MD_UT_SERVER, &TempString, METADATA_INHERIT | METADATA_SECURE )) { lstrcpyn(Password, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_PASSWORD; AQConfig.szMsgCatPassword = Password; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATPASSWORD, TempString.QueryStr()))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_DOMAIN, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(Domain, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_DOMAIN; AQConfig.szMsgCatDomain = Domain; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATDOMAIN, TempString.QueryStr()))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_NAMING_CONTEXT, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(NamingContext, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_NAMING_CONTEXT; AQConfig.szMsgCatNamingContext = NamingContext; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATNAMINGCONTEXT, TempString.QueryStr()))); } TempString.Reset(); if (mb.GetStr("RoutingSources", MD_SMTP_DS_TYPE, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(DsType, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_TYPE; AQConfig.szMsgCatType = DsType; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATDSTYPE, TempString.QueryStr()))); } AQConfig.cbVersion = sizeof(AQConfigInfo); TempString.Reset(); if(mb.GetStr("RoutingSources", MD_SMTP_DS_HOST, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(Host, TempString.QueryStr(), MAX_PATH); AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_HOST; AQConfig.szMsgCatHost = Host; _VERIFY(SUCCEEDED( m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_CATDSHOST, TempString.QueryStr()))); } // // Indicate that all the official "msgcat" keys that exist are set // AQConfig.dwAQConfigInfoFlags |= AQ_CONFIG_INFO_MSGCAT_DEFAULT; AQConfig.cbVersion = sizeof(AQConfigInfo); m_pIAdvQueueConfig->SetConfigInfo(&AQConfig); return TRUE; } BOOL SMTP_SERVER_INSTANCE::ReadRouteDomainIpSecList(MB& mb) { IMDCOM* pMBCom; HRESULT hRes; METADATA_RECORD mdRecord; DWORD dwRequiredLen; DWORD dwErr; BOOL fSuccess = TRUE; char *szValueName = ""; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::ReadRouteDomainIpSecList"); ResetRelayIpSecList(); pMBCom = (IMDCOM*)m_Service->QueryMDObject(); mdRecord.dwMDIdentifier = MD_SMTP_IP_RELAY_ADDRESSES; mdRecord.dwMDAttributes = METADATA_INHERIT; mdRecord.dwMDUserType = IIS_MD_UT_FILE; mdRecord.dwMDDataType = BINARY_METADATA; mdRecord.dwMDDataLen = 5000; mdRecord.pbMDData = (PBYTE) new char [5000]; if (mdRecord.pbMDData == NULL) { TraceFunctLeaveEx((LPARAM)this); return FALSE; } hRes = pMBCom->ComMDGetMetaData( mb.QueryHandle(), (LPBYTE)szValueName, &mdRecord, &dwRequiredLen ); if ( SUCCEEDED( hRes ) ) { SetRelayIpSecList(mdRecord.pbMDData, mdRecord.dwMDDataLen, mdRecord.dwMDDataTag ); } else { if (mdRecord.pbMDData) delete mdRecord.pbMDData; fSuccess = FALSE; dwErr = HRESULTTOWIN32( hRes ); } TraceFunctLeaveEx((LPARAM)this); return fSuccess; } void SMTP_SERVER_INSTANCE::BuildTurnTable(MULTISZ& msz, char * szDomainName) { const char * StartPtr = NULL; CTurnData * pTurnData = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::BuildTurnTable"); for (StartPtr = msz.First(); StartPtr != NULL; StartPtr = msz.Next( StartPtr )) { ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); DebugTrace((LPARAM)this , "%s is allowed to issue TURN", StartPtr); pTurnData = new CTurnData(StartPtr, szDomainName); if (pTurnData != NULL) { if (!m_TurnAccessList.InsertIntoTable((CHASH_ENTRY *) pTurnData)) { delete pTurnData; } } } TraceFunctLeaveEx((LPARAM)this); } //+--------------------------------------------------------------- // // Function: SetRouteDomainParameters // // Synopsis: Does the actual work of adding a domain from the metabase // //---------------------------------------------------------------- void SMTP_SERVER_INSTANCE::SetRouteDomainParameters(MB &mb, char *szDomainName, char *szRoutePath, char szActionType [MAX_PATH + 1], char szUserName [MAX_INTERNET_NAME + 1], char szEtrnDomain [MAX_INTERNET_NAME + 1], char szPassword [MAX_PATH + 1], char szTargetName [MAX_PATH + 1], DomainInfo *pLocalDomainInfo) { TraceFunctEnter("SMTP_SERVER_INSTANCE::SetRouteDomainParameters"); char * DomainPtr = NULL; char * UserNamePtr = NULL; char * PasswordPtr = NULL; char * TargetNamePtr = NULL; DWORD dwAction = 0; STR TempString; DWORD dwErr = NO_ERROR; DWORD dwLen = 0; BOOL RelayForAuth = TRUE; DWORD dwEtrnWaitTime = 0; BOOL WildCard = FALSE; CHAR pchAddr1[32] = ""; MULTISZ msz; if (szDomainName[0]) { ZeroMemory(pLocalDomainInfo, sizeof(DomainInfo)); pLocalDomainInfo->cbVersion = sizeof(DomainInfo); pLocalDomainInfo->cbDomainNameLength = lstrlen(szDomainName); pLocalDomainInfo->szDomainName = szDomainName; szActionType [0] = '\0'; szUserName [0] = '\0'; szPassword [0] = '\0'; szEtrnDomain[0] = '\0'; szTargetName[0] = '\0'; UserNamePtr = szUserName; PasswordPtr = szPassword; TargetNamePtr = szTargetName; dwAction = 0; DomainPtr = szDomainName; if (!mb.GetDword(szRoutePath, MD_ROUTE_ACTION, IIS_MD_UT_SERVER, &dwAction)) { dwAction = 0; } // If we don't recognize the actions we bail now - SMTP_DEFAULT // shound never be set on any RouteAction in the metabase if ((dwAction & ~SMTP_ALL_ROUTE_FLAGS) || (dwAction & SMTP_DEFAULT)) { TraceFunctLeave(); return; } // // It is not possible to configure the default routeaction // as drop. This will cause all mail to be NDR'd (the config // is pushed to AQ with no drop directory. // if ((0 == strcmp(szRoutePath, "")) && dwAction & SMTP_DROP) { ErrorTrace((LPARAM) this, "Default '*' RouteAction cannot be SMTP_DROP...Action was 0x%X now 0x%X", dwAction, dwAction & ~SMTP_DROP); dwAction &= ~SMTP_DROP; } TempString.Reset(); if (mb.GetStr(szRoutePath, MD_ROUTE_ACTION_TYPE, IIS_MD_UT_SERVER, &TempString, 0) && !TempString.IsEmpty()) { lstrcpyn(szActionType, TempString.QueryStr(), MAX_PATH); } TempString.Reset(); if (((dwAction & SMTP_SMARTHOST) || (dwAction & SMTP_SASL)) && mb.GetStr(szRoutePath, MD_ROUTE_USER_NAME, IIS_MD_UT_SERVER, &TempString) && !TempString.IsEmpty()) { lstrcpyn(szUserName, TempString.QueryStr(), MAX_INTERNET_NAME); } else { UserNamePtr = NULL; } TempString.Reset(); if (UserNamePtr != NULL) { if (mb.GetStr(szRoutePath, MD_ROUTE_PASSWORD, IIS_MD_UT_SERVER, &TempString, METADATA_SECURE | METADATA_INHERIT) && !TempString.IsEmpty()) { lstrcpyn(szPassword, TempString.QueryStr(), MAX_PATH); } } else { PasswordPtr = ""; } TempString.Reset(); if (TargetNamePtr != NULL) { if (mb.GetStr(szRoutePath, MD_ROUTE_AUTHTARGET, IIS_MD_UT_SERVER, &TempString, METADATA_SECURE | METADATA_INHERIT) && !TempString.IsEmpty()) { lstrcpyn(szTargetName, TempString.QueryStr(), MAX_PATH); } } else { TargetNamePtr = ""; } TempString.Reset(); if (mb.GetStr(szRoutePath, MD_SMTP_CSIDE_ETRN_DOMAIN, IIS_MD_UT_SERVER, &TempString, 0) && !TempString.IsEmpty()) { lstrcpyn(szEtrnDomain, TempString.QueryStr(), AB_MAX_DOMAIN); } dwEtrnWaitTime = ReadMetabaseDword(mb, MD_SMTP_CSIDE_ETRN_DELAY, 0); // // If the domain is set to explicitly allow relay for authenticated users.. // use that value. Otherwise, use the value set on the VSI (to maintain // backwards compatibility with W2K users). // if (dwAction & SMTP_AUTH_RELAY) RelayForAuth = TRUE; else RelayForAuth = m_fRelayForAuthUsers; if (dwAction & SMTP_SMARTHOST) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_REMOTE_SMARTHOST; if (dwAction & SMTP_SSL) pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_USE_SSL; if (szActionType[0] != '\0') { pLocalDomainInfo->szSmartHostDomainName = szActionType; pLocalDomainInfo->cbSmartHostDomainNameLength = lstrlen(szActionType); } else if (*szDomainName == '*') { //The domain names could be specified as "*" - an explicit smart //host must be used. pLocalDomainInfo->dwDomainInfoFlags &= ~DOMAIN_INFO_REMOTE_SMARTHOST; } else { //use the domain name as our smarthost pLocalDomainInfo->szSmartHostDomainName = szDomainName; pLocalDomainInfo->cbSmartHostDomainNameLength = lstrlen(pLocalDomainInfo->szSmartHostDomainName); } } //if we are doing SASL, make sure we have a username and password //and a correct authentication bits if (dwAction & SMTP_SASL) { // One of these must be set or else the SMTP_SASL flag doesn't // make sense if (!(dwAction & SMTP_AUTH_NTLM) && !(dwAction & SMTP_AUTH_CLEARTEXT) && !(dwAction & SMTP_AUTH_KERBEROS)) { // dbraun : we can be cleaner about handling this - bailing // leaves us with an incompletely configured domain TraceFunctLeave(); return; } if (UserNamePtr) { pLocalDomainInfo->szUserName = UserNamePtr; pLocalDomainInfo->cbUserNameLength = lstrlen (UserNamePtr); } if (PasswordPtr) { pLocalDomainInfo->szPassword = PasswordPtr; pLocalDomainInfo->cbPasswordLength = lstrlen(PasswordPtr); } if (dwAction & SMTP_AUTH_NTLM) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_USE_NTLM; } if (dwAction & SMTP_AUTH_CLEARTEXT) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_USE_PLAINTEXT; } if (dwAction & SMTP_AUTH_KERBEROS) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_USE_KERBEROS; if (TargetNamePtr) { pLocalDomainInfo->szAuthType = TargetNamePtr; pLocalDomainInfo->cbAuthTypeLength = lstrlen(TargetNamePtr); } } } char * szTmpDomain; //Check for the wildcard entries if (*DomainPtr == '*' && *(DomainPtr+1) == '.') { WildCard = TRUE; szTmpDomain = DomainPtr + 2; } else { szTmpDomain = DomainPtr; } if (szTmpDomain[0] != '*' || szTmpDomain[1] != 0) { if (!CAddr::ValidateDomainName(szTmpDomain)) { ErrorTrace((LPARAM)this , "%s is an invalid domain", DomainPtr); //bad domain name TraceFunctLeave(); return; } } if (dwAction & SMTP_DROP) { if (!CreateLayerDirectory(szActionType) && (dwErr = GetLastError()) != ERROR_ALREADY_EXISTS) { // Reflect the changes const CHAR *aszStrings[3]; CHAR szTemp1[MAX_PATH + 1]; CHAR szTemp2[MAX_PATH + 1]; aszStrings[0] = pchAddr1; aszStrings[1] = szTemp1; aszStrings[2] = szTemp2; lstrcpyn(szTemp1, szActionType, MAX_PATH); lstrcpyn(szTemp2, DomainPtr, MAX_PATH); SmtpLogEvent(SMTP_EVENT_NO_DROP_DIRECTORY, 3, aszStrings, dwErr); } dwLen = lstrlen(szActionType); if (dwLen > 0 && szActionType[dwLen-1] != '\\') lstrcat(szActionType, "\\"); pLocalDomainInfo->szDropDirectory = szActionType; pLocalDomainInfo->cbDropDirectoryLength = lstrlen (szActionType); pLocalDomainInfo->dwDomainInfoFlags = DOMAIN_INFO_LOCAL_DROP; } // Remove original one if it is there DeleteDomainEntry((const char *) DomainPtr); // // Insert local domains into the routing table. // if ((dwAction & SMTP_DELIVER) && (szActionType[0] == '\0')) { DebugTrace((LPARAM)this , "adding %s as a DELIVER domain", szDomainName); pLocalDomainInfo->dwDomainInfoFlags = DOMAIN_INFO_LOCAL_MAILBOX; } else if (dwAction & SMTP_ALIAS) { DebugTrace((LPARAM)this , "adding %s as a ALIAS domain", szDomainName); if (IsDefaultInRt()) { pLocalDomainInfo->dwDomainInfoFlags = DOMAIN_INFO_LOCAL_MAILBOX | DOMAIN_INFO_ALIAS; } else { pLocalDomainInfo->szDropDirectory = m_szMailDropDir; pLocalDomainInfo->cbDropDirectoryLength = lstrlen (m_szMailDropDir); pLocalDomainInfo->dwDomainInfoFlags = DOMAIN_INFO_LOCAL_DROP | DOMAIN_INFO_ALIAS; } } if (RelayForAuth) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_AUTH_RELAY; } if (dwAction & SMTP_DOMAIN_RELAY) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_DOMAIN_RELAY; } msz.Reset(); if ((dwAction & SMTP_ETRN_CMD) && mb.GetMultisz(szRoutePath, MD_SMTP_AUTHORIZED_TURN_LIST, IIS_MD_UT_SERVER, &msz) && !msz.IsEmpty()) { BuildTurnTable(msz, szTmpDomain); pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_TURN_ONLY; } if (dwAction & SMTP_ETRN_CMD && !(dwAction & SMTP_DISABLE_ETRN) ) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_ETRN_ONLY; }//if(dwAction & SMTP_ETRN_CMD) if (dwAction & SMTP_USE_HELO ) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_USE_HELO; }//if(dwAction & SMTP_USE_HELO) if (dwAction & SMTP_CSIDE_ETRN) { TempString.Reset(); if (mb.GetStr(szRoutePath, MD_SMTP_CSIDE_ETRN_DOMAIN, IIS_MD_UT_SERVER, &TempString, 0) && !TempString.IsEmpty()) { pLocalDomainInfo->szETRNDomainName = szEtrnDomain; lstrcpyn(pLocalDomainInfo->szETRNDomainName, TempString.QueryStr(), AB_MAX_DOMAIN); pLocalDomainInfo->cbETRNDomainNameLength = strlen(szEtrnDomain); pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_SEND_ETRN; } }//if(dwAction & SMTP_CSIDE_ETRN) if (dwAction & SMTP_CSIDE_TURN) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_SEND_TURN; pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_TURN_ON_EMPTY; }//if(dwAction & SMTP_CSIDE_TURN) if (dwAction & SMTP_DISABLE_BMIME) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_DISABLE_BMIME; }//if(dwAction & SMTP_DISABLE_BMIME) if (dwAction & SMTP_CHUNKING) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_USE_CHUNKING; }//if(dwAction & SMTP_CHUNKING) else if (dwAction & SMTP_DISABLE_CHUNK) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_DISABLE_CHUNKING; pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_DISABLE_BMIME; } if (dwAction & SMTP_DISABLE_DSN) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_DISABLE_DSN; }//if(dwAction & SMTP_DISBALE_DSN) if (dwAction & SMTP_DISABLE_PIPELINE) { pLocalDomainInfo->dwDomainInfoFlags |= DOMAIN_INFO_DISABLE_PIPELINE; }//if(dwAction & SMTP_DISABLE_PIPELINE) }//if (szDomainName[0]) TraceFunctLeave(); } //+--------------------------------------------------------------- // // Function: AddDomainEntry // // Synopsis: Add a domain into cached tables. // // Arguments: MB: already opened metabase. // DomainName: name of the domain to add. // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::AddDomainEntry (MB &mb, char * DomainName) { HRESULT hr = S_OK; BOOL fReturn = TRUE; char szDomainName [AB_MAX_DOMAIN + 1]; char szRoutePath [MAX_PATH + 1]; char szActionType [MAX_PATH + 1]; char szUserName [MAX_INTERNET_NAME + 1]; char szEtrnDomain [AB_MAX_DOMAIN + 1]; char szPassword [MAX_PATH + 1]; char szTargetName [MAX_PATH + 1]; DomainInfo LocalDomainInfo; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::AddNewEntry"); strcpy(szDomainName, DomainName); if (szDomainName[0]) { szRoutePath[0] = '\0'; wsprintf(szRoutePath, "/Domain/%s", szDomainName); SetRouteDomainParameters(mb, szDomainName, szRoutePath, szActionType, szUserName, szEtrnDomain, szPassword, szTargetName, &LocalDomainInfo); hr = m_pIAdvQueueConfig->SetDomainInfo(&LocalDomainInfo); if (FAILED(hr)) { fReturn = FALSE; } }//if (szDomainName[0]) TraceFunctLeaveEx((LPARAM)this); return fReturn; } //+--------------------------------------------------------------- // // Function: GetRouteDomains // // Synopsis: Get Routing domains from the metabase // // Arguments: MB: already opened metabase. // DomainName: name of the domain to add. // fRebuild: bool indicating if we need to rebuild // the whole table // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::GetRouteDomains(MB &mb, char * DomainName, BOOL fRebuild) { HRESULT hr = S_OK; DWORD ThreadId; DWORD error; BOOL fReturn = TRUE; DomainInfo LocalDomainInfo; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::GetRouteDomains"); ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); m_TurnAccessList.SetDupesAllowed(); // Only rebuild when fRebuild is TRUE. if (fRebuild) { // We need to clean m_hEnumDomainThreadHandle here // to avoid handle leak. if (m_hEnumDomainThreadHandle) { if (!m_fEnumThreadStarted) { // the thread has been created, but it has not been // able to grab the gencrit lock yet. This means // that it will be doing a full rebuild already, so // we can just bail goto cleanup; } // Set up signal and wait WaitForSingleObject(m_hEnumDomainThreadHandle, INFINITE); ErrorTrace((LPARAM)this, "EnumDomain Thread is dead"); CloseHandle(m_hEnumDomainThreadHandle); m_hEnumDomainThreadHandle = NULL; } m_fEnumThreadStarted = FALSE; m_hEnumDomainThreadHandle = CreateThread (NULL, 0, EnumAllDomains, this, 0, &ThreadId); if(m_hEnumDomainThreadHandle == NULL) { error = GetLastError(); ErrorTrace((LPARAM)this, "CreateThread failed for EnumAllDomains. err: %u", error); fReturn = FALSE; } else { DebugTrace((LPARAM)this , "EnumDomainThread is created, threadId [%ld]", ThreadId); } } else { if (!AddDomainEntry(mb, DomainName)) fReturn = FALSE; } cleanup: TraceFunctLeaveEx((LPARAM)this); return fReturn; } //+--------------------------------------------------------------- // // Function: EnumAllDomains // // Synopsis: Enumerate all domains in this SMTP instance. // // Arguments: ptr: pointer to the SMTP server instance. // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // //---------------------------------------------------------------- DWORD EnumAllDomains(VOID *ptr) { HRESULT hr = S_OK; char szDomainName [AB_MAX_DOMAIN + 1]; char szRoutePath [MAX_PATH + 1]; char szActionType [MAX_PATH + 1]; char szUserName [MAX_INTERNET_NAME + 1]; char szEtrnDomain [AB_MAX_DOMAIN + 1]; char szPassword [MAX_PATH + 1]; char szTargetName [MAX_PATH + 1]; char szStarDomain[] = "*"; DWORD dwTotal = 0; DomainInfo LocalDomainInfo; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); SMTP_SERVER_INSTANCE *pInstance = (SMTP_SERVER_INSTANCE *) ptr; TraceFunctEnterEx((LPARAM)pInstance, "EnumAllDomains"); pInstance->ExclusiveLockGenCrit(); pInstance->m_fEnumThreadStarted = TRUE; pInstance->m_TurnAccessList.SetDupesAllowed(); pInstance->m_TurnAccessList.RemoveAllEntries(); //Signal AQ that we are starting over pInstance->m_pIAdvQueueConfig->StartConfigUpdate(); if (pInstance->m_szMailDropDir[0] == '\0') { ZeroMemory(&LocalDomainInfo, sizeof(DomainInfo)); LocalDomainInfo.cbVersion = sizeof(DomainInfo); LocalDomainInfo.cbDomainNameLength = lstrlen(pInstance->m_szDefaultDomain); LocalDomainInfo.szDomainName = pInstance->m_szDefaultDomain; LocalDomainInfo.dwDomainInfoFlags = DOMAIN_INFO_LOCAL_MAILBOX; HRESULT hr; hr = pInstance->m_pIAdvQueueConfig->SetDomainInfo(&LocalDomainInfo); DebugTrace((LPARAM)pInstance , "default domain %s was added to the routing table", pInstance->m_szDefaultDomain); //signals default domain was added to routing table pInstance->m_fDefaultInRt = TRUE; } else { ZeroMemory(&LocalDomainInfo, sizeof(DomainInfo)); LocalDomainInfo.cbVersion = sizeof(DomainInfo); LocalDomainInfo.cbDomainNameLength = lstrlen(pInstance->m_szDefaultDomain); LocalDomainInfo.szDomainName = pInstance->m_szDefaultDomain; LocalDomainInfo.szDropDirectory = pInstance->m_szMailDropDir; LocalDomainInfo.cbDropDirectoryLength = lstrlen (pInstance->m_szMailDropDir); LocalDomainInfo.dwDomainInfoFlags = DOMAIN_INFO_LOCAL_DROP; hr = pInstance->m_pIAdvQueueConfig->SetDomainInfo(&LocalDomainInfo); } if( !mb.Open( pInstance->QueryMDPath(), METADATA_PERMISSION_READ ) ) { pInstance->ExclusiveUnLockGenCrit(); return FALSE; } while (!pInstance->IsShuttingDown() && mb.EnumObjects("/Domain", szDomainName, dwTotal++)) { if (szDomainName[0]) { szRoutePath[0] = '\0'; wsprintf(szRoutePath, "/Domain/%s", szDomainName); pInstance->SetRouteDomainParameters(mb, szDomainName, szRoutePath, szActionType, szUserName, szEtrnDomain, szPassword, szTargetName, &LocalDomainInfo); hr = pInstance->m_pIAdvQueueConfig->SetDomainInfo(&LocalDomainInfo); if (FAILED(hr)) { } }//if (szDomainName[0]) }//while (mb.EnumObjects("/RouteDomains", szDomainName, dw++)) pInstance->SetRouteDomainParameters(mb, szStarDomain, "", szActionType, szUserName, szEtrnDomain, szPassword, szTargetName, &LocalDomainInfo); if (pInstance->AlwaysUseSmartHost() && pInstance->m_szSmartHostName[0]) { LocalDomainInfo.dwDomainInfoFlags |= DOMAIN_INFO_REMOTE_SMARTHOST; LocalDomainInfo.szSmartHostDomainName = pInstance->m_szSmartHostName; LocalDomainInfo.cbSmartHostDomainNameLength = lstrlen(pInstance->m_szSmartHostName); } //check if we should always use SSL if (pInstance->GetDefaultRouteAction() & SMTP_SSL) { LocalDomainInfo.dwDomainInfoFlags |= DOMAIN_INFO_USE_SSL; } hr = pInstance->m_pIAdvQueueConfig->SetDomainInfo(&LocalDomainInfo); // jstamerj 1998/07/24 11:18:39: // For M2, AQueue requires a local domain entry " " in its // configuration to handle recipients without SMTP addresses // (such as an remote X400 address). This will be removed when // code is added to encapsualte non-SMTP addresses into SMTP. // Add this local domain here. // ZeroMemory(&LocalDomainInfo, sizeof(DomainInfo)); LocalDomainInfo.cbVersion = sizeof(DomainInfo); LocalDomainInfo.cbDomainNameLength = 1; LocalDomainInfo.szDomainName = " "; LocalDomainInfo.dwDomainInfoFlags = DOMAIN_INFO_LOCAL_MAILBOX; hr = pInstance->m_pIAdvQueueConfig->SetDomainInfo(&LocalDomainInfo); //Signal AQ that we are finished updating domain config pInstance->m_pIAdvQueueConfig->FinishConfigUpdate(); pInstance->ExclusiveUnLockGenCrit(); TraceFunctLeaveEx((LPARAM)pInstance); return TRUE; } BOOL SMTP_SERVER_INSTANCE::ReadIpSecList(void) { IMDCOM* pMBCom; METADATA_HANDLE hMB; HRESULT hRes; METADATA_RECORD mdRecord; DWORD dwRequiredLen; DWORD dwErr; BOOL fSuccess; m_GenLock.ExclusiveLock(); m_rfAccessCheck.Reset( (IMDCOM*)m_Service->QueryMDObject() ); pMBCom = (IMDCOM*)m_Service->QueryMDObject(); hRes = pMBCom->ComMDOpenMetaObject( METADATA_MASTER_ROOT_HANDLE, (BYTE *) QueryMDPath(), METADATA_PERMISSION_READ, 5000, &hMB ); if ( SUCCEEDED( hRes ) ) { mdRecord.dwMDIdentifier = MD_IP_SEC; mdRecord.dwMDAttributes = METADATA_INHERIT | METADATA_REFERENCE; mdRecord.dwMDUserType = IIS_MD_UT_FILE; mdRecord.dwMDDataType = BINARY_METADATA; mdRecord.dwMDDataLen = 0; mdRecord.pbMDData = (PBYTE)NULL; hRes = pMBCom->ComMDGetMetaData( hMB, (LPBYTE)"", &mdRecord, &dwRequiredLen ); if ( SUCCEEDED( hRes ) && mdRecord.dwMDDataTag ) { m_rfAccessCheck.Set( mdRecord.pbMDData, mdRecord.dwMDDataLen, mdRecord.dwMDDataTag ); } DBG_REQUIRE( SUCCEEDED(pMBCom->ComMDCloseMetaObject( hMB )) ); } else { fSuccess = FALSE; dwErr = HRESULTTOWIN32( hRes ); } m_GenLock.ExclusiveUnlock(); return TRUE; } BOOL IsNTFS(IN LPCSTR pszRealPath) /*++ Gets file system specific information for a given path. It uses GetVolumeInfomration() to query the file system type and file system flags. On success the flags and file system type are returned in passed in pointers. --*/ { #define MAX_FILE_SYSTEM_NAME_SIZE ( MAX_PATH) CHAR rgchBuf[MAX_FILE_SYSTEM_NAME_SIZE]; CHAR rgchRoot[MAX_FILE_SYSTEM_NAME_SIZE]; int i; DWORD dwReturn = ERROR_PATH_NOT_FOUND; if ( pszRealPath == NULL) { return FALSE; } if ( pszRealPath[0] == ('\\') && pszRealPath[1] == ('\\')) { return FALSE; } // else ZeroMemory( (void *) rgchRoot, sizeof(rgchRoot) ); // // This is non UNC name. // Copy just the root directory to rgchRoot for querying // for ( i = 0; i < 9 && pszRealPath[i] != '\0'; i++) { if ( (rgchRoot[i] = pszRealPath[i]) == ':') { break; } } // for if ( rgchRoot[i] != ':') { // // we could not find the root directory. // return with error value // return ( FALSE); } rgchRoot[i+1] = '\\'; // terminate the drive spec with a slash rgchRoot[i+2] = '\0'; // terminate the drive spec with null char // The rgchRoot should end with a "\" (slash) // otherwise, the call will fail. if ( GetVolumeInformation( rgchRoot, // lpRootPathName NULL, // lpVolumeNameBuffer 0, // len of volume name buffer NULL, // lpdwVolSerialNumber NULL, // lpdwMaxComponentLength NULL, // lpdwSystemFlags rgchBuf, // lpFileSystemNameBuff sizeof(rgchBuf)/sizeof(WCHAR) )) { if ( lstrcmp( rgchBuf, "NTFS") == 0) { return TRUE; } } return ( FALSE); } //+--------------------------------------------------------------- // // Function: SMTPCONFIG::ReadStartupRegParams // // Synopsis: Reads fixed (i.e. not modifiable on-the-fly) parameters // from the registry into the config class member variables. // // Arguments: None // // Returns: BOOL - TRUE on SUCCESS, FALSE on FAIL // // Note: Created by KeithLau on 7/15/96 // //---------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::ReadStartupRegParams(VOID) { BOOL fRet = TRUE; DWORD dwErr = NO_ERROR; DWORD dwAttr; char szValueName[MAX_PATH + 1]; MB mb( (IMDCOM*) g_pInetSvc->QueryMDObject() ); STR TempString; const CHAR * apszSubStrings[2]; CHAR pchAddr1[32] = ""; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::ReadStartupRegParams"); GetServerBindings(); m_GenLock.ExclusiveLock(); // // Read metabase data. // lstrcpy(szValueName, QueryMDPath()); _itoa(QueryInstanceId(), pchAddr1, 10); if ( !mb.Open( szValueName, METADATA_PERMISSION_READ | METADATA_PERMISSION_WRITE) ) { //UnLockConfig(); m_GenLock.ExclusiveUnlock(); return FALSE; } m_CmdLogFlags = ReadMetabaseDword(mb, MD_COMMAND_LOG_MASK, DEFAULT_CMD_LOG_FLAGS); m_fFlushMailFiles = !!ReadMetabaseDword(mb, MD_FLUSH_MAIL_FILE, TRUE); m_cMaxRoutingThreads = ReadMetabaseDword(mb, MD_ROUTING_THREADS, 8); StateTrace((LPARAM)this, "m_cMaxRoutingThreads = %u", m_cMaxRoutingThreads); m_fDisablePickupDotStuff = !!ReadMetabaseDword(mb, MD_SMTP_DISABLE_PICKUP_DOT_STUFF, FALSE); StateTrace((LPARAM)this, "m_fDisablePickupDotStuff = %u", m_fDisablePickupDotStuff); m_cMaxRemoteQThreads = ReadMetabaseDword(mb, MD_SMTP_MAX_REMOTEQ_THREADS, 1); StateTrace((LPARAM)this, "m_cMaxRoutingThreads = %u",m_cMaxRemoteQThreads); m_cMaxLocalQThreads = ReadMetabaseDword(mb, MD_SMTP_MAX_LOCALQ_THREADS, 1); StateTrace((LPARAM)this, "m_cMaxRoutingThreads = %u", m_cMaxLocalQThreads); TempString.Reset(); if (!mb.GetStr("", MD_MAIL_QUEUE_DIR, IIS_MD_UT_SERVER, &TempString, 0) || TempString.IsEmpty()) { // We had a problem reading the registry. // This is a very bad thing. We obviously cannot run without // a queue directory, so set fRet to FALSE so that we shut // back down (and log it so that an admin can fix it). m_szMailQueueDir[0] = '\0'; fRet = FALSE; apszSubStrings[0] = pchAddr1; apszSubStrings[1] = szMailQueueDir; SmtpLogEvent(SMTP_EVENT_CANNOT_READ_SVC_REGKEY, 2, apszSubStrings, 0); } else { lstrcpyn(m_szMailQueueDir, TempString.QueryStr(), MAX_PATH); if (!IsNTFS(m_szMailQueueDir)) { m_IsFileSystemNtfs = FALSE; DebugTrace((LPARAM)this, "Queue dir (%s) is not NTFS", m_szMailQueueDir); } // We found a path in the reg, so see if we can use it... dwAttr = GetFileAttributes(m_szMailQueueDir); if (dwAttr == 0xFFFFFFFF) { // The path doesn't exist yet, so we'll try to create it. if (!CreateLayerDirectory(m_szMailQueueDir) && (dwErr = GetLastError()) != ERROR_ALREADY_EXISTS) { // It doesn't exist and we couldn't create it, so // log it and bail with fRet = FALSE so that we shut // back down. ErrorTrace((LPARAM)this, "Unable to create mail queue directory (%s)", m_szMailQueueDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailQueueDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_QUEUE_DIR, 2, apszSubStrings, dwErr); m_szMailQueueDir[0] = '\0'; fRet = FALSE; } } else { if (!(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) { // The registry points to a file, so we're outta luck. // The directory doesn't exist, and we can't create it. // Because this is the queue directory, we're not going // to be able to start. Log it so it can be fixed, but // set fRet to FALSE so that we shut back down. ErrorTrace((LPARAM)this, "Mail queue directory (%s) already exists as a file", m_szMailQueueDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailQueueDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_QUEUE_DIR, 2, apszSubStrings, dwErr); m_szMailQueueDir[0] = '\0'; fRet = FALSE; } } } TempString.Reset(); if (! mb.GetStr("", MD_MAIL_DROP_DIR, IIS_MD_UT_SERVER, &TempString, 0) || TempString.IsEmpty()) { // We had a problem reading the metabase. // This is a very bad thing. We obviously cannot run without // a drop directory if we are in drop mode, so set fRet to // FALSE so that we shut back down (and log it so that an // admin can fix it). m_szMailDropDir[0] = '\0'; m_cchMailDropDir = 0; //don't start if we are disabled, since drop directory is not //configurable. if (!m_fIsRoutingTable) { fRet = FALSE; apszSubStrings[0] = pchAddr1; apszSubStrings[1] = szMailDropDir; SmtpLogEvent(SMTP_EVENT_CANNOT_READ_SVC_REGKEY, 2, apszSubStrings, 0); } } else { lstrcpyn(m_szMailDropDir, TempString.QueryStr(), MAX_PATH); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailDropDir; // We found a path in the reg, so see if we can use it... dwAttr = GetFileAttributes(m_szMailDropDir); if (dwAttr == 0xFFFFFFFF) { // The path doesn't exist yet, so we'll try to create it. if (!CreateLayerDirectory(m_szMailDropDir) && (dwErr = GetLastError()) != ERROR_ALREADY_EXISTS) { // It doesn't exist and we couldn't create it, so // log it and bail with fRet = FALSE so that we shut // back down. ErrorTrace((LPARAM)this, "Unable to create mail drop directory (%s)", m_szMailQueueDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailDropDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, dwErr); m_szMailDropDir[0] = '\0'; m_cchMailDropDir = 0; //don't start if we are disabled, since drop directory is not //configurable. if (!m_fIsRoutingTable) fRet = FALSE; } } else { if (!(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) { // The registry points to a file, so we're outta luck. // The directory doesn't exist, and we can't create it. // Because this is the queue directory, we're not going // to be able to start. Log it so it can be fixed, but // set fRet to FALSE so that we shut back down. ErrorTrace((LPARAM)this, "Mail drop directory (%s) already exists as a file", m_szMailDropDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailDropDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, dwErr); m_szMailDropDir[0] = '\0'; //don't start if we are disabled, since drop directory is not //configurable. if (!m_fIsRoutingTable) fRet = FALSE; } } // Looks like everything is going to be OK, so we add a backslash (if // necessary) because it makes things easier later. m_cchMailDropDir = lstrlen(m_szMailDropDir); if (m_cchMailDropDir > 0 && m_szMailDropDir[m_cchMailDropDir - 1] != '\\') { lstrcat(m_szMailDropDir, "\\"); m_cchMailDropDir++; } } // Looks like everything is going to be OK, so we add a backslash (if // necessary) because it makes things easier later. m_cchMailQueueDir = lstrlen(m_szMailQueueDir); if (m_cchMailQueueDir > 0 && m_szMailQueueDir[m_cchMailQueueDir - 1] != '\\') { lstrcat(m_szMailQueueDir, "\\"); m_cchMailQueueDir++; } m_fShouldPickupMail = !!ReadMetabaseDword(mb, MD_SHOULD_PICKUP_MAIL, TRUE); if (m_fShouldPickupMail) { m_cMaxDirBuffers = ReadMetabaseDword(mb, MD_MAX_DIR_BUFFERS, 2000); m_cMaxDirChangeIoSize = ReadMetabaseDword(mb, MD_MAX_DIR_CHANGE_IO_SIZE, 1000); m_cMaxDirPendingIos = ReadMetabaseDword(mb, MD_MAX_DIR_PENDING_IOS, 1); TempString.Reset(); if (!mb.GetStr("", MD_MAIL_PICKUP_DIR, IIS_MD_UT_SERVER, &TempString, 0) || TempString.IsEmpty()) { // We had a problem reading the registry. // This is a very bad thing. m_szMailPickupDir[0] = '\0'; m_fShouldPickupMail = 0; apszSubStrings[0] = pchAddr1; apszSubStrings[1] = szMailPickupDir; SmtpLogEvent(SMTP_EVENT_CANNOT_READ_SVC_REGKEY, 2, apszSubStrings, 0); } else { lstrcpyn(m_szMailPickupDir, TempString.QueryStr(), MAX_PATH); // We found a path in the reg, so see if we can use it... dwAttr = GetFileAttributes(m_szMailPickupDir); if (dwAttr == 0xFFFFFFFF) { // The path doesn't exist yet, so we'll try to create it. if (!CreateLayerDirectory(m_szMailPickupDir) && (dwErr = GetLastError()) != ERROR_ALREADY_EXISTS) { // It doesn't exist and we couldn't create it, so // log it and bail with fRet = FALSE so that we shut // back down. ErrorTrace((LPARAM)this, "Unable to create mail pickup queue directory (%s)", m_szMailPickupDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailPickupDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, dwErr); m_szMailPickupDir[0] = '\0'; m_fShouldPickupMail = 0; } } else { if (!(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) { // The registry points to a file, so we're outta luck. // The directory doesn't exist, and we can't create it. ErrorTrace((LPARAM)this, "Mail pickup queue directory (%s) already exists as a file", m_szMailPickupDir); apszSubStrings[0] = pchAddr1; apszSubStrings[1] = m_szMailPickupDir; SmtpLogEvent(SMTP_EVENT_INVALID_MAIL_DROP_DIR, 2, apszSubStrings, dwErr); m_szMailPickupDir[0] = '\0'; m_fShouldPickupMail = 0; } } } // Looks like everything is going to be OK, so we add a backslash (if // necessary) because it makes things easier later. m_cchMailPickupDir = lstrlen(m_szMailPickupDir); if (m_cchMailPickupDir > 0 && m_szMailPickupDir[m_cchMailPickupDir - 1] != '\\') { lstrcat(m_szMailPickupDir, "\\"); m_cchMailPickupDir++; } } if (!mb.GetStr("", MD_CONNECT_RESPONSE, IIS_MD_UT_SERVER, &TempString) || TempString.IsEmpty()) { // We had a problem reading the metabase // This is a very bad thing. lstrcpy (m_szConnectResponse, "Microsoft ESMTP MAIL Service, "); lstrcat (m_szConnectResponse, g_VersionString); lstrcat (m_szConnectResponse, " ready at "); // SmtpLogEvent(SMTP_EVENT_CANNOT_READ_SVC_REGKEY, 2, apszSubStrings, 0); } else { lstrcpyn(m_szConnectResponse, TempString.QueryStr(), MAX_PATH); } m_cchConnectResponse = lstrlenA(m_szConnectResponse); mb.Close(); m_GenLock.ExclusiveUnlock(); ReadIpSecList(); if (!fRet) SetLastError(ERROR_PATH_NOT_FOUND); TraceFunctLeaveEx((LPARAM)this); return fRet; } void SMTP_SERVER_INSTANCE::GetServerBindings(void) { DWORD dwErr = NO_ERROR; MB mb( (IMDCOM*)g_pInetSvc->QueryMDObject() ); TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::GetServerBindings"); m_GenLock.ExclusiveLock(); // // open metabase data. // if ( !mb.Open( QueryMDPath(), METADATA_PERMISSION_READ ) ) { m_GenLock.ExclusiveUnlock(); TraceFunctLeaveEx((LPARAM)this); return ; } m_ServerBindings.Reset(); if (!mb.GetMultisz("", MD_SERVER_BINDINGS, IIS_MD_UT_SERVER, &m_ServerBindings) || m_ServerBindings.IsEmpty()) { dwErr = GetLastError(); ErrorTrace((LPARAM)this, "Unable to read server bindings, error = %u", dwErr); } m_GenLock.ExclusiveUnlock(); TraceFunctLeaveEx((LPARAM)this); } VOID SMTP_SERVER_INSTANCE::MDChangeNotify( MD_CHANGE_OBJECT * pco ) /*++ This method handles the metabase change notification for this instance Arguments: hMDHandle - Metabase handle generating the change notification pcoChangeList - path and id that has changed --*/ { FIELD_CONTROL control = 0; BOOL fSslModified = FALSE; DWORD i; DWORD err; DWORD id; DWORD MdState; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::MDChangeNotify"); // // Tell our parent about the change notification first // LockThisForWrite(); IIS_SERVER_INSTANCE::MDChangeNotify( pco ); // Save change object m_pChangeObject = pco; for ( i = 0; i < pco->dwMDNumDataIDs; i++ ) { id = pco->pdwMDDataIDs[i]; switch ( id ) { case MD_REVERSE_NAME_LOOKUP: control |= FC_SMTP_INFO_REVERSE_LOOKUP; break; case MD_NTAUTHENTICATION_PROVIDERS: control |= FC_SMTP_INFO_NTAUTHENTICATION_PROVIDERS; break; case MD_MD_SERVER_SS_AUTH_MAPPING: case MD_SMTP_CLEARTEXT_AUTH_PROVIDER: control |= FC_SMTP_CLEARTEXT_AUTH_PROVIDER; break; case MD_AUTHORIZATION: control |= FC_SMTP_INFO_AUTHORIZATION; break; case MD_HOP_COUNT: control |= FC_SMTP_INFO_MAX_HOP_COUNT; break; case MD_MAX_SMTP_ERRORS: case MD_MAX_SMTP_AUTHLOGON_ERRORS: control |= FC_SMTP_INFO_MAX_ERRORS; break; case MD_MAX_MSG_SIZE: case MD_MAX_MSG_SIZE_B4_CLOSE: control |= FC_SMTP_INFO_MAX_SIZE; break; case MD_REMOTE_TIMEOUT: control |= FC_SMTP_INFO_REMOTE_TIMEOUT; break; case MD_MAX_OUTBOUND_CONNECTION: control |= FC_SMTP_INFO_MAX_OUTBOUND_CONN; break; case MD_MAX_RECIPIENTS: control |= FC_SMTP_INFO_MAX_RECIPS; break; case MD_ETRN_SUBDOMAINS: control |= FC_SMTP_INFO_ETRN_SUBDOMAINS; break; case MD_MAIL_DROP_DIR: control |= FC_SMTP_INFO_DEFAULT_DROP_DIR; control |= FC_SMTP_INFO_DOMAIN_ROUTING; break; case MD_LOCAL_RETRY_ATTEMPTS: case MD_LOCAL_RETRY_MINUTES: case MD_REMOTE_RETRY_ATTEMPTS: case MD_REMOTE_RETRY_MINUTES: // case MD_SHARE_RETRY_MINUTES: case MD_SMTP_REMOTE_RETRY_THRESHOLD: case MD_SMTP_REMOTE_PROGRESSIVE_RETRY_MINUTES: case MD_SMTP_EXPIRE_REMOTE_NDR_MIN: case MD_SMTP_EXPIRE_REMOTE_DELAY_MIN: case MD_SMTP_EXPIRE_LOCAL_NDR_MIN: case MD_SMTP_EXPIRE_LOCAL_DELAY_MIN: control |= FC_SMTP_INFO_RETRY; break; case MD_SHOULD_PIPELINE_OUT: case MD_SHOULD_PIPELINE_IN: control |= FC_SMTP_INFO_PIPELINE; break; case MD_SMTP_DS_TYPE: case MD_SMTP_DS_DATA_DIRECTORY: case MD_SMTP_DS_DEFAULT_MAIL_ROOT: case MD_SMTP_DS_BIND_TYPE: case MD_SMTP_DS_SCHEMA_TYPE: case MD_SMTP_DS_HOST: case MD_SMTP_DS_NAMING_CONTEXT: case MD_SMTP_DS_ACCOUNT: case MD_SMTP_DS_PASSWORD: case MD_SMTP_DS_DOMAIN: case MD_SMTP_DS_USE_CAT: case MD_SMTP_DS_PORT: case MD_SMTP_DS_FLAGS: control |= FC_SMTP_INFO_ROUTING; break; case MD_SEND_BAD_TO: case MD_SEND_NDR_TO: control |= FC_SMTP_INFO_SEND_TO_ADMIN; break; case MD_SMARTHOST_TYPE: case MD_SMARTHOST_NAME: control |= FC_SMTP_INFO_SMART_HOST; control |= FC_SMTP_INFO_DOMAIN_ROUTING; break; case MD_NAME_RESOLUTION_TYPE: case MD_BATCH_MSG_LIMIT: case MD_SMTP_IP_RELAY_ADDRESSES: case MD_SMTP_DISABLE_RELAY: case MD_SMTP_MAIL_NO_HELO: case MD_SMTP_HELO_NODOMAIN: case MD_SMTP_DISABLE_PICKUP_DOT_STUFF: case MD_SMTP_EVENTLOG_LEVEL: case MD_SMTP_DENIED_IP_ACTION: control |= FC_SMTP_INFO_COMMON_PARAMS; break; case MD_DEFAULT_DOMAIN_VALUE: control |= FC_SMTP_INFO_DEFAULT_DOMAIN; control |= FC_SMTP_INFO_DOMAIN_ROUTING; break; case MD_FQDN_VALUE: control |= FC_SMTP_INFO_FQDN; control |= FC_SMTP_INFO_DOMAIN_ROUTING; break; case MD_BAD_MAIL_DIR: control |= FC_SMTP_INFO_BAD_MAIL_DIR; break; case MD_DO_MASQUERADE: case MD_MASQUERADE_NAME: control |= FC_SMTP_INFO_MASQUERADE; break; case MD_REMOTE_SMTP_PORT: control |= FC_SMTP_INFO_REMOTE_PORT; break; case MD_LOCAL_DOMAINS: control |= FC_SMTP_INFO_LOCAL_DOMAINS; break; case MD_DOMAIN_ROUTING: control |= FC_SMTP_INFO_DOMAIN_ROUTING; break; case MD_POSTMASTER_EMAIL: case MD_POSTMASTER_NAME: control |= FC_SMTP_INFO_ADMIN_EMAIL_NAME; break; case MD_SMTP_SSL_REQUIRE_TRUSTED_CA: case MD_SMTP_SSL_CERT_HOSTNAME_VALIDATION: case MD_SSL_ACCESS_PERM: control |= FC_SMTP_INFO_SSL_PERM; break; case MD_SASL_LOGON_DOMAIN: control |= FC_SMTP_INFO_SASL_LOGON_DOMAIN; break; case MD_MAX_OUT_CONN_PER_DOMAIN: control |= FC_SMTP_INFO_MAX_OUT_CONN_PER_DOMAIN; break; case MD_INBOUND_COMMAND_SUPPORT_OPTIONS: case MD_OUTBOUND_COMMAND_SUPPORT_OPTIONS: case MD_ADD_NOHEADERS : control |= FC_SMTP_INFO_INBOUND_SUPPORT_OPTIONS; break; case MD_SERVER_BINDINGS: GetServerBindings(); break; case MD_IP_SEC: control |= FC_SMTP_INFO_DOMAIN_ROUTING; ReadIpSecList(); break; case MD_SMTP_RELAY_FOR_AUTH_USERS: control |= FC_SMTP_INFO_DOMAIN_ROUTING; control |= FC_SMTP_INFO_COMMON_PARAMS; break; case MD_DOMAIN_VALIDATION_FLAGS: control |= FC_SMTP_INFO_COMMON_PARAMS; break; case MD_ROUTE_ACTION: case MD_ROUTE_ACTION_TYPE: case MD_ROUTE_USER_NAME: case MD_ROUTE_PASSWORD: case MD_ROUTE_AUTHTARGET: case MD_SMTP_CSIDE_ETRN_DELAY: case MD_SMTP_CSIDE_ETRN_DOMAIN: case MD_SMTP_AUTHORIZED_TURN_LIST: control |= FC_SMTP_INFO_DOMAIN_ROUTING; break; case MD_SSL_CERT_HASH: case MD_SSL_CERT_CONTAINER: case MD_SSL_CERT_PROVIDER: case MD_SSL_CERT_OPEN_FLAGS: case MD_SSL_CERT_STORE_NAME: case MD_SSL_CTL_IDENTIFIER: case MD_SSL_CTL_CONTAINER: case MD_SSL_CTL_PROVIDER: case MD_SSL_CTL_PROVIDER_TYPE: case MD_SSL_CTL_OPEN_FLAGS: case MD_SSL_CTL_STORE_NAME: fSslModified = TRUE; break; default: break; } } //We need to react to domains being deleted // 9/29/98 - MikeSwa if (MD_CHANGE_TYPE_DELETE_OBJECT == pco->dwMDChangeType) { control |= FC_SMTP_INFO_DOMAIN_ROUTING; } // // If anything related to SSL has changed, call the function used to flush // the SSL/Schannel credential cache and reset the server certificate // if ( fSslModified && g_pSslKeysNotify ) { (g_pSslKeysNotify) ( SIMSSL_NOTIFY_MAPPER_SSLKEYS_CHANGED, this ); ResetSSLInfo(this); } MdState = QueryServerState( ); if ( (MdState == MD_SERVER_STATE_STOPPING) || (MdState == MD_SERVER_STATE_STOPPED)) { goto Done; } if ( control != 0 ) { if (!ReadRegParams(control, FALSE)) { err = GetLastError(); DBGPRINTF(( DBG_CONTEXT, "SMTP_SERVER_INSTANCE::MDChangeNotify() cannot read config, error %lx\n", err )); } } // Handle deleting domains if (control & FC_SMTP_INFO_DOMAIN_ROUTING) { // We are deleting objects if (pco->dwMDChangeType == MD_CHANGE_TYPE_DELETE_OBJECT) { if (!RemoveRegParams(NULL)) { err = GetLastError(); DBGPRINTF(( DBG_CONTEXT, "SMTP_SERVER_INSTANCE::MDChangeNotify() cannot remove config, error %lx\n", err )); } } } Done: m_pChangeObject = NULL; UnlockThis(); TraceFunctLeaveEx((LPARAM)this); } IIS_SSL_INFO* SMTP_SERVER_INSTANCE::QueryAndReferenceSSLInfoObj( VOID ) /*++ Description Returns SSL info for this instance; calls Reference() before returning We actually call GetCertificate( ) here, so the name is really incorrect, but changing it will involve asking IIS to change IISTYPES to add the new method as a vitual function. Arguments: None Returns: Ptr to SSL info object on success, NULL if failure --*/ { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::QueryAndReferenceSSLInfoObj"); IIS_SSL_INFO *pPtr = NULL; LockThisForRead(); // // If it's null, we may have to create it - unlock, lock for write and make sure it's // still NULL before creating it // if ( !m_pSSLInfo ) { UnlockThis(); LockThisForWrite(); // // Still null, so create it now // if ( !m_pSSLInfo ) { m_pSSLInfo = new IIS_SSL_INFO( (LPTSTR) QueryMDPath(), (IMDCOM *) g_pInetSvc->QueryMDObject() ); if ( m_pSSLInfo == NULL ) { SetLastError( ERROR_NOT_ENOUGH_MEMORY ); UnlockThis(); return NULL; } // // Acquire an internal reference // m_pSSLInfo->Reference(); // // Register for changes // IIS_SERVER_CERT *pCert = m_pSSLInfo->GetCertificate(); if ( pCert ) { // Do logging if we fail to get the certificate LogCertStatus(); } //NIMISHK**** Do I need CTL - maybe I can get rid of this call IIS_CTL *pCTL = m_pSSLInfo->GetCTL(); if ( pCTL ) { //Do logging if we fail to get the CTL LogCTLStatus(); } if ( g_pCAPIStoreChangeNotifier ) { if ( pCert && pCert->IsValid() ) { if (!g_pCAPIStoreChangeNotifier->RegisterStoreForChange( pCert->QueryStoreName(), pCert->QueryStoreHandle(), ResetSSLInfo, (PVOID) this ) ) { DebugTrace((LPARAM)this, "Failed to register for change event on store %s", pCert->QueryStoreName()); } } if ( pCTL && pCTL->IsValid() ) { if (!g_pCAPIStoreChangeNotifier->RegisterStoreForChange( pCTL->QueryStoreName(), pCTL->QueryOriginalStore(), ResetSSLInfo, (PVOID) this ) ) { DebugTrace((LPARAM)this, "Failed to register for change event on store %s", pCTL->QueryStoreName()); } } if ( ( pCert && pCert->IsValid()) || ( pCTL && pCTL->IsValid() ) ) { HCERTSTORE hRootStore = CertOpenStore( CERT_STORE_PROV_SYSTEM_A, 0, NULL, CERT_SYSTEM_STORE_LOCAL_MACHINE, "ROOT" ); if ( hRootStore ) { // // Watch for changes to the ROOT store // if ( !g_pCAPIStoreChangeNotifier->RegisterStoreForChange( "ROOT", hRootStore, ResetSSLInfo, (PVOID) this ) ) { DebugTrace((LPARAM)this, "Failed to register for change event on root store"); } CertCloseStore( hRootStore, 0 ); } else { DebugTrace((LPARAM)this, "Failed to open ROOT store, error 0x%d", GetLastError()); } } // if ( pCert || pCTL ) } // if (g_pCAPIStoreChangeNotifier) } // if ( !m_pSSLInfo ) } //if ( !m_pSSLInfo ) // // At this point, m_pSSLInfo should not be NULL anymore, so add the external reference // m_pSSLInfo->Reference(); pPtr = m_pSSLInfo; UnlockThis(); TraceFunctLeaveEx((LPARAM)this); return pPtr; } VOID SMTP_SERVER_INSTANCE::ResetSSLInfo( LPVOID pvParam ) /*++ Description: Wrapper function for function to call to notify of SSL changes Arguments: pvParam - pointer to instance for which SSL keys have changed Returns: Nothing --*/ { TraceFunctEnterEx((LPARAM)NULL, "SMTP_SERVER_INSTANCE::ResetSSLInfo"); // // Call function to flush credential cache etc // if ( g_pSslKeysNotify ) { g_pSslKeysNotify( SIMSSL_NOTIFY_MAPPER_SSLKEYS_CHANGED, pvParam ); } SMTP_SERVER_INSTANCE *pInst = (SMTP_SERVER_INSTANCE *) pvParam; pInst->LockThisForRead(); if ( pInst->m_pSSLInfo ) { pInst->UnlockThis(); pInst->LockThisForWrite(); if ( pInst->m_pSSLInfo ) { // // Stop watching for change notifications // IIS_SERVER_CERT *pCert = pInst->m_pSSLInfo->QueryCertificate(); IIS_CTL *pCTL = pInst->m_pSSLInfo->QueryCTL(); if ( g_pCAPIStoreChangeNotifier ) { // // Stop watching the store the cert came out of // if ( pCert && pCert->IsValid() ) { g_pCAPIStoreChangeNotifier->UnregisterStore( pCert->QueryStoreName(), ResetSSLInfo, (PVOID) pvParam ); } // // Stop watching the store the CTL came out of // if ( pCTL && pCTL->IsValid() ) { g_pCAPIStoreChangeNotifier->UnregisterStore( pCTL->QueryStoreName(), ResetSSLInfo, (PVOID) pvParam ); } // // Stop watching the ROOT store // g_pCAPIStoreChangeNotifier->UnregisterStore( "ROOT", ResetSSLInfo, (PVOID) pvParam ); } // // Release internal reference // IIS_SSL_INFO::Release( pInst->m_pSSLInfo ); // // Next call to QueryAndReferenceSSLObj() will create it again // pInst->m_pSSLInfo = NULL; } } pInst->UnlockThis(); TraceFunctLeaveEx((LPARAM)NULL); } VOID SMTP_SERVER_INSTANCE::LogCertStatus() /*++ Description: Writes system log event about status of server certificate if the cert is in some way not quite kosher eg expired, revoked, not signature-valid Arguments: None Returns: Nothing --*/ { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::LogCertStatus"); _ASSERT( m_pSSLInfo ); DWORD dwCertValidity = 0; // // If we didn't construct the cert fully, log an error // if ( !m_pSSLInfo->QueryCertificate()->IsValid() ) { CONST CHAR *apszMsgs[2]; CHAR achInstance[20]; CHAR achErrorNumber[20]; wsprintf( achInstance, "%lu", QueryInstanceId() ); wsprintf( achErrorNumber, "0x%x", GetLastError() ); apszMsgs[0] = achInstance; apszMsgs[1] = achErrorNumber; DWORD dwStatus = m_pSSLInfo->QueryCertificate()->Status(); DWORD dwStringID = 0; DebugTrace((LPARAM)this, "Couldn't retrieve server cert; status : %d", dwStatus); switch ( dwStatus ) { case CERT_ERR_MB: dwStringID = SSL_MSG_CERT_MB_ERROR; break; case CERT_ERR_CAPI: dwStringID = SSL_MSG_CERT_CAPI_ERROR; break; case CERT_ERR_CERT_NOT_FOUND: dwStringID = SSL_MSG_CERT_NOT_FOUND; break; default: dwStringID = SSL_MSG_CERT_INTERNAL_ERROR; break; } SmtpLogEvent(dwStringID, 2, apszMsgs, 0 ); TraceFunctLeaveEx((LPARAM)this); return; } // // If cert is invalid in some other way , write the appropriate log message // if ( m_pSSLInfo->QueryCertValidity( &dwCertValidity ) ) { const CHAR *apszMsgs[1]; CHAR achInstance[20]; wsprintfA( achInstance, "%lu", QueryInstanceId() ); apszMsgs[0] = achInstance; DWORD dwMsgID = 0; if ( ( dwCertValidity & CERT_TRUST_IS_NOT_TIME_VALID ) || ( dwCertValidity & CERT_TRUST_IS_NOT_TIME_NESTED ) || ( dwCertValidity & CERT_TRUST_CTL_IS_NOT_TIME_VALID ) ) { DebugTrace((LPARAM)this, "Server cert/CTL is not time-valid or time-nested"); dwMsgID = SSL_MSG_TIME_INVALID_SERVER_CERT; } if ( dwCertValidity & CERT_TRUST_IS_REVOKED ) { DebugTrace((LPARAM)this, "Server Cert is revoked"); dwMsgID = SSL_MSG_REVOKED_SERVER_CERT; } if ( ( dwCertValidity & CERT_TRUST_IS_UNTRUSTED_ROOT ) || ( dwCertValidity & CERT_TRUST_IS_PARTIAL_CHAIN ) ) { DebugTrace((LPARAM)this, "Server Cert doesn't chain up to a trusted root"); dwMsgID = SSL_MSG_UNTRUSTED_SERVER_CERT; } if ( ( dwCertValidity & CERT_TRUST_IS_NOT_SIGNATURE_VALID ) || ( dwCertValidity & CERT_TRUST_CTL_IS_NOT_SIGNATURE_VALID ) ) { DebugTrace((LPARAM)this, "Server Cert/CTL is not signature valid"); dwMsgID = SSL_MSG_SIGNATURE_INVALID_SERVER_CERT; } if ( dwMsgID ) { SmtpLogEvent( dwMsgID, 1, apszMsgs, 0 ); } } TraceFunctLeaveEx((LPARAM)this); } VOID SMTP_SERVER_INSTANCE::LogCTLStatus() /*++ Description: Writes system log event about status of server CTL if CTL isn't valid Arguments: None Returns: Nothing --*/ { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::LogCTLStatus"); _ASSERT( m_pSSLInfo ); // // If we didn't construct the CTL fully, log an error // if ( !m_pSSLInfo->QueryCTL()->IsValid() ) { CONST CHAR *apszMsgs[2]; CHAR achInstance[20]; CHAR achErrorNumber[20]; wsprintf( achInstance, "%lu", QueryInstanceId() ); wsprintf( achErrorNumber, "0x%x", GetLastError() ); apszMsgs[0] = achInstance; apszMsgs[1] = achErrorNumber; DWORD dwStatus = m_pSSLInfo->QueryCTL()->QueryStatus(); DWORD dwStringID = 0; DebugTrace((LPARAM)this, "Couldn't retrieve server CTL; status : %d\n", dwStatus); switch ( dwStatus ) { case CERT_ERR_MB: dwStringID = SSL_MSG_CTL_MB_ERROR; break; case CERT_ERR_CAPI: dwStringID = SSL_MSG_CTL_CAPI_ERROR; break; case CERT_ERR_CERT_NOT_FOUND: dwStringID = SSL_MSG_CTL_NOT_FOUND; break; default: dwStringID = SSL_MSG_CTL_INTERNAL_ERROR; break; } SmtpLogEvent( dwStringID, 2, apszMsgs, 0 ); TraceFunctLeaveEx((LPARAM)this); return; } TraceFunctLeaveEx((LPARAM)this); } BOOL SmtpMappingSupportFunction( PVOID pvInstance, PVOID pData, DWORD dwPropId) { if (dwPropId == SIMSSL_NOTIFY_MAPPER_SSLKEYS_CHANGED) { return (SetSslKeysNotify( (PFN_SF_NOTIFY) pData)); } else if (dwPropId == SIMSSL_NOTIFY_MAPPER_CERT11_CHANGED || dwPropId == SIMSSL_NOTIFY_MAPPER_CERTW_CHANGED) { return ( TRUE ); } else { return ( FALSE ); } } BOOL SetSslKeysNotify( PFN_SF_NOTIFY pFn ) /*++ Description Set the function called to notify SSL keys have changed Can be called only once Arguments: pFn - function to call to notify SSL keys change Returns: TRUE if function reference stored, FALSE otherwise --*/ { if ( g_pSslKeysNotify == NULL || pFn == NULL ) { g_pSslKeysNotify = pFn; return TRUE; } return FALSE; } CAddr * SMTP_SERVER_INSTANCE::AppendLocalDomain (CAddr * OldAddress) { char ReWriteAddr [MAX_INTERNET_NAME + 1]; CAddr * NewAddress = NULL; DWORD TotalSize = 0; //If there is no domain on this address, //then append the current domain to this //address m_GenLock.ShareLock(); TotalSize = lstrlen(OldAddress->GetAddress()) + lstrlen(GetDefaultDomain()) + 1; //+1 for the @ if ( TotalSize >= MAX_INTERNET_NAME) { m_GenLock.ShareUnlock(); SetLastError(ERROR_INVALID_DATA); return NULL; } lstrcpy(ReWriteAddr, OldAddress->GetAddress()); lstrcat(ReWriteAddr, "@"); lstrcat(ReWriteAddr, GetDefaultDomain()); m_GenLock.ShareUnlock(); //create a new CAddr NewAddress = CAddr::CreateKnownAddress (ReWriteAddr); return NewAddress; } BOOL SMTP_SERVER_INSTANCE::AppendLocalDomain(char * Address) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::AppendLocalDomain"); m_GenLock.ShareLock(); if ((lstrlen(Address) + lstrlen(GetDefaultDomain()) + 1) > MAX_INTERNET_NAME) { //Our concatanated name will be larger than allowed name ErrorTrace((LPARAM)this, "Generated address longer than allowed max"); m_GenLock.ShareUnlock(); return FALSE; } lstrcat(Address, "@"); lstrcat(Address, GetDefaultDomain()); m_GenLock.ShareUnlock(); TraceFunctLeaveEx((LPARAM)this); return TRUE; } CAddr * SMTP_SERVER_INSTANCE::MasqueradeDomain (CAddr * OldAddress) { char ReWriteAddr [MAX_INTERNET_NAME + 1]; CAddr * NewAddress = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::MasqueradeDomain"); m_GenLock.ShareLock(); if (m_fMasquerade) { //if there is a domain in the name, get rid of it //we are going to replace it with the masquerade if (OldAddress->GetDomainOffset()) { *(OldAddress->GetDomainOffset()) = '\0'; } lstrcpy(ReWriteAddr, OldAddress->GetAddress()); lstrcat(ReWriteAddr, "@"); lstrcat(ReWriteAddr, m_szMasqueradeName); //create a new CAddr NewAddress = CAddr::CreateKnownAddress (ReWriteAddr); } m_GenLock.ShareUnlock(); TraceFunctLeaveEx((LPARAM)this); return NewAddress; } BOOL SMTP_SERVER_INSTANCE::MasqueradeDomain(char * Address, char * DomainPtr) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::MasqueradeDomain"); m_GenLock.ShareLock(); if (m_fMasquerade) { //if there is a domain in the name, get rid of it //we are going to replace it with the masquerade if (DomainPtr) { if ((DomainPtr - Address + lstrlen(m_szMasqueradeName)) > MAX_INTERNET_NAME) { //Our concatanated name will be larger than allowed name ErrorTrace((LPARAM)this, "Generated address longer than allowed max"); m_GenLock.ShareUnlock(); return FALSE; } lstrcpy(DomainPtr, m_szMasqueradeName); } else { if ((lstrlen(Address) + lstrlen(m_szMasqueradeName) + 1) > MAX_INTERNET_NAME) { //Our concatanated name will be larger than allowed name ErrorTrace((LPARAM)this, "Generated address longer than allowed max"); m_GenLock.ShareUnlock(); return FALSE; } lstrcat(Address, "@"); lstrcat(Address, m_szMasqueradeName); } } m_GenLock.ShareUnlock(); TraceFunctLeaveEx((LPARAM)this); return TRUE; } extern BOOL IsIpInGlobalList(DWORD IpAddress); BOOL SMTP_SERVER_INSTANCE::CompareIpAddress(DWORD IpAddress) { BOOL fRet = TRUE; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::CompareIpAddress"); if (IpAddress != g_LoopBackAddr) { fRet = IsIpInGlobalList(IpAddress); } else { FatalTrace((LPARAM) this, "IpAddress %d is loopback - Failing connection", IpAddress); } TraceFunctLeaveEx((LPARAM)this); return fRet; } //Note : When connected port is passed in 0, it only does IP address comparison // BOOL SMTP_SERVER_INSTANCE::IsAddressMine(DWORD IpAddress, DWORD ConnectedPort) { char * Ptr = NULL; const char * StartPtr = NULL; const CHAR * ipAddressString = NULL; const CHAR * ipPortString = NULL; const CHAR * hostNameString = NULL; const CHAR * end = NULL; DWORD InetAddr = 0; DWORD CharIpSize = 0; CHAR temp[sizeof("123.123.123.123")]; INT length; LONG tempPort; BOOL BindingMatchFound = FALSE; BOOL IpSame = FALSE; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::IsAddressMine"); //grab the sharing lock m_GenLock.ShareLock(); if (m_ServerBindings.IsEmpty()) { DebugTrace((LPARAM) this, "Server bindings is emtpy - checking all addresses"); m_GenLock.ShareUnlock(); IpSame = CompareIpAddress(IpAddress); TraceFunctLeaveEx((LPARAM)this); return IpSame; } else { for (StartPtr = m_ServerBindings.First(); StartPtr != NULL; StartPtr = m_ServerBindings.Next( StartPtr )) { ipAddressString = StartPtr; ipPortString = strchr(StartPtr, ':'); if (ipPortString == NULL) { goto out; } ipPortString++; hostNameString = strchr( ipPortString, ':' ); if ( hostNameString == NULL ) { goto out; } hostNameString++; // // Validate and parse the IP address portion. // if ( *ipAddressString == ':' ) { InetAddr = INADDR_ANY; } else { length = DIFF(ipPortString - ipAddressString) - 1; if ( length >= sizeof(temp) ) { goto out; } CopyMemory( temp, ipAddressString, length); temp[length] = '\0'; InetAddr = (DWORD)inet_addr( temp ); if ( InetAddr == INADDR_NONE ) { goto out; } }//end else of if( *ipAddressString == ':' ) // // Validate and parse the port. // if ( *ipPortString == ':' ) { goto out; } length = (INT)(hostNameString - ipPortString); if ( length > sizeof(temp) ) { goto out; } CopyMemory(temp,ipPortString, length); temp[length] = '\0'; tempPort = strtol( temp, (CHAR **)&end, 0 ); if ( tempPort <= 0 || tempPort > 0xFFFF ) { goto out; } if ( *end != ':' ) { goto out; } if (InetAddr == INADDR_ANY) { IpSame = CompareIpAddress(IpAddress); if (IpSame && ((DWORD) tempPort == ConnectedPort) && (ConnectedPort > 0)) { BindingMatchFound = TRUE; goto out; } else if (IpSame && (ConnectedPort == 0)) { BindingMatchFound = TRUE; goto out; } } else if ( (IpAddress == InetAddr) && ((DWORD) tempPort == ConnectedPort) && (ConnectedPort > 0)) { FatalTrace((LPARAM) this, "IpAddress %d is one of mine - Failing connection", IpAddress); BindingMatchFound = TRUE; goto out; } else if ( (IpAddress == InetAddr) && (ConnectedPort == 0)) { BindingMatchFound = TRUE; goto out; } }//end for }//end else out: m_GenLock.ShareUnlock(); if (!BindingMatchFound) DebugTrace((LPARAM) this, "IpAddress %d is not one of mine", IpAddress); TraceFunctLeaveEx((LPARAM)this); return BindingMatchFound; } BOOL SMTP_SERVER_INSTANCE::MoveToBadMail ( IMailMsgProperties *pIMsg, BOOL fUseIMsg, char * MailFile, char * FilePath) { TCHAR BadMailDir[MAX_PATH + 1]; TCHAR MailPath [MAX_PATH + 1]; TCHAR StreamPath[MAX_PATH + 1]; char * pszSearch = NULL; BOOL f = TRUE; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::MoveToBadMail"); //concatenate the name of the stream to the file lstrcpy(MailPath, FilePath); lstrcat(MailPath, MailFile); m_GenLock.ShareLock(); f = SUCCEEDED(m_IAQ->HandleFailedMessage(pIMsg, fUseIMsg, MailPath, MESSAGE_FAILURE_BAD_PICKUP_DIR_FILE, E_FAIL)); m_GenLock.ShareUnlock(); TraceFunctLeaveEx((LPARAM)this); return f; } #define NTFS_STORE_DIRECTORY_REG_PATH _T("Software\\Microsoft\\Exchange\\StoreDriver\\Ntfs\\%u") #define NTFS_STORE_DIRECTORY_REG_NAME _T("StoreDir") #define NTFS_STORE_BACKSLASH _T("\\") #define NTFS_QUEUE_DIRECTORY_SUFFIX _T("\\Queue") #define NTFS_DROP_DIRECTORY_SUFFIX _T("\\Drop") /*++ Description: Initializes server configuration data from registry for SMTP Service. Some values are also initialized with constants. If invalid registry key or load data from registry fails, then use default values. Arguments: None Returns: TRUE if there are no errors. Limitations: No validity check is performed on the data present in registry. --*/ BOOL SMTP_SERVER_INSTANCE::InitFromRegistry(void) { LONG err = 0; DWORD dwErr = 0; DWORD SizeOfBuffer = 0; HKEY hkeyTcpipParam = NULL; BOOL fRet = FALSE; //char szHostName[MAX_PATH + 1]; // const CHAR * apszSubStrings[2]; CHAR pchAddr1[32] = ""; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::InitFromRegistry"); _itoa(QueryInstanceId(), pchAddr1, 10); // // This is added by KeithLau on 7/15/96 // This method loads the registry values that cannot be accessed // through the RPC's // if (!ReadStartupRegParams()) { ErrorTrace((LPARAM) this, "Read startup params failed."); goto Exit; } if (!StartAdvancedQueueing()) { DebugTrace((LPARAM)this, "Unable to load Advanced Queueing module\n"); goto Exit; } fInitializedAQ = TRUE; // // Metabase Structures... can change at runtime // Turn off FC_SMTP_INFO_DEFAULT_DROP_DIR since // we already read it in ReadStartupRegParams() if (!ReadRegParams(FC_SMTP_INFO_ALL & ~FC_SMTP_INFO_DEFAULT_DROP_DIR, TRUE, FALSE)) { ErrorTrace((LPARAM) this, "Read params failed."); goto Exit; } if (!InitQueues()) { err= GetLastError(); ErrorTrace((LPARAM) this, "can't init queue err = %d", err); if (err == NO_ERROR) err = ERROR_INVALID_PARAMETER; SetLastError(err); goto Exit; } fRet = TRUE; Exit: TraceFunctLeaveEx((LPARAM)this); return fRet; } /*++ Adds the new client connection to the list of client connections and increments the count of clients currently connected to server. If the count of max connections is exceeded, then the new connection is rejected. Arguments: pcc pointer to client connection to be added Returns: TRUE on success and FALSE if there is max Connections exceeded. --*/ CLIENT_CONNECTION * SMTP_SERVER_INSTANCE::CreateNewConnection( IN OUT PCLIENT_CONN_PARAMS ClientParam) { SMTP_CONNECTION * NewConnection = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::InsertNewConnection"); // We will not accept new connections until the intial queue is built if (!m_fShouldStartAcceptingConnections || IsShuttingDown()) { return (NULL); } NewConnection = SMTP_CONNECTION::CreateSmtpConnection (ClientParam, this); if (NewConnection != NULL) { BUMP_COUNTER(this, NumConnInOpen); LockConfig(); // Increment the count of connected users m_cCurrentConnections++; // Update the current maximum connections if ( m_cCurrentConnections > m_cMaxCurrentConnections) { m_cMaxCurrentConnections = m_cCurrentConnections; } //set the client unique ID NewConnection->SetClientId(m_dwNextInboundClientId); m_dwNextInboundClientId++; // // Insert into the list of connected users. // InsertTailList( &m_ConnectionsList, &NewConnection->QueryListEntry()); DebugTrace((LPARAM) this, "SMTP_SERVER_INSTANCE:InsertNewConnection succeeded"); UnLockConfig(); } TraceFunctLeaveEx((LPARAM)this); return ( NewConnection); } void SMTP_SERVER_INSTANCE::StopHint() { if ( g_pInetSvc && g_pInetSvc->QueryCurrentServiceState() == SERVICE_STOP_PENDING) { // 10/28/98 - MikeSwa //Use stop hint 30 seconds to avoid problems with commiting many //messages on shutdown. (the previous limit of 10 seconds seemed //to cause problems with the service control manager). g_pInetSvc->UpdateServiceStatus(SERVICE_STOP_PENDING, NO_ERROR, m_dwStopHint, 30000 ) ; m_dwStopHint++ ; } } /*++ Disconnects all user connections. --*/ VOID SMTP_SERVER_INSTANCE::DisconnectAllConnections( VOID) { int iteration = 0; int i = 0; int Count = 0; LONG cLastThreadCount = 0; DWORD dwLastAllocCount = 0; DWORD dwAllocCount; int AfterSleepCount = 0; DWORD dwStopHint = 2; DWORD dwTickCount; DWORD dwError = 0; PLIST_ENTRY pEntry = NULL; SMTP_CONNECTION * pConn = NULL; TraceFunctEnter("SMTP_SERVER_INSTANCE::DisconnectAllConnections( VOID)"); if (m_fShutdownCalled) { DebugTrace((LPARAM)this, "m_fShutdownCalled already -- leaving"); TraceFunctLeaveEx((LPARAM) this); } //set the global termination flag m_IsShuttingDown = TRUE; LockConfig(); TriggerStoreServerEvent(SMTP_STOREDRV_PREPSHUTDOWN_EVENT); m_fStoreDrvPrepShutDownEventCalled = TRUE; //close down all the active sockets. for ( pEntry = m_ConnectionsList.Flink; pEntry != &m_ConnectionsList; pEntry = pEntry->Flink) { //get the next connection object pConn = (SMTP_CONNECTION *) CONTAINING_RECORD( pEntry, CLIENT_CONNECTION, m_listEntry); _ASSERT( pConn != NULL); //call the disconnect routine. DisconnectClient() just closes the socket. //This will cause all pending I/Os to fail, and have the connections //bubble up to the completion routine where they will be removed from //the connection list and then distroyed pConn->DisconnectClient( ERROR_SERVER_DISABLED); } UnLockConfig(); DebugTrace((LPARAM)this, "Cancelling all outstanding SQL queries"); // // Wait for the users to die. // The connection objects should be automatically freed because the // socket has been closed. Subsequent requests will fail // and cause a blowaway of the connection objects. // looping is used to get out as early as possible when m_cCurrentConn == 0 // // // need to check Pool.GetAllocCount instead of InUseList.Empty // because alloc goes to zero during the delete operator // instead of during the destructor // // We sleep at most 120 seconds for a fixed user count. // dwTickCount = GetTickCount(); cLastThreadCount = GetProcessClientThreads(); for (i = 0; i < 180; i++) { DebugTrace((LPARAM)this, "Waiting for connections to die, i = %u", i); // dwAllocCount = SMTP_CONNECTION::Pool.GetAllocCount(); dwAllocCount = (DWORD) GetConnInAllocCount(); if (dwAllocCount == 0 && GetProcessClientThreads() == 0) { DebugTrace((LPARAM)this, "All SMTP_CONNECTIONs connections are gone!"); break; } Sleep(1000); // Update the stop hint checkpoint when we get within 1 second (1000 ms), of the timeout... if ((SERVICE_STOP_WAIT_HINT - 1000) < (GetTickCount() - dwTickCount) && g_pInetSvc && g_pInetSvc->QueryCurrentServiceState() == SERVICE_STOP_PENDING) { DebugTrace((LPARAM)this, "Updating stop hint, checkpoint = %u", dwStopHint); g_pInetSvc->UpdateServiceStatus(SERVICE_STOP_PENDING, NO_ERROR, dwStopHint, SERVICE_STOP_WAIT_HINT ) ; dwStopHint++ ; dwTickCount = GetTickCount(); } DebugTrace((LPARAM)this, "Alloc counts: current = %u, last = %u; Thread counts: current = %d, last = %d", dwAllocCount, dwLastAllocCount, GetProcessClientThreads(), cLastThreadCount); if (dwAllocCount < dwLastAllocCount || GetProcessClientThreads() < cLastThreadCount) { DebugTrace((LPARAM)this, "SMTP_CONNECTION Connections are going away, reseting i"); i = 0; } dwLastAllocCount = dwAllocCount; cLastThreadCount = GetProcessClientThreads(); } if (i == 180) { ErrorTrace((LPARAM) this, "%d users won't die", m_cCurrentConnections); // // once we're thru do it again to find any stray clients // LockConfig(); for (pEntry = m_ConnectionsList.Flink; pEntry != &m_ConnectionsList; pEntry = pEntry->Flink ) { // // get the next connection object // pConn = (SMTP_CONNECTION *)CONTAINING_RECORD(pEntry, CLIENT_CONNECTION, m_listEntry); _ASSERT(pConn != NULL); ErrorTrace( (LPARAM)pConn, "Stray client" ); } UnLockConfig(); } DebugTrace( (LPARAM)this, "SMTP_CONNECTION Count at end is: %d", SMTP_CONNECTION::Pool.GetAllocCount() ); TraceFunctLeaveEx((LPARAM) this); } /*++ Removes the current connection from the list of conenctions and decrements count of connected users Arguments: pcc pointer to client connection to be removed --*/ VOID SMTP_SERVER_INSTANCE::RemoveConnection( IN OUT CLIENT_CONNECTION * pConn) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::RemoveConnection"); _ASSERT( pConn != NULL); LockConfig(); // Remove from list of connections RemoveEntryList( &pConn->QueryListEntry()); // Decrement count of current users m_cCurrentConnections--; BUMP_COUNTER(this, NumConnInClose); UnLockConfig(); TraceFunctLeaveEx((LPARAM)this); } // SMTP_SERVER_INSTANCE::RemoveConnection() /*++ Adds the new client connection to the list of outbound client connections and increments the count of outbound clients currently connected to server. Arguments: pcc pointer to client connection to be added Returns: TRUE on success and FALSE if there is max Connections exceeded. --*/ BOOL SMTP_SERVER_INSTANCE::InsertNewOutboundConnection( IN OUT CLIENT_CONNECTION * pcc, BOOL ByPassLimitCheck) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::InsertNewOutboundConnection"); _ASSERT( pcc != NULL); m_OutLock.ExclusiveLock(); // Increment the count of connected users m_cCurrentOutConnections++; //set the client unique ID pcc->SetClientId(m_dwNextOutboundClientId); m_dwNextInboundClientId++; // Insert into the list of connected outbound users. InsertTailList( &m_OutConnectionsList, &pcc->QueryListEntry()); m_OutLock.ExclusiveUnlock(); BUMP_COUNTER(this, NumConnOutOpen); TraceFunctLeaveEx((LPARAM)this); return TRUE; } BOOL SMTP_SERVER_INSTANCE::InsertAsyncDnsObject( IN OUT CAsyncSmtpDns *pcc) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::InsertAsyncDnsObj"); _ASSERT( pcc != NULL); EnterCriticalSection( &m_csAsyncDns ) ; IncAsyncDnsObjs(); BUMP_COUNTER(this, RoutingTableLookups); // Insert into the list of connected outbound users. InsertTailList( &m_AsyncDnsList, &pcc->QueryListEntry()); LeaveCriticalSection( &m_csAsyncDns ) ; TraceFunctLeaveEx((LPARAM)this); return TRUE; } VOID SMTP_SERVER_INSTANCE::RemoveAsyncDnsObject( IN OUT CAsyncSmtpDns * pConn) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::RemoveAsyncDnsObject"); _ASSERT( pConn != NULL); EnterCriticalSection( &m_csAsyncDns ) ; // Remove from list of connections RemoveEntryList( &pConn->QueryListEntry()); DecAsyncDnsObjs(); DROP_COUNTER(this, RoutingTableLookups); LeaveCriticalSection( &m_csAsyncDns ) ; TraceFunctLeaveEx((LPARAM)this); } // SMTP_SERVER_INSTANCE::RemoveAsyncObject() VOID SMTP_SERVER_INSTANCE::DisconnectAllAsyncDnsConnections( VOID) { int iteration = 0; int i = 0; int Count = 0; DWORD dwLastAllocCount = 0; DWORD dwAllocCount; int AfterSleepCount = 0; DWORD dwStopHint = 2; DWORD dwTickCount; DWORD dwError = 0; PLIST_ENTRY pEntry = NULL; CAsyncSmtpDns * pConn = NULL; TraceFunctEnter("SMTP_SERVER_INSTANCE::DisconnectAllAsyncDnsConnections( VOID)"); if (m_fShutdownCalled) { DebugTrace((LPARAM)this, "m_fShutdownCalled already -- leaving"); TraceFunctLeaveEx((LPARAM) this); } EnterCriticalSection( &m_csAsyncDns ) ; //close down all the active sockets. for ( pEntry = m_AsyncDnsList.Flink; pEntry != &m_AsyncDnsList; pEntry = pEntry->Flink) { //get the next connection object pConn = (CAsyncSmtpDns *) CONTAINING_RECORD( pEntry, CAsyncSmtpDns, m_ListEntry); _ASSERT( pConn != NULL); //call the disconnect routine. DisconnectClient() just closes the socket. //This will cause all pending I/Os to fail, and have the connections //"bubble up to the completion routine where they will be removed from //the connection list and then distroyed pConn->DisconnectClient(); } LeaveCriticalSection( &m_csAsyncDns ) ; // // Wait for the users to die. // The connection objects should be automatically freed because the // socket has been closed. Subsequent requests will fail // and cause a blowaway of the connection objects. // looping is used to get out as early as possible when m_cCurrentConn == 0 // // // need to check Pool.GetAllocCount instead of InUseList.Empty // because alloc goes to zero during the delete operator // instead of during the destructor // // We sleep at most 120 seconds for a fixed user count. // dwTickCount = GetTickCount(); for (i = 0; i < 500; i++) { DebugTrace((LPARAM)this, "Waiting for connections to die, i = %u", i); dwAllocCount = (DWORD) GetAsyncDnsAllocCount (); if (dwAllocCount == 0) { DebugTrace((LPARAM)this, "All SMTP DNS connections are gone!"); break; } Sleep(1000); // Update the stop hint checkpoint when we get within 1 second (1000 ms), of the timeout... if ((SERVICE_STOP_WAIT_HINT - 1000) < (GetTickCount() - dwTickCount) && g_pInetSvc && g_pInetSvc->QueryCurrentServiceState() == SERVICE_STOP_PENDING) { DebugTrace((LPARAM)this, "Updating stop hint, checkpoint = %u", dwStopHint); g_pInetSvc->UpdateServiceStatus(SERVICE_STOP_PENDING, NO_ERROR, dwStopHint, SERVICE_STOP_WAIT_HINT ) ; dwStopHint++ ; dwTickCount = GetTickCount(); } DebugTrace((LPARAM)this, "Alloc counts: current = %u, last = %u", dwAllocCount, dwLastAllocCount); if (dwAllocCount < dwLastAllocCount) { DebugTrace((LPARAM)this, "SMTP DNS Connections are going away, reseting i"); i = 0; } dwLastAllocCount = dwAllocCount; } if (i == 500) { ErrorTrace((LPARAM) this, "%d users won't die", m_cCurrentConnections); // // once we're thru do it again to find any stray clients // EnterCriticalSection( &m_csAsyncDns ) ; for (pEntry = m_AsyncDnsList.Flink; pEntry != &m_AsyncDnsList; pEntry = pEntry->Flink ) { // // get the next connection object // pConn = (CAsyncSmtpDns *)CONTAINING_RECORD(pEntry, CAsyncSmtpDns, m_ListEntry); _ASSERT(pConn != NULL); ErrorTrace( (LPARAM)pConn, "Stray client" ); } LeaveCriticalSection( &m_csAsyncDns ) ; } DebugTrace( (LPARAM)this, "SMTP DNS Count at end is: %d", CAsyncSmtpDns::Pool.GetAllocCount() ); TraceFunctLeaveEx((LPARAM) this); } BOOL SMTP_SERVER_INSTANCE::InsertAsyncObject( IN OUT CAsyncMx *pcc) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::InsertNewAsyncConnection"); _ASSERT( pcc != NULL); m_OutLock.ExclusiveLock(); IncAsyncMxOutObjs(); // Insert into the list of connected outbound users. InsertTailList( &m_AsynConnectList, &pcc->QueryListEntry()); m_OutLock.ExclusiveUnlock(); TraceFunctLeaveEx((LPARAM)this); return TRUE; } VOID SMTP_SERVER_INSTANCE::RemoveAsyncObject( IN OUT CAsyncMx * pConn) { TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::RemoveAsyncObject"); _ASSERT( pConn != NULL); m_OutLock.ExclusiveLock(); // Remove from list of connections RemoveEntryList( &pConn->QueryListEntry()); DecAsyncMxOutObjs(); m_OutLock.ExclusiveUnlock(); TraceFunctLeaveEx((LPARAM)this); } // SMTP_SERVER_INSTANCE::RemoveAsyncObject() VOID SMTP_SERVER_INSTANCE::DisconnectAllAsyncConnections( VOID) { PLIST_ENTRY pEntry = NULL; CAsyncMx * pConn = NULL; int iteration = 0; int i = 0; int Count = 0; LONG cLastThreadCount = 0; DWORD dwLastAllocCount = 0; DWORD dwAllocCount; int AfterSleepCount = 0; DWORD dwStopHint = 2; DWORD dwTickCount; DWORD dwError = 0; TraceFunctEnter("SMTP_SERVER_INSTANCE::DisconnectAllAsynConnections( VOID)"); #if 0 //close down all the active sockets. for ( pEntry = m_AsynConnectList.Flink; pEntry != &m_AsynConnectList; pEntry = pEntry->Flink) { //get the next connection object pConn = (CAsyncMx *) CONTAINING_RECORD( pEntry, CAsyncMx, m_ListEntry); _ASSERT( pConn != NULL); pConn->SignalObject( ); } #endif // // Wait for the users to die. // The connection objects should be automatically freed because the // socket has been closed. Subsequent requests will fail // and cause a blowaway of the connection objects. // looping is used to get out as early as possible when m_cCurrentConn == 0 // // // need to check Pool.GetAllocCount instead of InUseList.Empty // because alloc goes to zero during the delete operator // instead of during the destructor // // We sleep at most 120 seconds for a fixed user count. // dwTickCount = GetTickCount(); for (i = 0; i < 500; i++) { DebugTrace((LPARAM)this, "Waiting for async connections to die, i = %u", i); dwAllocCount = (DWORD) GetAsyncMxOutAllocCount(); if (dwAllocCount == 0) { DebugTrace((LPARAM)this, "All CASYNCMXs connections are gone!"); break; } Sleep(1000); // Update the stop hint checkpoint when we get within 1 second (1000 ms), of the timeout... if ((SERVICE_STOP_WAIT_HINT - 1000) < (GetTickCount() - dwTickCount) && g_pInetSvc && g_pInetSvc->QueryCurrentServiceState() == SERVICE_STOP_PENDING) { DebugTrace((LPARAM)this, "Updating stop hint, checkpoint = %u", dwStopHint); g_pInetSvc->UpdateServiceStatus(SERVICE_STOP_PENDING, NO_ERROR, dwStopHint, SERVICE_STOP_WAIT_HINT ) ; dwStopHint++ ; dwTickCount = GetTickCount(); } DebugTrace((LPARAM)this, "Alloc counts: current = %u, last = %u; Thread counts: current = %d, last = %d", dwAllocCount, dwLastAllocCount, GetProcessClientThreads(), cLastThreadCount); if (dwAllocCount < dwLastAllocCount) { DebugTrace((LPARAM)this, "CASYNCMX Connections are going away, resetting i"); i = 0; } dwLastAllocCount = dwAllocCount; } if (i == 500) { ErrorTrace((LPARAM) this, "%d users won't die", m_cNumAsyncObjsAlloced); // // once we're thru do it again to find any stray clients // m_OutLock.ExclusiveLock(); for (pEntry = m_ConnectionsList.Flink; pEntry != &m_ConnectionsList; pEntry = pEntry->Flink ) { // // get the next connection object // pConn = (CAsyncMx *)CONTAINING_RECORD(pEntry, CAsyncMx, m_ListEntry); _ASSERT(pConn != NULL); ErrorTrace( (LPARAM)pConn, "Stray client" ); } m_OutLock.ExclusiveUnlock(); } DebugTrace( (LPARAM)this, "CASYNCMX Count at end is: %d", m_cNumAsyncObjsAlloced); TraceFunctLeaveEx((LPARAM) this); } /*++ Disconnects all user connections. --*/ VOID SMTP_SERVER_INSTANCE::DisconnectAllOutboundConnections( VOID) { int iteration = 0; int i = 0; int Count = 0; DWORD dwLastAllocCount = 0; DWORD dwAllocCount; int AfterSleepCount = 0; DWORD dwStopHint = 2; DWORD dwTickCount; PLIST_ENTRY pEntry = NULL; CLIENT_CONNECTION * pConn = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::DisconnectAllOutboundConnections( VOID)"); if (m_fShutdownCalled) { DebugTrace((LPARAM)this, "m_fShutdownCalled already -- leaving"); TraceFunctLeaveEx((LPARAM) this); } m_OutLock.ExclusiveLock(); //close down all the active sockets. for ( pEntry = m_OutConnectionsList.Flink; pEntry != &m_OutConnectionsList; pEntry = pEntry->Flink) { //get the next connection object pConn = CONTAINING_RECORD( pEntry, CLIENT_CONNECTION, m_listEntry); _ASSERT( pConn != NULL); //call the disconnect routine. DisconnectClient() just closes the socket. //This will cause all pending I/Os to fail, and have the connections //"bubble up to the completion routine where they will be removed from //the connection list and then distroyed pConn->DisconnectClient( ERROR_SERVER_DISABLED); } m_OutLock.ExclusiveUnlock(); // // Wait for the users to die. // The connection objects should be automatically freed because the // socket has been closed. Subsequent requests will fail // and cause a blowaway of the connection objects. // looping is used to get out as early as possible when m_cCurrentConn == 0 // // // Wait for the users to die. // The connection objects should be automatically freed because the // socket has been closed. Subsequent requests will fail // and cause a blowaway of the connection objects. // looping is used to get out as early as possible when m_cCurrentConn == 0 // // // need to check Pool.GetAllocCount instead of InUseList.Empty // because alloc goes to zero during the delete operator // instead of during the destructor // // We sleep at most 180 seconds for a fixed user count. // dwTickCount = GetTickCount(); for (i = 0; i < 180; i++) { DebugTrace((LPARAM)this, "Waiting for connections to die, i = %u", i); dwAllocCount = (DWORD) GetConnOutAllocCount(); if (dwAllocCount == 0) { DebugTrace((LPARAM)this, "All SMTP_CONNOUTs connections are gone!"); break; } Sleep(1000); // Update the stop hint checkpoint when we get within 1 second (1000 ms), of the timeout... if ((SERVICE_STOP_WAIT_HINT - 1000) < (GetTickCount() - dwTickCount) && g_pInetSvc && g_pInetSvc->QueryCurrentServiceState() == SERVICE_STOP_PENDING) { DebugTrace((LPARAM)this, "Updating stop hint, checkpoint = %u", dwStopHint); g_pInetSvc->UpdateServiceStatus(SERVICE_STOP_PENDING, NO_ERROR, dwStopHint, SERVICE_STOP_WAIT_HINT ) ; dwStopHint++ ; dwTickCount = GetTickCount(); } DebugTrace((LPARAM)this, "Alloc counts: current = %u, last = %u", dwAllocCount, dwLastAllocCount); if (dwAllocCount < dwLastAllocCount) { DebugTrace((LPARAM)this, "SMTP_CONNECTION Connections are going away, resetting i"); i = 0; } dwLastAllocCount = dwAllocCount; } DebugTrace( (LPARAM)this, "SMTP_CONNOUT Count at end is: %d", SMTP_CONNECTION::Pool.GetAllocCount() ); TraceFunctLeaveEx((LPARAM)this); } /*++ Removes the current connection from the list of conenctions and decrements count of connected users Arguments: pcc pointer to client connection to be removed --*/ VOID SMTP_SERVER_INSTANCE::RemoveOutboundConnection( IN OUT CLIENT_CONNECTION * pConn) { _ASSERT( pConn != NULL); m_OutLock.ExclusiveLock(); // Remove from list of connections RemoveEntryList( &pConn->QueryListEntry()); // Decrement count of current users m_cCurrentOutConnections--; BUMP_COUNTER(this, NumConnOutClose); m_OutLock.ExclusiveUnlock(); } // SMTP_SERVER_INSTANCE::RemoveConnection() BOOL SMTP_SERVER_INSTANCE::InitDirectoryNotification(void) { char MailPickUp[MAX_PATH + 1]; TraceFunctEnterEx((LPARAM)this, "InitDirectoryNotification"); lstrcpy(MailPickUp, GetMailPickupDir()); //get rid of the \\ at the end of the name if (MailPickUp [GetMailPickupDirLength() - 1] == '\\') MailPickUp [GetMailPickupDirLength() - 1] = '\0'; //create an outbound connectiong_SmtpConfig->GetMailPickupDir() SmtpDir = SMTP_DIRNOT::CreateSmtpDirNotification(MailPickUp, SMTP_DIRNOT::ReadDirectoryCompletion, this); if (SmtpDir == NULL) { TraceFunctLeaveEx((LPARAM)this); return FALSE; } //now that both queues are initialized, create the //thread that goes off and finds any mail files //that were not sent and builds up both queues DWORD ThreadId; DWORD error; DirPickupThreadHandle = CreateThread (NULL, 0, SMTP_DIRNOT::PickupInitialFiles, SmtpDir, 0, &ThreadId); if (DirPickupThreadHandle == NULL) { error = GetLastError(); ErrorTrace((LPARAM)this, "CreateThread failed for SMTP_DIRNOT::PickupInitialFiles. err: %u", error); } TraceFunctLeaveEx((LPARAM)this); return TRUE; } void SMTP_SERVER_INSTANCE::DestroyDirectoryNotification(void) { TraceFunctEnterEx((LPARAM)this, "DestroyDirectoryNotification(void)"); //wait for the initial pickup thread //to die if (DirPickupThreadHandle != NULL) { if (SmtpDir) SmtpDir->SetPickupRetryQueueEvent(); WaitForSingleObject(DirPickupThreadHandle, INFINITE); ErrorTrace((LPARAM)this, "Initial pickup thread is dead"); CloseHandle(DirPickupThreadHandle); DirPickupThreadHandle = NULL; } //Just close the handle to the //directory. This will cause //the notification to come back //with an error, and it will be //deleted. if (SmtpDir) { SmtpDir->CloseDirHandle(); } if (SmtpDir) { delete SmtpDir; SmtpDir = NULL; CloseHandle(StopHandle); StopHandle = NULL; } TraceFunctLeaveEx((LPARAM)this); } extern char g_UserName[]; extern char g_DomainName[]; extern char g_Password[]; void SMTP_SERVER_INSTANCE::SinkSmtpServerStartHintFunc(void) { if (g_pInetSvc) { ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); } } void SMTP_SERVER_INSTANCE::SinkSmtpServerStopHintFunc(void) { StopHint(); } void SmtpServerStartHintFunc(PVOID ThisPtr) { ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); } void SmtpServerStopHintFunc(PVOID ThisPtr) { ((SMTP_SERVER_INSTANCE *) ThisPtr)->StopHint(); } BOOL SMTP_SERVER_INSTANCE::StartAdvancedQueueing(void) { TraceFunctEnterEx((LPARAM) this, "LoadAdvancedQueueing"); HRESULT hr = S_OK; ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); hr = (g_pfnInitializeAQ)(m_ComSmtpServer, QueryInstanceId(), g_UserName, g_DomainName, g_Password, SmtpServerStartHintFunc, (PVOID) this, &m_IAQ, &m_ICM, &m_pIAdvQueueConfig, &m_pvAQInstanceContext); ErrorTrace((LPARAM)this, "Advanced Queuing returned status code %x", hr); TraceFunctLeaveEx((LPARAM) this); return !FAILED(hr); } BOOL SMTP_SERVER_INSTANCE::StopQDrivers(void) { TraceFunctEnterEx((LPARAM) this, "StopQDrivers"); if (m_pIAdvQueueConfig) { m_pIAdvQueueConfig->Release(); m_pIAdvQueueConfig = NULL; } if (m_IAQ != NULL) { m_IAQ->Release(); m_IAQ = NULL; } if (m_ICM != NULL) { m_ICM->Release(); m_ICM = NULL; } if (fInitializedAQ && g_pfnDeinitializeAQ && m_pvAQInstanceContext) { (g_pfnDeinitializeAQ)(m_pvAQInstanceContext, SmtpServerStopHintFunc, (PVOID) this); m_pvAQInstanceContext = NULL; fInitializedAQ = FALSE; } TraceFunctLeaveEx((LPARAM) this); return TRUE; } extern void VerifyFQDNWithGlobalIp(DWORD InstanceId,char * szFQDomainName); void SMTP_SERVER_INSTANCE::VerifyFQDNWithBindings(void) { CONST CHAR *apszMsgs[2]; CHAR achInstance[20]; CHAR achIPAddr[20]; char * FQDNValue = NULL; char * IpAddress = NULL; in_addr UNALIGNED * P_Addr = NULL; PHOSTENT pH = NULL; char * Ptr = NULL; const char * StartPtr = NULL; const CHAR * ipAddressString = NULL; const CHAR * ipPortString = NULL; const CHAR * end = NULL; DWORD InetAddr = 0; DWORD CharIpSize = 0; CHAR temp[sizeof("123.123.123.123")]; INT length; BOOL fGlobalListChecked = FALSE; //Get the current instnace id wsprintf( achInstance, "%lu", QueryInstanceId() ); apszMsgs[1] = achInstance; //grab the sharing lock if (m_ServerBindings.IsEmpty()) { if (!fGlobalListChecked) { VerifyFQDNWithGlobalIp(QueryInstanceId(), m_szFQDomainName); fGlobalListChecked = TRUE; } } else { for (StartPtr = m_ServerBindings.First(); StartPtr != NULL; StartPtr = m_ServerBindings.Next( StartPtr )) { ipAddressString = StartPtr; ipPortString = strchr(StartPtr, ':'); if (ipPortString == NULL) { wsprintf( achIPAddr,"%s","0.0.0.0"); apszMsgs[0] = achIPAddr; SmtpLogEvent( SMTP_EVENT_UNRESOLVED_FQDN,2,apszMsgs,0 ); } // Validate and parse the IP address portion. // if ( *ipAddressString == ':' ) { if (!fGlobalListChecked) { VerifyFQDNWithGlobalIp(QueryInstanceId(),m_szFQDomainName); fGlobalListChecked = TRUE; } } else { length = DIFF(ipPortString - ipAddressString) - 1; if ( length >= sizeof(temp) ) { wsprintf( achIPAddr,"%s", temp); apszMsgs[0] = achIPAddr; SmtpLogEvent( SMTP_EVENT_UNRESOLVED_FQDN,1,apszMsgs,0 ); } else { CopyMemory( temp, ipAddressString, length); temp[length] = '\0'; InetAddr = inet_addr( temp ); if ( InetAddr != INADDR_NONE ) { //For IP address find the host name ((PSMTP_IIS_SERVICE) g_pInetSvc)->StartHintFunction(); pH = gethostbyaddr((char*)(&InetAddr), 4, PF_INET ); if (pH && pH->h_name) { if (_strcmpi(pH->h_name,m_szFQDomainName)) { wsprintf( achIPAddr,"%s",temp); apszMsgs[0] = achIPAddr; SmtpLogEvent( SMTP_EVENT_UNRESOLVED_FQDN,2,apszMsgs,0 ); } } else { wsprintf( achIPAddr,"%s",temp); apszMsgs[0] = achIPAddr; SmtpLogEvent( SMTP_EVENT_UNRESOLVED_FQDN,2,apszMsgs,0 ); } } else { wsprintf( achIPAddr,"%s", temp); apszMsgs[0] = achIPAddr; SmtpLogEvent( SMTP_EVENT_UNRESOLVED_FQDN,2,apszMsgs,0 ); } } }//end else of if( *ipAddressString == ':' ) }//end for }//end else return; } BOOL SMTP_SERVER_INSTANCE::RegisterServicePrincipalNames(BOOL fLock) { if (fLock) m_GenLock.ExclusiveLock(); if (!m_fHaveRegisteredPrincipalNames) { PSMTP_IIS_SERVICE pService = (PSMTP_IIS_SERVICE) g_pInetSvc; if (pService->ResetServicePrincipalNames()) { m_fHaveRegisteredPrincipalNames = CSecurityCtx::RegisterServicePrincipalNames( SMTP_SERVICE_NAME, m_szFQDomainName); } } if (fLock) m_GenLock.ExclusiveUnlock(); return( m_fHaveRegisteredPrincipalNames ); } //+------------------------------------------------------------ // // Function: SMTP_SERVER_INSTANCE::HrSetWellKnownIServerProps // // Synopsis: Take info from member variables and set them in the // IServer property bag // // Arguments: NONE // // Returns: // S_OK: Success // error from CMailMsgLoggingPropertyBag // // History: // jstamerj 1998/11/17 16:41:08: Created. // //------------------------------------------------------------- HRESULT SMTP_SERVER_INSTANCE::HrSetWellKnownIServerProps() { HRESULT hr; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::HrSetWellKnownIServerProps"); _ASSERT(m_pSmtpInfo); hr = m_InstancePropertyBag.PutDWORD( PE_ISERVID_DW_INSTANCE, m_pSmtpInfo->dwInstanceId); if(FAILED(hr)) goto CLEANUP; hr = m_InstancePropertyBag.PutStringA( PE_ISERVID_SZ_DEFAULTDOMAIN, m_szDefaultDomain); if(FAILED(hr)) goto CLEANUP; CLEANUP: DebugTrace((LPARAM)this, "returning hr %08lx", hr); TraceFunctLeaveEx((LPARAM)this); return hr; } //---[ SMTP_SERVER_INSTANCE::IsDropDirQuotaExceeded ]-------------------------- // // // Description: // Checks to see if we are past our drop dir quota (if enforced). The // quota is defined to be 11 times the max messages size (or 22 MB if // there is no max message size). If we are within 1 max message size // of the quota (2MB is no max message size), we will assume that this // message will push us over the quota. // Parameters: // - // Returns: // TRUE We are past our quota (or this message will do it) // FALSE We are still under drop dir quota. // History: // 10/28/1999 - MikeSwa Created // //----------------------------------------------------------------------------- BOOL SMTP_SERVER_INSTANCE::IsDropDirQuotaExceeded() { TraceFunctEnterEx((LPARAM) this, "SMTP_SERVER_INSTANCE::IsDropDirQuotaExceeded"); DWORD cbMaxMsgSize = GetMaxMsgSize(); BOOL fQuotaExceeded = FALSE; HANDLE hDropDirFind = INVALID_HANDLE_VALUE; LARGE_INTEGER LIntDropDirSize; LARGE_INTEGER LIntDropQuota; LARGE_INTEGER LIntCurrentFile; WIN32_FIND_DATA FileInfo; CHAR szDropDirSearch[sizeof(m_szMailDropDir) + sizeof("*")]; ZeroMemory(&FileInfo, sizeof(FileInfo)); LIntDropDirSize.QuadPart = 0; LIntDropQuota.QuadPart = 0; if (!IsDropDirQuotaCheckingEnabled() || !GetMailDropDir(szDropDirSearch)) { fQuotaExceeded = FALSE; goto Exit; } if (!cbMaxMsgSize) cbMaxMsgSize = 2*1024*1024; //Default to 2 MB //Set drop dir quota to be 10 times the max message size, if we //exceed this, then we are within 1 max message size of the "true" quota LIntDropQuota.QuadPart = 10*cbMaxMsgSize; //Build up a search so we can loop over the file names lstrcat(szDropDirSearch, "*"); hDropDirFind = FindFirstFileEx(szDropDirSearch, FindExInfoStandard, &FileInfo, FindExSearchNameMatch, NULL, 0); if (INVALID_HANDLE_VALUE == hDropDirFind) { ErrorTrace((LPARAM) this, "Unable to open drop dir for quota checking - 0x%X", GetLastError()); //If we cannot open the directory assume we are over quota fQuotaExceeded = TRUE; goto Exit; } //Loop over all the files we have found do { LIntCurrentFile.LowPart = FileInfo.nFileSizeLow; LIntCurrentFile.HighPart = FileInfo.nFileSizeHigh; LIntDropDirSize.QuadPart += LIntCurrentFile.QuadPart; if (LIntDropQuota.QuadPart <= LIntDropDirSize.QuadPart) { fQuotaExceeded = TRUE; goto Exit; } } while (FindNextFile(hDropDirFind, &FileInfo)); if (LIntDropQuota.QuadPart <= LIntDropDirSize.QuadPart) fQuotaExceeded = TRUE; else fQuotaExceeded = FALSE; Exit: if (INVALID_HANDLE_VALUE != hDropDirFind) FindClose(hDropDirFind); TraceFunctLeave(); return fQuotaExceeded; } //----------------------------------------------------------------------------- // Description: // Given a GUID identifying a DNS-list returned by a DNS sink, this // function searches the list of GUID/DNS-lists till a matching GUID is // found, and then returns the DNS-list associated with the GUID. // Arguments: // IN GUID *pGuidNetwork - Identifies a "network" and the set of DNS // servers to resolve names on that network. // Returns: // CTcpRegIpList* corresponding to the DNS-list for the GUID passed in. // NULL if there is no matching GUID. // Notes: // See the documentation for UpdateDnsServerInfo for a general overview // of this feature. //----------------------------------------------------------------------------- CTcpRegIpList *SMTP_SERVER_INSTANCE::GetDnsServerInfo(GUID *pGuidNetwork) { INT iRet = 0; ULONG i = 0; CTcpRegIpList *pTcpRegIpList = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::GetDnsServerInfo"); m_slServerList.ShareLock(); // // Loop through lists of servers, find a GUID match and return a pointer to // the CTcpRegIpList struct for the list. // for(i = 0; i < m_cServerList; i++) { iRet = memcmp( &(m_rgpServerList[i]->m_guid), pGuidNetwork, sizeof(GUID)); if(0 == iRet) { DebugTrace((LPARAM)this, "Matched GUID #%d", i); pTcpRegIpList = &(m_rgpServerList[i]->m_TcpRegIpList); break; } } m_slServerList.ShareUnlock(); TraceFunctLeaveEx((LPARAM)this); return pTcpRegIpList; } //----------------------------------------------------------------------------- // Description: // Returns the DNS list associated with a GUID. If the GUID is not already // in the table of GUID/lists stored in the VSI object, it is entered into // the table and the DNS list is returned. // Arguments: // DNS_SERVER_INFO pDnsServerInfo - New GUID/list to add, unless the GUID // is already present in the table in which case the new list is used // to update the list in the table. // Returns: // CTcpRegIpList* to the list associated with the GUID // NULL if a failure occurred (typically out of memory) // Overview: // SMTP fires the DNS resolver event each time it tries to resolve a // remote-domain. Through this event, the DNS-sink can control (among // other things), the set of DNS servers that SMTP should use to resolve // the remote-domain. // // This functionality is used to provide the "DMZ" feature in SMTP. If // a mail server is in the DMZ between an internal network and the // Internet, and it's function is to relay mail between the internal // network and the Internet: then remote-hosts on the Internet must // be resolved using a different set of DNS servers than remote-hosts // on the intranet. This is because they have different name spaces. // // In such a situation, the DNS sink determines whether the target // host should be resolved using an alternate set of DNS servers, // and returns the list of DNS servers to be used each time the // DNS event if fired. // // Now, we keep track of DNS server state across different outbound // connections, so we would like to keep the DNS list around and // keep tabulating the connection failure/success statistics, and // only update the DNS list when there is a change in the servers // being used. // // Since this feature may also be used by a DNS sink to route mail // through multiple (> 2) networks, there may be multiple distinct // lists of DNS servers in use at any time. It is therefore important // to distinguish between when the list of DNS servers has changed, // and when a totally different list (for a different) network is // being returned. // // For this purpose, each network/list is identified by a GUID. Each // time the DNS sink event returns a list of DNS servers and the GUID // for the list, this function checks to see if that GUID/list has // already been added to our set of GUID/lists. If not, the new // GUID/list is added. If it already exists, we check to see if any // of the DNS servers have changed, and update them if needed. // // Note: // The number of DNS lists are assumed to be quite small. Also, DNS // lists can only be added. Once added there is no way to remove a // DNS list (other than shutting down the VSI). The idea is that each // DNS list corresponds to a "network" and there are only 2-3 different // networks. Connections to different networks cause different DNS // lists to be used. //----------------------------------------------------------------------------- CTcpRegIpList *SMTP_SERVER_INSTANCE::UpdateDnsServerInfo(DNS_SERVER_INFO *pDnsServerInfo) { BOOL fRet = FALSE; INT iRet = 0; BOOL fGUIDFound = FALSE; DNS_SERVER_LIST *pServerList = NULL; CTcpRegIpList *pTcpRegIpList = NULL; TraceFunctEnterEx((LPARAM)this, "SMTP_SERVER_INSTANCE::UpdateDnsServerInfo"); pTcpRegIpList = GetDnsServerInfo(&(pDnsServerInfo->guidNetwork)); // Found a match: update the server-list if(pTcpRegIpList) { fRet = pTcpRegIpList->UpdateIfChanged((PIP_ARRAY)(pDnsServerInfo->pipServers)); TraceFunctLeaveEx((LPARAM)this); return pTcpRegIpList; } // // This DNS server list does not exist. Add it to the list. // if(m_cServerList >= MAX_DNS_SERVER_LISTS) { ErrorTrace((LPARAM)this, "Exceeded maximum DNS list limit"); TraceFunctLeaveEx((LPARAM)this); return NULL; } pServerList = new DNS_SERVER_LIST(); if(!pServerList) { pTcpRegIpList = NULL; fRet = FALSE; goto Cleanup; } fRet = (pServerList->m_TcpRegIpList).Update((PIP_ARRAY)pDnsServerInfo->pipServers); if(!fRet) { ErrorTrace((LPARAM)this, "Update failed"); pTcpRegIpList = NULL; fRet = FALSE; goto Cleanup; } CopyMemory(&(pServerList->m_guid), &(pDnsServerInfo->guidNetwork), sizeof(GUID)); m_slServerList.ExclusiveLock(); DebugTrace((LPARAM)this, "Added GUID %d to the list", m_cServerList); m_rgpServerList[m_cServerList] = pServerList; m_cServerList++; m_slServerList.ExclusiveUnlock(); pTcpRegIpList = &(pServerList->m_TcpRegIpList); fRet = TRUE; Cleanup: if(!fRet && pServerList) delete pServerList; TraceFunctLeaveEx((LPARAM)this); return pTcpRegIpList; }