diff options
author | Rozhuk Ivan <rozhuk.im@gmail.com> | 2014-11-29 19:52:48 +0000 |
---|---|---|
committer | Rozhuk Ivan <rozhuk.im@gmail.com> | 2014-11-29 19:52:48 +0000 |
commit | 2b003dac85649ee02d06fa11b8d0e8fe523c916d (patch) | |
tree | d145ebf993df846e18d038fab22a8c119b11c453 /plugins/Exchange/src/MirandaExchange.cpp | |
parent | 46f7dbfa99111c48e3e5b01ff61394e6d0d29742 (diff) |
Exchange: code refactoring. Review requred: suspect lost res free on errors.
git-svn-id: http://svn.miranda-ng.org/main/trunk@11159 1316c22d-e87f-b044-9b9b-93d7a3e3ba9c
Diffstat (limited to 'plugins/Exchange/src/MirandaExchange.cpp')
-rw-r--r-- | plugins/Exchange/src/MirandaExchange.cpp | 1896 |
1 files changed, 924 insertions, 972 deletions
diff --git a/plugins/Exchange/src/MirandaExchange.cpp b/plugins/Exchange/src/MirandaExchange.cpp index 6589754671..0c0828383a 100644 --- a/plugins/Exchange/src/MirandaExchange.cpp +++ b/plugins/Exchange/src/MirandaExchange.cpp @@ -1,973 +1,925 @@ -/*
-Exchange notifier plugin for Miranda IM
-
-Copyright © 2006 Cristian Libotean, Attila Vajda
-
-This program is free software; you can redistribute it and/or
-modify it under the terms of the GNU General Public License
-as published by the Free Software Foundation; either version 2
-of the License, or (at your option) any later version.
-
-This program is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with this program; if not, write to the Free Software
-Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
-*/
-
-/********************************************************************
- created: 2006/04/11
- created: 11:4:2006 17:28
- filename: MirandaExchange.cpp
- file base: MirandaExchange
- file ext: cpp
- author: Attila Vajda
-
- purpose: Miranda Exchange Plugin
-*********************************************************************/
-
-#include "stdafx.h"
-#include "MirandaExchange.h"
-#include "commonheaders.h"
-#include "utils.h"
-
-#include <time.h>
-
-#ifndef NO_EXCHANGE_TEST
-
-//////////////////////////////////////////////////////////////////////////
-HRESULT HrMAPIFindDefaultMsgStore( // RETURNS: return code
- IN LPMAPISESSION lplhSession, // session pointer
- OUT ULONG *lpcbeid, // count of bytes in entry ID
- OUT LPENTRYID *lppeid) // entry ID of default store
-{
- HRESULT hr = NOERROR;
- HRESULT hrT = NOERROR;
- SCODE sc = 0;
- LPMAPITABLE lpTable = NULL;
- LPSRowSet lpRows = NULL;
- LPENTRYID lpeid = NULL;
- ULONG cbeid = 0;
- ULONG cRows = 0;
- ULONG i = 0;
-
- SizedSPropTagArray(2, rgPropTagArray)={2,{PR_DEFAULT_STORE,PR_ENTRYID}};
-
- // Get the list of available message stores from MAPI
- hrT = MAPICALL(lplhSession)->GetMsgStoresTable( 0, &lpTable);
-
- if (!FAILED(hrT))
- {
- // Get the row count for the message recipient table
- hrT = MAPICALL(lpTable)->GetRowCount(0, &cRows);
- if (!FAILED(hrT))
- {
- // Set the columns to return
- hrT = MAPICALL(lpTable)->SetColumns((LPSPropTagArray)&rgPropTagArray, 0);
- if (!FAILED(hrT))
- {
- // Go to the beginning of the recipient table for the envelope
- hrT = MAPICALL(lpTable)->SeekRow( BOOKMARK_BEGINNING, 0, NULL);
- if (!FAILED(hrT))
- {
- // Read all the rows of the table
- hrT = MAPICALL(lpTable)->QueryRows( cRows, 0, &lpRows);
- if (SUCCEEDED(hrT) && (lpRows != NULL) && (lpRows->cRows == 0))
- {
- FreeProws(lpRows);
- lpRows = NULL;
- hrT = MAPI_E_NOT_FOUND;
- }
- }
- }
- }
- }
-
- if ( !FAILED(hrT) )
- {
- bool bGetOut = false;
- for(i = 0; (i < cRows) && (!bGetOut); i++)
- {
- if (lpRows->aRow[i].lpProps[0].Value.b == TRUE)
- {
- cbeid = lpRows->aRow[i].lpProps[1].Value.bin.cb;
-
- sc = MAPIAllocateBuffer(cbeid, (void **)&lpeid);
-
- if (FAILED(sc))
- {
- cbeid = 0;
- lpeid = NULL;
- bGetOut = true;
- }
- else
- {
- // Copy entry ID of message store
- CopyMemory(lpeid,lpRows->aRow[i].lpProps[1].Value.bin.lpb,cbeid);
- bGetOut = true;
- }
- }
- }
- }
-
- if (lpRows != NULL)
- {
- FreeProws(lpRows);
- }
-
- UlRelease(lpTable);
- *lpcbeid = cbeid;
- *lppeid = lpeid;
- return hr;
-}
-
-
-CKeeper::CKeeper( LPTSTR szSender, LPTSTR szSubject, LPSTR szEntryID)
-{
- m_szSender = NULL ;
- m_szSubject = NULL ;
- m_szEntryID = NULL ;
- m_nSizeSender = 0 ;
- m_nSizeSubject = 0 ;
- m_nSizeEntryID = 0 ;
-
- if (NULL != szSender) {
- m_nSizeSender = (UINT)_tcslen(szSender)+1;
- m_szSender = new TCHAR[ m_nSizeSender ];
- memset(m_szSender, 0, m_nSizeSender * sizeof(TCHAR));
- _tcscpy(m_szSender, szSender);
- }
-
- if (NULL != szSubject) {
- m_nSizeSubject = (UINT)_tcslen(szSubject) +1;
- m_szSubject = new TCHAR[m_nSizeSubject];
- memset(m_szSubject, 0, m_nSizeSubject * sizeof(TCHAR));
- _tcscpy(m_szSubject, szSubject);
- }
-
- if (NULL != szEntryID) {
- m_nSizeEntryID = (UINT)strlen( szEntryID ) +1;
- m_szEntryID = new char[m_nSizeEntryID];
- memset(m_szEntryID, 0, m_nSizeEntryID * sizeof(char));
- strcpy(m_szEntryID, szEntryID );
- }
-}
-
-CKeeper::~CKeeper()
-{
- if ( m_nSizeSender>0 && NULL != m_szSender )
- {
- m_nSizeSender =0;
- delete[] m_szSender;
- m_szSender = NULL;
- }
-
- if ( m_nSizeSubject>0 && NULL != m_szSubject )
- {
- m_nSizeSender =0;
- delete[] m_szSubject;
- m_szSubject = NULL;
- }
-
- if ( m_nSizeEntryID>0 && NULL != m_szEntryID )
- {
- m_nSizeEntryID = 0;
- delete[] m_szEntryID;
- m_szEntryID = NULL;
- }
-}
-
-CMirandaExchange::CMirandaExchange()
-{
- UINT nSize = 0;
- short nSizeOfTCHAR = sizeof( TCHAR );
-
- m_szUsername = NULL ;
- m_szPassword = NULL ;
- m_szExchangeServer = NULL ;
- m_lpMAPISession = NULL ;
- m_lpInbox = NULL ;
- m_lpMDB = NULL;
- m_bLoginOK = false ;
- m_bFolderInboxOK = false ;
- m_nNumberOfHeaders = 0 ;
-}
-
-CMirandaExchange::~CMirandaExchange()
-{
- if ( NULL != m_szUsername )
- {
- delete[] m_szUsername;
- m_szUsername = NULL;
- }
-
- if ( NULL != m_szPassword )
- {
- delete[] m_szPassword;
- m_szPassword = NULL;
- }
-
- if ( NULL != m_szExchangeServer )
- {
- delete[] m_szExchangeServer;
- m_szExchangeServer = NULL;
- }
-
- if ( NULL != m_lpInbox )
- {
- UlRelease(m_lpInbox);
- m_lpInbox = NULL;
- }
-
- if ( NULL != m_lpMDB )
- {
- UlRelease(m_lpMDB );
- m_lpMDB = NULL;
- }
-
- if ( NULL!= m_lpMAPISession )
- {
- m_lpMAPISession->Logoff(NULL,NULL,NULL);
- UlRelease(m_lpMAPISession );
- m_lpMAPISession = NULL;
- }
-
- if ( m_nNumberOfHeaders>0 && NULL != m_HeadersKeeper )
- {
- for( UINT i=0; i<m_nNumberOfHeaders; i++ )
- {
- if ( NULL != m_HeadersKeeper[i])
- {
- delete m_HeadersKeeper[i];
- m_HeadersKeeper[i] = NULL;
- }
- }
-
- m_nNumberOfHeaders =0 ;
- }
-
- //MAPIUninitialize();
-}
-
-
-HRESULT CallOpenEntry( LPMDB lpMDB, LPADRBOOK lpAB, LPMAPICONTAINER lpContainer, LPMAPISESSION lpMAPISession,
- ULONG cbEntryID, LPENTRYID lpEntryID, ULONG ulFlags, ULONG* ulObjTypeRet, LPUNKNOWN* lppUnk)
-{
- if (!lppUnk) return MAPI_E_INVALID_PARAMETER;
- HRESULT hRes = S_OK;
- ULONG ulObjType = NULL;
- LPUNKNOWN lpUnk = NULL;
- ULONG ulNoCacheFlags = NULL;
-
- *lppUnk = NULL;
-
- //ulFlags |= MAPI_NO_CACHE;
- //in case we need to retry without MAPI_NO_CACHE - do not add MAPI_NO_CACHE to ulFlags after this point
- //if (MAPI_NO_CACHE & ulFlags) ulNoCacheFlags = ulFlags & ~MAPI_NO_CACHE;
- ulNoCacheFlags = ulFlags;
-
- if (lpMDB)
- {
- //Log(_T("CallOpenEntry: Calling OpenEntry on MDB with ulFlags = 0x%X\n"),ulFlags);
- lpMDB->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulFlags,
- &ulObjType,
- &lpUnk);
- if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags)
- {
- hRes = S_OK;
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- (lpMDB->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulNoCacheFlags,
- &ulObjType,
- &lpUnk));
- }
- if (FAILED(hRes))
- {
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- }
- }
- if (lpAB && !lpUnk)
- {
- hRes = S_OK;
- //Log(_T("CallOpenEntry: Calling OpenEntry on AB with ulFlags = 0x%X\n"),ulFlags);
- (lpAB->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulFlags,
- &ulObjType,
- &lpUnk));
- if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags)
- {
- hRes = S_OK;
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- (lpAB->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulNoCacheFlags,
- &ulObjType,
- &lpUnk));
- }
- if (FAILED(hRes))
- {
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- }
- }
-
- if (lpContainer && !lpUnk)
- {
- hRes = S_OK;
- //Log(_T("CallOpenEntry: Calling OpenEntry on Container with ulFlags = 0x%X\n"),ulFlags);
- (lpContainer->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulFlags,
- &ulObjType,
- &lpUnk));
- if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags)
- {
- hRes = S_OK;
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- (lpContainer->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulNoCacheFlags,
- &ulObjType,
- &lpUnk));
- }
- if (FAILED(hRes))
- {
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- }
- }
-
- if (lpMAPISession && !lpUnk)
- {
- hRes = S_OK;
- //Log(_T("CallOpenEntry: Calling OpenEntry on Session with ulFlags = 0x%X\n"),ulFlags);
- (lpMAPISession->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulFlags,
- &ulObjType,
- &lpUnk));
- if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags)
- {
- hRes = S_OK;
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- (lpMAPISession->OpenEntry(
- cbEntryID,
- lpEntryID,
- NULL,//no interface
- ulNoCacheFlags,
- &ulObjType,
- &lpUnk));
- }
- if (FAILED(hRes))
- {
- if (lpUnk) (lpUnk)->Release();
- lpUnk = NULL;
- }
- }
-
- if (lpUnk)
- {
- //Log(_T("OnOpenEntryID: Got object (0x%08X) of type 0x%08X = %s\n"),lpUnk,ulObjType,ObjectTypeToString(ulObjType));
- *lppUnk = lpUnk;
- }
- if (ulObjTypeRet) *ulObjTypeRet = ulObjType;
- return hRes;
-}
-
-HRESULT CallOpenEntry( LPMDB lpMDB, LPADRBOOK lpAB, LPMAPICONTAINER lpContainer, LPMAPISESSION lpMAPISession,
- LPSBinary lpSBinary, ULONG ulFlags, ULONG* ulObjTypeRet, LPUNKNOWN* lppUnk)
-{
- return CallOpenEntry( lpMDB, lpAB, lpContainer, lpMAPISession, lpSBinary?lpSBinary->cb:0,
- (LPENTRYID)(lpSBinary?lpSBinary->lpb:0), ulFlags, ulObjTypeRet, lppUnk);
-}
-
-HRESULT CMirandaExchange::InitializeAndLogin( LPCTSTR szUsername, LPCTSTR szPassword, LPCTSTR szExchangeServer )
-{
- _popupUtil("Connecting to Exchange ...");
- UINT nSize = 0;
- short nSizeOfTCHAR = sizeof( TCHAR );
-
- if (m_szUsername == NULL && NULL != szUsername) {
- nSize = (UINT)_tcslen(szUsername);
- if (nSize > 0) {
- nSize++;
- m_szUsername = new TCHAR[nSize];
- memset ( m_szUsername, 0, nSize * nSizeOfTCHAR );
- _tcscpy( m_szUsername, szUsername );
- }
- }
-
- if (m_szPassword == NULL && NULL != szPassword) {
- nSize = (UINT)_tcslen(szPassword);
- if (nSize > 0) {
- nSize++;
- m_szPassword = new TCHAR[nSize];
- memset(m_szPassword, 0, nSize * nSizeOfTCHAR);
- _tcscpy(m_szPassword, szPassword);
- }
- }
-
- if (m_szExchangeServer == NULL && NULL != szExchangeServer) {
- nSize = (UINT)_tcslen(szExchangeServer);
- if (nSize > 0) {
- nSize++;
- m_szExchangeServer = new TCHAR[nSize];
- memset(m_szExchangeServer, 0, nSize * nSizeOfTCHAR);
- _tcscpy(m_szExchangeServer, szExchangeServer);
- }
- }
-
- if (!m_bLoginOK || m_lpInbox || NULL == m_lpMAPISession) {
- HRESULT hr = S_OK;
- LPMDB lpMDB = NULL;
- MAPIINIT_0 mapiInit = { MAPI_INIT_VERSION , MAPI_MULTITHREAD_NOTIFICATIONS };
-
- if ( !m_bNoInitAgain) {
- m_bNoInitAgain = true;
- hr = MAPIInitialize( &mapiInit) ;
- }
-
- if ( SUCCEEDED(hr)) {
- TCHAR szPIDandName[128];
- TCHAR szPID[20];
-
- _tstrtime(szPID);
- _tcscpy(szPIDandName, m_szUsername);
- _tcscat(szPIDandName, szPID);
-
- hr = CreateProfile(szPIDandName);
- if ( HR_FAILED(hr)) {
- //Log("Create profile failed: 0x%08X", hr);
- return hr;
- }
-
- DWORD dwFlags = MAPI_EXPLICIT_PROFILE|MAPI_EXTENDED|MAPI_NEW_SESSION|MAPI_NO_MAIL ;
-
- hr = MAPILogonEx( 0, (LPTSTR)mir_t2a(szPIDandName), (LPTSTR)mir_t2a(m_szPassword), dwFlags, &m_lpMAPISession );
-
- if (FAILED(hr)) {
- //Log( _T("MAPI Logon failed: 0x%08X"), hr );
- return hr;
- }
-
- LPPROFADMIN pProfAdmin = NULL;
- hr = MAPIAdminProfiles( 0, &pProfAdmin );
-
- if ((FAILED(hr)) || (NULL == pProfAdmin))
- {
- //Log("Admin profile interface creation failed: 0x%08X", hr);
- }
- else {
- hr = pProfAdmin->DeleteProfile( (LPTSTR)mir_t2a(szPIDandName), 0 );
- if ( FAILED(hr) )
- {
- //Log( "Failed to delete the profile: 0x%08X", hr );
- }
- }
-
- if (pProfAdmin)
- pProfAdmin->Release();
-
- ULONG cbDefStoreEid = 0;
-
- CMAPIBuffer< LPENTRYID> pDefStoreEid = NULL;
- hr = HrMAPIFindDefaultMsgStore(m_lpMAPISession, &cbDefStoreEid, &pDefStoreEid );
- if (FAILED(hr))
- return hr;
-
- // Open default message store
- LPMDB pDefMsgStore = NULL;
- hr = m_lpMAPISession->OpenMsgStore(0, cbDefStoreEid, pDefStoreEid, NULL,
- MAPI_BEST_ACCESS, &pDefMsgStore);
-
- HRESULT hRes = S_OK;
- ULONG cbInboxEID = NULL;
- CMAPIBuffer< LPENTRYID> lpInboxEID = NULL;
-
- if (NULL == pDefMsgStore )
- return hr;
-
- hRes = pDefMsgStore->GetReceiveFolder( _T("IPM"), NULL, &cbInboxEID, &lpInboxEID, NULL);
- m_lpMDB = pDefMsgStore;
- if (cbInboxEID && lpInboxEID) {
- hRes = CallOpenEntry( pDefMsgStore, NULL, NULL, NULL, cbInboxEID, lpInboxEID, MAPI_BEST_ACCESS, NULL, (LPUNKNOWN*)&m_lpInbox);
-
- if ( m_lpInbox && hRes == S_OK)
- m_bFolderInboxOK = true;
- }
- }
- }
-
- return S_OK;
-}
-
-HRESULT CMirandaExchange::CreateProfile( LPTSTR szProfileName )
-{
- HRESULT hr = S_OK;
- CMAPIInterface<LPPROFADMIN> pProfAdmin = NULL;
- CMAPIInterface<LPSERVICEADMIN> pMsgSvcAdmin = NULL;
- CMAPIInterface<LPMAPITABLE> pMsgSvcTable = NULL;
- LPSRowSet pRows = NULL;
- ULONG ulFlags=0;
- enum {iSvcName, iSvcUID, cptaSvc};
-
- SizedSPropTagArray(cptaSvc, sptCols) =
- {
- cptaSvc,
- PR_SERVICE_NAME,
- PR_SERVICE_UID
- };
- ulFlags &= ~MAPI_UNICODE;
- hr = MAPIAdminProfiles(ulFlags, &pProfAdmin);
-
- if (!(FAILED(hr)) || (pProfAdmin))
- {
- hr = pProfAdmin->CreateProfile((LPTSTR)mir_t2a(szProfileName), NULL, NULL, ulFlags);
-
- if (!FAILED(hr))
- {
- hr = pProfAdmin->AdminServices( (LPTSTR)mir_t2a(szProfileName), NULL, NULL, ulFlags, &pMsgSvcAdmin);
-
- if ( !(FAILED(hr)) || (pMsgSvcAdmin) )
- {
- hr = pMsgSvcAdmin->CreateMsgService((LPTSTR)("MSEMS"), (LPTSTR)("")/*"Microsoft Exchange Server"*/, NULL, 0);
-
- if (!FAILED(hr))
- {
- hr = pMsgSvcAdmin->GetMsgServiceTable(0, &pMsgSvcTable);
- if ( !(FAILED(hr)) || ( pMsgSvcTable) )
- {
- SRestriction sres;
- sres.rt = RES_CONTENT;
- sres.res.resContent.ulFuzzyLevel = FL_FULLSTRING;
- sres.res.resContent.ulPropTag = PR_SERVICE_NAME_A;
- SPropValue spv;
- sres.res.resContent.lpProp = &spv;
- spv.ulPropTag = PR_SERVICE_NAME_A;
- spv.Value.lpszA = (LPSTR)"MSEMS";
-
- hr = HrQueryAllRows(pMsgSvcTable,
- (LPSPropTagArray) &sptCols,
- &sres,
- NULL,
- 0,
- &pRows);
-
- if (!FAILED(hr)) {
- UINT nSize = (UINT)_tcslen(m_szUsername)+2;
-
- TCHAR* szUniqName = new TCHAR[nSize];
-
- memset( szUniqName, 0, nSize*sizeof(TCHAR) );
- _tcscpy( szUniqName,_T("="));
- _tcscat( szUniqName, m_szUsername);
-
- // Set values for PR_PROFILE_UNRESOLVED_NAME and PR_PROFILE_UNRESOLVED_SERVER
- SPropValue spval[2];
- spval[0].ulPropTag = PR_PROFILE_UNRESOLVED_NAME;
- spval[0].Value.lpszA = mir_t2a(szUniqName);
- spval[1].ulPropTag = PR_PROFILE_UNRESOLVED_SERVER;
- spval[1].Value.lpszA = mir_t2a(m_szExchangeServer);
-
-
- // Configure msg service
- /*hr =*/ pMsgSvcAdmin->ConfigureMsgService(
- (LPMAPIUID) pRows->aRow->lpProps[iSvcUID].Value.bin.lpb,
- 0, NULL, 2, spval);
-
- delete[] szUniqName;
- }
- }
- }
- }
- }
- else
- {
- pProfAdmin->DeleteProfile((LPTSTR)mir_t2a(szProfileName), ulFlags);
- }
- }
-
- if (pRows)
- {
- FreeProws(pRows);
- }
-
- return hr;
-}
-
-HRESULT CMirandaExchange::isMapiSessionOK( LPMAPISESSION lpSession )
-{
- return S_OK;
-}
-
-HRESULT CMirandaExchange::CheckForNewMails( int &nNewMails)
-{
- HRESULT hRes;
- if ( m_nNumberOfHeaders>0 && NULL != m_HeadersKeeper )
- {
- for( UINT i=0; i<m_nNumberOfHeaders; i++ )
- {
- if ( NULL != m_HeadersKeeper[i])
- {
- delete m_HeadersKeeper[i];
- m_HeadersKeeper[i] = NULL;
- }
- }
-
- m_nNumberOfHeaders =0 ;
- }
-
- m_nNumberOfHeaders = 0;
-
- try
- {
- if ( m_lpMAPISession != NULL && (isMapiSessionOK(m_lpMAPISession)== S_OK) && m_lpInbox != NULL && m_bFolderInboxOK )
- {
- hRes= CheckInFolder( m_lpInbox );
- }
- else
- {
- m_bLoginOK = 0;
- hRes = InitializeAndLogin(NULL,NULL,NULL);
-
- if (hRes == S_OK)
- {
- hRes = CheckInFolder( m_lpInbox );
- }
- }
-
- if (hRes == S_OK)
- {
- nNewMails = m_nNumberOfHeaders;
- }
- }
- catch (...)
- {
-
- }
-
- return hRes;
-}
-
-HRESULT CMirandaExchange::LogOFF()
-{
- try
- {
- if (NULL != m_lpInbox)
- {
- UlRelease(m_lpInbox);
- m_lpInbox = NULL;
- }
-
- if (NULL != m_lpMDB)
- {
- UlRelease(m_lpMDB);
- m_lpMDB = NULL;
- }
-
- if ( NULL!= m_lpMAPISession )
- {
- m_lpMAPISession->Logoff( NULL, NULL, NULL );
- m_lpMAPISession->Release();
- m_lpMAPISession = NULL;
- }
-
- if ( m_nNumberOfHeaders>0 && NULL != m_HeadersKeeper )
- {
- for( UINT i=0; i<m_nNumberOfHeaders; i++ )
- {
- if ( NULL != m_HeadersKeeper[i])
- {
- delete m_HeadersKeeper[i];
- m_HeadersKeeper[i] = NULL;
- }
- }
-
- m_nNumberOfHeaders =0 ;
- }
- }
- catch (...)
- {
-
- }
-
- return S_OK;
-}
-
-HRESULT CMirandaExchange::MarkAsRead( LPTSTR szEntryID )
-{
- LPMESSAGE lpMessage = NULL ;
-
- SizedSPropTagArray(3,sptaFlags) =
- {
- 3,
- { PR_ENTRYID ,
- PR_MESSAGE_FLAGS,
- PR_SENDER_NAME
- }
- };
-
- LPBYTE lpData = NULL ;
- ULONG ulC = 0 ;
-
- HexToBin(szEntryID, ulC, lpData);
-
- CallOpenEntry( m_lpMDB, NULL, NULL, m_lpMAPISession, ulC, (LPENTRYID) lpData, MAPI_BEST_ACCESS, NULL, (LPUNKNOWN*)&lpMessage);
- delete lpData;
-
- if ( NULL != lpMessage)
- {
- lpMessage->SetReadFlag( 0 );
- lpMessage->SaveChanges(FORCE_SAVE);
-
- lpMessage->Release();
- lpMessage = NULL;
- }
-
- return 0;
-}
-
-
-HRESULT CMirandaExchange::CheckInFolder( LPMAPIFOLDER lpFolder )
-{
- HRESULT hr = NOERROR;
-
- if ( lpFolder != NULL && m_bFolderInboxOK )
- {
-
- CMAPIInterface<LPMAPITABLE> lpTable = NULL ;
- LPSRowSet lpRow = NULL ;
- LPSPropValue lpRowProp = NULL ;
- ULONG i = 0L ;
- ULONG *lpcbeid = 0 ;
-
- SizedSPropTagArray(5,sptaDETAILS) =
- {
- 5,
- {
- PR_ENTRYID,
- PR_MESSAGE_FLAGS,
- PR_SENDER_NAME,
- PR_ORIGINAL_SENDER_EMAIL_ADDRESS,
- PR_SUBJECT
- }
- };
-
- CMAPIInterface<LPMAPITABLE> lpMessageTable;
-
- hr = lpFolder->GetContentsTable( 0, &lpMessageTable );
- if ( HR_FAILED( hr ) )
- {
- return -1;
- }
-
- LPSRowSet lpRowsR = NULL;
-
- //////////////////////////////////////////////////////////////////////////
- SRestriction srRoot;
- srRoot.rt = RES_BITMASK;
- srRoot.res.resBitMask.relBMR = BMR_EQZ;
- srRoot.res.resBitMask.ulPropTag = PR_MESSAGE_FLAGS;
- srRoot.res.resBitMask.ulMask = MSGFLAG_READ;
-
- SizedSSortOrderSet( 1, sso ) = { 1, 0, 0, { PR_MESSAGE_DELIVERY_TIME, TABLE_SORT_DESCEND } };
-
- hr = HrQueryAllRows( lpMessageTable, (LPSPropTagArray) &sptaDETAILS,&srRoot,(LPSSortOrderSet) & sso,0L, &lpRowsR );
-
- //////////////////////////////////////////////////////////////////////////
-
- if ( HR_FAILED( hr ) )
- {
- return -1;
- }
- else
- {
- TCHAR* szSenderName = NULL;
- TCHAR* szSubject = NULL;
- LPSTR szEntryID = NULL;
-
- for( i = 0; ( i < lpRowsR->cRows) && ( m_nNumberOfHeaders < MAX_NUMBER_OF_HEADERS ); ++i )
- {
- if ( !(lpRowsR->aRow[ i ].lpProps[ 1 ].Value.l & MSGFLAG_READ) )
- {
-
- if ( !FAILED(lpRowsR->aRow[i].lpProps[2].Value.err) )
- {
- szSenderName = lpRowsR->aRow[i].lpProps[2].Value.lpszW;
- }
-
- if ( NULL == szSenderName)
- {
- if ( !FAILED(lpRowsR->aRow[i].lpProps[3].Value.err))
- {
- szSenderName = lpRowsR->aRow[i].lpProps[3].Value.lpszW;
- }
- }
-
-
- if ( !FAILED(lpRowsR->aRow[i].lpProps[4].Value.err) )
- {
- szSubject = lpRowsR->aRow[i].lpProps[4].Value.lpszW;
- }
-
- szEntryID = BinToHex( lpRowsR->aRow[i].lpProps[0].Value.bin.cb, lpRowsR->aRow[i].lpProps[0].Value.bin.lpb );
- m_HeadersKeeper[m_nNumberOfHeaders] = new CKeeper( szSenderName, szSubject, szEntryID );
- m_nNumberOfHeaders++;
-
- delete[] szEntryID;
-
- szEntryID = NULL;
- szSubject = NULL;
- szSenderName = NULL;
- }
- }
-
- if (lpRowsR)
- {
- FreeProws(lpRowsR);
- }
- }
-
- if (m_nNumberOfHeaders<MAX_NUMBER_OF_HEADERS)
- {
- const enum {IDISPNAME, IENTRYID, ICHILDCOUNT};
-
- static SizedSPropTagArray ( 3, rgColProps) =
- {
- 3,
- {
- PR_DISPLAY_NAME_A,
- PR_ENTRYID,
- PR_FOLDER_CHILD_COUNT
- }
- };
-
- ULONG ulObjType = 0L;
-
- hr = MAPICALL( lpFolder)->GetHierarchyTable( MAPI_DEFERRED_ERRORS, &lpTable);
- if (!FAILED(hr))
- {
- hr = HrQueryAllRows( lpTable, (LPSPropTagArray)&rgColProps, NULL, NULL, 0L, &lpRow);
- if (!FAILED(hr))
- {
- for(i = 0; i < lpRow->cRows; i++)
- {
- lpRowProp = lpRow->aRow[i].lpProps;
- CMAPIInterface<LPMAPIFOLDER> lpSubFolder = NULL;
- hr = CallOpenEntry( m_lpMDB, NULL, NULL, NULL, lpRowProp[IENTRYID].Value.bin.cb, (LPENTRYID)lpRowProp[IENTRYID].Value.bin.lpb, MAPI_BEST_ACCESS, NULL, (LPUNKNOWN*)&lpSubFolder );
- if ( !FAILED(hr) )
- {
- hr = CheckInFolder( lpSubFolder );
- //if (FAILED(hr) ){//Log("failed checkinfolder for %s\n",lpRowProp[IDISPNAME].Value.lpszA );}
- }
- }
- }
-
- if (NULL != lpRow)
- {
- FreeProws(lpRow);
- lpRow = NULL;
- }
- }
- }
- }
-
- return hr;
-}
-
-HRESULT CMirandaExchange::OpenTheMessage( LPTSTR szEntryID )
-{
- //(Default)//// HKEY_CLASSES_ROOT\mailto\shell\open\command
- HKEY hTheKey;
- HRESULT hRes = E_FAIL;
-
- TCHAR szRegValue[ 512 ];
- DWORD dwLength = 512 ;
- DWORD dwType = REG_SZ;
-
- if ( RegOpenKeyEx(HKEY_CLASSES_ROOT,
- _T("mailto\\shell\\open\\command"),
- 0,
- KEY_ALL_ACCESS | KEY_EXECUTE | KEY_QUERY_VALUE ,
- &hTheKey) == ERROR_SUCCESS
- )
- {
- LONG lResult = RegQueryValueEx( hTheKey, NULL, NULL, (LPDWORD)&dwType, (LPBYTE)szRegValue, &dwLength);
- RegCloseKey( hTheKey );
-
- if ( lResult != ERROR_SUCCESS )
- {
- hRes = E_FAIL;
- }
- else
- {
-
- TCHAR* szTheEnd = _tcsstr( szRegValue,_T(".EXE") );
-
- if ( NULL != szTheEnd )
- {
- szRegValue[ _tcslen(szRegValue) - _tcslen(szTheEnd) +5 ] = _T('\0');
- _tcscat( szRegValue, _T(" /recycle") );
- STARTUPINFO si;
- PROCESS_INFORMATION pi;
-
- DWORD dwCode = 0;
-
- ZeroMemory ( &si, sizeof ( STARTUPINFO));
-
- si.cb = sizeof ( STARTUPINFO);
- si.dwFlags = STARTF_USESHOWWINDOW;
- si.wShowWindow = SW_SHOWNORMAL;
-
- if ( CreateProcess ( NULL,
- szRegValue,
- NULL,
- NULL,
- 0,
- NORMAL_PRIORITY_CLASS,
- NULL,
- NULL,
- &si,
- &pi
- ))
- {
- hRes = S_OK;
- }
- }
- }
- }
- else
- {
- hRes = E_FAIL;
- }
-
- return hRes;
-
-}
+/* +Exchange notifier plugin for Miranda IM + +Copyright © 2006 Cristian Libotean, Attila Vajda + +This program is free software; you can redistribute it and/or +modify it under the terms of the GNU General Public License +as published by the Free Software Foundation; either version 2 +of the License, or (at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. + +You should have received a copy of the GNU General Public License +along with this program; if not, write to the Free Software +Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. +*/ + +/******************************************************************** + created: 2006/04/11 + created: 11:4:2006 17:28 + filename: MirandaExchange.cpp + file base: MirandaExchange + file ext: cpp + author: Attila Vajda + + purpose: Miranda Exchange Plugin +*********************************************************************/ + +#include "stdafx.h" +#include "MirandaExchange.h" +#include "commonheaders.h" +#include "utils.h" + +#include <time.h> + +#ifndef NO_EXCHANGE_TEST + +////////////////////////////////////////////////////////////////////////// +HRESULT HrMAPIFindDefaultMsgStore( // RETURNS: return code + IN LPMAPISESSION lplhSession, // session pointer + OUT ULONG *lpcbeid, // count of bytes in entry ID + OUT LPENTRYID *lppeid) // entry ID of default store +{ + HRESULT hr = NOERROR; + HRESULT hrT = NOERROR; + SCODE sc = 0; + LPMAPITABLE lpTable = NULL; + LPSRowSet lpRows = NULL; + LPENTRYID lpeid = NULL; + ULONG cbeid = 0; + ULONG cRows = 0; + ULONG i = 0; + + SizedSPropTagArray(2, rgPropTagArray)={2,{PR_DEFAULT_STORE,PR_ENTRYID}}; + + // Get the list of available message stores from MAPI + hrT = MAPICALL(lplhSession)->GetMsgStoresTable( 0, &lpTable); + if (FAILED(hrT)) + goto err_out; + // Get the row count for the message recipient table + hrT = MAPICALL(lpTable)->GetRowCount(0, &cRows); + if (FAILED(hrT)) + goto err_out; + // Set the columns to return + hrT = MAPICALL(lpTable)->SetColumns((LPSPropTagArray)&rgPropTagArray, 0); + if (FAILED(hrT)) + goto err_out; + // Go to the beginning of the recipient table for the envelope + hrT = MAPICALL(lpTable)->SeekRow( BOOKMARK_BEGINNING, 0, NULL); + if (FAILED(hrT)) + goto err_out; + // Read all the rows of the table + hrT = MAPICALL(lpTable)->QueryRows( cRows, 0, &lpRows); + if (SUCCEEDED(hrT) && lpRows != NULL && lpRows->cRows == 0) { + hrT = MAPI_E_NOT_FOUND; + goto err_out; + } + + if (FAILED(hrT)) + goto err_out; + for (i = 0; i < cRows; i ++) { + if (lpRows->aRow[i].lpProps[0].Value.b == FALSE) + continue; + cbeid = lpRows->aRow[i].lpProps[1].Value.bin.cb; + sc = MAPIAllocateBuffer(cbeid, (void **)&lpeid); + if (FAILED(sc)) { + cbeid = 0; + lpeid = NULL; + } else { + // Copy entry ID of message store + CopyMemory(lpeid, lpRows->aRow[i].lpProps[1].Value.bin.lpb, cbeid); + } + break; + } + +err_out: + if (lpRows != NULL) + FreeProws(lpRows); + UlRelease(lpTable); + *lpcbeid = cbeid; + *lppeid = lpeid; + + return hr; +} + + +CKeeper::CKeeper( LPTSTR szSender, LPTSTR szSubject, LPSTR szEntryID) +{ + m_szSender = NULL ; + m_szSubject = NULL ; + m_szEntryID = NULL ; + m_nSizeSender = 0 ; + m_nSizeSubject = 0 ; + m_nSizeEntryID = 0 ; + + if (NULL != szSender) { + m_nSizeSender = (UINT)_tcslen(szSender)+1; + m_szSender = new TCHAR[ m_nSizeSender ]; + memset(m_szSender, 0, m_nSizeSender * sizeof(TCHAR)); + _tcscpy(m_szSender, szSender); + } + + if (NULL != szSubject) { + m_nSizeSubject = (UINT)_tcslen(szSubject) +1; + m_szSubject = new TCHAR[m_nSizeSubject]; + memset(m_szSubject, 0, m_nSizeSubject * sizeof(TCHAR)); + _tcscpy(m_szSubject, szSubject); + } + + if (NULL != szEntryID) { + m_nSizeEntryID = (UINT)strlen( szEntryID ) +1; + m_szEntryID = new char[m_nSizeEntryID]; + memset(m_szEntryID, 0, m_nSizeEntryID * sizeof(char)); + strcpy(m_szEntryID, szEntryID ); + } +} + +CKeeper::~CKeeper() +{ + if ( m_nSizeSender>0 && NULL != m_szSender ) + { + m_nSizeSender =0; + delete[] m_szSender; + m_szSender = NULL; + } + + if ( m_nSizeSubject>0 && NULL != m_szSubject ) + { + m_nSizeSender =0; + delete[] m_szSubject; + m_szSubject = NULL; + } + + if ( m_nSizeEntryID>0 && NULL != m_szEntryID ) + { + m_nSizeEntryID = 0; + delete[] m_szEntryID; + m_szEntryID = NULL; + } +} + +CMirandaExchange::CMirandaExchange() +{ + UINT nSize = 0; + short nSizeOfTCHAR = sizeof( TCHAR ); + + m_szUsername = NULL ; + m_szPassword = NULL ; + m_szExchangeServer = NULL ; + m_lpMAPISession = NULL ; + m_lpInbox = NULL ; + m_lpMDB = NULL; + m_bLoginOK = false ; + m_bFolderInboxOK = false ; + m_nNumberOfHeaders = 0 ; +} + +CMirandaExchange::~CMirandaExchange() +{ + if ( NULL != m_szUsername ) + { + delete[] m_szUsername; + m_szUsername = NULL; + } + + if ( NULL != m_szPassword ) + { + delete[] m_szPassword; + m_szPassword = NULL; + } + + if ( NULL != m_szExchangeServer ) + { + delete[] m_szExchangeServer; + m_szExchangeServer = NULL; + } + + if ( NULL != m_lpInbox ) + { + UlRelease(m_lpInbox); + m_lpInbox = NULL; + } + + if ( NULL != m_lpMDB ) + { + UlRelease(m_lpMDB ); + m_lpMDB = NULL; + } + + if ( NULL!= m_lpMAPISession ) + { + m_lpMAPISession->Logoff(NULL,NULL,NULL); + UlRelease(m_lpMAPISession ); + m_lpMAPISession = NULL; + } + + if ( m_nNumberOfHeaders>0 && NULL != m_HeadersKeeper ) + { + for( UINT i=0; i<m_nNumberOfHeaders; i++ ) + { + if ( NULL != m_HeadersKeeper[i]) + { + delete m_HeadersKeeper[i]; + m_HeadersKeeper[i] = NULL; + } + } + + m_nNumberOfHeaders =0 ; + } + + //MAPIUninitialize(); +} + + +HRESULT CallOpenEntry( LPMDB lpMDB, LPADRBOOK lpAB, LPMAPICONTAINER lpContainer, LPMAPISESSION lpMAPISession, + ULONG cbEntryID, LPENTRYID lpEntryID, ULONG ulFlags, ULONG* ulObjTypeRet, LPUNKNOWN* lppUnk) +{ + if (!lppUnk) return MAPI_E_INVALID_PARAMETER; + HRESULT hRes = S_OK; + ULONG ulObjType = NULL; + LPUNKNOWN lpUnk = NULL; + ULONG ulNoCacheFlags = NULL; + + *lppUnk = NULL; + + //ulFlags |= MAPI_NO_CACHE; + //in case we need to retry without MAPI_NO_CACHE - do not add MAPI_NO_CACHE to ulFlags after this point + //if (MAPI_NO_CACHE & ulFlags) ulNoCacheFlags = ulFlags & ~MAPI_NO_CACHE; + ulNoCacheFlags = ulFlags; + + if (lpMDB) + { + //Log(_T("CallOpenEntry: Calling OpenEntry on MDB with ulFlags = 0x%X\n"),ulFlags); + lpMDB->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulFlags, + &ulObjType, + &lpUnk); + if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags) + { + hRes = S_OK; + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + (lpMDB->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulNoCacheFlags, + &ulObjType, + &lpUnk)); + } + if (FAILED(hRes)) + { + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + } + } + if (lpAB && !lpUnk) + { + hRes = S_OK; + //Log(_T("CallOpenEntry: Calling OpenEntry on AB with ulFlags = 0x%X\n"),ulFlags); + (lpAB->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulFlags, + &ulObjType, + &lpUnk)); + if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags) + { + hRes = S_OK; + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + (lpAB->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulNoCacheFlags, + &ulObjType, + &lpUnk)); + } + if (FAILED(hRes)) + { + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + } + } + + if (lpContainer && !lpUnk) + { + hRes = S_OK; + //Log(_T("CallOpenEntry: Calling OpenEntry on Container with ulFlags = 0x%X\n"),ulFlags); + (lpContainer->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulFlags, + &ulObjType, + &lpUnk)); + if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags) + { + hRes = S_OK; + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + (lpContainer->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulNoCacheFlags, + &ulObjType, + &lpUnk)); + } + if (FAILED(hRes)) + { + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + } + } + + if (lpMAPISession && !lpUnk) + { + hRes = S_OK; + //Log(_T("CallOpenEntry: Calling OpenEntry on Session with ulFlags = 0x%X\n"),ulFlags); + (lpMAPISession->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulFlags, + &ulObjType, + &lpUnk)); + if (MAPI_E_UNKNOWN_FLAGS == hRes && ulNoCacheFlags) + { + hRes = S_OK; + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + (lpMAPISession->OpenEntry( + cbEntryID, + lpEntryID, + NULL,//no interface + ulNoCacheFlags, + &ulObjType, + &lpUnk)); + } + if (FAILED(hRes)) + { + if (lpUnk) (lpUnk)->Release(); + lpUnk = NULL; + } + } + + if (lpUnk) + { + //Log(_T("OnOpenEntryID: Got object (0x%08X) of type 0x%08X = %s\n"),lpUnk,ulObjType,ObjectTypeToString(ulObjType)); + *lppUnk = lpUnk; + } + if (ulObjTypeRet) *ulObjTypeRet = ulObjType; + return hRes; +} + +HRESULT CallOpenEntry( LPMDB lpMDB, LPADRBOOK lpAB, LPMAPICONTAINER lpContainer, LPMAPISESSION lpMAPISession, + LPSBinary lpSBinary, ULONG ulFlags, ULONG* ulObjTypeRet, LPUNKNOWN* lppUnk) +{ + return CallOpenEntry( lpMDB, lpAB, lpContainer, lpMAPISession, lpSBinary?lpSBinary->cb:0, + (LPENTRYID)(lpSBinary?lpSBinary->lpb:0), ulFlags, ulObjTypeRet, lppUnk); +} + +HRESULT CMirandaExchange::InitializeAndLogin( LPCTSTR szUsername, LPCTSTR szPassword, LPCTSTR szExchangeServer ) +{ + _popupUtil("Connecting to Exchange ..."); + UINT nSize = 0; + short nSizeOfTCHAR = sizeof( TCHAR ); + + if (m_szUsername == NULL && NULL != szUsername) { + nSize = (UINT)_tcslen(szUsername); + if (nSize > 0) { + nSize++; + m_szUsername = new TCHAR[nSize]; + memset ( m_szUsername, 0, nSize * nSizeOfTCHAR ); + _tcscpy( m_szUsername, szUsername ); + } + } + + if (m_szPassword == NULL && NULL != szPassword) { + nSize = (UINT)_tcslen(szPassword); + if (nSize > 0) { + nSize++; + m_szPassword = new TCHAR[nSize]; + memset(m_szPassword, 0, nSize * nSizeOfTCHAR); + _tcscpy(m_szPassword, szPassword); + } + } + + if (m_szExchangeServer == NULL && NULL != szExchangeServer) { + nSize = (UINT)_tcslen(szExchangeServer); + if (nSize > 0) { + nSize++; + m_szExchangeServer = new TCHAR[nSize]; + memset(m_szExchangeServer, 0, nSize * nSizeOfTCHAR); + _tcscpy(m_szExchangeServer, szExchangeServer); + } + } + + if (!m_bLoginOK || m_lpInbox || NULL == m_lpMAPISession) { + HRESULT hr = S_OK; + LPMDB lpMDB = NULL; + MAPIINIT_0 mapiInit = { MAPI_INIT_VERSION , MAPI_MULTITHREAD_NOTIFICATIONS }; + + if ( !m_bNoInitAgain) { + m_bNoInitAgain = true; + hr = MAPIInitialize( &mapiInit) ; + } + + if ( SUCCEEDED(hr)) { + TCHAR szPIDandName[128]; + TCHAR szPID[20]; + + _tstrtime(szPID); + _tcscpy(szPIDandName, m_szUsername); + _tcscat(szPIDandName, szPID); + + hr = CreateProfile(szPIDandName); + if ( HR_FAILED(hr)) { + //Log("Create profile failed: 0x%08X", hr); + return hr; + } + + DWORD dwFlags = MAPI_EXPLICIT_PROFILE|MAPI_EXTENDED|MAPI_NEW_SESSION|MAPI_NO_MAIL ; + + hr = MAPILogonEx( 0, (LPTSTR)mir_t2a(szPIDandName), (LPTSTR)mir_t2a(m_szPassword), dwFlags, &m_lpMAPISession ); + + if (FAILED(hr)) { + //Log( _T("MAPI Logon failed: 0x%08X"), hr ); + return hr; + } + + LPPROFADMIN pProfAdmin = NULL; + hr = MAPIAdminProfiles( 0, &pProfAdmin ); + + if ((FAILED(hr)) || (NULL == pProfAdmin)) + { + //Log("Admin profile interface creation failed: 0x%08X", hr); + } + else { + hr = pProfAdmin->DeleteProfile( (LPTSTR)mir_t2a(szPIDandName), 0 ); + if ( FAILED(hr) ) + { + //Log( "Failed to delete the profile: 0x%08X", hr ); + } + } + + if (pProfAdmin) + pProfAdmin->Release(); + + ULONG cbDefStoreEid = 0; + + CMAPIBuffer< LPENTRYID> pDefStoreEid = NULL; + hr = HrMAPIFindDefaultMsgStore(m_lpMAPISession, &cbDefStoreEid, &pDefStoreEid ); + if (FAILED(hr)) + return hr; + + // Open default message store + LPMDB pDefMsgStore = NULL; + hr = m_lpMAPISession->OpenMsgStore(0, cbDefStoreEid, pDefStoreEid, NULL, + MAPI_BEST_ACCESS, &pDefMsgStore); + + HRESULT hRes = S_OK; + ULONG cbInboxEID = NULL; + CMAPIBuffer< LPENTRYID> lpInboxEID = NULL; + + if (NULL == pDefMsgStore ) + return hr; + + hRes = pDefMsgStore->GetReceiveFolder( _T("IPM"), NULL, &cbInboxEID, &lpInboxEID, NULL); + m_lpMDB = pDefMsgStore; + if (cbInboxEID && lpInboxEID) { + hRes = CallOpenEntry( pDefMsgStore, NULL, NULL, NULL, cbInboxEID, lpInboxEID, MAPI_BEST_ACCESS, NULL, (LPUNKNOWN*)&m_lpInbox); + + if ( m_lpInbox && hRes == S_OK) + m_bFolderInboxOK = true; + } + } + } + + return S_OK; +} + +HRESULT CMirandaExchange::CreateProfile( LPTSTR szProfileName ) +{ + HRESULT hr = S_OK; + CMAPIInterface<LPPROFADMIN> pProfAdmin = NULL; + CMAPIInterface<LPSERVICEADMIN> pMsgSvcAdmin = NULL; + CMAPIInterface<LPMAPITABLE> pMsgSvcTable = NULL; + LPSRowSet pRows = NULL; + ULONG ulFlags = 0; + SRestriction sres; + SIZE_T nSize; + TCHAR* szUniqName; + enum {iSvcName, iSvcUID, cptaSvc}; + + SizedSPropTagArray(cptaSvc, sptCols) = + { + cptaSvc, + PR_SERVICE_NAME, + PR_SERVICE_UID + }; + ulFlags &= ~MAPI_UNICODE; + hr = MAPIAdminProfiles(ulFlags, &pProfAdmin); + if (FAILED(hr) || pProfAdmin == NULL) + return hr; + hr = pProfAdmin->CreateProfile((LPTSTR)mir_t2a(szProfileName), NULL, NULL, ulFlags); + + if (!FAILED(hr)) { + pProfAdmin->DeleteProfile((LPTSTR)mir_t2a(szProfileName), ulFlags); + return hr; + } + hr = pProfAdmin->AdminServices( (LPTSTR)mir_t2a(szProfileName), NULL, NULL, ulFlags, &pMsgSvcAdmin); + + if (FAILED(hr) || pMsgSvcAdmin == NULL) + return hr; + hr = pMsgSvcAdmin->CreateMsgService((LPTSTR)("MSEMS"), (LPTSTR)("")/*"Microsoft Exchange Server"*/, NULL, 0); + + if (FAILED(hr)) + return hr; + hr = pMsgSvcAdmin->GetMsgServiceTable(0, &pMsgSvcTable); + if (FAILED(hr) || pMsgSvcTable == NULL) + return hr; + + sres.rt = RES_CONTENT; + sres.res.resContent.ulFuzzyLevel = FL_FULLSTRING; + sres.res.resContent.ulPropTag = PR_SERVICE_NAME_A; + SPropValue spv; + sres.res.resContent.lpProp = &spv; + spv.ulPropTag = PR_SERVICE_NAME_A; + spv.Value.lpszA = (LPSTR)"MSEMS"; + + hr = HrQueryAllRows(pMsgSvcTable, + (LPSPropTagArray) &sptCols, + &sres, + NULL, + 0, + &pRows); + + if (FAILED(hr)) + return hr; + nSize = _tcslen(m_szUsername); + szUniqName = (TCHAR*)mir_alloc(sizeof(TCHAR) * (nSize + 4)); + if (szUniqName != NULL) { + memcpy(szUniqName, _T("="), sizeof(TCHAR)); + memcpy((szUniqName + 1), m_szUsername, (sizeof(TCHAR) * (nSize + 1))); + // Set values for PR_PROFILE_UNRESOLVED_NAME and PR_PROFILE_UNRESOLVED_SERVER + SPropValue spval[2]; + spval[0].ulPropTag = PR_PROFILE_UNRESOLVED_NAME; + spval[0].Value.lpszA = mir_t2a(szUniqName); + spval[1].ulPropTag = PR_PROFILE_UNRESOLVED_SERVER; + spval[1].Value.lpszA = mir_t2a(m_szExchangeServer); + + // Configure msg service + /*hr =*/ pMsgSvcAdmin->ConfigureMsgService( + (LPMAPIUID) pRows->aRow->lpProps[iSvcUID].Value.bin.lpb, + 0, NULL, 2, spval); + + mir_free(szUniqName); + } + FreeProws(pRows); + + return hr; +} + +HRESULT CMirandaExchange::isMapiSessionOK( LPMAPISESSION lpSession ) +{ + return S_OK; +} + +HRESULT CMirandaExchange::CheckForNewMails( int &nNewMails) +{ + HRESULT hRes; + if ( m_nNumberOfHeaders>0 && NULL != m_HeadersKeeper ) + { + for( UINT i=0; i<m_nNumberOfHeaders; i++ ) + { + if ( NULL != m_HeadersKeeper[i]) + { + delete m_HeadersKeeper[i]; + m_HeadersKeeper[i] = NULL; + } + } + + m_nNumberOfHeaders =0 ; + } + + m_nNumberOfHeaders = 0; + + try + { + if ( m_lpMAPISession != NULL && (isMapiSessionOK(m_lpMAPISession)== S_OK) && m_lpInbox != NULL && m_bFolderInboxOK ) + { + hRes= CheckInFolder( m_lpInbox ); + } + else + { + m_bLoginOK = 0; + hRes = InitializeAndLogin(NULL,NULL,NULL); + + if (hRes == S_OK) + { + hRes = CheckInFolder( m_lpInbox ); + } + } + + if (hRes == S_OK) + { + nNewMails = m_nNumberOfHeaders; + } + } + catch (...) + { + + } + + return hRes; +} + +HRESULT CMirandaExchange::LogOFF() +{ + try + { + if (NULL != m_lpInbox) + { + UlRelease(m_lpInbox); + m_lpInbox = NULL; + } + + if (NULL != m_lpMDB) + { + UlRelease(m_lpMDB); + m_lpMDB = NULL; + } + + if ( NULL!= m_lpMAPISession ) + { + m_lpMAPISession->Logoff( NULL, NULL, NULL ); + m_lpMAPISession->Release(); + m_lpMAPISession = NULL; + } + + if ( m_nNumberOfHeaders>0 && NULL != m_HeadersKeeper ) + { + for( UINT i=0; i<m_nNumberOfHeaders; i++ ) + { + if ( NULL != m_HeadersKeeper[i]) + { + delete m_HeadersKeeper[i]; + m_HeadersKeeper[i] = NULL; + } + } + + m_nNumberOfHeaders =0 ; + } + } + catch (...) + { + + } + + return S_OK; +} + +HRESULT CMirandaExchange::MarkAsRead( LPTSTR szEntryID ) +{ + LPMESSAGE lpMessage = NULL ; + + SizedSPropTagArray(3,sptaFlags) = + { + 3, + { PR_ENTRYID , + PR_MESSAGE_FLAGS, + PR_SENDER_NAME + } + }; + + LPBYTE lpData = NULL ; + ULONG ulC = 0 ; + + HexToBin(szEntryID, ulC, lpData); + + CallOpenEntry( m_lpMDB, NULL, NULL, m_lpMAPISession, ulC, (LPENTRYID) lpData, MAPI_BEST_ACCESS, NULL, (LPUNKNOWN*)&lpMessage); + delete lpData; + + if ( NULL != lpMessage) + { + lpMessage->SetReadFlag( 0 ); + lpMessage->SaveChanges(FORCE_SAVE); + + lpMessage->Release(); + lpMessage = NULL; + } + + return 0; +} + + +HRESULT CMirandaExchange::CheckInFolder( LPMAPIFOLDER lpFolder ) +{ + HRESULT hr = NOERROR; + CMAPIInterface<LPMAPITABLE> lpTable = NULL; + LPSRowSet lpRow = NULL; + LPSPropValue lpRowProp = NULL; + ULONG i = 0L; + ULONG *lpcbeid = NULL; + TCHAR* szSenderName = NULL; + TCHAR* szSubject = NULL; + LPSTR szEntryID = NULL; + + if ( lpFolder == NULL || !m_bFolderInboxOK ) + return hr; + + SizedSPropTagArray(5,sptaDETAILS) = + { + 5, + { + PR_ENTRYID, + PR_MESSAGE_FLAGS, + PR_SENDER_NAME, + PR_ORIGINAL_SENDER_EMAIL_ADDRESS, + PR_SUBJECT + } + }; + + CMAPIInterface<LPMAPITABLE> lpMessageTable; + + hr = lpFolder->GetContentsTable( 0, &lpMessageTable ); + if ( HR_FAILED( hr ) ) + { + return -1; + } + + LPSRowSet lpRowsR = NULL; + + ////////////////////////////////////////////////////////////////////////// + SRestriction srRoot; + srRoot.rt = RES_BITMASK; + srRoot.res.resBitMask.relBMR = BMR_EQZ; + srRoot.res.resBitMask.ulPropTag = PR_MESSAGE_FLAGS; + srRoot.res.resBitMask.ulMask = MSGFLAG_READ; + + SizedSSortOrderSet( 1, sso ) = { 1, 0, 0, { PR_MESSAGE_DELIVERY_TIME, TABLE_SORT_DESCEND } }; + + hr = HrQueryAllRows( lpMessageTable, (LPSPropTagArray) &sptaDETAILS,&srRoot,(LPSSortOrderSet) & sso, 0L, &lpRowsR ); + + ////////////////////////////////////////////////////////////////////////// + + if (HR_FAILED(hr)) + return -1; + + for( i = 0; ( i < lpRowsR->cRows) && ( m_nNumberOfHeaders < MAX_NUMBER_OF_HEADERS ); ++i ) + { + if ( !(lpRowsR->aRow[ i ].lpProps[ 1 ].Value.l & MSGFLAG_READ) ) + { + + if ( !FAILED(lpRowsR->aRow[i].lpProps[2].Value.err) ) + { + szSenderName = lpRowsR->aRow[i].lpProps[2].Value.lpszW; + } + + if ( NULL == szSenderName) + { + if ( !FAILED(lpRowsR->aRow[i].lpProps[3].Value.err)) + { + szSenderName = lpRowsR->aRow[i].lpProps[3].Value.lpszW; + } + } + + + if ( !FAILED(lpRowsR->aRow[i].lpProps[4].Value.err) ) + { + szSubject = lpRowsR->aRow[i].lpProps[4].Value.lpszW; + } + + szEntryID = BinToHex( lpRowsR->aRow[i].lpProps[0].Value.bin.cb, lpRowsR->aRow[i].lpProps[0].Value.bin.lpb ); + m_HeadersKeeper[m_nNumberOfHeaders] = new CKeeper(szSenderName, szSubject, szEntryID ); + m_nNumberOfHeaders++; + + delete[] szEntryID; + + szEntryID = NULL; + szSubject = NULL; + szSenderName = NULL; + } + } + FreeProws(lpRowsR); + + + if (m_nNumberOfHeaders < MAX_NUMBER_OF_HEADERS) { + const enum {IDISPNAME, IENTRYID, ICHILDCOUNT}; + + static SizedSPropTagArray ( 3, rgColProps) = + { + 3, + { + PR_DISPLAY_NAME_A, + PR_ENTRYID, + PR_FOLDER_CHILD_COUNT + } + }; + + ULONG ulObjType = 0L; + + hr = MAPICALL( lpFolder)->GetHierarchyTable( MAPI_DEFERRED_ERRORS, &lpTable); + if (!FAILED(hr)) { + hr = HrQueryAllRows( lpTable, (LPSPropTagArray)&rgColProps, NULL, NULL, 0L, &lpRow); + if (!FAILED(hr)) { + for(i = 0; i < lpRow->cRows; i ++) { + lpRowProp = lpRow->aRow[i].lpProps; + CMAPIInterface<LPMAPIFOLDER> lpSubFolder = NULL; + hr = CallOpenEntry( m_lpMDB, NULL, NULL, NULL, lpRowProp[IENTRYID].Value.bin.cb, (LPENTRYID)lpRowProp[IENTRYID].Value.bin.lpb, MAPI_BEST_ACCESS, NULL, (LPUNKNOWN*)&lpSubFolder ); + if ( !FAILED(hr) ) + { + hr = CheckInFolder( lpSubFolder ); + //if (FAILED(hr) ){//Log("failed checkinfolder for %s\n",lpRowProp[IDISPNAME].Value.lpszA );} + } + } + FreeProws(lpRow); + lpRow = NULL; + } + } + } + + return hr; +} + +HRESULT CMirandaExchange::OpenTheMessage( LPTSTR szEntryID ) +{ + //(Default)//// HKEY_CLASSES_ROOT\mailto\shell\open\command + HKEY hTheKey; + HRESULT hRes = E_FAIL; + + TCHAR szRegValue[ 512 ]; + DWORD dwLength = 512 ; + DWORD dwType = REG_SZ; + + if ( RegOpenKeyEx(HKEY_CLASSES_ROOT, + _T("mailto\\shell\\open\\command"), + 0, + KEY_ALL_ACCESS | KEY_EXECUTE | KEY_QUERY_VALUE , + &hTheKey) == ERROR_SUCCESS + ) + { + LONG lResult = RegQueryValueEx( hTheKey, NULL, NULL, (LPDWORD)&dwType, (LPBYTE)szRegValue, &dwLength); + RegCloseKey( hTheKey ); + + if ( lResult != ERROR_SUCCESS ) + { + hRes = E_FAIL; + } + else + { + + TCHAR* szTheEnd = _tcsstr( szRegValue,_T(".EXE") ); + + if ( NULL != szTheEnd ) + { + szRegValue[ _tcslen(szRegValue) - _tcslen(szTheEnd) +5 ] = _T('\0'); + _tcscat( szRegValue, _T(" /recycle") ); + STARTUPINFO si; + PROCESS_INFORMATION pi; + + DWORD dwCode = 0; + + ZeroMemory ( &si, sizeof ( STARTUPINFO)); + + si.cb = sizeof ( STARTUPINFO); + si.dwFlags = STARTF_USESHOWWINDOW; + si.wShowWindow = SW_SHOWNORMAL; + + if ( CreateProcess ( NULL, + szRegValue, + NULL, + NULL, + 0, + NORMAL_PRIORITY_CLASS, + NULL, + NULL, + &si, + &pi + )) + { + hRes = S_OK; + } + } + } + } + else + { + hRes = E_FAIL; + } + + return hRes; + +} #endif //NO_EXCHANGE_TEST
\ No newline at end of file |