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.

252 lines
7.4 KiB

  1. // ServiceSpecification.cpp: implementation of the CServiceSpecification class.
  2. //
  3. // Copyright (c) 1997-2001 Microsoft Corporation, All Rights Reserved
  4. //
  5. //////////////////////////////////////////////////////////////////////
  6. #include "precomp.h"
  7. #include "ServiceSpecification.h"
  8. #include "ExtendString.h"
  9. #include "ExtendQuery.h"
  10. //////////////////////////////////////////////////////////////////////
  11. // Construction/Destruction
  12. //////////////////////////////////////////////////////////////////////
  13. CServiceSpecification::CServiceSpecification(CRequestObject *pObj, IWbemServices *pNamespace,
  14. IWbemContext *pCtx):CGenericClass(pObj, pNamespace, pCtx)
  15. {
  16. }
  17. CServiceSpecification::~CServiceSpecification()
  18. {
  19. }
  20. HRESULT CServiceSpecification::CreateObject(IWbemObjectSink *pHandler, ACTIONTYPE atAction)
  21. {
  22. HRESULT hr = WBEM_S_NO_ERROR;
  23. MSIHANDLE hView = NULL;
  24. MSIHANDLE hRecord = NULL;
  25. int i = -1;
  26. WCHAR wcBuf[BUFF_SIZE];
  27. WCHAR wcProductCode[39];
  28. DWORD dwBufSize;
  29. bool bMatch = false;
  30. UINT uiStatus;
  31. bool bGotID = false;
  32. WCHAR wcServiceInstall[BUFF_SIZE];
  33. WCHAR wcTestCode[39];
  34. //These will change from class to class
  35. bool bDiskID;
  36. SetSinglePropertyPath(L"CheckID");
  37. //improve getobject performance by optimizing the query
  38. if(atAction != ACTIONTYPE_ENUM)
  39. {
  40. // we are doing GetObject so we need to be reinitialized
  41. hr = WBEM_E_NOT_FOUND;
  42. BSTR bstrCompare;
  43. int iPos = -1;
  44. bstrCompare = SysAllocString ( L"CheckID" );
  45. if ( bstrCompare )
  46. {
  47. if(FindIn(m_pRequest->m_Property, bstrCompare, &iPos))
  48. {
  49. if ( ::SysStringLen ( m_pRequest->m_Value[iPos] ) < BUFF_SIZE )
  50. {
  51. //Get the action we're looking for
  52. wcscpy(wcBuf, m_pRequest->m_Value[iPos]);
  53. // safe operation if wcslen ( wcBuf ) > 38
  54. if ( wcslen ( wcBuf ) > 38 )
  55. {
  56. wcscpy(wcTestCode, &(wcBuf[(wcslen(wcBuf) - 38)]));
  57. }
  58. else
  59. {
  60. // we are not good to go, they have sent us longer string
  61. SysFreeString ( bstrCompare );
  62. throw hr;
  63. }
  64. // safe because lenght has been tested already in condition
  65. RemoveFinalGUID(m_pRequest->m_Value[iPos], wcServiceInstall);
  66. bGotID = true;
  67. }
  68. else
  69. {
  70. // we are not good to go, they have sent us longer string
  71. SysFreeString ( bstrCompare );
  72. throw hr;
  73. }
  74. }
  75. SysFreeString ( bstrCompare );
  76. }
  77. else
  78. {
  79. throw CHeap_Exception(CHeap_Exception::E_ALLOCATION_ERROR);
  80. }
  81. }
  82. Query wcQuery;
  83. wcQuery.Append ( 1, L"select distinct `ServiceInstall`, `Component_`, `Name`, `DisplayName`, `ServiceType`, `StartType`, `ErrorControl`, `LoadOrderGroup`, `Dependencies`, `StartName`, `Password` from ServiceInstall" );
  84. //optimize for GetObject
  85. if ( bGotID )
  86. {
  87. wcQuery.Append ( 3, L" where `ServiceInstall`=\'", wcServiceInstall, L"\'" );
  88. }
  89. LPWSTR Buffer = NULL;
  90. LPWSTR dynBuffer = NULL;
  91. DWORD dwDynBuffer = 0L;
  92. while(!bMatch && m_pRequest->Package(++i) && (hr != WBEM_E_CALL_CANCELLED))
  93. {
  94. // safe operation:
  95. // Package ( i ) returns NULL ( tested above ) or valid WCHAR [39]
  96. wcscpy(wcProductCode, m_pRequest->Package(i));
  97. if((atAction == ACTIONTYPE_ENUM) || (bGotID && (_wcsicmp(wcTestCode, wcProductCode) == 0))){
  98. //Open our database
  99. try
  100. {
  101. if ( GetView ( &hView, wcProductCode, wcQuery, L"ServiceInstall", TRUE, FALSE ) )
  102. {
  103. uiStatus = g_fpMsiViewFetch(hView, &hRecord);
  104. while(!bMatch && (uiStatus != ERROR_NO_MORE_ITEMS) && (hr != WBEM_E_CALL_CANCELLED)){
  105. CheckMSI(uiStatus);
  106. if(FAILED(hr = SpawnAnInstance(&m_pObj))) throw hr;
  107. //----------------------------------------------------
  108. dwBufSize = BUFF_SIZE;
  109. GetBufferToPut ( hRecord, 1, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, Buffer );
  110. PutProperty(m_pObj, pID, Buffer);
  111. PutKeyProperty ( m_pObj, pCheckID, Buffer, &bDiskID, m_pRequest, 1, wcProductCode );
  112. if ( dynBuffer && dynBuffer [ 0 ] != 0 )
  113. {
  114. dynBuffer [ 0 ] = 0;
  115. }
  116. //====================================================
  117. dwBufSize = BUFF_SIZE;
  118. GetBufferToPut ( hRecord, 2, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, Buffer );
  119. if ( ValidateComponentName ( msidata.GetDatabase(), wcProductCode, Buffer ) )
  120. {
  121. if ( dynBuffer && dynBuffer [ 0 ] != 0 )
  122. {
  123. dynBuffer [ 0 ] = 0;
  124. }
  125. dwBufSize = BUFF_SIZE;
  126. PutPropertySpecial ( hRecord, 3, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, pName );
  127. dwBufSize = BUFF_SIZE;
  128. PutPropertySpecial ( hRecord, 4, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, FALSE, 3, pDisplayName, pCaption, pDescription );
  129. PutProperty(m_pObj, pServiceType, g_fpMsiRecordGetInteger(hRecord, 5));
  130. PutProperty(m_pObj, pStartType, g_fpMsiRecordGetInteger(hRecord, 6));
  131. PutProperty(m_pObj, pErrorControl, g_fpMsiRecordGetInteger(hRecord, 7));
  132. dwBufSize = BUFF_SIZE;
  133. PutPropertySpecial ( hRecord, 8, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, pLoadOrderGroup );
  134. dwBufSize = BUFF_SIZE;
  135. PutPropertySpecial ( hRecord, 9, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, pDependencies );
  136. dwBufSize = BUFF_SIZE;
  137. PutPropertySpecial ( hRecord, 10, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, pStartName );
  138. dwBufSize = BUFF_SIZE;
  139. PutPropertySpecial ( hRecord, 11, dwBufSize, wcBuf, dwDynBuffer, dynBuffer, pPassword );
  140. //----------------------------------------------------
  141. if(bDiskID) bMatch = true;
  142. if((atAction != ACTIONTYPE_GET) || bMatch){
  143. hr = pHandler->Indicate(1, &m_pObj);
  144. }
  145. }
  146. else
  147. {
  148. if ( dynBuffer && dynBuffer [ 0 ] != 0 )
  149. {
  150. dynBuffer [ 0 ] = 0;
  151. }
  152. }
  153. m_pObj->Release();
  154. m_pObj = NULL;
  155. g_fpMsiCloseHandle(hRecord);
  156. uiStatus = g_fpMsiViewFetch(hView, &hRecord);
  157. }
  158. }
  159. }
  160. catch(...)
  161. {
  162. if ( dynBuffer )
  163. {
  164. delete [] dynBuffer;
  165. dynBuffer = NULL;
  166. }
  167. g_fpMsiCloseHandle(hRecord);
  168. g_fpMsiViewClose(hView);
  169. g_fpMsiCloseHandle(hView);
  170. msidata.CloseDatabase ();
  171. if(m_pObj)
  172. {
  173. m_pObj->Release();
  174. m_pObj = NULL;
  175. }
  176. throw;
  177. }
  178. g_fpMsiCloseHandle(hRecord);
  179. g_fpMsiViewClose(hView);
  180. g_fpMsiCloseHandle(hView);
  181. msidata.CloseDatabase ();
  182. }
  183. }
  184. if ( dynBuffer )
  185. {
  186. delete [] dynBuffer;
  187. dynBuffer = NULL;
  188. }
  189. return hr;
  190. }