514 lines
13 KiB
C++
514 lines
13 KiB
C++
|
#include "StdAfx.h"
|
||
|
#include "ADMTScript.h"
|
||
|
#include "UserMigration.h"
|
||
|
|
||
|
#include "Error.h"
|
||
|
#include "VarSetOptions.h"
|
||
|
#include "VarSetAccountOptions.h"
|
||
|
#include "VarSetSecurity.h"
|
||
|
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
// CUserMigration
|
||
|
//---------------------------------------------------------------------------
|
||
|
|
||
|
|
||
|
CUserMigration::CUserMigration() :
|
||
|
m_lDisableOption(admtEnableTarget),
|
||
|
m_lSourceExpiration(-1),
|
||
|
m_bMigrateSids(false),
|
||
|
m_bTranslateRoamingProfile(false),
|
||
|
m_bUpdateUserRights(false),
|
||
|
m_bMigrateGroups(false),
|
||
|
m_bUpdatePreviouslyMigratedObjects(false),
|
||
|
m_bFixGroupMembership(true),
|
||
|
m_bMigrateServiceAccounts(false)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
CUserMigration::~CUserMigration()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
// IUserMigration Implementation --------------------------------------------
|
||
|
|
||
|
|
||
|
// DisableOption Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_DisableOption(long lOption)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
if (IsDisableOptionValid(lOption))
|
||
|
{
|
||
|
m_lDisableOption = lOption;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = AdmtSetError(CLSID_Migration, IID_IUserMigration, E_INVALIDARG, IDS_E_DISABLE_OPTION_INVALID);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_DisableOption(long* plOption)
|
||
|
{
|
||
|
*plOption = m_lDisableOption;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// SourceExpiration Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_SourceExpiration(long lExpiration)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
if (IsSourceExpirationValid(lExpiration))
|
||
|
{
|
||
|
m_lSourceExpiration = lExpiration;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
hr = AdmtSetError(CLSID_Migration, IID_IUserMigration, E_INVALIDARG, IDS_E_SOURCE_EXPIRATION_INVALID);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_SourceExpiration(long* plExpiration)
|
||
|
{
|
||
|
*plExpiration = m_lSourceExpiration;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// MigrateSIDs Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_MigrateSIDs(VARIANT_BOOL bMigrate)
|
||
|
{
|
||
|
m_bMigrateSids = bMigrate ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_MigrateSIDs(VARIANT_BOOL* pbMigrate)
|
||
|
{
|
||
|
*pbMigrate = m_bMigrateSids ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// TranslateRoamingProfile Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_TranslateRoamingProfile(VARIANT_BOOL bTranslate)
|
||
|
{
|
||
|
m_bTranslateRoamingProfile = bTranslate ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_TranslateRoamingProfile(VARIANT_BOOL* pbTranslate)
|
||
|
{
|
||
|
*pbTranslate = m_bTranslateRoamingProfile ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// UpdateUserRights Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_UpdateUserRights(VARIANT_BOOL bUpdate)
|
||
|
{
|
||
|
m_bUpdateUserRights = bUpdate ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_UpdateUserRights(VARIANT_BOOL* pbUpdate)
|
||
|
{
|
||
|
*pbUpdate = m_bUpdateUserRights ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// MigrateGroups Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_MigrateGroups(VARIANT_BOOL bMigrate)
|
||
|
{
|
||
|
m_bMigrateGroups = bMigrate ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_MigrateGroups(VARIANT_BOOL* pbMigrate)
|
||
|
{
|
||
|
*pbMigrate = m_bMigrateGroups ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// UpdatePreviouslyMigratedObjects Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_UpdatePreviouslyMigratedObjects(VARIANT_BOOL bUpdate)
|
||
|
{
|
||
|
m_bUpdatePreviouslyMigratedObjects = bUpdate ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_UpdatePreviouslyMigratedObjects(VARIANT_BOOL* pbUpdate)
|
||
|
{
|
||
|
*pbUpdate = m_bUpdatePreviouslyMigratedObjects ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// FixGroupMembership Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_FixGroupMembership(VARIANT_BOOL bFix)
|
||
|
{
|
||
|
m_bFixGroupMembership = bFix ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_FixGroupMembership(VARIANT_BOOL* pbFix)
|
||
|
{
|
||
|
*pbFix = m_bFixGroupMembership ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// MigrateServiceAccounts Property
|
||
|
|
||
|
STDMETHODIMP CUserMigration::put_MigrateServiceAccounts(VARIANT_BOOL bMigrate)
|
||
|
{
|
||
|
m_bMigrateServiceAccounts = bMigrate ? true : false;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP CUserMigration::get_MigrateServiceAccounts(VARIANT_BOOL* pbMigrate)
|
||
|
{
|
||
|
*pbMigrate = m_bMigrateServiceAccounts ? VARIANT_TRUE : VARIANT_FALSE;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
// Migrate Method
|
||
|
|
||
|
STDMETHODIMP CUserMigration::Migrate(long lOptions, VARIANT vntInclude, VARIANT vntExclude)
|
||
|
{
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
MutexWait();
|
||
|
|
||
|
bool bLogOpen = _Module.OpenLog();
|
||
|
|
||
|
try
|
||
|
{
|
||
|
_Module.Log(ErrI, IDS_STARTED_USER_MIGRATION);
|
||
|
|
||
|
InitSourceDomainAndContainer();
|
||
|
InitTargetDomainAndContainer();
|
||
|
|
||
|
SetDefaultExcludedSystemProperties();
|
||
|
|
||
|
VerifyInterIntraForest();
|
||
|
VerifyCallerDelegated();
|
||
|
ValidateMigrationParameters();
|
||
|
|
||
|
if (m_bMigrateSids)
|
||
|
{
|
||
|
VerifyCanAddSidHistory();
|
||
|
}
|
||
|
|
||
|
VerifyPasswordOption();
|
||
|
|
||
|
DoOption(lOptions, vntInclude, vntExclude);
|
||
|
}
|
||
|
catch (_com_error& ce)
|
||
|
{
|
||
|
_Module.Log(ErrE, IDS_E_CANT_MIGRATE_USERS, ce);
|
||
|
hr = AdmtSetError(CLSID_Migration, IID_IUserMigration, ce, IDS_E_CANT_MIGRATE_USERS);
|
||
|
}
|
||
|
catch (...)
|
||
|
{
|
||
|
_Module.Log(ErrE, IDS_E_CANT_MIGRATE_USERS, _com_error(E_FAIL));
|
||
|
hr = AdmtSetError(CLSID_Migration, IID_IUserMigration, E_FAIL, IDS_E_CANT_MIGRATE_USERS);
|
||
|
}
|
||
|
|
||
|
if (bLogOpen)
|
||
|
{
|
||
|
_Module.CloseLog();
|
||
|
}
|
||
|
|
||
|
MutexRelease();
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
// Implementation -----------------------------------------------------------
|
||
|
|
||
|
|
||
|
// ValidateMigrationParameters Method
|
||
|
|
||
|
void CUserMigration::ValidateMigrationParameters()
|
||
|
{
|
||
|
bool bIntraForest = m_spInternal->IntraForest ? true : false;
|
||
|
|
||
|
if (bIntraForest)
|
||
|
{
|
||
|
// validate conflict option
|
||
|
|
||
|
long lConflictOptions = m_spInternal->ConflictOptions;
|
||
|
long lConflictOption = lConflictOptions & 0x0F;
|
||
|
|
||
|
if (lConflictOption == admtReplaceConflicting)
|
||
|
{
|
||
|
AdmtThrowError(GUID_NULL, GUID_NULL, E_INVALIDARG, IDS_E_INTRA_FOREST_REPLACE);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// DoNames Method
|
||
|
|
||
|
void CUserMigration::DoNames()
|
||
|
{
|
||
|
CDomainAccounts aUsers;
|
||
|
|
||
|
m_SourceDomain.QueryUsers(GetSourceContainer(), m_setIncludeNames, m_setExcludeNames, aUsers);
|
||
|
|
||
|
DoUsers(aUsers, GetTargetContainer());
|
||
|
}
|
||
|
|
||
|
|
||
|
// DoDomain Method
|
||
|
|
||
|
void CUserMigration::DoDomain()
|
||
|
{
|
||
|
CContainer& rSource = GetSourceContainer();
|
||
|
CContainer& rTarget = GetTargetContainer();
|
||
|
|
||
|
if (m_nRecurseMaintain == 2)
|
||
|
{
|
||
|
rTarget.CreateContainerHierarchy(rSource);
|
||
|
}
|
||
|
|
||
|
DoContainers(rSource, rTarget);
|
||
|
}
|
||
|
|
||
|
|
||
|
// DoContainers Method
|
||
|
|
||
|
void CUserMigration::DoContainers(CContainer& rSource, CContainer& rTarget)
|
||
|
{
|
||
|
DoUsers(rSource, rTarget);
|
||
|
|
||
|
if (m_nRecurseMaintain == 2)
|
||
|
{
|
||
|
ContainerVector aContainers;
|
||
|
rSource.QueryContainers(aContainers);
|
||
|
|
||
|
for (ContainerVector::iterator it = aContainers.begin(); it != aContainers.end(); it++)
|
||
|
{
|
||
|
DoContainers(*it, rTarget.GetContainer(it->GetName()));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// DoUsers Method
|
||
|
|
||
|
void CUserMigration::DoUsers(CContainer& rSource, CContainer& rTarget)
|
||
|
{
|
||
|
CDomainAccounts aUsers;
|
||
|
rSource.QueryUsers(m_nRecurseMaintain == 1, m_setExcludeNames, aUsers);
|
||
|
|
||
|
DoUsers(aUsers, rTarget);
|
||
|
}
|
||
|
|
||
|
|
||
|
// DoUsers Method
|
||
|
|
||
|
void CUserMigration::DoUsers(CDomainAccounts& rUsers, CContainer& rTarget)
|
||
|
{
|
||
|
if (rUsers.size() > 0)
|
||
|
{
|
||
|
if (!m_bMigrateServiceAccounts)
|
||
|
{
|
||
|
RemoveServiceAccounts(rUsers);
|
||
|
}
|
||
|
|
||
|
if (rUsers.size() > 0)
|
||
|
{
|
||
|
CVarSet aVarSet;
|
||
|
|
||
|
SetOptions(rTarget.GetPath(), aVarSet);
|
||
|
SetAccountOptions(aVarSet);
|
||
|
|
||
|
VerifyRenameConflictPrefixSuffixValid();
|
||
|
|
||
|
FillInVarSetForUsers(rUsers, aVarSet);
|
||
|
|
||
|
rUsers.clear();
|
||
|
|
||
|
#ifdef _DEBUG
|
||
|
aVarSet.Dump();
|
||
|
#endif
|
||
|
|
||
|
PerformMigration(aVarSet);
|
||
|
|
||
|
SaveSettings(aVarSet);
|
||
|
|
||
|
if ((m_nRecurseMaintain == 2) && m_bMigrateGroups)
|
||
|
{
|
||
|
FixObjectsInHierarchy(_T("group"));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// RemoveServiceAccounts Method
|
||
|
|
||
|
void CUserMigration::RemoveServiceAccounts(CDomainAccounts& rUsers)
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
CVarSet varset;
|
||
|
|
||
|
IIManageDBPtr spDatabase(__uuidof(IManageDB));
|
||
|
IUnknownPtr spunkVarSet(varset.GetInterface());
|
||
|
IUnknown* punkVarset = spunkVarSet;
|
||
|
spDatabase->GetServiceAccount(_bstr_t(_T("")), &punkVarset);
|
||
|
|
||
|
long lCount = varset.Get(_T("ServiceAccountEntries"));
|
||
|
|
||
|
if (lCount > 0)
|
||
|
{
|
||
|
StringSet setNames;
|
||
|
|
||
|
for (long lIndex = 0; lIndex < lCount; lIndex++)
|
||
|
{
|
||
|
setNames.insert(_bstr_t(varset.Get(_T("ServiceAccount.%ld"), lIndex)));
|
||
|
}
|
||
|
|
||
|
_bstr_t strDomain = m_SourceDomain.NameFlat();
|
||
|
|
||
|
for (CDomainAccounts::iterator itUser = rUsers.begin(); itUser != rUsers.end(); )
|
||
|
{
|
||
|
bool bFound = false;
|
||
|
|
||
|
_bstr_t strAccountName = strDomain + _T("\\") + itUser->GetSamAccountName();
|
||
|
_bstr_t strUserPrincipalName = itUser->GetUserPrincipalName();
|
||
|
|
||
|
PCTSTR pszAccountName = strAccountName;
|
||
|
PCTSTR pszUserPrincipalName = strUserPrincipalName;
|
||
|
|
||
|
for (StringSet::iterator itName = setNames.begin(); itName != setNames.end(); itName++)
|
||
|
{
|
||
|
PCTSTR pszName = *itName;
|
||
|
|
||
|
if (pszName)
|
||
|
{
|
||
|
if (pszAccountName && (_tcsicmp(pszName, pszAccountName) == 0))
|
||
|
{
|
||
|
bFound = true;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (pszUserPrincipalName && (_tcsicmp(pszName, pszUserPrincipalName) == 0))
|
||
|
{
|
||
|
bFound = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (bFound)
|
||
|
{
|
||
|
itUser = rUsers.erase(itUser);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
++itUser;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
catch (_com_error& ce)
|
||
|
{
|
||
|
AdmtThrowError(GUID_NULL, GUID_NULL, ce, IDS_E_REMOVING_SERVICE_ACCOUNTS);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// SetOptions Method
|
||
|
|
||
|
void CUserMigration::SetOptions(_bstr_t strTargetOu, CVarSet& rVarSet)
|
||
|
{
|
||
|
CVarSetOptions aOptions(rVarSet);
|
||
|
|
||
|
aOptions.SetTest(m_spInternal->TestMigration ? true : false);
|
||
|
|
||
|
aOptions.SetUndo(false);
|
||
|
aOptions.SetWizard(_T("user"));
|
||
|
|
||
|
aOptions.SetIntraForest(m_spInternal->IntraForest ? true : false);
|
||
|
aOptions.SetSourceDomain(m_SourceDomain.NameFlat(), m_SourceDomain.NameDns(), m_SourceDomain.Sid());
|
||
|
aOptions.SetTargetDomain(m_TargetDomain.NameFlat(), m_TargetDomain.NameDns());
|
||
|
aOptions.SetTargetOu(strTargetOu);
|
||
|
|
||
|
if (m_bMigrateSids || (m_spInternal->PasswordOption == admtCopyPassword))
|
||
|
{
|
||
|
aOptions.SetTargetServer(
|
||
|
m_TargetDomain.DomainControllerNameFlat(),
|
||
|
m_TargetDomain.DomainControllerNameDns()
|
||
|
);
|
||
|
}
|
||
|
|
||
|
aOptions.SetRenameOptions(m_spInternal->RenameOption, m_spInternal->RenamePrefixOrSuffix);
|
||
|
}
|
||
|
|
||
|
|
||
|
// SetAccountOptions Method
|
||
|
|
||
|
void CUserMigration::SetAccountOptions(CVarSet& rVarSet)
|
||
|
{
|
||
|
CVarSetAccountOptions aOptions(rVarSet);
|
||
|
|
||
|
aOptions.SetPasswordOption(m_spInternal->PasswordOption, m_spInternal->PasswordServer);
|
||
|
aOptions.SetPasswordFile(m_spInternal->PasswordFile);
|
||
|
aOptions.SetConflictOptions(m_spInternal->ConflictOptions, m_spInternal->ConflictPrefixOrSuffix);
|
||
|
|
||
|
aOptions.SetDisableOption(m_lDisableOption);
|
||
|
aOptions.SetSourceExpiration(m_lSourceExpiration);
|
||
|
aOptions.SetMigrateSids(m_bMigrateSids);
|
||
|
aOptions.SetUserMigrationOptions(m_bMigrateGroups, m_bUpdatePreviouslyMigratedObjects);
|
||
|
aOptions.SetFixGroupMembership(m_bFixGroupMembership);
|
||
|
aOptions.SetUpdateUserRights(m_bUpdateUserRights);
|
||
|
aOptions.SetTranslateRoamingProfile(m_bTranslateRoamingProfile);
|
||
|
|
||
|
aOptions.SetExcludedUserProps(m_spInternal->UserPropertiesToExclude);
|
||
|
aOptions.SetExcludedInetOrgPersonProps(m_spInternal->InetOrgPersonPropertiesToExclude);
|
||
|
|
||
|
if (m_bMigrateGroups)
|
||
|
{
|
||
|
aOptions.SetExcludedGroupProps(m_spInternal->GroupPropertiesToExclude);
|
||
|
}
|
||
|
}
|