mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1275 lines
42 KiB
1275 lines
42 KiB
//*********************************************************************
|
|
//* Microsoft Windows **
|
|
//* Copyright(c) Microsoft Corp., 1994-1995 **
|
|
//*********************************************************************
|
|
|
|
//
|
|
// MAPICALL.C - Functions to call MAPI for internet mail profile configuration
|
|
//
|
|
//
|
|
|
|
// HISTORY:
|
|
//
|
|
// 1/25/95 jeremys Created.
|
|
// 96/03/09 markdu Added a wait cursor during loading of MAPI
|
|
//
|
|
|
|
#include "mapicall.h"
|
|
#include "obcomglb.h"
|
|
#include "resource.h"
|
|
|
|
/**********************************************************************
|
|
|
|
Terminology:
|
|
|
|
profile - a collection of settings for Exchange that determine
|
|
the services that are used and the address book and message store
|
|
service - a MAPI plug-in that talks to a mail back end
|
|
(or address book or message store)
|
|
|
|
There can be a number of profiles installed on a particular machine.
|
|
Each profile contains a set of services.
|
|
|
|
Stategy:
|
|
|
|
To configure Microsoft Exchange, we need to do the following:
|
|
|
|
1) Establish a profile to modify
|
|
- If any profiles currently exist, find the default profile.
|
|
Otherwise create a profile, which will be initially empty.
|
|
|
|
2) Populate the profile with the required services.
|
|
- The profile needs to contain the Internet Mail service, an
|
|
address book and a message store. If any of these items are
|
|
not present, we add them to the profile.
|
|
|
|
3) Configure the internet mail service for this profile.
|
|
- stick in the user's email address, email server, etc.
|
|
|
|
**********************************************************************/
|
|
|
|
// instance handle must be in per-instance data segment
|
|
#pragma data_seg(DATASEG_PERINSTANCE)
|
|
#define SMALL_BUF_LEN 48 // convenient size for small text buffers
|
|
|
|
HINSTANCE hInstMAPIDll=NULL; // handle to MAPI dll we load explicitly
|
|
DWORD dwMAPIRefCount = 0;
|
|
static const WCHAR cszNull[] = L"";
|
|
static const WCHAR sz0[] = L"0";
|
|
static const WCHAR sz1[] = L"1";
|
|
static const WCHAR szNull[] = L"";
|
|
static const WCHAR szSlash[] = L"\\";
|
|
LPMAPIINITIALIZE lpMAPIInitialize = NULL;
|
|
LPMAPIADMINPROFILES lpMAPIAdminProfiles = NULL;
|
|
LPMAPIUNINITIALIZE lpMAPIUninitialize = NULL;
|
|
LPMAPIALLOCATEBUFFER lpMAPIAllocateBuffer = NULL;
|
|
LPMAPIFREEBUFFER lpMAPIFreeBuffer = NULL;
|
|
LPHRQUERYALLROWS lpHrQueryAllRows = NULL;
|
|
// global function pointers for MAPI apis
|
|
|
|
//////////////////////////////////////////////////////
|
|
// MAPI api function names
|
|
//////////////////////////////////////////////////////
|
|
static const CHAR szMAPIInitialize[] = "MAPIInitialize";
|
|
static const CHAR szMAPIUninitialize[] = "MAPIUninitialize";
|
|
static const CHAR szMAPIAdminProfiles[] = "MAPIAdminProfiles";
|
|
static const CHAR szMAPIAllocateBuffer[] = "MAPIAllocateBuffer";
|
|
static const CHAR szMAPIFreeBuffer[] = "MAPIFreeBuffer";
|
|
static const CHAR szHrQueryAllRows[] = "HrQueryAllRows@24";
|
|
|
|
|
|
#define NUM_MAPI_PROCS 6
|
|
// API table for function addresses to fetch
|
|
APIFCN MAPIApiList[NUM_MAPI_PROCS] = {
|
|
{ (PVOID *) &lpMAPIInitialize, szMAPIInitialize},
|
|
{ (PVOID *) &lpMAPIUninitialize, szMAPIUninitialize},
|
|
{ (PVOID *) &lpMAPIAdminProfiles, szMAPIAdminProfiles},
|
|
{ (PVOID *) &lpMAPIAllocateBuffer, szMAPIAllocateBuffer},
|
|
{ (PVOID *) &lpMAPIFreeBuffer, szMAPIFreeBuffer},
|
|
{ (PVOID *) &lpHrQueryAllRows, szHrQueryAllRows}};
|
|
|
|
#pragma data_seg(DATASEG_DEFAULT)
|
|
|
|
// function prototypes
|
|
HRESULT GetProfileList(LPPROFADMIN lpProfAdmin, LPSRowSet * ppRowSet);
|
|
HRESULT GetServicesList(LPSERVICEADMIN lpServiceAdmin, LPSRowSet *ppRowSet);
|
|
HRESULT CreateProfileIfNecessary(LPPROFADMIN lpProfAdmin, WCHAR * pszSelProfileName);
|
|
HRESULT InstallRequiredServices(LPSERVICEADMIN pServiceAdmin,
|
|
LPSRowSet pServiceRowSet);
|
|
VOID FreeSRowSet(LPSRowSet prws);
|
|
BOOL ValidateProperty(LPSPropValue pval, ULONG cVal, ULONG ulPropTag);
|
|
BOOL DoesFileExist(WCHAR * pszPath, WCHAR * pszFileName);
|
|
HRESULT ConfigInternetService(MAILCONFIGINFO * pMailConfigInfo,
|
|
LPSERVICEADMIN lpServiceAdmin);
|
|
HRESULT GetServiceUID(WCHAR * pszName, LPSERVICEADMIN lpServiceAdmin,
|
|
LPMAPIUID *ppMapiUID);
|
|
BOOL MakeUniqueFilename(UINT uIDFilename, UINT uIDAltFilename,
|
|
WCHAR * pszFilename, DWORD cbFilename);
|
|
extern BOOL GetApiProcAddresses(HMODULE hModDLL, APIFCN * pApiProcList,
|
|
UINT nApiProcs);
|
|
HRESULT ConfigNewService(LPSERVICEADMIN lpServiceAdmin, LPMAPIUID lpMapiUID,
|
|
UINT uIDFilename, UINT uIDFilename1,UINT uPropValID);
|
|
|
|
// enums
|
|
enum { ivalDisplayName, ivalServiceName, ivalResourceFlags, ivalServiceDllName,
|
|
ivalServiceEntryName, ivalServiceUID, ivalServiceSupportFiles,
|
|
cvalMsgSrvMax };
|
|
|
|
BOOL gfMAPIActive = FALSE;
|
|
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: GetApiProcAddresses
|
|
|
|
SYNOPSIS: Gets proc addresses for a table of functions
|
|
|
|
EXIT: returns TRUE if successful, FALSE if unable to retrieve
|
|
any proc address in table
|
|
|
|
HISTORY:
|
|
96/02/28 markdu If the api is not found in the module passed in,
|
|
try the backup (RNAPH.DLL)
|
|
|
|
********************************************************************/
|
|
BOOL GetApiProcAddresses(HMODULE hModDLL, APIFCN * pApiProcList,UINT nApiProcs)
|
|
{
|
|
|
|
UINT nIndex;
|
|
// cycle through the API table and get proc addresses for all the APIs we
|
|
// need
|
|
for (nIndex = 0;nIndex < nApiProcs;nIndex++)
|
|
{
|
|
if (!(*pApiProcList[nIndex].ppFcnPtr = (PVOID) GetProcAddress(hModDLL,
|
|
pApiProcList[nIndex].pszName)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
/*******************************************************************
|
|
|
|
NAME: InitMAPI
|
|
|
|
SYNOPSIS: Loads the MAPI dll, gets proc addresses and initializes
|
|
MAPI
|
|
|
|
EXIT: TRUE if successful, or FALSE if fails. Displays its
|
|
own error message upon failure.
|
|
|
|
NOTES: We load MAPI explicitly because the DLL may not be installed
|
|
when we begin the wizard... we may have to install it and
|
|
then load it.
|
|
|
|
|
|
********************************************************************/
|
|
BOOL InitMAPI(HWND hWnd)
|
|
{
|
|
HRESULT hr;
|
|
|
|
// load MAPI only if not already loaded... otherwise just increment
|
|
// reference count
|
|
|
|
if (!hInstMAPIDll) {
|
|
|
|
// get the filename (MAPI32.DLL) out of resource
|
|
// load the MAPI dll
|
|
hInstMAPIDll = LoadLibrary(L"MAPI32.DLL");
|
|
if (!hInstMAPIDll) {
|
|
UINT uErr = GetLastError();
|
|
//DisplayErrorMessage(hWnd, IDS_ERRLoadMAPIDll1,uErr,ERRCLS_STANDARD,
|
|
// MB_ICONSTOP, szMAPIDll);
|
|
return FALSE;
|
|
}
|
|
|
|
// cycle through the API table and get proc addresses for all the APIs we
|
|
// need
|
|
BOOL fSuccess = GetApiProcAddresses(hInstMAPIDll, MAPIApiList,NUM_MAPI_PROCS);
|
|
|
|
if (!fSuccess) {
|
|
DeInitMAPI();
|
|
return FALSE;
|
|
}
|
|
|
|
// initialize MAPI
|
|
assert(lpMAPIInitialize);
|
|
hr = lpMAPIInitialize(NULL);
|
|
if (HR_FAILED(hr)) {
|
|
DeInitMAPI();
|
|
return FALSE;
|
|
}
|
|
|
|
gfMAPIActive = TRUE;
|
|
}
|
|
|
|
dwMAPIRefCount ++;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: DeInitMAPI
|
|
|
|
SYNOPSIS: Uninitializes MAPI and unloads MAPI dlls
|
|
|
|
********************************************************************/
|
|
VOID DeInitMAPI(VOID)
|
|
{
|
|
// decrease reference count
|
|
if (dwMAPIRefCount) {
|
|
dwMAPIRefCount--;
|
|
}
|
|
|
|
// shut down and unload MAPI if reference count hits zero
|
|
if (!dwMAPIRefCount) {
|
|
// uninitialize MAPI
|
|
if (gfMAPIActive && lpMAPIUninitialize) {
|
|
lpMAPIUninitialize();
|
|
gfMAPIActive = FALSE;
|
|
}
|
|
|
|
// free the MAPI dll
|
|
if (hInstMAPIDll) {
|
|
FreeLibrary(hInstMAPIDll);
|
|
hInstMAPIDll = NULL;
|
|
}
|
|
|
|
// set function pointers to NULL
|
|
for (UINT nIndex = 0;nIndex<NUM_MAPI_PROCS;nIndex++)
|
|
*MAPIApiList[nIndex].ppFcnPtr = NULL;
|
|
}
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: SetMailProfileInformation
|
|
|
|
SYNOPSIS: Sets up MAPI profile for internet mail and sets
|
|
user information in profile.
|
|
|
|
ENTRY: pMailConfigInfo - pointer to struct with configuration info
|
|
|
|
EXIT: returns an HRESULT
|
|
|
|
NOTES: See strategy statement above
|
|
|
|
********************************************************************/
|
|
HRESULT SetMailProfileInformation(MAILCONFIGINFO * pMailConfigInfo)
|
|
{
|
|
HRESULT hr;
|
|
LPPROFADMIN pProfAdmin=NULL; // interface to administer profiles
|
|
LPSERVICEADMIN pServiceAdmin=NULL; // interface to administer services
|
|
LPSRowSet pServiceRowSet=NULL;
|
|
WCHAR szSelProfileName[cchProfileNameMax+1]=L"";
|
|
|
|
assert(pMailConfigInfo);
|
|
|
|
// get a pointer to the interface to administer profiles
|
|
assert(lpMAPIAdminProfiles);
|
|
hr = lpMAPIAdminProfiles(0, &pProfAdmin);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"MAPIAdminProfiles returned 0x%lx", hr);
|
|
return (hr);
|
|
}
|
|
assert(pProfAdmin);
|
|
// release this interface when we leave the function
|
|
RELEASE_ME_LATER ReleaseProfAdminLater(pProfAdmin);
|
|
|
|
// get profile name from passed-in struct, if specified
|
|
if (pMailConfigInfo->pszProfileName && lstrlen(pMailConfigInfo->pszProfileName)) {
|
|
lstrcpy(szSelProfileName, pMailConfigInfo->pszProfileName);
|
|
} else {
|
|
// no profile specified, use default name
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), IDS_DEFAULT_PROFILE_NAME, szSelProfileName,MAX_CHARS_IN_BUFFER(szSelProfileName));
|
|
}
|
|
|
|
// create profile if we need to
|
|
hr = CreateProfileIfNecessary(pProfAdmin, szSelProfileName);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
|
|
// set this profile as default if appropriate
|
|
if (pMailConfigInfo->fSetProfileAsDefault) {
|
|
hr = pProfAdmin->SetDefaultProfile(szSelProfileName, 0);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
}
|
|
|
|
assert(lstrlen(szSelProfileName)); // should have profile name at this point
|
|
//DEBUGMSG(L"Modifying MAPI profile: %s", szSelProfileName);
|
|
|
|
// get a pointer to the interface to administer services for this profile
|
|
hr = pProfAdmin->AdminServices(szSelProfileName, NULL,NULL,0,
|
|
&pServiceAdmin);
|
|
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
assert(pServiceAdmin);
|
|
// release pServiceAdmin interface when done
|
|
RELEASE_ME_LATER rlServiceAdmin(pServiceAdmin);
|
|
|
|
// get a list of services for this profile
|
|
hr = GetServicesList(pServiceAdmin, &pServiceRowSet);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
assert(pServiceRowSet);
|
|
|
|
// install any services we need which aren't already present in the profile
|
|
hr = InstallRequiredServices(pServiceAdmin, pServiceRowSet);
|
|
// done with profile row set, free the table
|
|
FreeSRowSet(pServiceRowSet);
|
|
pServiceRowSet = NULL;
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
|
|
// configure the internet mail service with the passed in email name,
|
|
// server, etc.
|
|
hr = ConfigInternetService(pMailConfigInfo, pServiceAdmin);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"ConfigInternetService returned 0x%x" , hr);
|
|
return hr;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: GetProfileList
|
|
|
|
SYNOPSIS: retrieves a list of MAPI profiles
|
|
|
|
ENTRY: lpProfAdmin - pointer to profile admin interface
|
|
ppRowSet - pointer to an SRowSet pointer that is filled in
|
|
|
|
EXIT: returns an HRESULT. If successful, *ppRowSet contains
|
|
pointer to SRowSet with profile list.
|
|
|
|
NOTES: Cloned from MAPI profile control panel code.
|
|
Caller MUST call MAPIFreeBuffer to free *ppRowSet when done.
|
|
|
|
********************************************************************/
|
|
HRESULT GetProfileList(LPPROFADMIN lpProfAdmin, LPSRowSet * ppRowSet)
|
|
{
|
|
HRESULT hr;
|
|
LPMAPITABLE pMapiTable=NULL;
|
|
SPropTagArray TagArray= {3,
|
|
{PR_DISPLAY_NAME,
|
|
PR_COMMENT,
|
|
PR_DEFAULT_PROFILE}};
|
|
|
|
assert(lpProfAdmin);
|
|
assert(ppRowSet);
|
|
|
|
// call the lpProfAdmin interface to get the MAPI profile table
|
|
hr = lpProfAdmin->GetProfileTable(0, &pMapiTable);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
assert(pMapiTable);
|
|
// release this interface when we leave the function
|
|
RELEASE_ME_LATER ReleaseMapiTableLater(pMapiTable);
|
|
|
|
// set properties of table columns
|
|
hr = pMapiTable->SetColumns(&TagArray, 0);
|
|
if (!HR_FAILED(hr)) {
|
|
// get row set information from table
|
|
hr = pMapiTable->QueryRows(4000, 0,ppRowSet);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: GetServicesList
|
|
|
|
SYNOPSIS: retrieves a list of MAPI services in a profile
|
|
|
|
ENTRY: lpProfAdmin - pointer to service admin interface
|
|
ppRowSet - pointer to an SRowSet pointer that is filled in
|
|
|
|
EXIT: returns an HRESULT. If successful, *ppRowSet contains
|
|
pointer to SRowSet with service list.
|
|
|
|
NOTES: Cloned from MAPI profile control panel code.
|
|
Caller MUST call MAPIFreeBuffer to free *ppRowSet when done.
|
|
|
|
********************************************************************/
|
|
HRESULT GetServicesList(LPSERVICEADMIN lpServiceAdmin, LPSRowSet *ppRowSet)
|
|
{
|
|
HRESULT hr;
|
|
ULONG iRow;
|
|
LPMAPITABLE pMapiTable = NULL;
|
|
SCODE sc = S_OK;
|
|
static SPropTagArray taga = {7,
|
|
{ PR_DISPLAY_NAME,
|
|
PR_SERVICE_NAME,
|
|
PR_RESOURCE_FLAGS,
|
|
PR_SERVICE_DLL_NAME,
|
|
PR_SERVICE_ENTRY_NAME,
|
|
PR_SERVICE_UID,
|
|
PR_SERVICE_SUPPORT_FILES }};
|
|
|
|
*ppRowSet = NULL;
|
|
|
|
hr = lpServiceAdmin->GetMsgServiceTable(0, &pMapiTable);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
// free this interface when function exits
|
|
RELEASE_ME_LATER rlTable(pMapiTable);
|
|
|
|
|
|
hr = pMapiTable->SetColumns(&taga, 0);
|
|
if (!HR_FAILED(hr)) {
|
|
// BUGBUG get rid of 'magic number' (appears in MAPI
|
|
// ctrl panel code, need to find out what it is) jeremys 1/30/95
|
|
hr = pMapiTable->QueryRows(4000, 0,ppRowSet);
|
|
}
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
|
|
for(iRow = 0; iRow < (*ppRowSet)->cRows; iRow++)
|
|
{
|
|
// make sure properties are valid, if not then slam something in
|
|
ValidateProperty((*ppRowSet)->aRow[iRow].lpProps, 0, PR_DISPLAY_NAME);
|
|
ValidateProperty((*ppRowSet)->aRow[iRow].lpProps, 1, PR_SERVICE_NAME);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: CreateProfileIfNecessary
|
|
|
|
SYNOPSIS: Creates profile if it doesn't already exist
|
|
|
|
ENTRY: lpProfAdmin - pointer to profile admin interface
|
|
pszSelProfileName - name of profile to create
|
|
|
|
EXIT: returns an HRESULT.
|
|
|
|
********************************************************************/
|
|
HRESULT CreateProfileIfNecessary(LPPROFADMIN pProfAdmin, WCHAR * pszSelProfileName)
|
|
{
|
|
HRESULT hr = hrSuccess;
|
|
LPWSTR lpProfileName=NULL;
|
|
BOOL fDefault;
|
|
|
|
assert(pProfAdmin);
|
|
assert(pszSelProfileName);
|
|
|
|
ENUM_MAPI_PROFILE EnumMAPIProfile;
|
|
|
|
// walk through the profile names, see if we have a match
|
|
while (EnumMAPIProfile.Next(&lpProfileName, &fDefault)) {
|
|
assert(lpProfileName);
|
|
|
|
if (!lstrcmpi(lpProfileName, pszSelProfileName)) {
|
|
return hrSuccess; // found a match, nothing to do
|
|
}
|
|
}
|
|
|
|
// no match, need to create profile
|
|
//DEBUGMSG(L"Creating MAPI profile: %s", pszSelProfileName);
|
|
// call MAPI to create the profile
|
|
hr = pProfAdmin->CreateProfile(pszSelProfileName,
|
|
NULL, (ULONG) 0, (ULONG) 0);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: InstallRequiredServices
|
|
|
|
SYNOPSIS: Installs the 3 services we need (message store,
|
|
address book, internet mail) in the profile
|
|
if they're not already present. Calls functions to configure
|
|
message store and address book (they both need a filename
|
|
to use) if we're adding them.
|
|
|
|
ENTRY: lpServiceAdmin - pointer to service admin interface
|
|
pServiceRowSet - MAPI table with list of installed services
|
|
|
|
EXIT: returns an HRESULT.
|
|
|
|
NOTES: We deliberately don't configure internet mail service here--
|
|
we do that in the main routine. The reason is that we
|
|
need to configure internet mail whether it's already installed
|
|
or not, address book and message store we only need to configure
|
|
if they're brand new.
|
|
|
|
********************************************************************/
|
|
HRESULT InstallRequiredServices(LPSERVICEADMIN pServiceAdmin,
|
|
LPSRowSet pServiceRowSet)
|
|
{
|
|
ULONG iRow, iService;
|
|
WCHAR szServiceName[SMALL_BUF_LEN+1];
|
|
LPMAPIUID pMapiUID=NULL;
|
|
HRESULT hr=hrSuccess;
|
|
|
|
// table for MAPI services we need to make sure are installed in profile
|
|
MSGSERVICE MAPIServiceList[NUM_SERVICES] = {
|
|
{ FALSE, IDS_INTERNETMAIL_SERVICENAME, IDS_INTERNETMAIL_DESCRIPTION, FALSE, 0,0,0},
|
|
{ FALSE, IDS_MESSAGESTORE_SERVICENAME, IDS_MESSAGESTORE_DESCRIPTION, TRUE,
|
|
IDS_MESSAGESTORE_FILENAME, IDS_MESSAGESTORE_FILENAME1,PR_PST_PATH},
|
|
{ FALSE, IDS_ADDRESSBOOK_SERVICENAME, IDS_ADDRESSBOOK_DESCRIPTION, TRUE,
|
|
IDS_ADDRESSBOOK_FILENAME, IDS_ADDRESSBOOK_FILENAME1,PR_PAB_PATH}};
|
|
|
|
// walk through the list of services
|
|
for (iRow = 0;iRow < pServiceRowSet->cRows;iRow ++) {
|
|
//DEBUGMSG(L"Profile contains service: %s (%s)",
|
|
// pServiceRowSet->aRow[iRow].lpProps[ivalDisplayName].Value.LPSZ,
|
|
// pServiceRowSet->aRow[iRow].lpProps[ivalServiceName].Value.LPSZ);
|
|
|
|
// for each service, walk through our array of required services,
|
|
// and see if there's a match
|
|
for (iService = 0;iService < NUM_SERVICES;iService ++) {
|
|
// load the name for this service out of resource
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), MAPIServiceList[iService].uIDServiceName,
|
|
szServiceName, MAX_CHARS_IN_BUFFER(szServiceName));
|
|
|
|
// compare it to the service name in the table of
|
|
// installed services for this profile
|
|
if (!lstrcmpi(szServiceName,
|
|
pServiceRowSet->aRow[iRow].lpProps[ivalServiceName].Value.LPSZ)) {
|
|
// this is a match!
|
|
MAPIServiceList[iService].fPresent = TRUE;
|
|
break; // break the inner 'for' loop
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// install any services we need which are not already present
|
|
for (iService = 0;iService < NUM_SERVICES;iService ++) {
|
|
|
|
if (!MAPIServiceList[iService].fPresent) {
|
|
WCHAR szServiceDesc[MAX_RES_LEN+1];
|
|
MSGSERVICE * pMsgService = &MAPIServiceList[iService];
|
|
|
|
// load the service name and description
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), pMsgService->uIDServiceName,
|
|
szServiceName, MAX_CHARS_IN_BUFFER(szServiceName));
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), pMsgService->uIDServiceDescription,
|
|
szServiceDesc, MAX_CHARS_IN_BUFFER(szServiceDesc));
|
|
//DEBUGMSG(L"Adding service: %s (%s)",
|
|
// szServiceDesc, szServiceName);
|
|
|
|
// create the service
|
|
hr = pServiceAdmin->CreateMsgService(szServiceName,
|
|
szServiceDesc, 0,0);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
|
|
// call a creation-time config procedure if specified
|
|
if (pMsgService->fNeedConfig) {
|
|
|
|
// get the UID (identifier) for this service
|
|
// based on service name, APIs downstream need this
|
|
hr = GetServiceUID(szServiceName, pServiceAdmin,
|
|
&pMapiUID);
|
|
if (HR_FAILED(hr))
|
|
return hr;
|
|
assert(pMapiUID);
|
|
|
|
// call the proc to configure newly-created service
|
|
hr = ConfigNewService(pServiceAdmin, pMapiUID,
|
|
pMsgService->uIDStoreFilename, pMsgService->uIDStoreFilename1,
|
|
pMsgService->uPropID);
|
|
|
|
// free the buffer with the UID
|
|
assert(lpMAPIFreeBuffer);
|
|
lpMAPIFreeBuffer(pMapiUID);
|
|
pMapiUID = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
#define NUM_MAIL_PROPS 11
|
|
/*******************************************************************
|
|
|
|
NAME: ConfigInternetService
|
|
|
|
SYNOPSIS: Configures the Internet Mail service (route 66) with
|
|
user's email name, email server, etc.
|
|
|
|
ENTRY: pMailConfigInfo - pointer to struct with configuration info
|
|
pServiceAdmin - pointer to service admin interface
|
|
|
|
EXIT: returns an HRESULT
|
|
|
|
NOTES: will stomp any existing settings for properties that it
|
|
sets.
|
|
|
|
********************************************************************/
|
|
HRESULT ConfigInternetService(MAILCONFIGINFO * pMailConfigInfo,
|
|
LPSERVICEADMIN pServiceAdmin)
|
|
{
|
|
HRESULT hr;
|
|
SPropValue PropValue[NUM_MAIL_PROPS];
|
|
WCHAR szServiceName[SMALL_BUF_LEN+1];
|
|
LPMAPIUID pMapiUID=NULL;
|
|
UINT nProps = NUM_MAIL_PROPS;
|
|
|
|
assert(pMailConfigInfo);
|
|
assert(pServiceAdmin);
|
|
|
|
// get service UID for internet mail service
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), IDS_INTERNETMAIL_SERVICENAME, szServiceName,MAX_CHARS_IN_BUFFER(szServiceName));
|
|
hr = GetServiceUID(szServiceName, pServiceAdmin,&pMapiUID);
|
|
if (HR_FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
assert(pMapiUID);
|
|
|
|
|
|
// set the property value for each property. Note that the order
|
|
// of items in the array doesn't mattter. The ulPropTag member indicates
|
|
// what property the PropValue item is for, and the lpszW, b or l member
|
|
// contains the data for that property.
|
|
|
|
// need to "encrypt" mail account password with xor bit mask. Mail client
|
|
// expects it to be thusly "encrypted" when it reads it out. It's stored
|
|
// in the registry in this securely "encrypted" format. Somebody pretty
|
|
// smart must have thought of this.
|
|
|
|
// configure mail service properties
|
|
PropValue[0].ulPropTag = PR_CFG_EMAIL_ADDRESS;
|
|
PropValue[0].Value.lpszW = pMailConfigInfo->pszEmailAddress;
|
|
PropValue[1].ulPropTag = PR_CFG_EMAIL_DISPLAY_NAME;
|
|
PropValue[1].Value.lpszW = pMailConfigInfo->pszEmailDisplayName;
|
|
PropValue[2].ulPropTag = PR_CFG_SERVER_PATH;
|
|
PropValue[2].Value.lpszW = pMailConfigInfo->pszEmailServer;
|
|
PropValue[3].ulPropTag = PR_CFG_EMAIL_ACCOUNT;
|
|
PropValue[3].Value.lpszW = pMailConfigInfo->pszEmailAccountName;
|
|
PropValue[4].ulPropTag = PR_CFG_PASSWORD;
|
|
PropValue[4].Value.lpszW = (LPWSTR) szNull;
|
|
PropValue[5].ulPropTag = PR_CFG_REMEMBER;
|
|
PropValue[5].Value.b = (USHORT) TRUE;
|
|
// configure for RNA or LAN as appropriate
|
|
PropValue[6].ulPropTag = PR_CFG_RNA_PROFILE;
|
|
PropValue[7].ulPropTag = PR_CFG_CONN_TYPE;
|
|
PropValue[8].ulPropTag = PR_CFG_DELIVERY_OPTIONS;
|
|
if (pMailConfigInfo->pszConnectoidName &&
|
|
lstrlen(pMailConfigInfo->pszConnectoidName)) {
|
|
PropValue[6].Value.lpszW = pMailConfigInfo->pszConnectoidName;
|
|
PropValue[7].Value.l = (long) CONNECT_TYPE_REMOTE;
|
|
// set transfer mode for "selective"..
|
|
PropValue[8].Value.l = DOWNLOAD_OPTION_HEADERS;
|
|
} else {
|
|
PropValue[6].Value.lpszW = (LPWSTR) szNull;
|
|
PropValue[7].Value.l = (long) CONNECT_TYPE_LAN;
|
|
// set automatic transfer mode... mail guys made up the weird
|
|
// define name, not me!
|
|
PropValue[8].Value.l = DOWNLOAD_OPTION_MAIL_DELETE;
|
|
}
|
|
PropValue[9].ulPropTag = PR_CFG_REMOTE_USERNAME;
|
|
PropValue[9].Value.lpszW = pMailConfigInfo->pszEmailAccountName;
|
|
PropValue[10].ulPropTag = PR_CFG_REMOTE_PASSWORD;
|
|
PropValue[10].Value.lpszW = pMailConfigInfo->pszEmailAccountPwd;
|
|
|
|
// call the service admin interface to configure the service with these
|
|
// properties
|
|
hr = pServiceAdmin->ConfigureMsgService(pMapiUID, NULL,0,
|
|
nProps, PropValue);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"ConfigureMsgService returned 0x%x", hr);
|
|
}
|
|
|
|
// free the buffer with the UID
|
|
assert(lpMAPIFreeBuffer);
|
|
lpMAPIFreeBuffer(pMapiUID);
|
|
pMapiUID = NULL;
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: ConfigMessageStore
|
|
|
|
SYNOPSIS: Generates a unique filename and sets it as the
|
|
message store
|
|
|
|
ENTRY: lpServiceAdmin - pointer to service admin interface
|
|
lpMapiUID - UID for this service (message store)
|
|
|
|
EXIT: returns an HRESULT
|
|
|
|
NOTES: This code expects to be called only when the service is
|
|
newly created. Calling it on an existing service will
|
|
cause it to stomp existing settings.
|
|
|
|
********************************************************************/
|
|
HRESULT ConfigNewService(LPSERVICEADMIN lpServiceAdmin, LPMAPIUID lpMapiUID,
|
|
UINT uIDFilename, UINT uIDFilename1,UINT uPropValID)
|
|
{
|
|
WCHAR szMsgStorePath[MAX_PATH+1];
|
|
HRESULT hr=hrSuccess;
|
|
|
|
assert(lpServiceAdmin);
|
|
assert(lpMapiUID);
|
|
|
|
// build a path for the message store
|
|
if (!MakeUniqueFilename(uIDFilename, uIDFilename1,
|
|
szMsgStorePath, MAX_CHARS_IN_BUFFER(szMsgStorePath) ))
|
|
{
|
|
//DEBUGTRAP(L"Unable to create unique filename");
|
|
return MAPI_E_COLLISION;
|
|
}
|
|
//(L"Creating MAPI store %s", szMsgStorePath);
|
|
|
|
// set this filename for the message store
|
|
SPropValue PropVal;
|
|
PropVal.ulPropTag = uPropValID;
|
|
PropVal.Value.lpszW = szMsgStorePath;
|
|
hr = lpServiceAdmin->ConfigureMsgService(lpMapiUID, NULL,0,1,&PropVal);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"ConfigureMsgService returned 0x%x", hr);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: FindInternetMailService
|
|
|
|
SYNOPSIS: Detects if internet mail is installed, returns
|
|
email address and email server if it is.
|
|
|
|
********************************************************************/
|
|
BOOL FindInternetMailService(WCHAR * pszEmailAddress, DWORD cbEmailAddress,
|
|
WCHAR * pszEmailServer, DWORD cbEmailServer)
|
|
{
|
|
assert(pszEmailAddress);
|
|
assert(pszEmailServer);
|
|
|
|
if (!hInstMAPIDll && !InitMAPI(NULL))
|
|
return FALSE;
|
|
|
|
// look through all profiles. For each profile, look through all
|
|
// services. If we find an instance of the internet mail service,
|
|
// then return email address and password to caller. If there is
|
|
// more than one profile with the internet mail service, we
|
|
// will return the first one we find.
|
|
|
|
ENUM_MAPI_PROFILE EnumMAPIProfile;
|
|
LPWSTR lpProfileName, lpServiceName;
|
|
BOOL fDefault;
|
|
// walk through the list of profiles...
|
|
while (EnumMAPIProfile.Next(&lpProfileName, &fDefault)) {
|
|
assert(lpProfileName);
|
|
|
|
//DEBUGMSG(L"Found profile: %s", lpProfileName);
|
|
|
|
// for each profile, walk through the list of services...
|
|
ENUM_MAPI_SERVICE EnumMAPIService(lpProfileName);
|
|
while (EnumMAPIService.Next(&lpServiceName)) {
|
|
WCHAR szSmallBuf[SMALL_BUF_LEN+1];
|
|
|
|
//DEBUGMSG(L"Found service: %s", lpServiceName);
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), IDS_INTERNETMAIL_SERVICENAME,
|
|
szSmallBuf, MAX_CHARS_IN_BUFFER(szSmallBuf));
|
|
|
|
if (!lstrcmpi(lpServiceName, szSmallBuf)) {
|
|
|
|
//BUGBUG 21-May-1995 jeremys Get e-mail server & address from MAPI profile
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
ENUM_MAPI_PROFILE::ENUM_MAPI_PROFILE(VOID)
|
|
{
|
|
LPPROFADMIN pProfAdmin=NULL; // interface to administer profiles
|
|
HRESULT hr;
|
|
|
|
//assert(gfMAPIActive, L"MAPI not initialized!");
|
|
|
|
_pProfileRowSet = NULL;
|
|
_nEntries = 0;
|
|
_iRow = 0;
|
|
|
|
// get a pointer to the interface to administer profiles
|
|
assert(lpMAPIAdminProfiles);
|
|
hr = lpMAPIAdminProfiles(0, &pProfAdmin);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"MAPIAdminProfiles returned 0x%lx", hr);
|
|
return;
|
|
}
|
|
assert(pProfAdmin);
|
|
// release this interface when we leave the function
|
|
RELEASE_ME_LATER ReleaseProfAdminLater(pProfAdmin);
|
|
|
|
// get the rows in the profile table
|
|
hr = GetProfileList(pProfAdmin, &_pProfileRowSet);
|
|
if (HR_FAILED(hr))
|
|
return;
|
|
assert(_pProfileRowSet);
|
|
|
|
_nEntries = _pProfileRowSet->cRows;
|
|
|
|
}
|
|
|
|
ENUM_MAPI_PROFILE::~ENUM_MAPI_PROFILE(VOID)
|
|
{
|
|
if (_pProfileRowSet) {
|
|
// done with profile row set, free the table
|
|
FreeSRowSet(_pProfileRowSet);
|
|
_pProfileRowSet = NULL;
|
|
}
|
|
}
|
|
|
|
BOOL ENUM_MAPI_PROFILE::Next(LPWSTR * ppProfileName, BOOL * pfDefault)
|
|
{
|
|
assert(pfDefault);
|
|
|
|
if (!_pProfileRowSet)
|
|
return FALSE;
|
|
|
|
if (_iRow < _pProfileRowSet->cRows) {
|
|
LPSPropValue pPropVal = _pProfileRowSet->aRow[_iRow].lpProps;
|
|
assert(pPropVal);
|
|
|
|
// get pointer to profile name
|
|
*ppProfileName = pPropVal[0].Value.LPSZ;
|
|
assert(*ppProfileName);
|
|
// set 'this profile is default' flag
|
|
*pfDefault = pPropVal[2].Value.b;
|
|
|
|
_iRow++;
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
ENUM_MAPI_SERVICE::ENUM_MAPI_SERVICE(LPWSTR pszProfileName)
|
|
{
|
|
LPPROFADMIN pProfAdmin=NULL; // interface to administer profiles
|
|
LPSERVICEADMIN pServiceAdmin=NULL; // interface to administer services
|
|
HRESULT hr;
|
|
|
|
assert(pszProfileName);
|
|
//assertSZ(gfMAPIActive, L"MAPI not initialized!");
|
|
|
|
_pServiceRowSet = NULL;
|
|
_nEntries = 0;
|
|
_iRow = 0;
|
|
|
|
// get a pointer to the interface to administer profiles
|
|
assert(lpMAPIAdminProfiles);
|
|
hr = lpMAPIAdminProfiles(0, &pProfAdmin);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"MAPIAdminProfiles returned 0x%lx", hr);
|
|
return;
|
|
}
|
|
assert(pProfAdmin);
|
|
// release this interface when we leave the function
|
|
RELEASE_ME_LATER ReleaseProfAdminLater(pProfAdmin);
|
|
|
|
// get a pointer to the interface to administer services for this profile
|
|
hr = pProfAdmin->AdminServices(pszProfileName, NULL,NULL,0,
|
|
&pServiceAdmin);
|
|
if (HR_FAILED(hr)) {
|
|
//DEBUGMSG(L"AdminServices returned 0x%lx", hr);
|
|
return;
|
|
}
|
|
// release this interface when we leave the function
|
|
RELEASE_ME_LATER ReleaseServiceAdminLater(pServiceAdmin);
|
|
|
|
// get the rows in the profile table
|
|
hr = GetServicesList(pServiceAdmin, &_pServiceRowSet);
|
|
if (HR_FAILED(hr))
|
|
return;
|
|
assert(_pServiceRowSet);
|
|
|
|
_nEntries = _pServiceRowSet->cRows;
|
|
|
|
}
|
|
|
|
ENUM_MAPI_SERVICE::~ENUM_MAPI_SERVICE(VOID)
|
|
{
|
|
if (_pServiceRowSet) {
|
|
// done with profile row set, free the table
|
|
FreeSRowSet(_pServiceRowSet);
|
|
_pServiceRowSet = NULL;
|
|
}
|
|
}
|
|
|
|
BOOL ENUM_MAPI_SERVICE::Next(LPWSTR * ppServiceName)
|
|
{
|
|
if (!_pServiceRowSet)
|
|
return FALSE;
|
|
|
|
if (_iRow < _pServiceRowSet->cRows) {
|
|
LPSPropValue pPropVal = _pServiceRowSet->aRow[_iRow].lpProps;
|
|
assert(pPropVal);
|
|
|
|
// get pointer to profile name
|
|
*ppServiceName = pPropVal[ivalServiceName].Value.LPSZ;
|
|
assert(*ppServiceName);
|
|
|
|
_iRow++;
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: MakeUniqueFilename
|
|
|
|
SYNOPSIS: Generates a filename in the Windows directory that
|
|
does not already exist
|
|
|
|
ENTRY: uIDFilename - ID of string resource for desired name
|
|
for the file
|
|
uIDAltFilename - ID of string resource with template
|
|
for filename to use if file with uIDFilename's name
|
|
already exists. Template should contain %u into
|
|
which numbers will be inserted to make filename unique.
|
|
pszFilename - buffer to return path and filename into
|
|
cbFilename - size of pszFilename buffer
|
|
|
|
EXIT: returns TRUE if successful, FALSE if couldn't make
|
|
unique filename within MAX_FILENAME_TRIES tries
|
|
|
|
********************************************************************/
|
|
// number of times we'll try to create a unique filename before giving up
|
|
#define MAX_MAKEFILENAME_TRIES 20
|
|
BOOL MakeUniqueFilename(UINT uIDFilename, UINT uIDAltFilename,
|
|
WCHAR * pszFilename, DWORD cchFilename)
|
|
{
|
|
WCHAR szFileName[SMALL_BUF_LEN+1];
|
|
BOOL fSuccess = FALSE;
|
|
|
|
assert(pszFilename);
|
|
|
|
// build a path for the filename
|
|
UINT uRet=GetWindowsDirectory(pszFilename, cchFilename);
|
|
//assertSZ(uRet, L"GetWindowsDirectory failed");
|
|
|
|
// choose a file name that doesn't already exist
|
|
|
|
// first, try using the string resource specified by uIDFilename
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), uIDFilename, szFileName,MAX_CHARS_IN_BUFFER(szFileName));
|
|
if (DoesFileExist(pszFilename, szFileName)) {
|
|
|
|
// if that file exists, then use the string resource uIDAltFilename
|
|
// which has a replacable parameter. We'll try adding numbers to
|
|
// the filename to make it unique.
|
|
|
|
WCHAR szFileFmt[SMALL_BUF_LEN+1];
|
|
LoadString( GetModuleHandle(L"msobcomm.dll"), uIDAltFilename, szFileFmt,MAX_CHARS_IN_BUFFER(szFileFmt));
|
|
|
|
for (UINT nIndex = 0; nIndex < MAX_MAKEFILENAME_TRIES; nIndex ++) {
|
|
// make a name e.g. "mailbox4.pst"
|
|
wsprintf(szFileName, szFileFmt,nIndex);
|
|
if (!DoesFileExist(pszFilename, szFileName)) {
|
|
// OK, found a filename that doesn't exist
|
|
fSuccess = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
// first try succeeded
|
|
fSuccess = TRUE;
|
|
}
|
|
|
|
if (fSuccess) {
|
|
// now we have unique filename, build the full path
|
|
|
|
lstrcat(pszFilename, szSlash);
|
|
lstrcat(pszFilename, szFileName);
|
|
}
|
|
|
|
return fSuccess;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: DoesFileExist
|
|
|
|
SYNOPSIS: Checks to see whether the specified file exists
|
|
|
|
ENTRY: pszPath - path to directory
|
|
pszFilename - name of file
|
|
|
|
EXIT: returns TRUE if file exists, FALSE if it doesn't
|
|
|
|
********************************************************************/
|
|
BOOL DoesFileExist(WCHAR * pszPath, WCHAR * pszFileName)
|
|
{
|
|
WCHAR szPath[MAX_PATH+1];
|
|
OFSTRUCT of;
|
|
|
|
assert(pszPath);
|
|
assert(pszFileName);
|
|
|
|
// concatenate the path and file name
|
|
lstrcpy(szPath, pszPath);
|
|
lstrcat(szPath, szSlash);
|
|
lstrcat(szPath, pszFileName);
|
|
|
|
// find out if this file exists
|
|
WIN32_FIND_DATA fd;
|
|
HANDLE hfd = FindFirstFile(szPath, &fd);
|
|
if (INVALID_HANDLE_VALUE == hfd)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
CloseHandle(hfd);
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: GetServiceUID
|
|
|
|
SYNOPSIS: Given a MAPI service name, gets the MAPIUID associated
|
|
with it.
|
|
|
|
ENTRY: pszServiceName - name of MAPI service (e.g. "IMAIL", "MSPST AB")
|
|
lpServiceAdmin - pointer to service admin interface
|
|
ppMapiUID - pointer to pointer for MAPIUID struct
|
|
|
|
EXIT: returns an HRESULT
|
|
|
|
NOTES: Cloned from MAPI profile wizard code, if you think this
|
|
function is big and ugly now you should have seen it before
|
|
I cleaned it up.
|
|
|
|
This function allocates a MAPIUID, the caller is responsible
|
|
for freeing this (use MAPIFreeBuffer) when done.
|
|
|
|
********************************************************************/
|
|
HRESULT GetServiceUID(WCHAR * pszServiceName, LPSERVICEADMIN lpServiceAdmin,
|
|
LPMAPIUID *ppMapiUID)
|
|
{
|
|
HRESULT hr =hrSuccess;
|
|
LPSPropValue pTblProp =NULL;
|
|
DWORD iRow, iColumn;
|
|
LPMAPITABLE pTable =NULL;
|
|
LPSRowSet pRowSet =NULL;
|
|
LPSRow pRow =NULL;
|
|
int nFound =0;
|
|
LPMAPIUID pMapiUID =NULL;
|
|
BOOL fContinue = TRUE;
|
|
SizedSPropTagArray(2, Tbltaga) = { 2, { PR_SERVICE_NAME,
|
|
PR_SERVICE_UID }};
|
|
|
|
assert(pszServiceName);
|
|
assert(lpServiceAdmin);
|
|
assert(ppMapiUID);
|
|
|
|
// get table of message services
|
|
hr = lpServiceAdmin->GetMsgServiceTable(0, &pTable);
|
|
if (HR_FAILED(hr))
|
|
{
|
|
//DEBUGMSG(L"GetMsgServiceTable returned 0x%x", hr);
|
|
return hr;
|
|
}
|
|
assert(pTable);
|
|
// release this table when we exit this function
|
|
RELEASE_ME_LATER rlTable(pTable);
|
|
|
|
assert(lpHrQueryAllRows);
|
|
hr = lpHrQueryAllRows(pTable, (LPSPropTagArray) &Tbltaga, NULL, NULL, 0, &pRowSet);
|
|
if (HR_FAILED(hr))
|
|
{
|
|
//DEBUGMSG(L"HrQueryAllRows returned 0x%x", hr);
|
|
return hr;
|
|
}
|
|
assert(pRowSet);
|
|
|
|
iRow =0;
|
|
while (fContinue && iRow< pRowSet->cRows)
|
|
{
|
|
pRow = &pRowSet->aRow[iRow];
|
|
pTblProp = pRow->lpProps;
|
|
nFound = 0;
|
|
for (iColumn=0; iColumn<pRow->cValues; iColumn++)
|
|
{ //Check each property
|
|
if (pTblProp->ulPropTag ==PR_SERVICE_UID)
|
|
{
|
|
nFound++;
|
|
assert(lpMAPIAllocateBuffer);
|
|
lpMAPIAllocateBuffer(pTblProp->Value.bin.cb, (LPVOID FAR *) &pMapiUID);
|
|
if (!pMapiUID)
|
|
{
|
|
hr = MAPI_E_NOT_ENOUGH_MEMORY;
|
|
fContinue = FALSE;
|
|
break;
|
|
}
|
|
memcpy(pMapiUID, pTblProp->Value.bin.lpb, (size_t) pTblProp->Value.bin.cb);
|
|
*ppMapiUID = pMapiUID;
|
|
}
|
|
else if ((pTblProp->ulPropTag ==PR_SERVICE_NAME) &&
|
|
!lstrcmpi(pTblProp->Value.lpszW, pszServiceName))
|
|
{
|
|
nFound++;
|
|
}
|
|
pTblProp++;
|
|
|
|
if (nFound == 2) {
|
|
// found it!
|
|
fContinue = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
iRow++;
|
|
|
|
if (nFound < 2) {
|
|
// if one but not both items matched above, then deallocate buffer
|
|
if (pMapiUID) {
|
|
assert(lpMAPIFreeBuffer);
|
|
lpMAPIFreeBuffer(pMapiUID);
|
|
pMapiUID =NULL;
|
|
}
|
|
if (*ppMapiUID)
|
|
*ppMapiUID = NULL;
|
|
}
|
|
}
|
|
|
|
if (HR_FAILED(hr) || nFound < 2) {
|
|
// free buffer if we didn't find the UID
|
|
if (pMapiUID) {
|
|
assert(lpMAPIFreeBuffer);
|
|
lpMAPIFreeBuffer(pMapiUID);
|
|
}
|
|
if (*ppMapiUID)
|
|
*ppMapiUID = NULL;
|
|
}
|
|
|
|
if (pRowSet)
|
|
FreeSRowSet(pRowSet);
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*******************************************************************
|
|
|
|
NAME: FreeSRowSet
|
|
|
|
SYNOPSIS: Frees an SRowSet structure and the rows therein
|
|
|
|
ENTRY: prws - the row set to free
|
|
|
|
NOTES: Cloned from MAPI profile ctrl panel code
|
|
|
|
********************************************************************/
|
|
VOID FreeSRowSet(LPSRowSet prws)
|
|
{
|
|
ULONG irw;
|
|
|
|
if (!prws)
|
|
return;
|
|
|
|
assert(lpMAPIFreeBuffer);
|
|
|
|
// Free each row
|
|
for (irw = 0; irw < prws->cRows; irw++)
|
|
lpMAPIFreeBuffer(prws->aRow[irw].lpProps);
|
|
|
|
// Free the top level structure
|
|
lpMAPIFreeBuffer(prws);
|
|
}
|
|
|
|
/*
|
|
* ValidateProperty
|
|
*
|
|
* Purpose:
|
|
* Given a string prop, make sure it contains a valid string.
|
|
*
|
|
* Arguments:
|
|
* pval
|
|
* cVal
|
|
* ulPropTag
|
|
*
|
|
* Returns:
|
|
* BOOL
|
|
*/
|
|
WCHAR szUnk[] = L"???";
|
|
BOOL ValidateProperty(LPSPropValue pval, ULONG cVal, ULONG ulPropTag)
|
|
{
|
|
if(pval[cVal].ulPropTag != ulPropTag)
|
|
{
|
|
// make sure we're not stomping on good properties.
|
|
assert(PROP_TYPE(pval[cVal].ulPropTag) == PT_ERROR);
|
|
|
|
pval[cVal].ulPropTag = ulPropTag;
|
|
pval[cVal].Value.LPSZ = szUnk;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
#pragma data_seg(DATASEG_READONLY)
|
|
// note: this array depends on errors in rc file being in this order
|
|
// starting with S_OK. Don't rearrange one without rearranging the other.
|
|
static SCODE mpIdsScode[] =
|
|
{
|
|
S_OK,
|
|
MAPI_E_NO_ACCESS,
|
|
E_NOINTERFACE,
|
|
E_INVALIDARG,
|
|
MAPI_E_CALL_FAILED,
|
|
MAPI_E_NOT_FOUND,
|
|
MAPI_E_NO_SUPPORT,
|
|
MAPI_W_ERRORS_RETURNED,
|
|
MAPI_W_PARTIAL_COMPLETION,
|
|
MAPI_E_BAD_CHARWIDTH,
|
|
MAPI_E_BAD_VALUE,
|
|
MAPI_E_BUSY,
|
|
MAPI_E_COLLISION,
|
|
MAPI_E_COMPUTED,
|
|
MAPI_E_CORRUPT_DATA,
|
|
MAPI_E_CORRUPT_STORE,
|
|
MAPI_E_DISK_ERROR,
|
|
MAPI_E_HAS_FOLDERS,
|
|
MAPI_E_HAS_MESSAGES,
|
|
MAPI_E_INVALID_ENTRYID,
|
|
MAPI_E_INVALID_OBJECT,
|
|
MAPI_E_LOGON_FAILED,
|
|
MAPI_E_NETWORK_ERROR,
|
|
MAPI_E_NON_STANDARD,
|
|
MAPI_E_NOT_ENOUGH_DISK,
|
|
MAPI_E_NOT_ENOUGH_MEMORY,
|
|
MAPI_E_NOT_ENOUGH_RESOURCES,
|
|
MAPI_E_NOT_IN_QUEUE,
|
|
MAPI_E_OBJECT_CHANGED,
|
|
MAPI_E_OBJECT_DELETED,
|
|
MAPI_E_STRING_TOO_LONG,
|
|
MAPI_E_SUBMITTED,
|
|
MAPI_E_TOO_BIG,
|
|
MAPI_E_UNABLE_TO_ABORT,
|
|
MAPI_E_UNCONFIGURED,
|
|
MAPI_E_UNEXPECTED_TYPE,
|
|
MAPI_E_UNKNOWN_FLAGS,
|
|
MAPI_E_USER_CANCEL,
|
|
MAPI_E_VERSION
|
|
};
|
|
#pragma data_seg()
|
|
|
|
|
|
|