Leaked source code of windows server 2003
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.
 
 
 
 
 
 

1846 lines
54 KiB

/*===================================================================
Microsoft IIS
Microsoft Confidential.
Copyright 1997 Microsoft Corporation. All Rights Reserved.
Component: WAMREG
File: Auxfunc.cpp
implementation of supporting functions for WAMREG, including
interface to Register and Unregister WAM CLSID in registry,
interface to Create Package with MTS,
Owner: LeiJin
Note:
===================================================================*/
#include "common.h"
#include "auxfunc.h"
#include "iiscnfgp.h"
#include "dbgutil.h"
#include "iwamreg.h"
#include <inetinfo.h>
//==================================================================
// Global data definitions
//
//==================================================================
//
// string contains the physical path of wamreg.dll, ie. c:\winnt\system32\inetsrv\wam.dll
//
CHAR WamRegRegistryConfig::m_szWamDllPath[MAX_PATH];
//
// the permission to access the registry
//
const REGSAM WamRegRegistryConfig::samDesired = KEY_READ | KEY_WRITE;
//
// A thread will first grab a token, and wait until the token matches the m_dwServiceNum,
// In such way, the order of threads making the requests are perserved.
// m_hWriteLock (Event) is used for the blocking other threads
// m_csWAMREGLock is used for access the m_dwServiceToken and m_dwServiceNum
//
// private global, static variables for WamAdmLock
WamAdmLock WamRegGlobal::m_WamAdmLock;
WamRegGlobal g_WamRegGlobal;
WamRegRegistryConfig g_RegistryConfig;
//
// Defined at /LM/W3SVC/
// Default package ID(IIS In-Process Application) and the default WAMCLSID(IISWAM.W3SVC).
//
const WCHAR WamRegGlobal::g_szIISInProcPackageID[] =
W3_INPROC_PACKAGE_ID;
const WCHAR WamRegGlobal::g_szInProcWAMCLSID[] =
W3_INPROC_WAM_CLSID;
const WCHAR WamRegGlobal::g_szInProcWAMProgID[] = L"IISWAM.W3SVC";
const WCHAR WamRegGlobal::g_szIISOOPPoolPackageID[] =
W3_OOP_POOL_PACKAGE_ID;
const WCHAR WamRegGlobal::g_szOOPPoolWAMCLSID[] =
W3_OOP_POOL_WAM_CLSID;
const WCHAR WamRegGlobal::g_szOOPPoolWAMProgID[] = L"IISWAM.OutofProcessPool";
const WCHAR WamRegGlobal::g_szIISInProcPackageName[] = DEFAULT_PACKAGENAME;
const WCHAR WamRegGlobal::g_szIISOOPPoolPackageName[] = L"IIS Out-Of-Process Pooled Applications";
const WCHAR WamRegGlobal::g_szMDAppPathPrefix[] = L"/LM/W3SVC/";
const DWORD WamRegGlobal::g_cchMDAppPathPrefix =
(sizeof(L"/LM/W3SVC/")/sizeof(WCHAR)) - 1;
const WCHAR WamRegGlobal::g_szMDW3SVCRoot[] = L"/LM/W3SVC";
const DWORD WamRegGlobal::g_cchMDW3SVCRoot = (sizeof(L"/LM/W3SVC")/sizeof(WCHAR)) - 1;
#ifndef DBGERROR
#define DBGERROR(args) ((void)0) /* Do Nothing */
#endif
BOOL WamRegGlobal::Init(VOID)
{
return m_WamAdmLock.Init();
}
BOOL WamRegGlobal::UnInit(VOID)
{
return m_WamAdmLock.UnInit();
}
WamAdmLock::WamAdmLock()
: m_dwServiceToken(0),
m_dwServiceNum(0),
m_hWriteLock((HANDLE)NULL)
{
}
/*===================================================================
Init
Init certain variables for this supporting module of WAMREG.
Return: NONE
===================================================================*/
BOOL WamAdmLock::Init()
{
BOOL fReturn = TRUE;
INITIALIZE_CRITICAL_SECTION(&m_csLock);
m_hWriteLock = IIS_CREATE_EVENT(
"WamAdmLock::m_hWriteLock",
&m_hWriteLock,
TRUE,
TRUE
);
if (m_hWriteLock == NULL)
{
DBGPRINTF((DBG_CONTEXT, "Failed to create m_hWriteLock(Event). error = %08x",
GetLastError()));
fReturn = FALSE;
}
return fReturn;
}
/*===================================================================
UnInit
Uninit certain variables for this supporting module of WAMREG.
Return: NONE
===================================================================*/
BOOL WamAdmLock::UnInit()
{
BOOL fReturn = TRUE;
DeleteCriticalSection(&m_csLock);
if (m_hWriteLock)
{
BOOL fTemp = CloseHandle(m_hWriteLock);
if ( fTemp == FALSE)
{
DBGPRINTF((DBG_CONTEXT, "error in CloseHandle. errno = %d\n", GetLastError()));
fReturn = FALSE;
}
m_hWriteLock = (HANDLE)0;
}
return fReturn;
}
/*===================================================================
GetNextServiceToken
Obtain the next service token when a thread enters WAMREG, if the token
obtained by the thread is not the same as m_dwServiceNum, the thread has
to wait until the token it owns is same as m_dwServiceNum.
The function returns a token value.
Return: DWORD, Next Service Token
===================================================================*/
DWORD WamAdmLock::GetNextServiceToken( )
{
DWORD dwToken;
Lock();
dwToken = m_dwServiceToken;
m_dwServiceToken++;
UnLock();
return dwToken;
}
/*===================================================================
FAppPathAllowConfig
Test to see if we can make configuration changes(Delete/Create) on a path. Currently,
this function is used to block changes to the default application/package. The default
in proc package should not be deleted/altered at runtime.
Parameter:
pwszMetabasePath
Return: BOOL
Side affect: TRUE if we can configure the app on the app path.
===================================================================*/
BOOL WamRegGlobal::FAppPathAllowConfig
(
IN LPCWSTR pwszMetabasePath
)
{
BOOL fReturn = TRUE;
DWORD cchMDPath = 0;
DBG_ASSERT(pwszMetabasePath);
// Since szMDPath has a path that always starts with "/LM/W3SVC/", the input size must be
// greater that length of "/LM/W3SVC/", This check is necessary to protect that the default
// IIS (inproc) package being deleted by accident.
cchMDPath = wcslen(pwszMetabasePath);
if (cchMDPath <= WamRegGlobal::g_cchMDAppPathPrefix)
{
fReturn = FALSE;
}
return fReturn;
}
/*===================================================================
FIsW3SVCRoot
Test to see the MetabasePath is same as L"/LM/W3SVC".
Parameter:
pwszMetabasePath
Return: BOOL
Side affect: TRUE if we can configure the app on the app path.
===================================================================*/
BOOL WamRegGlobal::FIsW3SVCRoot
(
IN LPCWSTR wszMetabasePath
)
{
INT iReturn;
DBG_ASSERT(wszMetabasePath != NULL);
iReturn = _wcsnicmp(wszMetabasePath, WamRegGlobal::g_szMDW3SVCRoot, WamRegGlobal::g_cchMDW3SVCRoot+1);
return (iReturn == 0 ? TRUE : FALSE);
}
/*===================================================================
AcquireLock
Get a write lock, there can only be one thread doing work via DCOM interface, (i.e. has the write lock).
All other threads calling WamAdmin interfaces are blocked in this function. After returning from
this call, the thread is guaranteed to be a "Critical Section".
A simple CriticalSection only solve half of the problem. It guarantees the mutual exclusive condition.
But once a thread leaves the CS, the CS can not control which blocking threads can access CS next.
Parameter:
NONE.
Return: HRESULT
Side affect: Once returned, thread is in a "Critical Section".
===================================================================*/
VOID WamAdmLock::AcquireWriteLock( )
{
DWORD dwWaitReturn = WAIT_OBJECT_0;
DWORD dwMyToken = GetNextServiceToken();
BOOL fIsMyTurn = FALSE; // Assume it is not my turn before we try to acquire the lock.
DBG_ASSERT(m_hWriteLock);
do {
dwWaitReturn = WaitForSingleObject(m_hWriteLock, INFINITE);
//
// Check for successful return.
//
if (dwWaitReturn == WAIT_OBJECT_0)
{
Lock();
if (dwMyToken == m_dwServiceNum)
{
fIsMyTurn = TRUE;
}
UnLock();
}
else
{
//
// A failure down this code path might cause a busy loop...
// However, such failure is very unlikely. Attach a debugger can tell the story immediately.
//
DBGPRINTF((DBG_CONTEXT, "WaitForSingleObject failed, function returns %08x, errno = %08x\n",
dwWaitReturn,
GetLastError()));
DBG_ASSERT(FALSE);
}
} while (FALSE == fIsMyTurn);
ResetEvent(m_hWriteLock);
IF_DEBUG(WAMREG_MTS)
{
DBGPRINTF((DBG_CONTEXT, "Thread %08x acquired the WriteLock of WAMREG, ServiceNum is %d.\n",
GetCurrentThreadId(),
dwMyToken));
}
}
/*===================================================================
ReleaseLock
Release a write lock. See comments in CWmRgSrv::AcquireLock.
Parameter:
NONE.
Return: HRESULT
Side affect: Leave "Critical Section".
===================================================================*/
VOID WamAdmLock::ReleaseWriteLock( )
{
//CONSIDER: m_dwServerNum out-of-bound
Lock();
IF_DEBUG(WAMREG_MTS)
{
DBGPRINTF((DBG_CONTEXT, "Thread %08x released the WriteLock of WAMREG, ServiceNum is %d.\n",
GetCurrentThreadId(),
m_dwServiceNum));
}
m_dwServiceNum++;
SetEvent(m_hWriteLock);
UnLock();
}
/*===================================================================
RegisterCLSID
Register a WAM CLSID and a ProgID..After a successful registerCLSID call, you
should have
My Computer\HKEY_CLASSES_ROOT\CLSID\{szCLSIDEntryIn}
\InProcServer32 "...physical location of wam.dll"
\ProgID szProgIDIn
\VersionIndependentProgID "IISWAM.W3SVC"
Parameter:
szCLSIDEntryIn: CLSID for a WAM.
szProgIDIn: ProgID for the WAM.
fSetVIProgID: TRUE if this function needs to set the Version Independent ProgID.
FALSE, Otherwise.
Return: HRESULT
NOTE: Registry API should use ANSI version, otherwise, it will cause trouble on Win95.
===================================================================*/
HRESULT WamRegRegistryConfig::RegisterCLSID
(
IN LPCWSTR szCLSIDEntryIn,
IN LPCWSTR szProgIDIn,
IN BOOL fSetVIProgID
)
{
static const CHAR szWAMDLL[] = "wam.dll";
static const CHAR szClassDesc[] = "Web Application Manager Object";
static const CHAR szThreadingModel[] = "ThreadingModel";
static const CHAR szInprocServer32[] = "InprocServer32";
static const CHAR szTEXT_VIProgID[] = "VersionIndependentProgID";
static const CHAR szTEXT_ProgID[] = "ProgID";
static const CHAR szTEXT_Clsid[] = "Clsid";
static const CHAR szFreeThreaded[] = "Free";
static const CHAR szVIProgID[] = "IISWAM.Application";
HRESULT hr = E_FAIL;
HKEY hkeyT = NULL, hkey2 = NULL;
CHAR szCLSIDPath[100] = "CLSID\\"; // CLSID\\{....} , less that 100.
CHAR szCLSIDEntry[uSizeCLSID]; // ANSI version of CLSID.
CHAR* szProgID = NULL; // a pointer to ANSI version of ProgID.
DWORD dwSizeofProgID = 0; // # of char of ProgID.
DBG_ASSERT(szProgIDIn);
DBG_ASSERT(szCLSIDEntryIn);
//
// Make a clsid ID.
//
WideCharToMultiByte(CP_ACP, 0, szCLSIDEntryIn, -1, szCLSIDEntry, uSizeCLSID, NULL, NULL);
strncat(szCLSIDPath, szCLSIDEntry, uSizeCLSID);
//
// Make a Prog ID.
//
// *2 for DBCS enabling for App MD path
dwSizeofProgID = wcslen(szProgIDIn)*2 + 1;
szProgID = new CHAR[dwSizeofProgID];
if (NULL == szProgID)
{
hr = E_OUTOFMEMORY;
goto LErrExit;
}
WideCharToMultiByte(CP_ACP, 0, szProgIDIn, -1, szProgID, dwSizeofProgID, NULL, NULL);
// install the CLSID key
// Setting the value of the description creates the key for the clsid
//
if ((RegSetValueA(HKEY_CLASSES_ROOT, szCLSIDPath, REG_SZ, szClassDesc,
strlen(szClassDesc)) != ERROR_SUCCESS))
goto LErrExit;
//
// Open the CLSID key so we can set values on it
//
if (RegOpenKeyExA(HKEY_CLASSES_ROOT, szCLSIDPath, 0, samDesired, &hkeyT) != ERROR_SUCCESS)
goto LErrExit;
//
// install the InprocServer32 key and open the sub-key to set the named value
//
if ((RegSetValueA(hkeyT, szInprocServer32, REG_SZ, m_szWamDllPath, strlen(m_szWamDllPath)) != ERROR_SUCCESS) ||
(RegOpenKeyExA(hkeyT, szInprocServer32, 0, samDesired, &hkey2) != ERROR_SUCCESS))
goto LErrExit;
//
// install the ProgID key and version independent ProgID key
//
if ((RegSetValueA(hkeyT, szTEXT_ProgID, REG_SZ, szProgID, strlen(szProgID)) != ERROR_SUCCESS) ||
(RegSetValueA(hkeyT, szTEXT_VIProgID, REG_SZ, szVIProgID, strlen(szVIProgID)) != ERROR_SUCCESS))
goto LErrExit;
if (RegCloseKey(hkeyT) != ERROR_SUCCESS)
goto LErrExit;
hkeyT = hkey2;
hkey2 = NULL;
//
// install the ThreadingModel named value
//
if (RegSetValueExA(hkeyT, szThreadingModel, 0, REG_SZ, (const BYTE *)szFreeThreaded,
strlen(szFreeThreaded)) != ERROR_SUCCESS)
goto LErrExit;
if (RegCloseKey(hkeyT) != ERROR_SUCCESS)
goto LErrExit;
else
hkeyT = NULL;
// Set up ProgID key
if ((RegSetValueA(HKEY_CLASSES_ROOT, szProgID, REG_SZ, szClassDesc,
strlen(szClassDesc)) != ERROR_SUCCESS))
goto LErrExit;
if (RegOpenKeyExA(HKEY_CLASSES_ROOT, szProgID, 0, samDesired, &hkeyT) != ERROR_SUCCESS)
goto LErrExit;
if ((RegSetValueA(hkeyT, szTEXT_Clsid, REG_SZ, szCLSIDEntry, strlen(szCLSIDEntry)) != ERROR_SUCCESS))
goto LErrExit;
if (RegCloseKey(hkeyT) != ERROR_SUCCESS)
goto LErrExit;
else
hkeyT = NULL;
// Set up Version Independent key only at setup IIS default time
if (fSetVIProgID)
{
if ((RegSetValueA(HKEY_CLASSES_ROOT, szVIProgID, REG_SZ, szClassDesc,
strlen(szClassDesc)) != ERROR_SUCCESS))
goto LErrExit;
if (RegOpenKeyExA(HKEY_CLASSES_ROOT, szVIProgID, 0, samDesired, &hkeyT) != ERROR_SUCCESS)
goto LErrExit;
if ((RegSetValueA(hkeyT, szTEXT_Clsid, REG_SZ, szCLSIDEntry, strlen(szCLSIDEntry)) != ERROR_SUCCESS))
goto LErrExit;
if (RegCloseKey(hkeyT) != ERROR_SUCCESS)
goto LErrExit;
else
hkeyT = NULL;
}
hr = NOERROR;
LErrExit:
if (szProgID)
{
delete [] szProgID;
szProgID = NULL;
}
if (hkeyT)
{
RegCloseKey(hkeyT);
}
if (hkey2)
{
RegCloseKey(hkey2);
}
return hr;
}
/*===================================================================
UnRegisterCLSID
UnRegister a WAM CLSID & a corresponding WAM PROG ID.
Parameter:
szCLSIDEntryIn: [in] CLSID for a WAM.
fDeleteVIProgID: TRUE, to delete the version independent prog id, FALSE, not touch VI progID.
Return: HRESULT
===================================================================*/
HRESULT WamRegRegistryConfig::UnRegisterCLSID
(
IN LPCWSTR wszCLSIDEntryIn,
IN BOOL fDeleteVIProgID
)
{
HRESULT hr = E_FAIL;
HKEY hkey = NULL;
CHAR szCLSIDEntry[uSizeCLSID];
CHAR szCLSIDPath[100] = "CLSID\\";
WCHAR *szProgID = NULL;
CLSID Clsid_WAM;
static const WCHAR szVIProgID[] = L"IISWAM.Application";
DBG_ASSERT(wszCLSIDEntryIn);
//
// Make a clsid ID.
//
WideCharToMultiByte(CP_ACP, 0, wszCLSIDEntryIn, -1, szCLSIDEntry, uSizeCLSID, NULL, NULL);
strncat(szCLSIDPath, szCLSIDEntry, uSizeCLSID);
//
// UnRegister ProgID and Version Independent Prog ID.
//
hr = CLSIDFromString((WCHAR *)wszCLSIDEntryIn, &Clsid_WAM);
if (SUCCEEDED(hr))
{
hr = ProgIDFromCLSID(Clsid_WAM, &szProgID);
if (SUCCEEDED(hr))
{
hr = UnRegisterProgID(szProgID);
CoTaskMemFree(szProgID);
szProgID = NULL;
}
else
{
DBGPRINTF((DBG_CONTEXT, "error = %08x\n", hr));
}
}
else
{
DBGPRINTF((DBG_CONTEXT, "error = %08x\n", hr));
}
if (fDeleteVIProgID)
{
hr = UnRegisterProgID(szVIProgID);
if (FAILED(hr))
{
DBGPRINTF((DBG_CONTEXT, "error = %08x\n", hr));
}
}
DWORD dwReg;
//
// Open the HKEY_CLASSES_ROOT\CLSID\{...} key so we can delete its subkeys
//
dwReg = RegOpenKeyExA(HKEY_CLASSES_ROOT, szCLSIDPath, 0, samDesired, &hkey);
if (dwReg == ERROR_SUCCESS)
{
DWORD iKey = 0;
CHAR szKeyName[MAX_PATH];
DWORD cbKeyName;
//
// Enumerate all its subkeys, and delete them
// for (iKey=0;;iKey++) might not work with multiple sub keys, the last interation has iKey >
// the actually number of subkeys left. Set iKey = 0, so that we can always delete them all.
//
while(TRUE)
{
cbKeyName = sizeof(szKeyName);
if (RegEnumKeyExA(hkey, iKey, szKeyName, &cbKeyName, 0, NULL, 0, NULL) != ERROR_SUCCESS)
break;
if (RegDeleteKeyA(hkey, szKeyName) != ERROR_SUCCESS)
break;
}
// Close the key, and then delete it
dwReg = RegCloseKey(hkey);
if ( dwReg != ERROR_SUCCESS)
{
DBGPRINTF((DBG_CONTEXT, "error = %08x\n", HRESULT_FROM_WIN32(dwReg)));
}
}
dwReg = RegDeleteKeyA(HKEY_CLASSES_ROOT, szCLSIDPath);
if ( dwReg != ERROR_SUCCESS)
{
DBGPRINTF((DBG_CONTEXT, "error = %08x\n", HRESULT_FROM_WIN32(dwReg)));
}
//
// Return hr Result
//
if (SUCCEEDED(hr))
{
if (dwReg != ERROR_SUCCESS)
{
hr = HRESULT_FROM_WIN32(dwReg);
}
}
else
{
DBG_ASSERT((DBG_CONTEXT, "Failed to UnRegisterCLSID, %S, fDeleteVIProgID = %s\n",
wszCLSIDEntryIn,
(fDeleteVIProgID ? "TRUE" : "FALSE")));
}
return hr;
}
/*===================================================================
UnRegisterProgID
UnRegister a Prog ID.
Parameter:
szProgID: [in] Prog ID can be a version independent Prog ID.
Return: HRESULT
===================================================================*/
HRESULT WamRegRegistryConfig::UnRegisterProgID
(
IN LPCWSTR szProgIDIn
)
{
HKEY hkey = NULL;
DWORD iKey;
DWORD cbKeyName;
DWORD dwSizeofProgID;
CHAR szKeyName[255];
CHAR* szProgID = NULL;
HRESULT hr = E_FAIL;
DBG_ASSERT(szProgIDIn);
//
// Make a Prog ID.
//
// DBCS enabling *2
dwSizeofProgID = wcslen(szProgIDIn)*2 + 1;
szProgID = new CHAR[dwSizeofProgID];
if (NULL == szProgID)
{
hr = E_OUTOFMEMORY;
goto LErrExit;
}
WideCharToMultiByte(CP_ACP, 0, szProgIDIn, -1, szProgID, dwSizeofProgID, NULL, NULL);
// Open the HKEY_CLASSES_ROOT\szProgID key so we can delete its subkeys
if (RegOpenKeyExA(HKEY_CLASSES_ROOT, szProgID, 0, samDesired, &hkey) != ERROR_SUCCESS)
goto LErrExit;
// Enumerate all its subkeys, and delete them
for (iKey=0;;iKey++)
{
cbKeyName = sizeof(szKeyName);
if (RegEnumKeyExA(hkey, iKey, szKeyName, &cbKeyName, 0, NULL, 0, NULL) != ERROR_SUCCESS)
break;
if (RegDeleteKeyA(hkey, szKeyName) != ERROR_SUCCESS)
goto LErrExit;
}
// Close the key, and then delete it
if (RegCloseKey(hkey) != ERROR_SUCCESS)
goto LErrExit;
else
hkey = NULL;
if (RegDeleteKeyA(HKEY_CLASSES_ROOT, szProgID) != ERROR_SUCCESS)
goto LErrExit;
hr = NOERROR;
LErrExit:
if (szProgID)
delete [] szProgID;
if (hkey)
RegCloseKey(hkey);
return hr;
}
/*===================================================================
LoadWamDllPath
Read Wam Dll Path from Registry. This function is implemented in ANSI version
of Registry API(Win95 compatibility).
Parameter:
Return: HRESULT
Side Affect:
NONE.
===================================================================*/
HRESULT WamRegRegistryConfig::LoadWamDllPath
(
void
)
{
LONG lReg = 0;
HKEY hKey = NULL;
HRESULT hr = NOERROR;
m_szWamDllPath[0] = '\0';
lReg = RegOpenKeyExA(HKEY_LOCAL_MACHINE,
"Software\\Microsoft\\InetStp",
0,
KEY_READ,
&hKey);
if (lReg == ERROR_SUCCESS)
{
LONG lErr = 0;
DWORD dwType;
CHAR szWamDllName[] = "\\wam.dll";
DWORD cbData = (sizeof(m_szWamDllPath) - sizeof(szWamDllName));
lErr = RegQueryValueExA(hKey,
"InstallPath",
0,
&dwType,
(LPBYTE)m_szWamDllPath,
&cbData);
if (lErr == ERROR_SUCCESS)
{
if (dwType == REG_SZ)
{
strncat(m_szWamDllPath, szWamDllName, sizeof(szWamDllName));
hr = NOERROR;
}
else
{
hr = E_UNEXPECTED;
DBGPRINTF((DBG_CONTEXT, "Wrong Type for InstallPath registry key.dwType = %d\n",
dwType));
}
}
else
{
hr = HRESULT_FROM_WIN32(lErr);
}
RegCloseKey(hKey);
}
else
{
hr = HRESULT_FROM_WIN32(lReg);
}
return hr;
}
/*===================================================================
SzWamProgID
Make a WAM Prog ID, if the MetabasePath is null, assume it is the default,
that is WAM de
fault, so, it will be IISWAM.W3SVC. Otherwise, the format is
IISWAM.__1__Application__Path where application path is \\LM\w3svc\1\
Application_path.
Parameter:
szMetabasePath: [in] MetabasePath.
Return:
TYPE: LPWSTR, a string contains ProgID
NULL, if failed.
Side Affect:
Allocate memory for return result use new. Caller needs to free szWamProgID
use delete[].
===================================================================*/
HRESULT WamRegGlobal::SzWamProgID
(
IN LPCWSTR pwszMetabasePath,
OUT LPWSTR* ppszWamProgID
)
{
HRESULT hr = NOERROR;
static WCHAR wszIISProgIDPreFix[] = L"IISWAM.";
WCHAR *pwszResult = NULL;
WCHAR *pwszApplicationPath = NULL;
UINT uPrefixLen = (sizeof(wszIISProgIDPreFix) / sizeof(WCHAR));
DBG_ASSERT(pwszMetabasePath);
*ppszWamProgID = NULL;
//
// Make a new WAM Prog ID based on pwszMetabasepath
//
WCHAR *pStr, *pResult;
UINT uSize = 0;
//
// CONSIDER: use (sizeof(L"/LM/W3SVC/")/sizeof(WCHAR) - 1) for 10
// CONSIDER: use global const for L"/LM/W3SVC/"
// Since all paths start with /LM/W3SVC/, omit the prefix.
//
if (_wcsnicmp(pwszMetabasePath, L"\\LM\\W3SVC\\", 10) == 0 ||
_wcsnicmp(pwszMetabasePath, L"/LM/W3SVC/", 10) == 0)
{
pwszApplicationPath = (WCHAR *)(pwszMetabasePath + 10);
}
else
{
*ppszWamProgID = NULL;
hr = HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER);
}
if (SUCCEEDED(hr))
{
pStr = pwszApplicationPath;
//
// Calculate uSize for allocation
//
while(*pStr != NULL)
{
//
// '/' or '\\' will be converted to '__', from 1 char to 2 chars.
//
if (*pStr == '\\' || *pStr == '/')
uSize ++;
pStr++;
uSize++;
}
DBG_ASSERT(uSize > 0);
uSize += uPrefixLen;
// uSize takes the null terminator into count.
pwszResult = new WCHAR[uSize];
if (pwszResult != NULL)
{
wcsncpy(pwszResult, wszIISProgIDPreFix, uPrefixLen);
pStr = pwszApplicationPath;
pResult = pwszResult + uPrefixLen - 1;
while(*pStr != NULL)
{
if (*pStr == '\\' || *pStr == '/')
{
*pResult++ = '_';
*pResult++ = '_';
pStr++;
}
else
{
*pResult++ = *pStr++;
}
}
// NULL Terminating the result
pwszResult[uSize-1] = '\0';
}
else
{
hr = E_OUTOFMEMORY;
}
}
if (SUCCEEDED(hr))
{
*ppszWamProgID = pwszResult;
}
return hr;
}
/*===================================================================
GetViperPackageName
Make a Package Name. Follow the naming conversion, "IIS-{web site name/
application name}"
Parameter:
szMetabasePath: [in] MetabasePath.
Return: HRESULT
Side Affect:
Allocate memory for return result use new. Caller needs to free
szPackageName using delete[].
===================================================================*/
HRESULT WamRegGlobal::GetViperPackageName
(
IN LPCWSTR wszMetabasePath,
OUT LPWSTR* ppszPackageNameOut
)
{
static WCHAR wszPackageNamePreFix[] = L"IIS-{";
static WCHAR wszPackageNamePostFix[] = L"}";
WCHAR wszServerName[500];
WCHAR* pwszPackageName;
WCHAR *wszResult = NULL;
WCHAR *pStr, *pResult;
UINT cPackageName = 0;
UINT cServerName = 0;
UINT uSize = 0;
HRESULT hr = NOERROR;
WamRegMetabaseConfig MDConfig;
if ((_wcsnicmp(wszMetabasePath, WamRegGlobal::g_szMDAppPathPrefix, WamRegGlobal::g_cchMDAppPathPrefix) == 0) ||
(_wcsnicmp(wszMetabasePath, WamRegGlobal::g_szMDAppPathPrefix, WamRegGlobal::g_cchMDAppPathPrefix) == 0))
{
hr = MDConfig.GetWebServerName(wszMetabasePath, wszServerName, sizeof(wszServerName));
if (SUCCEEDED(hr))
{
cServerName = wcslen(wszServerName);
}
}
else
{
hr = E_FAIL;
DBGPRINTF((DBG_CONTEXT, "Unknown metabase path %S\n", wszMetabasePath));
DBG_ASSERT(FALSE); // Confused ??? MetabasePath has other format? not start with /LM/W3SVC/ ???
}
if (SUCCEEDED(hr))
{
pwszPackageName = (WCHAR *)(wszMetabasePath + 10);
// Explanation: skip the 1 at /LM/W3SVC/1/, 1 is the virtual server, the
// naming conversion
// will replace the number 1 with some nice name(from GetWebServerName call).
while(*pwszPackageName != NULL)
{
if (*pwszPackageName == L'\\' || *pwszPackageName == L'/')
break;
pwszPackageName++;
}
DBG_ASSERT(pwszPackageName != NULL); // We must be able find '\\' or '/' before we scan the whole path.
cPackageName = wcslen(pwszPackageName);
pStr = pwszPackageName;
// 8 = wcslen(TEXT("IIS-{")) + wcslen(TEXT("}")) + '/' + null terminator
uSize = 8 + cPackageName + cServerName;
wszResult = new WCHAR [uSize];
if (wszResult != NULL)
{
//
// IIS-{
//
pResult = wszResult;
wcsncpy(wszResult, wszPackageNamePreFix, sizeof(wszPackageNamePreFix) / sizeof(WCHAR));
pResult += sizeof(wszPackageNamePreFix) / sizeof(WCHAR) - 1;
//
// IIS-{ Web Sever Name
//
wcsncpy(pResult, wszServerName, cServerName + 1);
pResult += cServerName;
//
// IIS-{ Web Server Name /
//
wcsncpy(pResult, L"/", sizeof(L"/"));
pResult += 1; // sizeof(TEXT("/")) == 2
//
// IIS-{ Web Server Name / PackageName(ApplicationName)
//
wcsncpy(pResult, pwszPackageName, cPackageName + 1);
pResult += cPackageName;
//
// IIS-{ Web Server Name / PackageName(ApplicationName) } \n
//
wcsncpy(pResult, wszPackageNamePostFix, sizeof(wszPackageNamePostFix) / sizeof(WCHAR));
}
else
{
hr = E_OUTOFMEMORY;
}
}
if (FAILED(hr))
{
if (wszResult)
{
delete[] wszResult;
wszResult = NULL;
}
*ppszPackageNameOut = NULL;
}
else
{
DBG_ASSERT(wszResult);
*ppszPackageNameOut = wszResult;
}
return hr;
}
/*===================================================================
ConstructFullPath
When use GetDataPaths call, it only returns patial path relative to a metabase path.
(sub node of a metabase path). This fuction will contruct the complete metabase path to
a sub node.
Parameter:
pwszMetabasePathPrefix: [in] MetabasePath.
pwszPartialPath
ppwszResult result buffer
Return: HRESULT
Side Affect:
Allocate memory for return result use new. Caller needs to free
*ppwszResult using delete[].
===================================================================*/
HRESULT WamRegGlobal::ConstructFullPath
(
IN LPCWSTR pwszMetabasePathPrefix,
IN DWORD dwcPrefix,
IN LPCWSTR pwszPartialPath,
OUT LPWSTR* ppwszResult
)
{
HRESULT hr = NOERROR;
DWORD cchPrefix = dwcPrefix;
DWORD cchPartialPath = 0;
DWORD cchFullPath = 0;
WCHAR *pResult = NULL;
BOOL fHasEndSlash = FALSE;
DBG_ASSERT(pwszPartialPath != NULL);
if (pwszMetabasePathPrefix[dwcPrefix-1] == L'\\' ||
pwszMetabasePathPrefix[dwcPrefix-1] == L'/')
{
cchPrefix--;
}
cchPartialPath = wcslen(pwszPartialPath);
// Skip the ending '/' of pwszPartialPath if thereis any.
if (cchPartialPath > 0 &&
(pwszPartialPath[cchPartialPath-1] == L'/'
|| pwszPartialPath[cchPartialPath-1] == L'\\'))
{
cchPartialPath--;
fHasEndSlash=TRUE;
}
cchFullPath = cchPrefix + cchPartialPath + 1;
pResult = new WCHAR [cchFullPath];
if (pResult)
{
memcpy(pResult, pwszMetabasePathPrefix, cchPrefix*sizeof(WCHAR));
memcpy(pResult+cchPrefix, pwszPartialPath, cchPartialPath*sizeof(WCHAR));
pResult[cchFullPath-1] = L'\0';
}
else
{
hr = E_OUTOFMEMORY;
}
*ppwszResult = pResult;
return hr;
}
/*===================================================================
GetNewSzGUID
Generate a new GUID and put into *ppszGUID.
Parameter:
LPWSTR *ppszGUID a pointer to an array, allocated in this function
and freed by caller.
Return: HRESULT
===================================================================*/
HRESULT WamRegGlobal::GetNewSzGUID(OUT LPWSTR *ppszGUID)
{
GUID GUID_Temp;
HRESULT hr;
DBG_ASSERT(ppszGUID);
// Create a new WAM CLSID
hr = CoCreateGuid(&GUID_Temp);
if (SUCCEEDED(hr))
{
hr = StringFromCLSID(GUID_Temp, ppszGUID);
if (FAILED(hr))
{
*ppszGUID = NULL;
}
}
return hr;
}
/*===================================================================
CreatePooledApp
Register a WAM CLSID.
Parameter:
szMetabasePath: [in] MetabaseKey.
Return: HRESULT
===================================================================*/
HRESULT WamRegGlobal::CreatePooledApp
(
IN LPCWSTR szMetabasePath,
IN BOOL fInProc,
IN BOOL fRecover
)
{
HRESULT hr = NOERROR;
WamRegMetabaseConfig MDConfig;
DBG_ASSERT(szMetabasePath);
if (SUCCEEDED(hr))
{
MDPropItem rgProp[IWMDP_MAX];
MDConfig.InitPropItemData(&rgProp[0]);
// Update APPRoot
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_ROOT, szMetabasePath);
//Update AppIsolated
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_ISOLATED,
(fInProc) ? static_cast<DWORD>(eAppRunInProc)
: static_cast<DWORD>(eAppRunOutProcInDefaultPool));
//
// in case this is an recover operation, we do not remove App Friendly Name.
//
if (!fRecover)
{
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_FRIENDLY_NAME, L"");
}
hr = MDConfig.UpdateMD( rgProp, METADATA_INHERIT, szMetabasePath );
}
if (FAILED(hr))
{
HRESULT hrT = NOERROR;
DBGPRINTF((DBG_CONTEXT, "Failed to create in proc application. path = %S, error = %08x\n",
szMetabasePath,
hr));
}
return hr;
}
/*===================================================================
CreateOutProcApp
Create an out prop application.
Parameter:
szMetabasePath: [in] MetabaseKey.
fRecover [in] if TRUE, we recover/recreate the application.
fSaveMB [in] if TRUE, save metabase immediately
Return: HRESULT
===================================================================*/
HRESULT WamRegGlobal::CreateOutProcApp(
IN LPCWSTR szMetabasePath,
IN BOOL fRecover, /* = FALSE */
IN BOOL fSaveMB /* = TRUE */
)
{
WCHAR *szWAMCLSID = NULL;
WCHAR *szPackageID = NULL;
WCHAR *szNameForNewPackage = NULL;
HRESULT hr;
HRESULT hrRegister = E_FAIL;
HRESULT hrPackage = E_FAIL;
HRESULT hrMetabase = E_FAIL;
WCHAR szIdentity[MAX_PATH];
WCHAR szPwd[MAX_PATH];
WamRegMetabaseConfig MDConfig;
WamRegPackageConfig PackageConfig;
DBG_ASSERT(szMetabasePath);
hr = GetNewSzGUID(&szWAMCLSID);
if (FAILED(hr))
{
DBGPRINTF((DBG_CONTEXT, "Failed to create a new szGUID. error = %08x\n", hr));
return hr;
}
else
{
WCHAR *szProgID = NULL;
// Do WAM CLSID registration
//
hr = SzWamProgID(szMetabasePath, &szProgID);
if (FAILED(hr))
{
DBGPRINTF((DBG_CONTEXT, "Failed to Create WAM ProgID, hr = %08x\n",
hr));
}
else
{
hr = g_RegistryConfig.RegisterCLSID(szWAMCLSID, szProgID, FALSE);
hrRegister = hr;
delete [] szProgID;
szProgID = NULL;
if (FAILED(hrRegister))
{
DBGPRINTF((DBG_CONTEXT, "Failed to registerCLSID. error %08x\n", hr));
}
}
}
if (SUCCEEDED(hr))
{
WCHAR szLastOutProcPackageID[uSizeCLSID];
//
// When an outproc package gets deleted, it might/might not removed from the MTS,
// the next time, same app path marked as out-proc again, we try to reuse the outproc
// package. Therefore, we need to save the OutprogPackageID as LastOutProcPackageID.
//
szLastOutProcPackageID[0] = NULL;
MDConfig.MDGetLastOutProcPackageID(szMetabasePath, szLastOutProcPackageID);
if (szLastOutProcPackageID[0] == NULL)
{
hr = GetNewSzGUID(&szPackageID);
}
else
{
szPackageID = (WCHAR *)CoTaskMemAlloc(uSizeCLSID*sizeof(WCHAR));
if (szPackageID == NULL)
{
hr = E_OUTOFMEMORY;
}
else
{
wcsncpy(szPackageID, szLastOutProcPackageID, uSizeCLSID);
}
}
}
if (SUCCEEDED(hr))
{
hr = GetViperPackageName(szMetabasePath, &szNameForNewPackage);
}
if (SUCCEEDED(hr))
{
hr = MDConfig.MDGetIdentity(szIdentity, sizeof(szIdentity), szPwd, sizeof(szPwd));
}
if (SUCCEEDED(hr))
{
//
// Create the catalog object etc for MTS configuration
//
hr = PackageConfig.CreateCatalog();
if (FAILED(hr))
{
DBGPRINTF((DBG_CONTEXT, "Failed to call MTS Admin API. error %08x\n", hr));
}
else
{
hr = PackageConfig.CreatePackage(
szPackageID,
szNameForNewPackage,
szIdentity,
szPwd );
if (SUCCEEDED(hr))
{
hr = PackageConfig.AddComponentToPackage(
szPackageID,
szWAMCLSID);
if (FAILED(hr))
{
PackageConfig.RemovePackage(szPackageID);
}
}
}
hrPackage = hr;
}
if (SUCCEEDED(hr))
{
MDPropItem rgProp[IWMDP_MAX];
MDConfig.InitPropItemData(&rgProp[0]);
// Update WAMCLSID
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_WAMCLSID, szWAMCLSID);
// Update APPRoot
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_ROOT, szMetabasePath);
//Update AppIsolated
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_ISOLATED, 1);
//Update AppPackageName
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_PACKAGE_NAME, szNameForNewPackage);
//Update AppPackageID
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_PACKAGEID, szPackageID);
//
// in case this is an recover operation, we do not remove App Friendly Name.
//
if (!fRecover)
{
// It doesn't really make sense to set this on every isolated application.
// This will be much easier to administer globally if we allow it to be set
// at a higher level.
// MDConfig.MDSetPropItem(&rgProp[0], IWMDP_OOP_RECOVERLIMIT, APP_OOP_RECOVER_LIMIT_DEFAULT);
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_FRIENDLY_NAME, L"");
}
// Attempt to Save the metabase changes immediately. We want to ensure
// that the COM+ package is not orphaned.
hr = MDConfig.UpdateMD(rgProp, METADATA_INHERIT, szMetabasePath, fSaveMB );
if (FAILED(hr))
{
// Removed AbortUpdateMD call - we shouldn't wax the MB settings or
// we will orphan the COM+ package.
DBGPRINTF((
DBG_CONTEXT,
"Failed to set metabase properties on (%S). error == %08x\n",
szMetabasePath,
hr
));
}
hrMetabase = hr;
}
if (FAILED(hr))
{
HRESULT hrT = NOERROR;
DBGPRINTF((DBG_CONTEXT, "Failed to create out proc application. path = %S, error = %08x\n",
szMetabasePath,
hr));
if (SUCCEEDED(hrPackage))
{
hrT = PackageConfig.RemovePackage( szPackageID);
}
if (SUCCEEDED(hrRegister))
{
hrT = g_RegistryConfig.UnRegisterCLSID(szWAMCLSID, FALSE);
if (FAILED(hrT))
{
DBGPRINTF((DBG_CONTEXT, "Rollback: Failed to UnRegisterCLSID. error = %08x\n", hr));
}
}
}
if (szWAMCLSID)
{
CoTaskMemFree(szWAMCLSID);
szWAMCLSID = NULL;
}
if (szPackageID)
{
CoTaskMemFree(szPackageID);
szWAMCLSID = NULL;
}
if (szNameForNewPackage)
{
delete [] szNameForNewPackage;
szNameForNewPackage = NULL;
}
return hr;
}
HRESULT
WamRegGlobal::CreateOutProcAppReplica(
IN LPCWSTR szMetabasePath,
IN LPCWSTR szAppName,
IN LPCWSTR szWamClsid,
IN LPCWSTR szAppId
)
/*++
Function:
Called by the DeSerialize replication method to create a
new oop application.
Arguments:
szMetabasePath
szAppName
szWamClsid
szAppId
Return:
--*/
{
HRESULT hr = NOERROR;
DBG_ASSERT(szMetabasePath);
DBG_ASSERT(szWamClsid);
DBG_ASSERT(szAppId);
//
// Register wam.dll as a new component
//
WCHAR * szProgID = NULL;
BOOL fRegisteredWam = FALSE;
hr = SzWamProgID(szMetabasePath, &szProgID);
if( SUCCEEDED(hr) )
{
DBG_ASSERT( szProgID != NULL );
hr = g_RegistryConfig.RegisterCLSID( szWamClsid,
szProgID,
FALSE
);
if( SUCCEEDED(hr) )
{
fRegisteredWam = TRUE;
}
delete [] szProgID;
szProgID = NULL;
}
if( FAILED(hr) )
{
DBGERROR(( DBG_CONTEXT,
"Failed to register wam.dll. hr=%08x\n",
hr
));
}
//
// Get required application info
//
BOOL fGetCOMAppInfo = FALSE;
WCHAR szIdentity[MAX_PATH];
WCHAR szPwd[MAX_PATH];
WamRegMetabaseConfig MDConfig;
if( fRegisteredWam )
{
hr = MDConfig.MDGetIdentity( szIdentity,
sizeof(szIdentity),
szPwd,
sizeof(szPwd)
);
if( SUCCEEDED(hr) )
{
fGetCOMAppInfo = TRUE;
}
else
{
DBGERROR(( DBG_CONTEXT,
"Failed get required COM application info. hr=%08x\n",
hr
));
}
}
//
// Create the COM+ application
//
if( fGetCOMAppInfo )
{
WamRegPackageConfig PackageConfig;
hr = PackageConfig.CreateCatalog();
if( SUCCEEDED(hr) )
{
hr = PackageConfig.CreatePackage(
szAppId,
szAppName,
szIdentity,
szPwd );
if( SUCCEEDED(hr) )
{
hr = PackageConfig.AddComponentToPackage(
szAppId,
szWamClsid
);
}
// On failure we might want to cleanup, but I'm not sure
// that really makes sense.
}
// At this point, we would normally set the metabase properties
// but we will let the MB replication handle that for us.
// Note: if the MB replication fails, we will be left with
// a bunch of orphaned com applications
if( FAILED(hr) )
{
DBGERROR(( DBG_CONTEXT,
"COM+ App creation failed. AppId(%S) Name(%S) "
"Clsid(%S) hr=%08x\n",
szAppId,
szAppName,
szWamClsid,
hr
));
}
}
return hr;
}
/*===================================================================
DeleteApp
Register a WAM CLSID.
Parameter:
szMetabasePath: [in] MetabaseKey.
fDeletePackage: [in] when uninstall, this flag is true, we delete all IIS created packages.
fRemoveAppPool [in] Should the AppPoolId Property be removed
Return: HRESULT
===================================================================*/
HRESULT WamRegGlobal::DeleteApp
(
IN LPCWSTR szMetabasePath,
IN BOOL fRecover,
IN BOOL fRemoveAppPool
)
{
WCHAR szWAMCLSID[uSizeCLSID];
WCHAR szPackageID[uSizeCLSID];
DWORD dwAppMode = eAppRunInProc;
DWORD dwCallBack;
HRESULT hr, hrRegistry;
METADATA_HANDLE hMetabase = NULL;
WamRegMetabaseConfig MDConfig;
hr = MDConfig.MDGetDWORD(szMetabasePath, MD_APP_ISOLATED, &dwAppMode);
// return immediately, no application is defined, nothing to delete.
if (hr == MD_ERROR_DATA_NOT_FOUND)
{
return NOERROR;
}
if (FAILED(hr))
{
return hr;
}
//
//Set App State to be PAUSE/DISABLE, so that after we remove the application from
//run time WAM_DICTATOR lookup table, new request won't trigger the application to
//retstart.
//WAM_DICTATOR has code to check for this state.
//
hr = MDConfig.MDSetAppState(szMetabasePath, APPSTATUS_PAUSE);
if( FAILED(hr) )
{
DBGPRINTF(( DBG_CONTEXT,
"MDSetAppState Failed hr=%08x\n",
hr
));
}
hr = W3ServiceUtil(szMetabasePath, APPCMD_DELETE, &dwCallBack);
if( FAILED(hr) )
{
DBGPRINTF(( DBG_CONTEXT,
"W3ServiceUtil APPCMD_DELETE Failed on (%S) hr=%08x\n",
szMetabasePath,
hr
));
}
if (dwAppMode == eAppRunOutProcIsolated)
{
// Get WAM_CLSID, and PackageID.
hr = MDConfig.MDGetIDs(szMetabasePath, szWAMCLSID, szPackageID, dwAppMode);
// Remove the WAM from the package.
if (SUCCEEDED(hr))
{
WamRegPackageConfig PackageConfig;
HRESULT hrPackage;
hr = PackageConfig.CreateCatalog();
if ( FAILED( hr))
{
DBGPRINTF(( DBG_CONTEXT,
"Failed to Create MTS catalog hr=%08x\n",
hr));
}
else
{
hr = PackageConfig.RemoveComponentFromPackage(szPackageID,
szWAMCLSID,
dwAppMode);
if (FAILED(hr))
{
DBGPRINTF((DBG_CONTEXT, "Failed to remove component from package, \npackageid = %S, wamclsid = %S, hr = %08x\n",
szPackageID,
szWAMCLSID,
hr));
}
}
hrPackage = hr;
}
// Unregister WAM
hr = g_RegistryConfig.UnRegisterCLSID(szWAMCLSID, FALSE);
if (FAILED(hr))
{
DBGPRINTF((DBG_CONTEXT, "Failed to UnRegister WAMCLSID(%S), hr = %08x\n",
szWAMCLSID,
hr));
hrRegistry = hr;
}
}
if (SUCCEEDED(hr))
{
BOOL fChanged = FALSE;
MDPropItem rgProp[IWMDP_MAX];
MDConfig.InitPropItemData(&rgProp[0]);
if (dwAppMode == static_cast<DWORD>(eAppRunOutProcIsolated))
{
// Delete AppPackageName. (Inherited from W3SVC)
// Delete AppPackageID. (Inherited from W3SVC)
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_PACKAGE_NAME);
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_PACKAGEID);
// Delete WAMCLSID
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_WAMCLSID);
if (TsIsNtServer() || TsIsNtWksta())
{
MDConfig.MDSetPropItem(&rgProp[0], IWMDP_LAST_OUTPROC_PID, szPackageID);
}
fChanged = TRUE;
}
// If this is DeleteRecoverable mode, we do not delete APP_ROOT, APP_ISOLATED,
// OOP_RECOVERLIMIT and APP_STATE.
if (!fRecover)
{
// Delete AppFriendlyName
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_FRIENDLY_NAME);
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_ROOT);
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_ISOLATED);
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_APPSTATE);
if (fRemoveAppPool)
{
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_OOP_APP_APPPOOL_ID);
}
if (dwAppMode == static_cast<DWORD>(eAppRunOutProcIsolated))
{
// This will only be set for older isolated applications.
// Since we ignore the result of UpdateMD below, it is ok
// for us to try to delete the property.
MDConfig.MDDeletePropItem(&rgProp[0], IWMDP_OOP_RECOVERLIMIT);
}
fChanged = TRUE;
}
// For DeleteRecover operation, and the app is not outproc isolated,
// No property changes, therefore, no need to update metabase.
if (fChanged)
{
MDConfig.UpdateMD(rgProp, METADATA_NO_ATTRIBUTES, szMetabasePath);
}
}
return NOERROR;
}
/*===================================================================
RecoverApp
Recover an application based on MD_APP_ISOLATED property.
Parameter:
szMetabasePath: [in] MetabaseKey.
Return: HRESULT
===================================================================*/
HRESULT WamRegGlobal::RecoverApp
(
IN LPCWSTR szMetabasePath,
IN BOOL fForceRecover
)
{
HRESULT hr = NOERROR;
DWORD dwAppMode = 0;
WamRegMetabaseConfig MDConfig;
hr = MDConfig.MDGetDWORD(szMetabasePath, MD_APP_ISOLATED, &dwAppMode);
if (hr == MD_ERROR_DATA_NOT_FOUND)
{
hr = NOERROR;
}
else
{
if (SUCCEEDED(hr))
{
if (fForceRecover)
{
if (dwAppMode == static_cast<DWORD>(eAppRunOutProcInDefaultPool))
{
hr = CreatePooledApp(szMetabasePath, FALSE);
}
else if (dwAppMode == static_cast<DWORD>(eAppRunInProc))
{
hr = CreatePooledApp(szMetabasePath, TRUE);
}
else
{
hr = CreateOutProcApp(szMetabasePath);
}
}
if (SUCCEEDED(hr))
{
HRESULT hrT = NOERROR;
hrT = MDConfig.MDRemoveProperty(szMetabasePath, MD_APP_STATE, DWORD_METADATA);
if (FAILED(hrT))
{
if (hrT != MD_ERROR_DATA_NOT_FOUND)
{
DBGPRINTF((DBG_CONTEXT, "Failed to remove MD_APP_STATE from path %S, hr = %08x\n",
szMetabasePath,
hrT));
}
}
}
}
}
return hr;
}
/*============================================================================
W3ServiceUtil
sink function that unload/shutdown/querystatus of an application currently in the runtime
table.
Parameter:
szMDPath the application Path.
dwCommand The command.
pdwCallBackResult Contains the HRESULT from w3svc.dll.
==============================================================================*/
HRESULT WamRegGlobal::W3ServiceUtil
(
IN LPCWSTR szMDPath,
IN DWORD dwCommand,
OUT DWORD* pdwCallBackResult
)
{
HRESULT hr = NOERROR;
if (g_pfnW3ServiceSink)
{
#ifndef _IIS_6_0
// DBCS enabling for IIS 5.1
INT cSize = wcslen(szMDPath)*2 + 1;
CHAR *szPathT = new CHAR[cSize];
if (szPathT)
{
WideCharToMultiByte(0, 0, szMDPath, -1, szPathT, cSize, NULL, NULL);
hr = g_pfnW3ServiceSink(szPathT,
dwCommand,
pdwCallBackResult);
}
else
{
hr = E_OUTOFMEMORY;
}
delete [] szPathT;
#else
//
// IIS 6's implementation uses UNICODE directly, so
// we'll avoid the WideCharToMultiByte nonsense and
// just cast the path to fit the function arguments.
//
// We're not changing the function prototype purely
// because we are minimizing code churn in this module.
//
// IIS 6 gets the unicode directly
hr = g_pfnW3ServiceSink(reinterpret_cast<LPCSTR>(szMDPath),
dwCommand,
pdwCallBackResult);
#endif // _IIS_6_0
}
else
{
*pdwCallBackResult = APPSTATUS_NOW3SVC;
hr = NOERROR;
}
return hr;
}