|
|
// ADMTMsi.cpp : Defines the initialization routines for the DLL.
//
#include "stdafx.h"
#include <stdio.h>
#include <windows.h>
#include <winuser.h>
#include <lm.h>
#include <msi.h>
#include <msiquery.h>
#include "ADMTMsi.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__; #endif
//
// Note!
//
// If this DLL is dynamically linked against the MFC
// DLLs, any functions exported from this DLL which
// call into MFC must have the AFX_MANAGE_STATE macro
// added at the very beginning of the function.
//
// For example:
//
// extern "C" BOOL PASCAL EXPORT ExportedFunction()
// {
// AFX_MANAGE_STATE(AfxGetStaticModuleState());
// // normal function body here
// }
//
// It is very important that this macro appear in each
// function, prior to any calls into MFC. This means that
// it must appear as the first statement within the
// function, even before any object variable declarations
// as their constructors may generate calls into the MFC
// DLL.
//
// Please see MFC Technical Notes 33 and 58 for additional
// details.
//
/////////////////////////////////////////////////////////////////////////////
// CADMTMsiApp
BEGIN_MESSAGE_MAP(CADMTMsiApp, CWinApp) //{{AFX_MSG_MAP(CADMTMsiApp)
// NOTE - the ClassWizard will add and remove mapping macros here.
// DO NOT EDIT what you see in these blocks of generated code!
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CADMTMsiApp construction
CADMTMsiApp::CADMTMsiApp() { // TODO: add construction code here,
// Place all significant initialization in InitInstance
}
/////////////////////////////////////////////////////////////////////////////
// The one and only CADMTMsiApp object
CADMTMsiApp theApp; HWND installWnd = 0;
/********************
* Helper Functions * ********************/
/*********************************************************************
* * * Written by: Paul Thompson * * Date: 25 JAN 2001 * * * * This function is a callback function used by GetWndFromInstall* * to compare titles and store the found HWND globally. * * * *********************************************************************/
//BEGIN CheckTitle
BOOL CALLBACK CheckTitle(HWND hwnd, LPARAM lParam) { /* local variables */ WCHAR sText[MAX_PATH]; WCHAR * pTitle; BOOL bSuccess; int len;
/* function body */ pTitle = (WCHAR*)lParam; //get the title to compare
//get the title of this window
len = GetWindowText(hwnd, sText, MAX_PATH);
if ((len) && (pTitle)) { if (wcsstr(sText, pTitle)) { installWnd = hwnd; return FALSE; } } return TRUE; } //END CheckTitle
/*********************************************************************
* * * Written by: Paul Thompson * * Date: 25 JAN 2001 * * * * This function is responsible for getting the HWND of the * * current installation to be used to display a MessageBox tied to * * the install GUI. * * * *********************************************************************/
//BEGIN GetWndFromInstall
void GetWndFromInstall(MSIHANDLE hInstall) { /* local variables */ WCHAR szPropName[MAX_PATH]; UINT lret = ERROR_SUCCESS; WCHAR sTitle[MAX_PATH]; DWORD nCount = MAX_PATH;
/* function body */ //get the installation's title
wcscpy(szPropName, L"ProductName"); lret = MsiGetProperty(hInstall, szPropName, sTitle, &nCount); if (lret != ERROR_SUCCESS) wcscpy(sTitle, L"ADMT Password Migration DLL");
//get the window handle for the install GUI
EnumChildWindows(NULL, CheckTitle, (LPARAM)sTitle); if (!installWnd) installWnd = GetForegroundWindow(); } //END GetWndFromInstall
/**********************
* exported functions * **********************/
/*********************************************************************
* * * Written by: Paul Thompson * * Date: 22 DEC 2000 * * * * This function is responsible for saving current ADMT files in * * the %TEMP% folder prior to installing the new version. The * * installation will later call the restore function to restore the * * saved file. Currently this mechanism is used for saving the * * current protar.mdb database. * * * *********************************************************************/
//BEGIN SaveCurrentFiles
UINT __stdcall SaveCurrentFiles(MSIHANDLE hInstall) { /* local constants */ const int GETENVVAR_ERROR = 0; //this indicates an error from the "GetEnvironmentVariable" function
const WCHAR sDCValue[2] = L"1";
/* local variables */ WCHAR tempdir[MAX_PATH]; WCHAR filename[MAX_PATH]; WCHAR newfilename[MAX_PATH]; int length; UINT lret = ERROR_SUCCESS; WCHAR sPropName[MAX_PATH]; WCHAR sDir[MAX_PATH]; DWORD nCount = MAX_PATH; HANDLE hFile; WIN32_FIND_DATA fDat; BOOL bSuccess;
/* function body */ //initialize these strings
wcscpy(sPropName, L"INSTALLDIR");
//if INSTALLDIR was not retrieved, set to default
if (MsiGetProperty(hInstall, sPropName, sDir, &nCount) != ERROR_SUCCESS) { length = GetEnvironmentVariable( L"ProgramFiles", sDir, MAX_PATH); if (length != GETENVVAR_ERROR) wcscat(sDir, L"\\Active Directory Migration Tool\\"); else wcscpy(sDir, L"C:\\Program Files\\Active Directory Migration Tool\\"); }
//find the temp dir
length = GetEnvironmentVariable( L"temp", tempdir, MAX_PATH); if (length == GETENVVAR_ERROR) wcscpy(tempdir, L"C:\\Temp");
//copy files to temp
wcscpy(filename, sDir); wcscat(filename, L"Protar.mdb"); wcscpy(newfilename, tempdir); wcscat(newfilename, L"\\Protar.mdb"); hFile = FindFirstFile(filename, &fDat); //if found, copy it
if (hFile != INVALID_HANDLE_VALUE) { FindClose(hFile); bSuccess = CopyFile(filename, newfilename, FALSE);
if (bSuccess) { lret = ERROR_SUCCESS; wcscpy(sPropName, L"bMDBSaved"); lret = MsiSetProperty(hInstall, sPropName, sDCValue); } else lret = ERROR_INSTALL_FAILURE; } else { wcscpy(sPropName, L"bMDBNotPresent"); lret = MsiSetProperty(hInstall, sPropName, sDCValue); }
return lret; } //END SaveCurrentFiles
/*********************************************************************
* * * Written by: Paul Thompson * * Date: 22 DEC 2000 * * * * This function is responsible for restoring ADMT files * * previously stored by a call to "SaveCurrentFiles". Currently this* * mechanism is used for saving the current protar.mdb database. * * * *********************************************************************/
//BEGIN RestoreFiles
UINT __stdcall RestoreFiles(MSIHANDLE hInstall) { /* local constants */ const int GETENVVAR_ERROR = 0; //this indicates an error from the "GetEnvironmentVariable" function
const WCHAR sDCValue[2] = L"1";
/* local variables */ WCHAR sDir[MAX_PATH]; WCHAR tempdir[MAX_PATH]; WCHAR filename[MAX_PATH]; WCHAR newfilename[MAX_PATH]; UINT lret = ERROR_SUCCESS; BOOL bSuccess; WCHAR sPropName[MAX_PATH]; DWORD nCount = MAX_PATH; int length;
/* function body */ //get the dir where we saved the files previously
wcscpy(sPropName, L"INSTALLDIR");
//if not retrieved, set to default
if (MsiGetProperty(hInstall, sPropName, sDir, &nCount) != ERROR_SUCCESS) { length = GetEnvironmentVariable( L"ProgramFiles", sDir, MAX_PATH); if (length != GETENVVAR_ERROR) wcscat(sDir, L"\\Active Directory Migration Tool\\"); else wcscpy(sDir, L"C:\\Program Files\\Active Directory Migration Tool\\"); }
//get the dir where we saved the files previously
length = GetEnvironmentVariable( L"temp", tempdir, MAX_PATH); if (length == GETENVVAR_ERROR) wcscpy(tempdir, L"C:\\Temp");
//copy files back
wcscpy(filename, tempdir); wcscat(filename, L"\\Protar.mdb"); wcscpy(newfilename, sDir); wcscat(newfilename, L"Protar.mdb"); bSuccess = CopyFile(filename, newfilename, FALSE);
if (bSuccess) { wcscpy(sPropName, L"bMDBRestored"); lret = MsiSetProperty(hInstall, sPropName, sDCValue); } else { lret = ERROR_INSTALL_FAILURE; } return lret; } //END RestoreFiles
/*********************************************************************
* * * Written by: Paul Thompson * * Date: 12 SEPT 2000 * * * * This function is responsible for displaying a message box. * * * *********************************************************************/
//BEGIN DisplayExiting
UINT __stdcall DisplayExiting(MSIHANDLE hInstall) { /* local variables */ WCHAR sPropName[MAX_PATH]; UINT lret = ERROR_SUCCESS; WCHAR sTitle[MAX_PATH] = L""; WCHAR sMsg[MAX_PATH] = L""; DWORD nCount = MAX_PATH;
/* function body */ //initialize these strings
wcscpy(sPropName, L"bMDBSaved");
//if this is not a DC, get its messages
if (MsiGetProperty(hInstall, sPropName, sMsg, &nCount) == ERROR_SUCCESS) { if (!wcscmp(sMsg, L"0")) { //get the leave messagebox msg string and title for not being able to save protar.mdb
wcscpy(sPropName, L"MDBLeaveMsg"); lret = MsiGetProperty(hInstall, sPropName, sMsg, &nCount); if (lret != ERROR_SUCCESS) wcscpy(sMsg, L"ADMT's internal database, protar.mdb, could not be saved. The installation cannot continue."); wcscpy(sPropName, L"MDBLeaveTitle"); lret = MsiGetProperty(hInstall, sPropName, sTitle, &nCount); if (lret != ERROR_SUCCESS) wcscpy(sTitle, L"Protar.mdb Not Saved!"); } else { //get the leave messagebox msg string and title for not being able to restore protar.mdb
wcscpy(sPropName, L"MDB2LeaveMsg"); lret = MsiGetProperty(hInstall, sPropName, sMsg, &nCount); if (lret != ERROR_SUCCESS) { wcscpy(sMsg, L"ADMT's internal database, protar.mdb, could not be restored. Manually restore"); wcscat(sMsg, L" it from the, environment variable, TEMP directory."); } wcscpy(sPropName, L"MDB2LeaveTitle"); lret = MsiGetProperty(hInstall, sPropName, sTitle, &nCount); if (lret != ERROR_SUCCESS) wcscpy(sTitle, L"Protar.mdb Not Restored!"); } }
GetWndFromInstall(hInstall); MessageBox(installWnd, sMsg, sTitle, MB_ICONSTOP | MB_OK); return lret; } //END DisplayExiting
/*********************************************************************
* * * Written by: Paul Thompson * * Date: 14 JAN 2000 * * * * This function is responsible for displaying a message box. * * * *********************************************************************/
//BEGIN IsUpgrade
UINT __stdcall IsUpgrade(MSIHANDLE hInstall) { /* local constants */ const int GETENVVAR_ERROR = 0; //this indicates an error from the "GetEnvironmentVariable" function
const WCHAR sExit[2] = L"1";
/* local variables */ WCHAR sPropName[MAX_PATH]; UINT lret = ERROR_SUCCESS; WCHAR sTitle[MAX_PATH] = L""; WCHAR sMsg[MAX_PATH] = L""; WCHAR sDir[MAX_PATH] = L""; WCHAR sKey[MAX_PATH] = L""; DWORD nCount = MAX_PATH; long lrtn = ERROR_SUCCESS; HKEY hADMTKey; int length;
/* function body */ /* see if ADMT V1.0 is installed by looking at the registry and find
out where it is installed at */ //open the ADMT Registry key, if it exists
wcscpy(sKey, L"SOFTWARE\\Mission Critical Software\\DomainAdmin"); if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, sKey, 0, KEY_READ, &hADMTKey) != ERROR_SUCCESS) return lret;
//get the current install path
wcscpy(sPropName, L"Directory"); nCount = MAX_PATH; if (RegQueryValueEx(hADMTKey, sPropName, NULL, NULL, (LPBYTE)sDir, &nCount) != ERROR_SUCCESS) { length = GetEnvironmentVariable( L"ProgramFiles", sDir, MAX_PATH); if (length != GETENVVAR_ERROR) wcscat(sDir, L"\\Active Directory Migration Tool\\"); else wcscpy(sDir, L"C:\\Program Files\\Active Directory Migration Tool\\"); } RegCloseKey(hADMTKey);
//now see if V1.0 is really installed (key exists)
wcscpy(sKey, L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall\\{76789332-34CD-11D3-9E6A-00A0C9AFE10F}"); if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, sKey, 0, KEY_READ, &hADMTKey) != ERROR_SUCCESS) return lret; RegCloseKey(hADMTKey);
//get the upgrade messagebox msg string and title
wcscpy(sPropName, L"UpgradeMsg"); lret = MsiGetProperty(hInstall, sPropName, sMsg, &nCount); if (lret != ERROR_SUCCESS) { wcscpy(sMsg, L"ADMT Version 1.00 is currently istalled. Would"); wcscat(sMsg, L" you like to upgrade to Version 2.00 Beta 2?"); } nCount = MAX_PATH; wcscpy(sPropName, L"UpgradeTitle"); lret = MsiGetProperty(hInstall, sPropName, sTitle, &nCount); if (lret != ERROR_SUCCESS) wcscpy(sTitle, L"Upgrade ADMT?");
//if they want to upgrade, save the install path
GetWndFromInstall(hInstall); if (MessageBox(installWnd, sMsg, sTitle, MB_ICONQUESTION | MB_YESNO) == IDYES) { wcscpy(sPropName, L"INSTALLDIR"); lret = MsiSetProperty(hInstall, sPropName, sDir); } else //else, set the flag to exit the install
{ wcscpy(sPropName, L"bUpgradeExit"); lret = MsiSetProperty(hInstall, sPropName, sExit); } return lret; } //END IsUpgrade
|