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.

1068 lines
26 KiB

  1. //---------------------------------------------------------------------------
  2. //
  3. // Microsoft Windows
  4. // Copyright (C) Microsoft Corporation, 1992 - 1995
  5. //
  6. // File: cnamesp.cxx
  7. //
  8. // Contents: Windows NT 3.5 Namespace Object
  9. //
  10. //
  11. // History: 01-30-95 krishnag Created.
  12. //
  13. //----------------------------------------------------------------------------
  14. #include "winnt.hxx"
  15. #pragma hdrstop
  16. DEFINE_IDispatch_Delegating_Implementation(CWinNTNamespace)
  17. DEFINE_IADsExtension_Implementation(CWinNTNamespace)
  18. DEFINE_IADs_Implementation(CWinNTNamespace)
  19. // Class CWinNTNamespace
  20. CWinNTNamespace::CWinNTNamespace()
  21. {
  22. VariantInit(&_vFilter);
  23. ENLIST_TRACKING(CWinNTNamespace);
  24. }
  25. HRESULT
  26. CWinNTNamespace::CreateNamespace(
  27. BSTR Parent,
  28. BSTR NamespaceName,
  29. DWORD dwObjectState,
  30. REFIID riid,
  31. CWinNTCredentials& Credentials,
  32. void **ppvObj
  33. )
  34. {
  35. CWinNTNamespace FAR * pNamespace = NULL;
  36. HRESULT hr = S_OK;
  37. hr = AllocateNamespaceObject(&pNamespace);
  38. BAIL_ON_FAILURE(hr);
  39. hr = pNamespace->InitializeCoreObject(
  40. Parent,
  41. NamespaceName,
  42. NAMESPACE_CLASS_NAME,
  43. NO_SCHEMA,
  44. CLSID_WinNTNamespace,
  45. dwObjectState
  46. );
  47. BAIL_ON_FAILURE(hr);
  48. pNamespace->_Credentials = Credentials;
  49. // check if the call is from UMI
  50. if(Credentials.GetFlags() & ADS_AUTH_RESERVED) {
  51. //
  52. // we do not pass riid to InitUmiObject below. This is because UMI object
  53. // does not support IDispatch. There are several places in ADSI code where
  54. // riid passed into this function is defaulted to IID_IDispatch -
  55. // IADsContainer::Create for example. To handle these cases, we always
  56. // request IID_IUnknown from the UMI object. Subsequent code within UMI
  57. // will QI for the appropriate interface.
  58. //
  59. // Namespace objects have no components in their ADsPath. Just set the
  60. // class for identification.
  61. pNamespace->_CompClasses[0] = L"Namespace";
  62. hr = pNamespace->InitUmiObject(
  63. pNamespace->_Credentials,
  64. NULL,
  65. 0,
  66. NULL,
  67. (IUnknown *)(INonDelegatingUnknown *) pNamespace,
  68. NULL,
  69. IID_IUnknown,
  70. ppvObj
  71. );
  72. BAIL_ON_FAILURE(hr);
  73. //
  74. // UMI object was created and the interface was obtained successfully.
  75. // UMI object now has a reference to the inner unknown of IADs, since
  76. // the call to Release() below is not going to be made in this case.
  77. //
  78. RRETURN(hr);
  79. }
  80. hr = pNamespace->QueryInterface(riid, ppvObj);
  81. BAIL_ON_FAILURE(hr);
  82. pNamespace->Release();
  83. RRETURN(hr);
  84. error:
  85. delete pNamespace;
  86. RRETURN_EXP_IF_ERR(hr);
  87. }
  88. CWinNTNamespace::~CWinNTNamespace( )
  89. {
  90. VariantClear(&_vFilter);
  91. delete _pDispMgr;
  92. }
  93. //----------------------------------------------------------------------------
  94. // Function: QueryInterface
  95. //
  96. // Synopsis: If this object is aggregated within another object, then
  97. // all calls will delegate to the outer object. Otherwise, the
  98. // non-delegating QI is called
  99. //
  100. // Arguments:
  101. //
  102. // iid interface requested
  103. // ppInterface Returns pointer to interface requested. NULL if interface
  104. // is not supported.
  105. //
  106. // Returns: S_OK on success. Error code otherwise.
  107. //
  108. // Modifies: *ppInterface to return interface pointer
  109. //
  110. //----------------------------------------------------------------------------
  111. STDMETHODIMP CWinNTNamespace::QueryInterface(
  112. REFIID iid,
  113. LPVOID *ppInterface
  114. )
  115. {
  116. if(_pUnkOuter != NULL)
  117. RRETURN(_pUnkOuter->QueryInterface(
  118. iid,
  119. ppInterface
  120. ));
  121. RRETURN(NonDelegatingQueryInterface(
  122. iid,
  123. ppInterface
  124. ));
  125. }
  126. //----------------------------------------------------------------------------
  127. // Function: AddRef
  128. //
  129. // Synopsis: IUnknown::AddRef. If this object is aggregated within
  130. // another, all calls will delegate to the outer object.
  131. // Otherwise, the non-delegating AddRef is called
  132. //
  133. // Arguments:
  134. //
  135. // None
  136. //
  137. // Returns: New reference count
  138. //
  139. // Modifies: Nothing
  140. //
  141. //----------------------------------------------------------------------------
  142. STDMETHODIMP_(ULONG) CWinNTNamespace::AddRef(void)
  143. {
  144. if(_pUnkOuter != NULL)
  145. RRETURN(_pUnkOuter->AddRef());
  146. RRETURN(NonDelegatingAddRef());
  147. }
  148. //----------------------------------------------------------------------------
  149. // Function: Release
  150. //
  151. // Synopsis: IUnknown::Release. If this object is aggregated within
  152. // another, all calls will delegate to the outer object.
  153. // Otherwise, the non-delegating Release is called
  154. //
  155. // Arguments:
  156. //
  157. // None
  158. //
  159. // Returns: New reference count
  160. //
  161. // Modifies: Nothing
  162. //
  163. //----------------------------------------------------------------------------
  164. STDMETHODIMP_(ULONG) CWinNTNamespace::Release(void)
  165. {
  166. if(_pUnkOuter != NULL)
  167. RRETURN(_pUnkOuter->Release());
  168. RRETURN(NonDelegatingRelease());
  169. }
  170. //----------------------------------------------------------------------------
  171. STDMETHODIMP
  172. CWinNTNamespace::NonDelegatingQueryInterface(REFIID iid, LPVOID FAR* ppv)
  173. {
  174. if (ppv == NULL) {
  175. RRETURN(E_POINTER);
  176. }
  177. if (IsEqualIID(iid, IID_IUnknown))
  178. {
  179. *ppv = (IADs FAR *)this;
  180. }else if (IsEqualIID(iid, IID_IDispatch))
  181. {
  182. *ppv = (IADs FAR *)this;
  183. }
  184. else if (IsEqualIID(iid, IID_ISupportErrorInfo))
  185. {
  186. *ppv = (ISupportErrorInfo FAR *) this;
  187. }
  188. else if (IsEqualIID(iid, IID_IADsContainer))
  189. {
  190. *ppv = (IADsContainer FAR *) this;
  191. }
  192. else if (IsEqualIID(iid, IID_IADs))
  193. {
  194. *ppv = (IADs FAR *) this;
  195. }
  196. else if (IsEqualIID(iid, IID_IADsOpenDSObject))
  197. {
  198. *ppv = (IADsOpenDSObject FAR *) this;
  199. }
  200. else if (IsEqualIID(iid, IID_IADsPathnameProvider))
  201. {
  202. *ppv = (IADsPathnameProvider FAR *) this;
  203. }
  204. else if( (_pDispatch != NULL) &&
  205. IsEqualIID(iid, IID_IADsExtension) )
  206. {
  207. *ppv = (IADsExtension *) this;
  208. }
  209. else
  210. {
  211. *ppv = NULL;
  212. return E_NOINTERFACE;
  213. }
  214. AddRef();
  215. return NOERROR;
  216. }
  217. /* ISupportErrorInfo method */
  218. STDMETHODIMP
  219. CWinNTNamespace::InterfaceSupportsErrorInfo(
  220. THIS_ REFIID riid
  221. )
  222. {
  223. if (IsEqualIID(riid, IID_IADs) ||
  224. IsEqualIID(riid, IID_IADsContainer) ||
  225. IsEqualIID(riid, IID_IADsOpenDSObject)) {
  226. RRETURN(S_OK);
  227. } else {
  228. RRETURN(S_FALSE);
  229. }
  230. }
  231. STDMETHODIMP
  232. CWinNTNamespace::SetInfo(THIS)
  233. {
  234. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  235. }
  236. STDMETHODIMP
  237. CWinNTNamespace::GetInfo(THIS)
  238. {
  239. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  240. }
  241. STDMETHODIMP
  242. CWinNTNamespace::ImplicitGetInfo(THIS)
  243. {
  244. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  245. }
  246. /* IADsContainer methods */
  247. STDMETHODIMP
  248. CWinNTNamespace::get_Count(long FAR* retval)
  249. {
  250. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  251. }
  252. STDMETHODIMP
  253. CWinNTNamespace::get_Filter(THIS_ VARIANT FAR* pVar)
  254. {
  255. HRESULT hr;
  256. VariantInit(pVar);
  257. hr = VariantCopy(pVar, &_vFilter);
  258. RRETURN_EXP_IF_ERR(hr);
  259. }
  260. STDMETHODIMP
  261. CWinNTNamespace::put_Filter(THIS_ VARIANT Var)
  262. {
  263. HRESULT hr;
  264. hr = VariantCopy(&_vFilter, &Var);
  265. RRETURN_EXP_IF_ERR(hr);
  266. }
  267. STDMETHODIMP
  268. CWinNTNamespace::put_Hints(THIS_ VARIANT Var)
  269. {
  270. RRETURN_EXP_IF_ERR( E_NOTIMPL);
  271. }
  272. STDMETHODIMP
  273. CWinNTNamespace::get_Hints(THIS_ VARIANT FAR* pVar)
  274. {
  275. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  276. }
  277. STDMETHODIMP
  278. CWinNTNamespace::GetObject(
  279. THIS_ BSTR ClassName,
  280. BSTR RelativeName,
  281. IDispatch * FAR* ppObject
  282. )
  283. {
  284. WCHAR szBuffer[MAX_PATH];
  285. DWORD dwLength = 0;
  286. HRESULT hr = S_OK;
  287. if (!RelativeName || !*RelativeName) {
  288. RRETURN_EXP_IF_ERR(E_ADS_UNKNOWN_OBJECT);
  289. }
  290. //
  291. // Make sure we are not going to overflow the string buffer.
  292. // +3 for // and \0
  293. //
  294. dwLength = wcslen(_ADsPath) + wcslen(RelativeName) + 3;
  295. if (dwLength > MAX_PATH) {
  296. BAIL_ON_FAILURE(hr = E_ADS_BAD_PARAMETER);
  297. }
  298. wcscpy(szBuffer, _ADsPath);
  299. wcscat(szBuffer, L"//");
  300. wcscat(szBuffer, RelativeName);
  301. if (ClassName) {
  302. //
  303. // +1 for the ",".
  304. //
  305. dwLength += wcslen(ClassName) + 1;
  306. if (dwLength > MAX_PATH) {
  307. BAIL_ON_FAILURE(hr = E_ADS_BAD_PARAMETER);
  308. }
  309. wcscat(szBuffer,L",");
  310. wcscat(szBuffer, ClassName);
  311. }
  312. hr = ::GetObject(
  313. szBuffer,
  314. (LPVOID *)ppObject,
  315. _Credentials
  316. );
  317. BAIL_ON_FAILURE(hr);
  318. error:
  319. RRETURN_EXP_IF_ERR(hr);
  320. }
  321. STDMETHODIMP
  322. CWinNTNamespace::get__NewEnum(THIS_ IUnknown * FAR* retval)
  323. {
  324. HRESULT hr;
  325. IUnknown FAR* punkEnum=NULL;
  326. IEnumVARIANT * penum = NULL;
  327. *retval = NULL;
  328. //
  329. // Create new enumerator for items currently
  330. // in collection and QI for IUnknown
  331. //
  332. hr = CWinNTNamespaceEnum::Create((CWinNTNamespaceEnum **)&penum, _vFilter,
  333. _Credentials);
  334. if (FAILED(hr)){
  335. goto error;
  336. }
  337. hr = penum->QueryInterface(
  338. IID_IUnknown,
  339. (VOID FAR* FAR*)retval
  340. );
  341. if (FAILED(hr)){
  342. goto error;
  343. }
  344. if (penum) {
  345. penum->Release();
  346. }
  347. return NOERROR;
  348. error:
  349. if (penum) {
  350. delete penum;
  351. }
  352. RRETURN_EXP_IF_ERR(hr);
  353. }
  354. STDMETHODIMP
  355. CWinNTNamespace::Create(
  356. THIS_ BSTR ClassName,
  357. BSTR RelativeName,
  358. IDispatch * FAR* ppObject
  359. )
  360. {
  361. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  362. }
  363. STDMETHODIMP
  364. CWinNTNamespace::Delete(
  365. THIS_ BSTR SourceName,
  366. BSTR Type
  367. )
  368. {
  369. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  370. }
  371. STDMETHODIMP
  372. CWinNTNamespace::CopyHere(
  373. THIS_ BSTR SourceName,
  374. BSTR NewName,
  375. IDispatch * FAR* ppObject
  376. )
  377. {
  378. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  379. }
  380. STDMETHODIMP
  381. CWinNTNamespace::MoveHere(
  382. THIS_ BSTR SourceName,
  383. BSTR NewName,
  384. IDispatch * FAR* ppObject
  385. )
  386. {
  387. RRETURN_EXP_IF_ERR(E_NOTIMPL);
  388. }
  389. HRESULT
  390. CWinNTNamespace::AllocateNamespaceObject(
  391. CWinNTNamespace ** ppNamespace
  392. )
  393. {
  394. CWinNTNamespace FAR * pNamespace = NULL;
  395. CAggregatorDispMgr FAR * pDispMgr = NULL;
  396. HRESULT hr = S_OK;
  397. pNamespace = new CWinNTNamespace();
  398. if (pNamespace == NULL) {
  399. hr = E_OUTOFMEMORY;
  400. }
  401. BAIL_ON_FAILURE(hr);
  402. pDispMgr = new CAggregatorDispMgr;
  403. if (pDispMgr == NULL) {
  404. hr = E_OUTOFMEMORY;
  405. }
  406. BAIL_ON_FAILURE(hr);
  407. hr = LoadTypeInfoEntry(
  408. pDispMgr,
  409. LIBID_ADs,
  410. IID_IADs,
  411. (IADs *)pNamespace,
  412. DISPID_REGULAR
  413. );
  414. BAIL_ON_FAILURE(hr);
  415. hr = LoadTypeInfoEntry(
  416. pDispMgr,
  417. LIBID_ADs,
  418. IID_IADsOpenDSObject,
  419. (IADsOpenDSObject *)pNamespace,
  420. DISPID_REGULAR
  421. );
  422. BAIL_ON_FAILURE(hr);
  423. hr = LoadTypeInfoEntry(
  424. pDispMgr,
  425. LIBID_ADs,
  426. IID_IADsContainer,
  427. (IADsContainer *)pNamespace,
  428. DISPID_NEWENUM
  429. );
  430. BAIL_ON_FAILURE(hr);
  431. pNamespace->_pDispMgr = pDispMgr;
  432. *ppNamespace = pNamespace;
  433. RRETURN(hr);
  434. error:
  435. delete pNamespace;
  436. delete pDispMgr;
  437. RRETURN(hr);
  438. }
  439. STDMETHODIMP
  440. CWinNTNamespace::OpenDSObject(
  441. BSTR lpszDNName,
  442. BSTR lpszUserName,
  443. BSTR lpszPassword,
  444. LONG lnReserved,
  445. IDispatch **ppADsObj
  446. )
  447. {
  448. HRESULT hr = S_OK;
  449. DWORD dwResult;
  450. IUnknown * pObject = NULL;
  451. *ppADsObj = NULL;
  452. // we need to block the invoke on UMI code
  453. lnReserved &= (~ADS_AUTH_RESERVED);
  454. CWinNTCredentials Credentials(lpszUserName, lpszPassword, lnReserved);
  455. hr = ::GetObject(lpszDNName, (LPVOID *)&pObject, Credentials);
  456. BAIL_ON_FAILURE(hr);
  457. // UMI objects do not implement IDispatch. Hence QI for IUnknown instead
  458. // of IDispatch.
  459. if(lnReserved & ADS_AUTH_RESERVED)
  460. // call is from UMI
  461. hr = pObject->QueryInterface(IID_IUnknown, (void **)ppADsObj);
  462. else
  463. hr = pObject->QueryInterface(IID_IDispatch, (void **)ppADsObj);
  464. BAIL_ON_FAILURE(hr);
  465. error:
  466. if (pObject)
  467. pObject->Release();
  468. RRETURN_EXP_IF_ERR(hr);
  469. }
  470. HRESULT
  471. CWinNTNamespace::ParsePath(
  472. BSTR bstrADsPath,
  473. DWORD dwType,
  474. PPATH_OBJECTINFO pObjectInfo
  475. )
  476. /*++
  477. Routine Description:
  478. Parse a path based on the type and return the information in pObjectInfo
  479. Arguments:
  480. bstrADsPath - ads path to be parsed
  481. dwType - the type of path to be parsed:
  482. ADS_PARSE_FULL
  483. ADS_PARSE_DN
  484. ADS_PARSE_COMPONENT
  485. pObjectInfo - the place where the parsed object is stored
  486. Return Value:
  487. S_OK on success, error code otherwise.
  488. --*/
  489. {
  490. HRESULT hr = S_OK;
  491. OBJECTINFO ObjInfo;
  492. POBJECTINFO pObjInfo = &ObjInfo;
  493. PWSTR szPath = NULL;
  494. memset(pObjInfo, 0, sizeof(OBJECTINFO));
  495. switch (dwType) {
  496. case ADS_PARSE_FULL:
  497. {
  498. CLexer Lexer(bstrADsPath);
  499. hr = Object(&Lexer, pObjInfo);
  500. BAIL_ON_FAILURE(hr);
  501. break;
  502. }
  503. case ADS_PARSE_DN:
  504. {
  505. WCHAR szToken[MAX_TOKEN_LENGTH];
  506. DWORD dwToken;
  507. CLexer Lexer(bstrADsPath);
  508. Lexer.SetAtDisabler(TRUE);
  509. hr = PathName(&Lexer,
  510. pObjInfo);
  511. BAIL_ON_FAILURE(hr);
  512. hr = Lexer.GetNextToken(szToken,
  513. &dwToken);
  514. BAIL_ON_FAILURE(hr);
  515. if (dwToken != TOKEN_END) {
  516. hr = E_ADS_BAD_PATHNAME;
  517. }
  518. break;
  519. }
  520. case ADS_PARSE_COMPONENT:
  521. {
  522. CLexer Lexer(bstrADsPath);
  523. Lexer.SetAtDisabler(TRUE);
  524. hr = Component(&Lexer,
  525. pObjInfo);
  526. BAIL_ON_FAILURE(hr);
  527. break;
  528. }
  529. default:
  530. break;
  531. }
  532. //
  533. // Setting new info
  534. //
  535. if (pObjInfo->ProviderName) {
  536. pObjectInfo->ProviderName = AllocADsStr(pObjInfo->ProviderName);
  537. if (!pObjectInfo->ProviderName) {
  538. hr = E_OUTOFMEMORY;
  539. BAIL_ON_FAILURE(hr);
  540. }
  541. }
  542. hr = SetObjInfoComponents(pObjInfo,
  543. pObjectInfo);
  544. BAIL_ON_FAILURE(hr);
  545. pObjectInfo->NumComponents = pObjInfo->NumComponents;
  546. pObjectInfo->dwPathType = ADS_PATHTYPE_ROOTFIRST;
  547. error:
  548. FreeObjectInfo(pObjInfo,TRUE);
  549. if (szPath != NULL) {
  550. FreeADsStr(szPath);
  551. }
  552. return (hr);
  553. }
  554. HRESULT
  555. CWinNTNamespace::SetObjInfoComponents(
  556. OBJECTINFO *pObjectInfo,
  557. PATH_OBJECTINFO *pObjectInfoTarget
  558. )
  559. /*++
  560. Routine Description:
  561. Set all the compoents in an objinfo from another objinfo. Assumes that the
  562. components in the target objinfo is empty. Users of this function can call
  563. FreeObjInfo to free that data prior to this function call.
  564. Arguments:
  565. Return Value:
  566. S_OK on success, error code otherwise.
  567. --*/
  568. {
  569. DWORD NumComponents;
  570. HRESULT hr = S_OK;
  571. NumComponents = 0;
  572. while (NumComponents < pObjectInfo->NumComponents) {
  573. if (pObjectInfo->ComponentArray[NumComponents]) {
  574. pObjectInfoTarget->ComponentArray[NumComponents].szComponent =
  575. AllocADsStr(pObjectInfo->ComponentArray[NumComponents]);
  576. if (pObjectInfoTarget->ComponentArray[NumComponents].szComponent == NULL) {
  577. pObjectInfoTarget->NumComponents = NumComponents;
  578. hr = E_OUTOFMEMORY;
  579. BAIL_ON_FAILURE(hr);
  580. }
  581. }
  582. if (pObjectInfo->DisplayComponentArray[NumComponents]) {
  583. pObjectInfoTarget->DisplayComponentArray[NumComponents].szComponent =
  584. AllocADsStr(pObjectInfo->DisplayComponentArray[NumComponents]);
  585. if (pObjectInfoTarget->DisplayComponentArray[NumComponents].szComponent == NULL) {
  586. pObjectInfoTarget->NumComponents = NumComponents;
  587. hr = E_OUTOFMEMORY;
  588. BAIL_ON_FAILURE(hr);
  589. }
  590. }
  591. NumComponents++;
  592. }
  593. pObjectInfoTarget->NumComponents = pObjectInfo->NumComponents;
  594. return hr;
  595. error:
  596. FreeObjInfoComponents(pObjectInfoTarget);
  597. RRETURN_EXP_IF_ERR(hr);
  598. }
  599. void
  600. CWinNTNamespace::FreeObjInfoComponents(
  601. PATH_OBJECTINFO *pObjectInfo
  602. )
  603. /*++
  604. Routine Description:
  605. Free all the compoents in an objinfo
  606. Arguments:
  607. Return Value:
  608. S_OK on success, error code otherwise.
  609. --*/
  610. {
  611. DWORD NumComponents;
  612. while (pObjectInfo->NumComponents > 0) {
  613. pObjectInfo->NumComponents--;
  614. NumComponents = pObjectInfo->NumComponents;
  615. if (pObjectInfo->ComponentArray[NumComponents].szComponent) {
  616. FreeADsStr(
  617. pObjectInfo->ComponentArray[NumComponents].szComponent);
  618. pObjectInfo->ComponentArray[NumComponents].szComponent = NULL;
  619. }
  620. if (pObjectInfo->ComponentArray[NumComponents].szValue) {
  621. FreeADsStr(
  622. pObjectInfo->ComponentArray[NumComponents].szValue);
  623. pObjectInfo->ComponentArray[NumComponents].szValue = NULL;
  624. }
  625. if (pObjectInfo->DisplayComponentArray[NumComponents].szComponent) {
  626. FreeADsStr(
  627. pObjectInfo->DisplayComponentArray[NumComponents].szComponent);
  628. pObjectInfo->DisplayComponentArray[NumComponents].szComponent = NULL;
  629. }
  630. if (pObjectInfo->DisplayComponentArray[NumComponents].szValue) {
  631. FreeADsStr(
  632. pObjectInfo->DisplayComponentArray[NumComponents].szValue);
  633. pObjectInfo->DisplayComponentArray[NumComponents].szValue = NULL;
  634. }
  635. }
  636. }
  637. void
  638. CWinNTNamespace::SetComponent(
  639. LPWSTR szReturn,
  640. DWORD cComponents,
  641. BOOL fEscaped
  642. )
  643. /*++
  644. Routine Description:
  645. Set an individual component in the pathname. For internal use only.
  646. Not exposed.
  647. Arguments:
  648. szReturn - the buffer to store the return value
  649. cComponents - the component number to be set
  650. Return Value:
  651. S_OK on success, error code otherwise.
  652. --*/
  653. {
  654. PATH_COMPONENT* pComponent = NULL;
  655. if (fEscaped) {
  656. pComponent = _pObjectInfo->DisplayComponentArray;
  657. }
  658. else {
  659. pComponent = _pObjectInfo->ComponentArray;
  660. }
  661. if (pComponent[cComponents].szValue) {
  662. //
  663. // If value exist, only show display value
  664. //
  665. wcscat(szReturn,
  666. pComponent[cComponents].szValue);
  667. }
  668. else {
  669. //
  670. // else value is only stored in Component
  671. //
  672. wcscat(szReturn,
  673. pComponent[cComponents].szComponent);
  674. }
  675. }
  676. HRESULT
  677. CWinNTNamespace::SetComponents(
  678. LPWSTR szReturn,
  679. LPWSTR chSeparator,
  680. DWORD dwType,
  681. BOOL fEscaped
  682. )
  683. /*++
  684. Routine Description:
  685. Set components in the pathname. For internal use only. Not exposed.
  686. Arguments:
  687. szReturn - the buffer to store the return value
  688. chSeparator - separator to be used
  689. dwType - the type to be set
  690. ADS_COMPONENT_LEAF
  691. ADS_COMPONENT_DN
  692. ADS_COMPONENT_PARENT
  693. Return Value:
  694. S_OK on success, error code otherwise.
  695. --*/
  696. {
  697. HRESULT hr = S_OK;
  698. long cComponents;
  699. long dwLimit;
  700. long dwOtherLimit = 0;
  701. if (dwType == ADS_COMPONENT_LEAF) {
  702. //
  703. // Only returns the leaf component
  704. //
  705. if (_pObjectInfo->dwPathType == ADS_PATHTYPE_ROOTFIRST) {
  706. if (_pObjectInfo->NumComponents > 0) {
  707. SetComponent(szReturn,
  708. _pObjectInfo->NumComponents - 1,
  709. fEscaped);
  710. }
  711. else {
  712. hr = E_ADS_BAD_PATHNAME;
  713. }
  714. }
  715. else {
  716. if (_pObjectInfo->NumComponents != 0) {
  717. SetComponent(szReturn,
  718. 0,
  719. fEscaped);
  720. }
  721. else {
  722. hr = E_ADS_BAD_PATHNAME;
  723. }
  724. }
  725. RRETURN(hr);
  726. }
  727. dwLimit = _pObjectInfo->NumComponents;
  728. if (dwType == ADS_COMPONENT_PARENT) {
  729. dwLimit--;
  730. }
  731. if (dwOtherLimit >= dwLimit) {
  732. hr = E_ADS_BAD_PATHNAME;
  733. goto error;
  734. }
  735. for (cComponents = dwOtherLimit; cComponents < dwLimit; cComponents++) {
  736. SetComponent(szReturn,
  737. cComponents,
  738. fEscaped);
  739. if (cComponents != dwLimit - 1) {
  740. wcscat(szReturn,
  741. chSeparator);
  742. }
  743. }
  744. error:
  745. RRETURN(S_OK);
  746. }
  747. DWORD CountPath(
  748. PPATH_OBJECTINFO pObjectInfo
  749. )
  750. {
  751. DWORD dwPath = 4; // Basic needs '://' and '/' for servername
  752. DWORD i;
  753. if (pObjectInfo->ProviderName) {
  754. dwPath += wcslen(pObjectInfo->ProviderName);
  755. }
  756. if (pObjectInfo->DisplayServerName) {
  757. dwPath += wcslen(pObjectInfo->DisplayServerName);
  758. }
  759. for (i=0;i<pObjectInfo->NumComponents;i++) {
  760. if (pObjectInfo->DisplayComponentArray[i].szComponent) {
  761. dwPath += wcslen(pObjectInfo->DisplayComponentArray[i].szComponent);
  762. }
  763. if (pObjectInfo->DisplayComponentArray[i].szValue) {
  764. dwPath += wcslen(pObjectInfo->DisplayComponentArray[i].szValue);
  765. }
  766. //
  767. // Add one for comma separator, one for equal sign
  768. //
  769. dwPath+=2;
  770. }
  771. return dwPath;
  772. }
  773. STDMETHODIMP
  774. CWinNTNamespace::ConstructPath(
  775. PPATH_OBJECTINFO pObjectInfo,
  776. DWORD dwFormatType,
  777. DWORD dwFlag,
  778. DWORD dwEscapedMode,
  779. BSTR *pbstrADsPath
  780. )
  781. {
  782. HRESULT hr = S_OK;
  783. PWSTR szReturn = NULL;
  784. long cComponents;
  785. DWORD dwPath = 0;
  786. BOOL fEscaped = FALSE;
  787. switch (dwEscapedMode) {
  788. case ADS_ESCAPEDMODE_OFF:
  789. case ADS_ESCAPEDMODE_OFF_EX:
  790. case ADS_ESCAPEDMODE_DEFAULT:
  791. fEscaped = FALSE;
  792. break;
  793. case ADS_ESCAPEDMODE_ON:
  794. fEscaped = TRUE;
  795. break;
  796. default:
  797. hr = E_INVALIDARG;
  798. goto error;
  799. }
  800. if (!pbstrADsPath) {
  801. hr = E_INVALIDARG;
  802. goto error;
  803. }
  804. dwPath = CountPath(pObjectInfo);
  805. szReturn = (PWSTR)AllocADsMem((dwPath + 1)* sizeof(WCHAR));
  806. if (szReturn == NULL) {
  807. hr = E_OUTOFMEMORY;
  808. goto error;
  809. }
  810. _pObjectInfo = pObjectInfo;
  811. wcscpy(szReturn,TEXT(""));
  812. switch (dwFormatType) {
  813. case ADS_FORMAT_WINDOWS:
  814. case ADS_FORMAT_WINDOWS_NO_SERVER:
  815. if (dwEscapedMode == ADS_ESCAPEDMODE_DEFAULT) {
  816. fEscaped = TRUE;
  817. }
  818. if (!pObjectInfo->ProviderName) {
  819. hr = E_FAIL; // Need Error Code
  820. goto error;
  821. }
  822. wcscat(szReturn,pObjectInfo->ProviderName);
  823. wcscat(szReturn,TEXT("://"));
  824. if (dwFormatType == ADS_FORMAT_WINDOWS) {
  825. if (pObjectInfo->DisplayServerName && (*(pObjectInfo->DisplayServerName))) {
  826. wcscat(szReturn,pObjectInfo->DisplayServerName);
  827. if (pObjectInfo->NumComponents>0) {
  828. wcscat(szReturn,TEXT("/"));
  829. }
  830. }
  831. }
  832. hr = SetComponents(szReturn,
  833. TEXT("/"),
  834. ADS_COMPONENT_DN,
  835. fEscaped);
  836. BAIL_ON_FAILURE(hr);
  837. break;
  838. case ADS_FORMAT_WINDOWS_DN:
  839. hr = SetComponents(szReturn,
  840. TEXT("/"),
  841. ADS_COMPONENT_DN,
  842. fEscaped);
  843. BAIL_ON_FAILURE(hr);
  844. break;
  845. case ADS_FORMAT_WINDOWS_PARENT:
  846. hr = SetComponents(szReturn,
  847. TEXT("/"),
  848. ADS_COMPONENT_PARENT,
  849. fEscaped);
  850. BAIL_ON_FAILURE(hr);
  851. break;
  852. case ADS_FORMAT_X500:
  853. case ADS_FORMAT_X500_NO_SERVER:
  854. case ADS_FORMAT_X500_DN:
  855. case ADS_FORMAT_X500_PARENT:
  856. hr = E_NOTIMPL;
  857. goto error;
  858. break;
  859. case ADS_FORMAT_LEAF:
  860. //
  861. // Reverse only if pathtype is X500. In that case, we need to get
  862. // the first element but not the last
  863. //
  864. hr = SetComponents(szReturn,
  865. NULL,
  866. ADS_COMPONENT_LEAF,
  867. fEscaped);
  868. BAIL_ON_FAILURE(hr);
  869. break;
  870. default:
  871. hr = E_INVALIDARG;
  872. goto error;
  873. }
  874. hr = ADsAllocString(szReturn, pbstrADsPath);
  875. error:
  876. if (szReturn) {
  877. FreeADsMem(szReturn);
  878. }
  879. RRETURN_EXP_IF_ERR(hr);
  880. }
  881. STDMETHODIMP
  882. CWinNTNamespace::GetEscapedElement(
  883. LONG lnReserved,
  884. BSTR bstrInStr,
  885. BSTR* pbstrOutStr
  886. )
  887. {
  888. RRETURN(E_NOTIMPL);
  889. }