2020-09-30 16:53:55 +02:00

373 lines
11 KiB
C++

/*++
Module Name:
IPStream.cpp
Abstract:
This module contains the implementation for CDfsSnapinScopeManager.
This class implements IPersistStream interface for the above.
--*/
#include "stdafx.h"
#include "DfsGUI.h"
#include "DfsScope.h"
#include "utils.h"
#include <lmdfs.h>
STDMETHODIMP
CDfsSnapinScopeManager::GetClassID(
OUT struct _GUID* o_pClsid
)
/*++
Routine Description:
Return the snapin CLSID.
Arguments:
o_pClsid - The clsid is returned here.
--*/
{
*o_pClsid = CLSID_DfsSnapinScopeManager;
return S_OK;
}
STDMETHODIMP
CDfsSnapinScopeManager::IsDirty(
)
/*++
Routine Description:
Use to check if the object has been changed since last save.
Returns S_OK if it has, otherwise returns S_FALSE
Return value:
S_OK, if the object has changed. i.e., dirty
S_FALSE, if the object has not changed, i.e., not dirty.
--*/
{
return m_pMmcDfsAdmin->GetDirty() ? S_OK : S_FALSE;
}
STDMETHODIMP
CDfsSnapinScopeManager::Load(
IN LPSTREAM i_pStream
)
/*++
Routine Description:
Used to load the snap-in from a saved file(.MSC file).
We set dirty to false(to disbale save), if load succeeds completely
Arguments:
i_pStream - Pointer to an IPersistStream object from which the saved information is to
be read.
--*/
{
RETURN_INVALIDARG_IF_NULL(i_pStream);
CWaitCursor WaitCursor;
// Get the size of data that was stored
ULONG ulDataLen = 0;
ULONG uBytesRead = 0;
HRESULT hr = i_pStream->Read(&ulDataLen, sizeof (ULONG), &uBytesRead);
RETURN_IF_FAILED(hr);
if (ulDataLen <= 0) // No use in continuing if no data is there
{
// do we have a local dfsroot?
TCHAR szLocalComputerName[MAX_COMPUTERNAME_LENGTH + 1] = {0};
DWORD dwSize = MAX_COMPUTERNAME_LENGTH + 1;
GetComputerName(szLocalComputerName, &dwSize);
ROOTINFOLIST DfsRootList;
if (S_OK == GetMultiDfsRoots(&DfsRootList, szLocalComputerName) && !DfsRootList.empty())
{
for (ROOTINFOLIST::iterator i = DfsRootList.begin(); i != DfsRootList.end(); i++)
{
CComPtr<IDfsRoot> pDfsRoot;
hr = CoCreateInstance(CLSID_DfsRoot, NULL, CLSCTX_INPROC_SERVER, IID_IDfsRoot, (void**) &pDfsRoot);
if(SUCCEEDED(hr))
{
hr = pDfsRoot->Initialize((*i)->bstrRootName);
if (S_OK == hr)
(void)m_pMmcDfsAdmin->AddDfsRootToList(pDfsRoot);
}
}
}
FreeRootInfoList(&DfsRootList);
/*
CComBSTR bstrRootEntryPath;
hr = IsHostingDfsRoot(szLocalComputerName, &bstrRootEntryPath);
if (S_OK == hr)
{
CComPtr<IDfsRoot> pDfsRoot;
hr = CoCreateInstance (CLSID_DfsRoot, NULL, CLSCTX_INPROC_SERVER, IID_IDfsRoot, (void**) &pDfsRoot);
if(SUCCEEDED(hr))
{
hr = pDfsRoot->Initialize(bstrRootEntryPath);
if (S_OK == hr)
(void)m_pMmcDfsAdmin->AddDfsRootToList(pDfsRoot);
}
} */
} else
{
bool bSomeLoadFailed = false;
BYTE* pStreamData = NULL;
do {
pStreamData = new BYTE [ulDataLen]; // Allocate memory for the data to be read
BREAK_OUTOFMEMORY_IF_NULL(pStreamData, &hr);
// Read the data from the stream
hr = i_pStream->Read(pStreamData, ulDataLen, &uBytesRead);
BREAK_IF_FAILED(hr);
BYTE* pData = pStreamData;
BYTE* pDataEnd = pStreamData + ulDataLen;
// Start by reading the first machines name
ULONG nVersion = 0;
TCHAR *lpszDfsName = (LPTSTR)pData;
if (*lpszDfsName == _T('\\'))
{
nVersion = 0;
} else
{
ULONG nVer = *(ULONG UNALIGNED *)pData;
if (nVer == 1)
{
nVersion = 1;
pData += sizeof(ULONG);
} else
{
hr = S_FALSE; // corrupted console file
break;
}
}
do
{
lpszDfsName = (LPTSTR)pData;
pData += sizeof(TCHAR) * (_tcslen(lpszDfsName) + 1);
CComPtr<IDfsRoot> pDfsRoot;
hr = CoCreateInstance (CLSID_DfsRoot, NULL, CLSCTX_INPROC_SERVER, IID_IDfsRoot, (void**) &pDfsRoot);
BREAK_IF_FAILED(hr);
// retrieve link filtering settings
ULONG ulMaxLimit = FILTERDFSLINKS_MAXLIMIT_DEFAULT;
FILTERDFSLINKS_TYPE lFilterType = FILTERDFSLINKS_TYPE_NO_FILTER;
TCHAR *pszFilterName = NULL;
if (nVersion == 1)
{
ulMaxLimit = *((ULONG UNALIGNED *)pData);
pData += sizeof(ULONG);
lFilterType = *((enum FILTERDFSLINKS_TYPE UNALIGNED *)pData);
pData += sizeof(lFilterType);
if (lFilterType != FILTERDFSLINKS_TYPE_NO_FILTER)
{
pszFilterName = (LPTSTR)pData;
pData += sizeof(TCHAR) * (_tcslen(pszFilterName) + 1);
}
}
hr = pDfsRoot->Initialize(lpszDfsName);
if (S_OK == hr)
{
CComBSTR bstrDfsRootEntryPath;
hr = pDfsRoot->get_RootEntryPath(&bstrDfsRootEntryPath);
if (SUCCEEDED(hr))
{
// If already present in the list, just ignore this entry
hr = m_pMmcDfsAdmin->IsAlreadyInList(bstrDfsRootEntryPath);
if (S_OK != hr)
{
(void) m_pMmcDfsAdmin->AddDfsRootToList(
pDfsRoot, ulMaxLimit, lFilterType, pszFilterName);
}
}
}
else
{
DisplayMessageBoxWithOK(IDS_MSG_FAILED_TO_INITIALIZE_DFSROOT, lpszDfsName);
bSomeLoadFailed = true; // Since we could not create a dfsroot
}
} while (pData < pDataEnd);
} while (false);
if (pStreamData)
delete [] pStreamData;
m_pMmcDfsAdmin->SetDirty(bSomeLoadFailed); // Cause we just read the whole dfsroot list from file
}
return hr;
}
STDMETHODIMP
CDfsSnapinScopeManager::Save(
OUT LPSTREAM o_pStream,
IN BOOL i_bClearDirty
)
/*++
Routine Description:
Used to save the snap-in to a .MSC file. This uses a IPersistStream object.
Arguments:
o_pStream - Pointer to an IPersistStream object to which the saved information is to
be written.
i_bClearDirty - A flag indication whether the dirty flag should be cleared
--*/
{
RETURN_INVALIDARG_IF_NULL(o_pStream);
ULONG nVersion = 1;
DFS_ROOT_LIST* lpDfsRootList = NULL;
HRESULT hr = m_pMmcDfsAdmin->GetList (&lpDfsRootList);
RETURN_IF_FAILED(hr);
ULONG ulDataLen = 0;
DFS_ROOT_LIST::iterator i;
for (i = lpDfsRootList->begin(); i != lpDfsRootList->end(); i++)
{
ulDataLen +=
((_tcslen((*i)->m_bstrRootEntryPath) + 1) * sizeof (TCHAR)) + // to hold RootEntryPath
sizeof(ULONG) + // to hold LinkFilterMaxLimit
sizeof(enum FILTERDFSLINKS_TYPE); // to hold LinkFilterType
if ((*i)->m_pMmcDfsRoot->get_LinkFilterType() != FILTERDFSLINKS_TYPE_NO_FILTER)
{
BSTR bstr = (*i)->m_pMmcDfsRoot->get_LinkFilterName();
ulDataLen += ((bstr ? _tcslen(bstr) : 0) + 1) * sizeof(TCHAR); // to hold LinkFilterName
}
}
if (!ulDataLen)
return hr; // no root to presist, return
ulDataLen += sizeof(nVersion); // to hold the version number
// Allocate data
BYTE* pStreamData = new BYTE [ulDataLen];
RETURN_OUTOFMEMORY_IF_NULL(pStreamData);
// Prepare the data
BYTE* pData = pStreamData;
ZeroMemory(pStreamData, ulDataLen);
// hold version number
memcpy(pData, &nVersion, sizeof(nVersion));
pData += sizeof(nVersion);
int len = 0;
for (i = lpDfsRootList->begin(); i != lpDfsRootList->end(); i++)
{
// hold RootEntryPath
len = (_tcslen((*i)->m_bstrRootEntryPath) + 1) * sizeof(TCHAR);
memcpy(pData, (*i)->m_bstrRootEntryPath, len);
pData += len;
// hold LinkFilterMaxLimit
ULONG ulLinkFilterMaxLimit = (*i)->m_pMmcDfsRoot->get_LinkFilterMaxLimit();
memcpy(pData, &ulLinkFilterMaxLimit, sizeof(ulLinkFilterMaxLimit));
pData += sizeof(ulLinkFilterMaxLimit);
// hold LinkFilterType
FILTERDFSLINKS_TYPE lLinkFilterType = (*i)->m_pMmcDfsRoot->get_LinkFilterType();
memcpy(pData, &lLinkFilterType, sizeof(lLinkFilterType));
pData += sizeof(lLinkFilterType);
// hold LinkFilterName
if (lLinkFilterType != FILTERDFSLINKS_TYPE_NO_FILTER)
{
BSTR bstr = (*i)->m_pMmcDfsRoot->get_LinkFilterName();
len = ((bstr ? _tcslen(bstr) : 0) + 1) * sizeof(TCHAR);
memcpy(pData, (bstr ? bstr : _T("")), len);
pData += len;
}
}
// Write the data length to the stream
ULONG uBytesWritten = 0;
hr = o_pStream->Write(&ulDataLen, sizeof(ulDataLen), &uBytesWritten);
if(SUCCEEDED(hr))
{
// Now write the data to the stream
hr = o_pStream->Write(pStreamData, ulDataLen, &uBytesWritten);
}
if (pStreamData)
delete [] pStreamData;
if (i_bClearDirty)
m_pMmcDfsAdmin->SetDirty(false);
return hr;
}
STDMETHODIMP
CDfsSnapinScopeManager::GetSizeMax(
OUT ULARGE_INTEGER* o_pulSize
)
/*++
Routine Description:
Return the size of the data we will write to the stream.
Arguments:
o_ulcbSize - Return the size of data in the low byte of this variable
--*/
{
RETURN_INVALIDARG_IF_NULL(o_pulSize);
DFS_ROOT_LIST* lpDfsRootList = NULL;
HRESULT hr = m_pMmcDfsAdmin->GetList (&lpDfsRootList);
RETURN_IF_FAILED(hr);
ULONG ulDataLen = 0;
for (DFS_ROOT_LIST::iterator i = lpDfsRootList->begin(); i != lpDfsRootList->end(); i++)
{
ulDataLen += (_tcslen ((*i)->m_bstrRootEntryPath) + 1) * sizeof (TCHAR);
}
o_pulSize->LowPart = ulDataLen; // Return the size in the low bit
o_pulSize->HighPart = 0;
return hr;
}