//+------------------------------------------------------------------------- // // AppWiz.cpl - "Add or Remove Programs" CPL. // Copyright (C) Microsoft // // File: Scripts.CPP // authomates running of TS application compatibility scripts // // History: Nov-14-2000 skuzin Created // //-------------------------------------------------------------------------- #include #include #include #include #include #include #include "Scripts.h" #include using namespace std; // //Struct to establish correspondence between app. name (key name) //and install and uninstall script names // class CAppScript { private: LPWSTR m_szKeyName; //key name that represents installed application LPWSTR m_szInstallScript; //install script name LPWSTR m_szUninstallScript; //uninstall script name DWORD m_bNeedReboot; //If set - then scripts must be run after reboot. BOOL m_bAlreadyInstalled; // public: // CAppScript() : m_szKeyName(NULL), m_szInstallScript(NULL), m_szUninstallScript(NULL), m_bNeedReboot(FALSE), m_bAlreadyInstalled(FALSE) { } ~CAppScript() { if(m_szKeyName) { LocalFree(m_szKeyName); } if(m_szInstallScript) { LocalFree(m_szInstallScript); } if(m_szUninstallScript) { LocalFree(m_szUninstallScript); } } BOOL Load(HKEY hKeyParent,LPCWSTR szKeyName); BOOL RunScriptIfApplicable(); private: BOOL RunScript(LPCWSTR szDir, LPCWSTR szScript); BOOL PrepareScriptForReboot(LPCWSTR szInstallDir, LPCWSTR szScript); //BUGBUG this function is public for test only }; //This class describes a list of pointers //to objects of class CAppScript class CAppScriptList : public list { public: //Deletes all CAppScript objects //before destroing the list itself. ~CAppScriptList() { CAppScriptList::iterator it; for(it=begin();it!=end(); it++) { delete (*it); } } }; class CAppScriptManager { private: CAppScriptList m_AppScriptList; public: CAppScriptManager(){}; ~CAppScriptManager(){}; BOOL Init(); BOOL RunScripts(); private: BOOL LoadSupportedAppList(); BOOL IsAppCompatOn(); }; //Functions - helpers. DWORD RegLoadString(HKEY hKey, LPCWSTR szValueName, LPWSTR *pszValue); DWORD RegLoadDWORD(HKEY hKey, LPCWSTR szValueName, DWORD *pdwValue); BOOL RegIsKeyExist(HKEY hKeyParent, LPCWSTR szKeyName); DWORD RegGetKeyInfo(HKEY hKey, LPDWORD pcSubKeys, LPDWORD pcMaxNameLen); DWORD RegKeyEnum(HKEY hKey, DWORD dwIndex, LPWSTR szSubKeyName, DWORD cSubKeyName); /////////////////////////////////////////////////////////////////////////////// //Exports /////////////////////////////////////////////////////////////////////////////// extern "C" LPVOID ScriptManagerInitScripts() { CAppScriptManager *pScriptManager = new CAppScriptManager(); if(pScriptManager) { if(!pScriptManager->Init()) { delete pScriptManager; pScriptManager = NULL; } } return pScriptManager; } extern "C" void ScriptManagerRunScripts( LPVOID *ppScriptManager) { if(*ppScriptManager) { CAppScriptManager *pScriptManager = reinterpret_cast(*ppScriptManager); pScriptManager->RunScripts(); delete pScriptManager; *ppScriptManager = NULL; } } /////////////////////////////////////////////////////////////////////////////// //class CAppScript /////////////////////////////////////////////////////////////////////////////// /****************************************************************************** CAppScript::Load() Purpose: Loads script information from the registry Sets m_bIsInitiallyInstalled to TRUE if app already installed ******************************************************************************/ BOOL CAppScript::Load( HKEY hKeyParent, LPCWSTR szKeyName) { DWORD err; HKEY hKey; err = RegOpenKeyExW(hKeyParent, szKeyName, 0, KEY_QUERY_VALUE, &hKey ); if(err == ERROR_SUCCESS) { RegLoadString(hKey, L"KeyName", &m_szKeyName); RegLoadString(hKey, L"InstallScript", &m_szInstallScript); RegLoadString(hKey, L"UninstallScript", &m_szUninstallScript); RegLoadDWORD(hKey, L"NeedReboot", &m_bNeedReboot); RegCloseKey(hKey); if(m_szKeyName) { m_bAlreadyInstalled = RegIsKeyExist(HKEY_LOCAL_MACHINE, m_szKeyName); return TRUE; } } return FALSE; } /****************************************************************************** CAppScript::RunScriptIfApplicable() Purpose: Checks if the application was installed or uninstalled and runs the script. If m_bNeedReboot flag is set - schedules the script to run after reboot. ******************************************************************************/ BOOL CAppScript::RunScriptIfApplicable() { KdPrint(("CAppScript::RunScriptIfApplicable() - ENTER\n")); const WCHAR szInstallSubDir[] = L"\\Application Compatibility Scripts\\Install"; const WCHAR szUninstallSubDir[] = L"\\Application Compatibility Scripts\\Uninstall"; static WCHAR szInstallDir[MAX_PATH+sizeof(szInstallSubDir)/sizeof(WCHAR)+1] = L""; static WCHAR szUninstallDir[MAX_PATH+sizeof(szUninstallSubDir)/sizeof(WCHAR)+1] = L""; if(!szInstallDir[0]) { //Get the scripts location //We need to do it only once //get Windows directory name if(!GetSystemWindowsDirectoryW(szInstallDir,MAX_PATH)) { KdPrint(("CAppScript::RunScriptIfApplicable() - GetWindowsDirectoryW() FAILED\n")); return FALSE; } wcscpy(szUninstallDir,szInstallDir); wcscat(szInstallDir,szInstallSubDir); wcscat(szUninstallDir,szUninstallSubDir); } if(!m_bAlreadyInstalled && RegIsKeyExist(HKEY_LOCAL_MACHINE, m_szKeyName) && m_szInstallScript) { //Application was installed if(m_bNeedReboot) { //Setup will continue after reboot //Create RunOnce entry to run script after system is rebooted KdPrint(("CAppScript::RunScriptIfApplicable() - PrepareScriptForReboot %ws\n",m_szInstallScript)); if(!PrepareScriptForReboot(szInstallDir, m_szInstallScript)) { KdPrint(("CAppScript::PrepareScriptForReboot() - FAILED\n",m_szInstallScript)); return FALSE; } } else { KdPrint(("CAppScript::RunScriptIfApplicable() - executing script %ws\n",m_szInstallScript)); if(!RunScript(szInstallDir,m_szInstallScript)) { KdPrint(("CAppScript::RunScriptIfApplicable() - executing script FAILED\n",m_szInstallScript)); return FALSE; } } m_bAlreadyInstalled = TRUE; } else { if(m_bAlreadyInstalled && !RegIsKeyExist(HKEY_LOCAL_MACHINE, m_szKeyName) && m_szUninstallScript) { //Application was uninstalled KdPrint(("CAppScript::RunScriptIfApplicable() - executing script %ws\n",m_szUninstallScript)); if(!RunScript(szUninstallDir,m_szUninstallScript)) { KdPrint(("CAppScript::RunScriptIfApplicable() - executing script FAILED\n",m_szUninstallScript)); return FALSE; } m_bAlreadyInstalled = FALSE; } } return TRUE; } /****************************************************************************** CAppScript::RunScript() Purpose: Runs script Waits untill script finishes ******************************************************************************/ BOOL CAppScript::RunScript( LPCWSTR szDir, LPCWSTR szScript) { WCHAR szCmdLineTemplate[] = L"cmd.exe /C %s"; DWORD dwSize = wcslen(szScript)*sizeof(WCHAR)+sizeof(szCmdLineTemplate); LPWSTR szCmdLine = (LPWSTR)LocalAlloc(LPTR,dwSize); if(!szCmdLine) { return FALSE; } swprintf(szCmdLine,szCmdLineTemplate,szScript); STARTUPINFO si; // startup information PROCESS_INFORMATION pi; // process information //Prepare startup info ZeroMemory(&si, sizeof(STARTUPINFO)); si.cb = sizeof(STARTUPINFO); BOOL bResult = CreateProcessW( NULL, // name of executable module szCmdLine, // command line string NULL, // SD NULL, // SD FALSE, // handle inheritance option CREATE_NEW_CONSOLE, // creation flags NULL, // new environment block szDir, // current directory name &si, // startup information &pi // process information ); LocalFree(szCmdLine); if(bResult) { WaitForSingleObject(pi.hProcess, INFINITE); CloseHandle(pi.hProcess); CloseHandle(pi.hThread); } return bResult; } /****************************************************************************** CAppScript::PrepareScriptForReboot() Purpose: Creates CMD file that will change current directory to ...\Application Compatibility Scripts\Install and then run the script. Schedules this CMD file to be run after reboot by creating an entry uder "HKCU\Software\Microsoft\Windows\CurrentVersion\RunOnce" key. ******************************************************************************/ BOOL CAppScript::PrepareScriptForReboot( LPCWSTR szInstallDir, LPCWSTR szScript) { WCHAR szFileNameTemplate[] = L"%s\\RunOnce.cmd"; LPWSTR szFullFileName = (LPWSTR)LocalAlloc(LPTR, wcslen(szInstallDir)*sizeof(WCHAR)+sizeof(szFileNameTemplate)); if(!szFullFileName) { return FALSE; } //Assemble full file name swprintf(szFullFileName,szFileNameTemplate,szInstallDir); HANDLE hFile = CreateFile( szFullFileName, // file name GENERIC_WRITE, // access mode 0, // share mode NULL, // SD CREATE_NEW, // how to create FILE_ATTRIBUTE_NORMAL, // file attributes NULL // handle to template file ); LocalFree(szFullFileName); //File did not exist if(hFile != INVALID_HANDLE_VALUE) { //File did not exist before //create code and write it into the file. //need code in ANSI char szCodeTemplate[] = "cd %S\r\n%%1\r\n"; DWORD nNumberOfBytesToWrite = wcslen(szInstallDir)+sizeof(szCodeTemplate); DWORD nNumberOfBytesWritten = 0; LPSTR szFileCode = (LPSTR)LocalAlloc(LPTR, nNumberOfBytesToWrite); if(!szFileCode) { CloseHandle(hFile); return FALSE; } sprintf(szFileCode, szCodeTemplate,szInstallDir); nNumberOfBytesToWrite = strlen(szFileCode); BOOL bResult = WriteFile( hFile, // handle to file (LPCVOID)szFileCode, // data buffer nNumberOfBytesToWrite, // number of bytes to write &nNumberOfBytesWritten, // number of bytes written NULL // overlapped buffer ); CloseHandle(hFile); LocalFree(szFileCode); if(!bResult || nNumberOfBytesToWrite != nNumberOfBytesWritten ) { return FALSE; } } else { if(GetLastError()!=ERROR_FILE_EXISTS) { return FALSE; } //If file already exists - do only registry changes. } //Registry changes: WCHAR szCommandTemplate[] = L"\"%s\\RunOnce.cmd\" %s"; DWORD cbCommandLen = (wcslen(szInstallDir)+wcslen(szScript))*sizeof(WCHAR)+sizeof(szCommandTemplate); LPWSTR szCommand = (LPWSTR)LocalAlloc(LPTR,cbCommandLen); if(!szCommand) { return FALSE; } swprintf(szCommand,szCommandTemplate,szInstallDir,szScript); cbCommandLen = (wcslen(szCommand)+1)*sizeof(WCHAR); HKEY hKey; LONG err; err = RegCreateKeyExW(HKEY_CURRENT_USER, L"Software\\Microsoft\\Windows\\CurrentVersion\\RunOnce", 0, NULL, 0, KEY_WRITE, NULL, &hKey, NULL); if(err == ERROR_SUCCESS) { err = RegSetValueExW(hKey, L"ZZZAppCompatScript", 0, REG_SZ, (CONST BYTE *)szCommand, cbCommandLen); RegCloseKey(hKey); } LocalFree(szCommand); if(err == ERROR_SUCCESS) { return TRUE; } else { return FALSE; } } /////////////////////////////////////////////////////////////////////////////// //class CAppScriptManager /////////////////////////////////////////////////////////////////////////////// /****************************************************************************** CAppScriptManager::Init() Purpose: Initialization. Returns FALSE if TS Application Compatibility if OFF or list of supported applications was not found in the registry. ******************************************************************************/ BOOL CAppScriptManager::Init() { //DebugBreak(); KdPrint(("CAppScriptManager::Init() - ENTER\n")); if(!IsAppCompatOn()) { KdPrint(("CAppScriptManager::Init() - TS App Compat is off!\n")); return FALSE; } if(!LoadSupportedAppList()) { KdPrint(("CAppScriptManager::Init() - LoadSupportedAppList() FAILED\n")); return FALSE; } KdPrint(("CAppScriptManager::Init() - OK\n")); return TRUE; } /****************************************************************************** CAppScriptManager::LoadSupportedAppList() Purpose: Loads from the registry the list of applications we care about along with their script names. Save this information in array of APP_SCRIPT structures. ******************************************************************************/ BOOL CAppScriptManager::LoadSupportedAppList() { HKEY hKey; LONG err; DWORD cSubKeys; DWORD cMaxSubKeyLen; KdPrint(("CAppScriptManager::LoadSupportedAppList() - ENTER\n")); err = RegOpenKeyExW(HKEY_LOCAL_MACHINE, L"Software\\Microsoft\\Windows NT\\CurrentVersion\\Terminal Server\\Compatibility\\Scripts", 0, KEY_READ, &hKey ); if(err == ERROR_SUCCESS) { err = RegGetKeyInfo(hKey, &cSubKeys, &cMaxSubKeyLen); if(err == ERROR_SUCCESS) { cMaxSubKeyLen+=1; //to include terminating NULL character KdPrint(("CAppScriptManager::LoadSupportedAppList() - %d apps supported\n",cSubKeys)); //Allocate buffer for subkey names LPWSTR szKeyName = (LPWSTR)LocalAlloc(LPTR,cMaxSubKeyLen*sizeof(WCHAR)); if(!szKeyName) { RegCloseKey(hKey); return FALSE; } CAppScript *pAppScript = NULL; for(DWORD i=0;iLoad(hKey, szKeyName)) { m_AppScriptList.push_back(pAppScript); } else { KdPrint(("CAppScriptManager::LoadSupportedAppList() - FAILED to load\n")); delete pAppScript; } } LocalFree(szKeyName); } RegCloseKey(hKey); } if(err != ERROR_SUCCESS) { return FALSE; } else { return TRUE; } } /****************************************************************************** CAppScriptManager::RunScripts() Purpose: Runs scripts for all installed or uninstalled applications. ******************************************************************************/ BOOL CAppScriptManager::RunScripts() { BOOL bInstallMode = FALSE; KdPrint(("CAppScriptManager::RunScripts() - ENTER\n")); //system must be now in INSTALL mode, set EXECUTE mode if(TermsrvAppInstallMode()) { bInstallMode = TRUE; KdPrint(("CAppScriptManager::RunScripts() - set EXECUTE mode\n")); if(!SetTermsrvAppInstallMode(FALSE)) { KdPrint(("CAppScriptManager::RunScripts() - SetTermsrvAppInstallMode() FAILED\n")); return FALSE; } } CAppScriptList::iterator it; for(it=m_AppScriptList.begin();it!=m_AppScriptList.end(); it++) { (*it)->RunScriptIfApplicable(); } if(bInstallMode) { //Restore INSTALL mode KdPrint(("CAppScriptManager::RunScripts() - return to INSTALL mode\n")); if(!SetTermsrvAppInstallMode(TRUE)) { KdPrint(("CAppScriptManager::RunScripts() - SetTermsrvAppInstallMode() FAILED\n")); return FALSE; } } KdPrint(("CAppScriptManager::RunScripts() - FINISH\n")); return TRUE; } /****************************************************************************** CAppScriptManager::IsAppCompatOn() Purpose: Checks if TS Application Compatibility mode is enabled. Returns TRUE if enabled, otherwise, as well as in case of any error, returns FALSE. ******************************************************************************/ BOOL CAppScriptManager::IsAppCompatOn() { HKEY hKey; DWORD dwData; BOOL fResult = FALSE; KdPrint(("CAppScriptManager::IsAppCompatOn() - ENTER\n")); if( RegOpenKeyExW(HKEY_LOCAL_MACHINE, REG_CONTROL_TSERVER, 0, KEY_QUERY_VALUE, &hKey) == ERROR_SUCCESS ) { if(RegLoadDWORD(hKey, L"TSAppCompat", &dwData) == ERROR_SUCCESS ) { KdPrint(("CAppScriptManager::IsAppCompatOn() - OK; Result=%d\n",dwData)); fResult = dwData; } RegCloseKey(hKey); } return fResult; } /////////////////////////////////////////////////////////////////////////////// //Functions - helpers. /////////////////////////////////////////////////////////////////////////////// /****************************************************************************** RegLoadString() Purpose: Loads a REG_SZ value from the registry Allocates buffer. Buffer then need to be freed using LocalFree function. ******************************************************************************/ DWORD RegLoadString( HKEY hKey, LPCWSTR szValueName, LPWSTR *pszValue) { DWORD cbData = 0; *pszValue = NULL; DWORD err = RegQueryValueExW( hKey, // handle to key szValueName, // value name NULL, // reserved NULL, // type buffer NULL, // data buffer &cbData // size of data buffer ); if(err == ERROR_SUCCESS) { *pszValue = (LPWSTR)LocalAlloc(LPTR,cbData); if(!*pszValue) { return GetLastError(); } err = RegQueryValueExW( hKey, // handle to key szValueName, // value name NULL, // reserved NULL, // type buffer (LPBYTE)*pszValue, // data buffer &cbData // size of data buffer ); if(err !=ERROR_SUCCESS) { LocalFree(*pszValue); *pszValue = NULL; } } return err; } /****************************************************************************** RegLoadDWORD() Purpose: Loads a REG_DWORD value from the registry ******************************************************************************/ DWORD RegLoadDWORD( HKEY hKey, LPCWSTR szValueName, DWORD *pdwValue) { DWORD cbData = sizeof(DWORD); return RegQueryValueExW( hKey, // handle to key szValueName, // value name NULL, // reserved NULL, // type buffer (LPBYTE)pdwValue, // data buffer &cbData // size of data buffer ); } /****************************************************************************** RegIsKeyExist() Purpose: Checks if key exists ******************************************************************************/ BOOL RegIsKeyExist( HKEY hKeyParent, LPCWSTR szKeyName) { LONG err; HKEY hKey; KdPrint(("RegIsKeyExist() - Opening key: hKeyParent=%d Key: %ws\n",hKeyParent,szKeyName)); err = RegOpenKeyExW(hKeyParent, szKeyName, 0, MAXIMUM_ALLOWED, &hKey ); if(err == ERROR_SUCCESS) { KdPrint(("RegIsKeyExist() - Key Exists!\n",err)); RegCloseKey(hKey); return TRUE; } else { KdPrint(("RegIsKeyExist() - err=%d\n",err)); return FALSE; } } /****************************************************************************** RegGetKeyInfo() Purpose: Gets key's number of sub keys and max sub key name length ******************************************************************************/ DWORD RegGetKeyInfo( HKEY hKey, LPDWORD pcSubKeys, LPDWORD pcMaxNameLen) { return RegQueryInfoKey( hKey, // handle to key NULL, // class buffer NULL, // size of class buffer NULL, // reserved pcSubKeys, // number of subkeys pcMaxNameLen, // longest subkey name (in TCHARs) NULL, // longest class string NULL, // number of value entries NULL, // longest value name NULL, // longest value data NULL, // descriptor length NULL // last write time ); } /****************************************************************************** RegKeyEnum() Purpose: Enumerates sub keys of the registry key ******************************************************************************/ DWORD RegKeyEnum( HKEY hKey, // handle to key to enumerate DWORD dwIndex, // subkey index LPWSTR szSubKeyName, // subkey name DWORD cSubKeyName) { FILETIME ftLastWriteTime; return RegEnumKeyExW( hKey, // handle to key to enumerate dwIndex, // subkey index szSubKeyName, // subkey name &cSubKeyName, // size of subkey buffer NULL, // reserved NULL, // class string buffer NULL, // size of class string buffer &ftLastWriteTime // last write time ); }