/*++ Copyright (c) 2000 Microsoft Corporation Module Name: schemaextensions.cpp Abstract: This file contains the implementation of the CSchemaExtensions class. This is the only class that talks to the catalog. Author: MarcelV Revision History: Mohit Srivastava 28-Nov-00 --*/ #include "iisprov.h" #include "schemaextensions.h" #include "metabase.hxx" LPWSTR g_wszDatabaseName = L"Metabase"; HRESULT GetMetabasePath(LPTSTR io_tszPath) /*++ Synopsis: This beast was copied and modified from \%sdxroot%\iis\svcs\infocomm\metadata\dll\metasub.cxx Arguments: [io_tszPath] - must be at least size MAX_PATH Return Value: --*/ { DBG_ASSERT(io_tszPath != NULL); HRESULT hresReturn = HRESULT_FROM_WIN32(ERROR_PATH_NOT_FOUND); TCHAR tszBuffer[MAX_PATH] = {0}; HKEY hkRegistryKey = NULL; DWORD dwRegReturn; DWORD dwType; DWORD dwSize = MAX_PATH * sizeof(TCHAR); dwRegReturn = RegOpenKey( HKEY_LOCAL_MACHINE, SETUP_REG_KEY, // TEXT("SOFTWARE\\Microsoft\\InetStp") &hkRegistryKey); if (dwRegReturn == ERROR_SUCCESS) { dwSize = MAX_PATH * sizeof(TCHAR); dwRegReturn = RegQueryValueEx(hkRegistryKey, INSTALL_PATH_VALUE, NULL, &dwType, (BYTE *) tszBuffer, &dwSize); if ((dwRegReturn == ERROR_SUCCESS) && dwType == (REG_SZ)) { hresReturn = HRESULT_FROM_WIN32(dwRegReturn); } RegCloseKey( hkRegistryKey ); } else { hresReturn = HRESULT_FROM_WIN32(dwRegReturn); } _tcscpy(io_tszPath, tszBuffer); if(FAILED(hresReturn)) { DBGPRINTF((DBG_CONTEXT, "Could not get metabase path, hr=0x%x\n", hresReturn)); } return hresReturn; } int __cdecl CompDBNames (const void * pDBMetaLHS, const void * pDBMetaRHS) { const tDATABASEMETARow *pLHS = static_cast (pDBMetaLHS ); const tDATABASEMETARow *pRHS = static_cast (pDBMetaRHS ); return _wcsicmp (pLHS->pInternalName, pRHS->pInternalName); } int __cdecl CompTableMeta (const void * pTableMetaLHS, const void * pTableMetaRHS) { const CTableMeta *pLHS = static_cast (pTableMetaLHS ); const CTableMeta *pRHS = static_cast (pTableMetaRHS ); return _wcsicmp (pLHS->TableMeta.pInternalName, pRHS->TableMeta.pInternalName); } int __cdecl CompTableDBName (const void * pTableMetaLHS, const void * pTableMetaRHS) { const CTableMeta *pLHS = static_cast (pTableMetaLHS ); const CTableMeta *pRHS = static_cast (pTableMetaRHS ); return _wcsicmp (pLHS->TableMeta.pDatabase, pRHS->TableMeta.pDatabase); } // sorted by table name and index int __cdecl CompColumnMetas (const void * pColumnMetaLHS, const void * pColumnMetaRHS) { const CColumnMeta *pLHS = static_cast (pColumnMetaLHS ); const CColumnMeta *pRHS = static_cast (pColumnMetaRHS ); int iCmp = _wcsicmp (pLHS->ColumnMeta.pTable, pRHS->ColumnMeta.pTable); if (iCmp != 0) { return iCmp; } return (*pLHS->ColumnMeta.pIndex) - (*pRHS->ColumnMeta.pIndex); } // sorted by table name and index int __cdecl CompTagMetas (const void * pTagMetaLHS, const void * pTagMetaRHS) { const tTAGMETARow *pLHS = static_cast (pTagMetaLHS ); const tTAGMETARow *pRHS = static_cast (pTagMetaRHS ); int iCmp = _wcsicmp (pLHS->pTable, pRHS->pTable); if (iCmp != 0) { return iCmp; } int iResult = (*pLHS->pColumnIndex) - (*pRHS->pColumnIndex); if (iResult != 0) { return iResult; } return (*pLHS->pValue) - (*pRHS->pValue); } CSchemaExtensions::CSchemaExtensions () { m_paTableMetas = 0; m_cNrTables = 0; m_paColumnMetas = 0; m_cNrColumns = 0; m_paTags = 0; m_cNrTags = 0; m_pQueryCells = 0; m_cQueryCells = 0; m_wszBinFileName = 0; m_tszBinFilePath = 0; m_bBinFileLoaded = false; } CSchemaExtensions::~CSchemaExtensions() { if(m_bBinFileLoaded) { m_spIMbSchemaComp->ReleaseBinFileName(m_wszBinFileName); } delete [] m_paTableMetas; delete [] m_paColumnMetas; delete [] m_paTags; delete [] m_pQueryCells; delete [] m_wszBinFileName; delete [] m_tszBinFilePath; } HRESULT CSchemaExtensions::Initialize(bool i_bUseExtensions) { HRESULT hr = S_OK; ULONG cch = 0; // // Get the dispenser // hr = DllGetSimpleObjectByIDEx(eSERVERWIRINGMETA_TableDispenser, IID_ISimpleTableDispenser2, (VOID**)&m_spDispenser, WSZ_PRODUCT_IIS); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Could not get dispenser, hr=0x%x\n", hr)); return hr; } // // Override the default logging mechanism // CComPtr spDispAdv; hr = m_spDispenser->QueryInterface( IID_IAdvancedTableDispenser, reinterpret_cast(&spDispAdv)); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Couldn't get Advanced Dispenser interface [0x%x]\n", hr)); return hr; } CComPtr spNullLogger = new NULL_Logger(); if(!spNullLogger) { DBGPRINTF((DBG_CONTEXT, "Out of memory\n")); return hr; } hr = spDispAdv->SetCatalogErrorLogger(spNullLogger); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Couldn't set logger [0x%x]\n", hr)); return hr; } // // Get the schema compiler interface from the dispenser which will help us // get the bin file name. // hr = m_spDispenser->QueryInterface(IID_IMetabaseSchemaCompiler, (LPVOID*)&m_spIMbSchemaComp); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Couldn't get SchemaCompiler interface, hr=0x%x\n", hr)); return hr; } // // Get the path of mbschema.xml // m_tszBinFilePath = new TCHAR[MAX_PATH+1]; if(m_tszBinFilePath == NULL) { return E_OUTOFMEMORY; } hr = GetMetabasePath(m_tszBinFilePath); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Couldn't get metabase path, hr=0x%x\n", hr)); return hr; } // // Convert it to Unicode, Set Bin Path // #ifndef UNICODE // we want to convert an MBCS string in lpszA int nLen = MultiByteToWideChar(CP_ACP, 0,m_tszBinFilePath, -1, NULL, NULL); LPWSTR lpszW = new WCHAR[nLen]; if(lpszW == NULL) { return E_OUTOFMEMORY; } if(MultiByteToWideChar(CP_ACP, 0, m_tszBinFilePath, -1, lpszW, nLen) == 0) { delete [] lpszW; hr = GetLastError(); return HRESULT_FROM_WIN32(hr); } hr = m_spIMbSchemaComp->SetBinPath(lpszW); delete [] lpszW; #else hr = m_spIMbSchemaComp->SetBinPath(m_tszBinFilePath); #endif if(FAILED(hr)) { return hr; } // // Get Bin FileName // hr = m_spIMbSchemaComp->GetBinFileName(NULL, &cch); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Couldn't get schema bin filename size, hr=0x%x\n", hr)); return hr; } m_wszBinFileName = new WCHAR[cch+1]; if(m_wszBinFileName == NULL) { return E_OUTOFMEMORY; } hr = m_spIMbSchemaComp->GetBinFileName(m_wszBinFileName, &cch); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "Couldn't get schema bin filename, hr=0x%x\n", hr)); return hr; } m_bBinFileLoaded = true; // // Set up the query cells // m_cQueryCells = 2; m_pQueryCells = new STQueryCell[m_cQueryCells]; if(m_pQueryCells == NULL) { return E_OUTOFMEMORY; } if(i_bUseExtensions) { m_pQueryCells[0].pData = (LPVOID)m_wszBinFileName; } else { m_pQueryCells[0].pData = (LPVOID)NULL; } m_pQueryCells[0].eOperator = eST_OP_EQUAL; m_pQueryCells[0].iCell = iST_CELL_SCHEMAFILE; m_pQueryCells[0].dbType = DBTYPE_WSTR; m_pQueryCells[0].cbSize = 0; m_pQueryCells[1].pData = (void *) g_wszDatabaseName; m_pQueryCells[1].eOperator = eST_OP_EQUAL; m_pQueryCells[1].iCell = iCOLLECTION_META_Database; m_pQueryCells[1].dbType = DBTYPE_WSTR; m_pQueryCells[1].cbSize = 0; hr = GenerateIt(); if(FAILED(hr)) { DBGPRINTF((DBG_CONTEXT, "GenerateIt failed, hr=0x%x\n", hr)); return hr; } return hr; } HRESULT CSchemaExtensions::GetMbSchemaTimeStamp( FILETIME* io_pFileTime) const { DBG_ASSERT(io_pFileTime != NULL); DBG_ASSERT(m_tszBinFilePath != NULL); HRESULT hr = S_OK; ULONG cchBinFilePath = _tcslen(m_tszBinFilePath); ULONG cchSchFileName = _tcslen(MD_SCHEMA_FILE_NAME); TCHAR* tszPathPlusName = new TCHAR[cchBinFilePath+1+cchSchFileName+1]; TCHAR* tszCurPos = tszPathPlusName; if(tszPathPlusName == NULL) { return E_OUTOFMEMORY; } // // Copy the path // tszCurPos = tszPathPlusName; memcpy(tszCurPos, m_tszBinFilePath, sizeof(TCHAR)*(cchBinFilePath+1)); // // Concat a \ if necessary, and the filename, // tszCurPos = tszPathPlusName + cchBinFilePath; if(m_tszBinFilePath[cchBinFilePath-1] != TEXT('\\')) { memcpy(tszCurPos, TEXT("\\"), sizeof(TCHAR)*2); tszCurPos++; } memcpy(tszCurPos, MD_SCHEMA_FILE_NAME, sizeof(TCHAR)*(cchSchFileName+1)); // // Now get the file info // { WIN32_FIND_DATA FindFileData; memset(&FindFileData, 0, sizeof(WIN32_FIND_DATA)); HANDLE hFindFile = FindFirstFile(tszPathPlusName, &FindFileData); if(hFindFile == INVALID_HANDLE_VALUE) { hr = GetLastError(); hr = HRESULT_FROM_WIN32(hr); goto exit; } FindClose(hFindFile); // // Set out parameters if everything succeeded // memcpy(io_pFileTime, &FindFileData.ftLastWriteTime, sizeof(FILETIME)); } exit: delete [] tszPathPlusName; return hr; } HRESULT CSchemaExtensions::GenerateIt () { HRESULT hr = S_OK; hr = GetTables (); if (FAILED (hr)) { return hr; } hr = GetColumns (); if (FAILED (hr)) { return hr; } hr = GetTags (); if (FAILED (hr)) { return hr; } hr = BuildInternalStructures (); if (FAILED (hr)) { return hr; } return hr; } HRESULT CSchemaExtensions::GetTables () { HRESULT hr = S_OK; ULONG one = 1; hr = m_spDispenser->GetTable (wszDATABASE_META, wszTABLE_TABLEMETA, m_pQueryCells, (void *)&one, eST_QUERYFORMAT_CELLS, 0, (void **) &m_spISTTableMeta); if (FAILED (hr)) { return hr; } hr = m_spISTTableMeta->GetTableMeta (0, 0, &m_cNrTables, 0); if (FAILED (hr)) { return hr; } if (m_cNrTables == 0) { return S_OK; } m_paTableMetas = new CTableMeta [m_cNrTables]; if (m_paTableMetas == 0) { return E_OUTOFMEMORY; } for (ULONG idx =0; idx < m_cNrTables; ++idx) { hr = m_spISTTableMeta->GetColumnValues (idx, sizeof (tTABLEMETARow)/sizeof (ULONG *), 0, 0, (void **) &m_paTableMetas[idx].TableMeta); if (FAILED (hr)) { return hr; } if (m_paTableMetas[idx].ColCount () > 0) { // set number of columns m_paTableMetas[idx].paColumns = new LPCColumnMeta[m_paTableMetas[idx].ColCount()]; if (m_paTableMetas[idx].paColumns == 0) { return E_OUTOFMEMORY; } } } // and sort them by table name qsort (m_paTableMetas, m_cNrTables, sizeof (CTableMeta), CompTableMeta); return hr; } HRESULT CSchemaExtensions::GetColumns () { HRESULT hr = S_OK; ULONG one = 1; hr = m_spDispenser->GetTable (wszDATABASE_META, wszTABLE_COLUMNMETA, m_pQueryCells, (void *)&one, eST_QUERYFORMAT_CELLS, 0, (void **) &m_spISTColumnMeta); if (FAILED (hr)) { return hr; } hr = m_spISTColumnMeta->GetTableMeta (0, 0, &m_cNrColumns, 0); if (FAILED (hr)) { return hr; } if (m_cNrColumns == 0) { return E_FAIL; } m_paColumnMetas = new CColumnMeta[m_cNrColumns]; if (m_paColumnMetas == 0) { return E_OUTOFMEMORY; } ULONG acbSizes[cCOLUMNMETA_NumberOfColumns]; for (ULONG idx =0; idx < m_cNrColumns; ++idx) { hr = m_spISTColumnMeta->GetColumnValues (idx, sizeof (tCOLUMNMETARow)/sizeof (ULONG *), 0, acbSizes, (void **) &m_paColumnMetas[idx].ColumnMeta); m_paColumnMetas[idx].cbDefaultValue = acbSizes[iCOLUMNMETA_DefaultValue]; if (FAILED (hr)) { return hr; } } qsort (m_paColumnMetas, m_cNrColumns, sizeof (CColumnMeta), CompColumnMetas); return hr; } HRESULT CSchemaExtensions::GetTags () { HRESULT hr = S_OK; ULONG one = 1; hr = m_spDispenser->GetTable (wszDATABASE_META, wszTABLE_TAGMETA, m_pQueryCells, (void *)&one, eST_QUERYFORMAT_CELLS, 0, (void **) &m_spISTTagMeta); if (FAILED (hr)) { return hr; } hr = m_spISTTagMeta->GetTableMeta (0, 0, &m_cNrTags, 0); if (FAILED (hr)) { return hr; } if (m_cNrTags == 0) { return E_FAIL; } m_paTags = new tTAGMETARow[m_cNrTags]; if (m_paTags == 0) { return E_OUTOFMEMORY; } for (ULONG idx =0; idx < m_cNrTags; ++idx) { hr = m_spISTTagMeta->GetColumnValues (idx, sizeof (tTAGMETARow)/sizeof (ULONG *), 0, 0, (void **) &m_paTags[idx]); if (FAILED (hr)) { return hr; } } qsort (m_paTags, m_cNrTags, sizeof (tTAGMETARow), CompTagMetas); return hr; } HRESULT CSchemaExtensions::BuildInternalStructures () { HRESULT hr = S_OK; // attach the tags to the tables ULONG idx = 0; while (idx < m_cNrTags) { // find the correct column CColumnMeta dummyColumnMeta; dummyColumnMeta.ColumnMeta.pTable = m_paTags[idx].pTable; dummyColumnMeta.ColumnMeta.pIndex = m_paTags[idx].pColumnIndex; // get column CColumnMeta *pColMeta = (CColumnMeta *) bsearch (&dummyColumnMeta, m_paColumnMetas, m_cNrColumns, sizeof (CColumnMeta), CompColumnMetas); DBG_ASSERT (pColMeta != NULL); if (NULL == pColMeta) { return E_FAIL; } DBG_ASSERT (wcscmp(pColMeta->ColumnMeta.pTable, m_paTags[idx].pTable) == 0 && *pColMeta->ColumnMeta.pIndex == *m_paTags[idx].pColumnIndex); // get count ULONG iStartIdx = idx; pColMeta->cNrTags = 1; idx++; // skip over this element while ((idx < m_cNrTags) && (wcscmp(pColMeta->ColumnMeta.pTable, m_paTags[idx].pTable) == 0) && (*pColMeta->ColumnMeta.pIndex == *m_paTags[idx].pColumnIndex)) { idx++; pColMeta->cNrTags += 1; } if (pColMeta->cNrTags > 0) { // allocate memory and copy the stuff pColMeta->paTags = new LPtTAGMETA[pColMeta->cNrTags]; if (pColMeta->paTags == 0) { return E_OUTOFMEMORY; } for (ULONG tagIdx = 0; tagIdx < pColMeta->cNrTags; ++tagIdx) { pColMeta->paTags[tagIdx] = &m_paTags[iStartIdx + tagIdx]; } } } // attach the columns to the tables for (idx=0; idx < m_cNrColumns; ++idx) { CTableMeta dummyTableMeta; dummyTableMeta.TableMeta.pInternalName = m_paColumnMetas[idx].ColumnMeta.pTable; // find table CTableMeta *pTableMeta = (CTableMeta *) bsearch (&dummyTableMeta, m_paTableMetas, m_cNrTables, sizeof (CTableMeta), CompTableMeta); DBG_ASSERT (pTableMeta != 0); DBG_ASSERT (wcscmp(pTableMeta->TableMeta.pInternalName, m_paColumnMetas[idx].ColumnMeta.pTable) == 0); // add Column to table ULONG iColumnIndex = *(m_paColumnMetas[idx].ColumnMeta.pIndex); DBG_ASSERT (iColumnIndex < pTableMeta->ColCount ()); pTableMeta->paColumns[iColumnIndex] = &m_paColumnMetas[idx]; } return hr; } CTableMeta* CSchemaExtensions::EnumTables(ULONG *io_idx) { DBG_ASSERT(io_idx != NULL); CTableMeta* pRet = NULL; while(1) { if(*io_idx < m_cNrTables) { pRet = &m_paTableMetas[*io_idx]; if( _wcsicmp(pRet->TableMeta.pDatabase, g_wszDatabaseName) == 0 && !(*pRet->TableMeta.pMetaFlags & fTABLEMETA_HIDDEN) ) { *io_idx = 1 + *io_idx; return pRet; } else { *io_idx = 1 + *io_idx; } } else { *io_idx = 1 + *io_idx; return NULL; } } return NULL; }