1887 lines
47 KiB
C++
1887 lines
47 KiB
C++
|
// Copyright (c) 1997-1999 Microsoft Corporation
|
||
|
//
|
||
|
// Tab state
|
||
|
//
|
||
|
// 03-31-98 sburns
|
||
|
// 10-05-00 jonn changed to CredUIGetPassword
|
||
|
|
||
|
|
||
|
|
||
|
#include "headers.hxx"
|
||
|
#include "state.hpp"
|
||
|
#include "resource.h"
|
||
|
#include "cred.hpp"
|
||
|
|
||
|
|
||
|
|
||
|
TCHAR const c_szWizardFilename[] = L"netplwiz.dll";
|
||
|
|
||
|
|
||
|
|
||
|
class Settings
|
||
|
{
|
||
|
public:
|
||
|
|
||
|
// default ctor, copy ctor, op=, dtor used
|
||
|
|
||
|
void
|
||
|
Refresh();
|
||
|
|
||
|
String ComputerDomainDnsName;
|
||
|
String DomainName;
|
||
|
String FullComputerName;
|
||
|
String PolicyDomainDnsName;
|
||
|
String ShortComputerName;
|
||
|
String NetbiosComputerName;
|
||
|
|
||
|
bool SyncDNSNames;
|
||
|
bool JoinedToWorkgroup;
|
||
|
bool NeedsReboot;
|
||
|
};
|
||
|
|
||
|
|
||
|
|
||
|
static State* instance = 0;
|
||
|
static bool machineIsDc = false;
|
||
|
static bool networkingInstalled = false;
|
||
|
static bool policyInEffect = false;
|
||
|
static bool mustReboot = false;
|
||
|
|
||
|
// no static initialization worries here, as these are rebuilt when the State
|
||
|
// instance is constructed/initialized/refreshed.
|
||
|
static Settings original;
|
||
|
static Settings current;
|
||
|
|
||
|
// not static String instances to avoid order of static initialization any
|
||
|
// problems
|
||
|
static const wchar_t* TCPIP_PARAMS_KEY =
|
||
|
L"System\\CurrentControlSet\\Services\\Tcpip\\Parameters";
|
||
|
|
||
|
static const wchar_t* SYNC_VALUE =
|
||
|
L"SyncDomainWithMembership";
|
||
|
|
||
|
static const wchar_t* NEW_HOSTNAME_VALUE = L"NV Hostname";
|
||
|
static const wchar_t* NEW_SUFFIX_VALUE = L"NV Domain";
|
||
|
|
||
|
|
||
|
bool
|
||
|
readSyncFlag()
|
||
|
{
|
||
|
bool retval = true;
|
||
|
|
||
|
do
|
||
|
{
|
||
|
RegistryKey key;
|
||
|
|
||
|
HRESULT hr = key.Open(HKEY_LOCAL_MACHINE, TCPIP_PARAMS_KEY);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
|
||
|
// default is to sync.
|
||
|
DWORD data = 1;
|
||
|
hr = key.GetValue(SYNC_VALUE, data);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
|
||
|
retval = data ? true : false;
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
|
||
|
// JonN 1/03/01 106601
|
||
|
// When the dns suffix checkbox is unchecked,
|
||
|
// domain join fails with a confusing message
|
||
|
//
|
||
|
// LevonE: When Join fails with ERROR_DS_COULDNT_UPDATE_SPNS the UI must check
|
||
|
// if (HKLM/System/CCS/Services/Tcpip/Parameters/SyncDomainWithMembership
|
||
|
// == 0x0 &&
|
||
|
// HKLM/System/CCS/Services/Tcpip/Parameters/NV Domain
|
||
|
// != AD_Domain_To_Be_Joined)
|
||
|
bool WarnDnsSuffix( const String& refNewDomainName )
|
||
|
{
|
||
|
if (readSyncFlag())
|
||
|
return false;
|
||
|
String strNVDomain;
|
||
|
RegistryKey key;
|
||
|
HRESULT hr2 = key.Open(HKEY_LOCAL_MACHINE, TCPIP_PARAMS_KEY);
|
||
|
if (!SUCCEEDED(hr2))
|
||
|
return false;
|
||
|
hr2 = key.GetValue(NEW_SUFFIX_VALUE, strNVDomain);
|
||
|
if (!SUCCEEDED(hr2))
|
||
|
return false;
|
||
|
return !!strNVDomain.icompare( refNewDomainName );
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
HRESULT
|
||
|
WriteSyncFlag(HWND dialog, bool flag)
|
||
|
{
|
||
|
LOG_FUNCTION(WriteSyncFlag);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
HRESULT hr = S_OK;
|
||
|
do
|
||
|
{
|
||
|
RegistryKey key;
|
||
|
|
||
|
hr = key.Create(HKEY_LOCAL_MACHINE, TCPIP_PARAMS_KEY);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
|
||
|
hr = key.SetValue(SYNC_VALUE, flag ? 1 : 0);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
IDS_CHANGE_SYNC_FLAG_FAILED);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// returns true if the machine is a domain controller running under ds repair
|
||
|
// mode, false if not
|
||
|
|
||
|
bool
|
||
|
IsDcInDsRepairMode()
|
||
|
{
|
||
|
LOG_FUNCTION(IsDcInDsRepairMode);
|
||
|
|
||
|
// We infer a DC in repair mode if we're told that the machine is a server
|
||
|
// and the safe boot option is ds repair, and the real product type is
|
||
|
// LanManNT.
|
||
|
//
|
||
|
// By "real" product type, I mean that which is written in the registry,
|
||
|
// not that which is reported by RtlGetNtProductType. The API gets the
|
||
|
// result from shared memory which is adjusted at boot to reflect the
|
||
|
// ds repair mode (from LanManNt to Server). The registry entry is not
|
||
|
// changed by repair mode.
|
||
|
//
|
||
|
// We have to check both because it is possible to boot a normal server
|
||
|
// in ds repair mode.
|
||
|
|
||
|
DWORD safeBoot = 0;
|
||
|
NT_PRODUCT_TYPE product = NtProductWinNt;
|
||
|
|
||
|
HRESULT hr = Computer::GetSafebootOption(HKEY_LOCAL_MACHINE, safeBoot);
|
||
|
|
||
|
// don't assert the result: the key may not be present
|
||
|
|
||
|
hr = Computer::GetProductTypeFromRegistry(HKEY_LOCAL_MACHINE, product);
|
||
|
ASSERT(SUCCEEDED(hr));
|
||
|
|
||
|
if (safeBoot == SAFEBOOT_DSREPAIR and product == NtProductLanManNt)
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
Settings::Refresh()
|
||
|
{
|
||
|
LOG_FUNCTION(Settings::Refresh);
|
||
|
|
||
|
String unknown = String::load(IDS_UNKNOWN);
|
||
|
ComputerDomainDnsName = unknown;
|
||
|
DomainName = unknown;
|
||
|
FullComputerName = unknown;
|
||
|
ShortComputerName = unknown;
|
||
|
PolicyDomainDnsName = unknown;
|
||
|
|
||
|
SyncDNSNames = readSyncFlag();
|
||
|
JoinedToWorkgroup = true;
|
||
|
|
||
|
// CODEWORK: we should reconcile this with the Computer object added
|
||
|
// to idpage.cpp
|
||
|
|
||
|
DSROLE_PRIMARY_DOMAIN_INFO_BASIC* info = 0;
|
||
|
HRESULT hr = MyDsRoleGetPrimaryDomainInformation(0, info);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (info->DomainNameDns)
|
||
|
{
|
||
|
DomainName = info->DomainNameDns;
|
||
|
}
|
||
|
else if (info->DomainNameFlat)
|
||
|
{
|
||
|
DomainName = info->DomainNameFlat;
|
||
|
}
|
||
|
|
||
|
// this is the workgroup name iff JoinedToWorkgroup == true
|
||
|
switch (info->MachineRole)
|
||
|
{
|
||
|
case DsRole_RoleBackupDomainController:
|
||
|
case DsRole_RolePrimaryDomainController:
|
||
|
{
|
||
|
machineIsDc = true;
|
||
|
JoinedToWorkgroup = false;
|
||
|
break;
|
||
|
}
|
||
|
case DsRole_RoleStandaloneWorkstation:
|
||
|
{
|
||
|
machineIsDc = false;
|
||
|
JoinedToWorkgroup = true;
|
||
|
|
||
|
if (DomainName.empty())
|
||
|
{
|
||
|
LOG(L"empty domain name, using default WORKGROUP");
|
||
|
|
||
|
DomainName = String::load(IDS_DEFAULT_WORKGROUP);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case DsRole_RoleStandaloneServer:
|
||
|
{
|
||
|
machineIsDc = false;
|
||
|
JoinedToWorkgroup = true;
|
||
|
|
||
|
// I wonder if we're really a DC booted in ds repair mode?
|
||
|
|
||
|
if (IsDcInDsRepairMode())
|
||
|
{
|
||
|
LOG(L"machine is in ds repair mode");
|
||
|
|
||
|
machineIsDc = true;
|
||
|
JoinedToWorkgroup = false;
|
||
|
|
||
|
// we can't determine the domain name (LSA won't tell
|
||
|
// us when running ds repair mode), so we fall back to
|
||
|
// unknown. This is better than "WORKGROUP" -- which is
|
||
|
// what info contains.
|
||
|
|
||
|
DomainName = unknown;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (DomainName.empty())
|
||
|
{
|
||
|
LOG(L"empty domain name, using default WORKGROUP");
|
||
|
|
||
|
DomainName = String::load(IDS_DEFAULT_WORKGROUP);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case DsRole_RoleMemberWorkstation:
|
||
|
case DsRole_RoleMemberServer:
|
||
|
{
|
||
|
machineIsDc = false;
|
||
|
JoinedToWorkgroup = false;
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
ASSERT(false);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
::DsRoleFreeMemory(info);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
popup.Error(
|
||
|
Win::GetDesktopWindow(),
|
||
|
hr,
|
||
|
String::load(IDS_ERROR_READING_MEMBERSHIP));
|
||
|
|
||
|
// fall back to other APIs to fill in the holes as best we can.
|
||
|
|
||
|
JoinedToWorkgroup = false;
|
||
|
machineIsDc = false;
|
||
|
|
||
|
// workstation, server, or DC? (imprescise, but better than a stick
|
||
|
// in the eye)
|
||
|
NT_PRODUCT_TYPE ntp = NtProductWinNt;
|
||
|
BOOLEAN result = ::RtlGetNtProductType(&ntp);
|
||
|
if (result)
|
||
|
{
|
||
|
switch (ntp)
|
||
|
{
|
||
|
case NtProductWinNt:
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
case NtProductServer:
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
case NtProductLanManNt:
|
||
|
{
|
||
|
machineIsDc = true;
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
ASSERT(false);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
networkingInstalled = IsNetworkingInstalled();
|
||
|
bool isTcpInstalled = networkingInstalled && IsTcpIpInstalled();
|
||
|
String activeFullName;
|
||
|
|
||
|
NetbiosComputerName = Computer::GetFuturePhysicalNetbiosName();
|
||
|
|
||
|
if (isTcpInstalled)
|
||
|
{
|
||
|
// When TCP/IP is installed on the computer, then we are interested
|
||
|
// in the computer DNS domain name suffix, and the short name is the
|
||
|
// computer's DNS hostname.
|
||
|
|
||
|
String activeShortName;
|
||
|
String futureShortName;
|
||
|
String activeDomainName;
|
||
|
String futureDomainName;
|
||
|
|
||
|
RegistryKey key;
|
||
|
hr = key.Open(HKEY_LOCAL_MACHINE, TCPIP_PARAMS_KEY);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
// Read these values without checking for failure, as empty string
|
||
|
// is ok.
|
||
|
|
||
|
activeShortName = key.GetString(L"Hostname");
|
||
|
activeDomainName = key.GetString(L"Domain");
|
||
|
futureShortName = key.GetString(NEW_HOSTNAME_VALUE);
|
||
|
|
||
|
ShortComputerName =
|
||
|
futureShortName.empty() ? activeShortName : futureShortName;
|
||
|
|
||
|
// here, check that the value was successfully read, because
|
||
|
// it may not be present.
|
||
|
|
||
|
hr = key.GetValue(NEW_SUFFIX_VALUE, futureDomainName);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
ComputerDomainDnsName = futureDomainName;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ComputerDomainDnsName = activeDomainName;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Determine if DNS domain name policy is in effect. This may change
|
||
|
// at any moment, asynchronously, so we save the result as a setting.
|
||
|
|
||
|
policyInEffect =
|
||
|
Computer::IsDnsSuffixPolicyInEffect(PolicyDomainDnsName);
|
||
|
|
||
|
// The full computer name is the short name + . + dns domain name
|
||
|
// if policy is in effect, the policy dns domain name takes precedence
|
||
|
// over the computer's dns domain name.
|
||
|
|
||
|
FullComputerName =
|
||
|
Computer::ComposeFullDnsComputerName(
|
||
|
ShortComputerName,
|
||
|
policyInEffect ? PolicyDomainDnsName : ComputerDomainDnsName);
|
||
|
activeFullName =
|
||
|
Computer::ComposeFullDnsComputerName(
|
||
|
activeShortName,
|
||
|
policyInEffect ? PolicyDomainDnsName : activeDomainName);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// 371944
|
||
|
|
||
|
activeFullName = Computer::GetActivePhysicalNetbiosName();
|
||
|
|
||
|
// when there is no TCP/IP, the short name is the NetBIOS name
|
||
|
|
||
|
ShortComputerName = NetbiosComputerName;
|
||
|
FullComputerName = ShortComputerName;
|
||
|
}
|
||
|
|
||
|
// This test does not take into account domain membership changes, as we
|
||
|
// have no prior membership info to compare the current membership to.
|
||
|
|
||
|
NeedsReboot = activeFullName != FullComputerName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::Delete()
|
||
|
{
|
||
|
LOG_FUNCTION(State::Delete);
|
||
|
|
||
|
delete instance;
|
||
|
instance = 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
State&
|
||
|
State::GetInstance()
|
||
|
{
|
||
|
ASSERT(instance);
|
||
|
|
||
|
return *instance;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::Init()
|
||
|
{
|
||
|
LOG_FUNCTION(State::Init);
|
||
|
ASSERT(!instance);
|
||
|
|
||
|
if (!instance)
|
||
|
{
|
||
|
instance = new State();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::Refresh()
|
||
|
{
|
||
|
LOG_FUNCTION(State::Refresh);
|
||
|
|
||
|
State::Delete();
|
||
|
State::Init();
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
State::State()
|
||
|
{
|
||
|
LOG_CTOR(State);
|
||
|
|
||
|
original.Refresh();
|
||
|
current = original;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
State::~State()
|
||
|
{
|
||
|
LOG_DTOR(State);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::NeedsReboot() const
|
||
|
{
|
||
|
return original.NeedsReboot;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::IsMachineDc() const
|
||
|
{
|
||
|
return machineIsDc;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::IsNetworkingInstalled() const
|
||
|
{
|
||
|
return networkingInstalled;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
String
|
||
|
State::GetFullComputerName() const
|
||
|
{
|
||
|
return current.FullComputerName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
String
|
||
|
State::GetDomainName() const
|
||
|
{
|
||
|
return current.DomainName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetDomainName(const String& name)
|
||
|
{
|
||
|
// LOG_FUNCTION2(State::SetDomainName, name);
|
||
|
|
||
|
current.DomainName = name;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::IsMemberOfWorkgroup() const
|
||
|
{
|
||
|
return current.JoinedToWorkgroup;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetIsMemberOfWorkgroup(bool yesNo)
|
||
|
{
|
||
|
current.JoinedToWorkgroup = yesNo;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
String
|
||
|
State::GetShortComputerName() const
|
||
|
{
|
||
|
return current.ShortComputerName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetShortComputerName(const String& name)
|
||
|
{
|
||
|
current.ShortComputerName = name;
|
||
|
if (!name.empty())
|
||
|
{
|
||
|
current.NetbiosComputerName = Dns::HostnameToNetbiosName(name);
|
||
|
SetFullComputerName();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// This avoids an assert in Dns::HostnameToNetbiosName and
|
||
|
// Computer::ComposeFullDnsComputerName. 119901
|
||
|
|
||
|
current.NetbiosComputerName = name;
|
||
|
current.FullComputerName = name;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::WasShortComputerNameChanged() const
|
||
|
{
|
||
|
return
|
||
|
original.ShortComputerName.icompare(current.ShortComputerName) != 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::WasNetbiosComputerNameChanged() const
|
||
|
{
|
||
|
return
|
||
|
original.NetbiosComputerName.icompare(current.NetbiosComputerName) != 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
String
|
||
|
State::GetComputerDomainDnsName() const
|
||
|
{
|
||
|
return current.ComputerDomainDnsName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetComputerDomainDnsName(const String& newName)
|
||
|
{
|
||
|
current.ComputerDomainDnsName = newName;
|
||
|
SetFullComputerName();
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetFullComputerName()
|
||
|
{
|
||
|
current.FullComputerName =
|
||
|
Computer::ComposeFullDnsComputerName(
|
||
|
current.ShortComputerName,
|
||
|
policyInEffect
|
||
|
? current.PolicyDomainDnsName
|
||
|
: current.ComputerDomainDnsName);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::WasMembershipChanged() const
|
||
|
{
|
||
|
if (current.DomainName.empty())
|
||
|
{
|
||
|
// this can happen when the domain name is not yet set or has been
|
||
|
// cleared by the user
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return
|
||
|
(Dns::CompareNames(
|
||
|
original.DomainName,
|
||
|
current.DomainName) != DnsNameCompareEqual) // 97064
|
||
|
|| original.JoinedToWorkgroup != current.JoinedToWorkgroup;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::ChangesNeedSaving() const
|
||
|
{
|
||
|
if (
|
||
|
original.ComputerDomainDnsName.icompare(
|
||
|
current.ComputerDomainDnsName) != 0
|
||
|
|| WasMembershipChanged()
|
||
|
|| WasShortComputerNameChanged()
|
||
|
|| SyncDNSNamesWasChanged())
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::GetSyncDNSNames() const
|
||
|
{
|
||
|
return current.SyncDNSNames;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetSyncDNSNames(bool yesNo)
|
||
|
{
|
||
|
current.SyncDNSNames = yesNo;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::SyncDNSNamesWasChanged() const
|
||
|
{
|
||
|
return original.SyncDNSNames != current.SyncDNSNames;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Prepend the domain name to the user name (making it a fully-qualified name
|
||
|
// in the form "domain\username") if the username does not appear to be an
|
||
|
// UPN, and the username does not appear to be fully-qualified already.
|
||
|
//
|
||
|
// domainName - netbios or DNS domain name.
|
||
|
//
|
||
|
// userName - user account name
|
||
|
//
|
||
|
// JonN 6/27/01 26151
|
||
|
// Attempting to join domain with username "someone\" gives a cryptic error
|
||
|
// returntype becomes HRESULT, userName becomes IN/OUT parameter
|
||
|
//
|
||
|
|
||
|
HRESULT
|
||
|
MassageUserName(const String& domainName, String& userName)
|
||
|
{
|
||
|
LOG_FUNCTION(MassageUserName);
|
||
|
// ASSERT(!userName.empty()); JonN 2/6/01 306520
|
||
|
|
||
|
static const String UPN_DELIMITER(L"@");
|
||
|
if (userName.find(UPN_DELIMITER) != String::npos)
|
||
|
{
|
||
|
// assume the name is a UPN: foouser@bar.com. This is not
|
||
|
// necessarily true, as account names may contain an '@' symbol.
|
||
|
// If that's the case, then they had better fully-qualify the name
|
||
|
// as domain\foo@bar....
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
if (!domainName.empty() && !userName.empty())
|
||
|
{
|
||
|
static const String DOMAIN_DELIMITER(L"\\");
|
||
|
size_t pos = userName.find(DOMAIN_DELIMITER);
|
||
|
|
||
|
if (pos == String::npos)
|
||
|
{
|
||
|
userName = domainName + DOMAIN_DELIMITER + userName;
|
||
|
}
|
||
|
//
|
||
|
// JonN 6/27/01 26151
|
||
|
// Attempting to join domain with username "someone\" gives a cryptic error
|
||
|
//
|
||
|
else if (pos == userName.length() - 1)
|
||
|
{
|
||
|
return HRESULT_FROM_WIN32(NERR_BadUsername);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Calls NetJoinDomain. The first call specifies the create computer account
|
||
|
// flag. If that fails with an access denied error, the call is repeated
|
||
|
// without the flag. (This is to cover the case where the domain
|
||
|
// administrator may have pre-created the computer account.)
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
//
|
||
|
// domain - domain to join. May be the netbios or DNS domain name.
|
||
|
//
|
||
|
// username - user account to be used. If empty, the currently logged in
|
||
|
// user's context is used.
|
||
|
//
|
||
|
// password - password for the above account. May be empty.
|
||
|
|
||
|
HRESULT
|
||
|
JoinDomain(
|
||
|
HWND dialog,
|
||
|
const String& domainName,
|
||
|
const String& username,
|
||
|
const EncryptedString& password,
|
||
|
const String& computerDomainDnsName, // 106601
|
||
|
bool deferSpn)
|
||
|
{
|
||
|
LOG_FUNCTION(JoinDomain);
|
||
|
ASSERT(!domainName.empty());
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
Win::CursorSetting cursor(IDC_WAIT);
|
||
|
|
||
|
// first attempt without create flag in case account was precreated
|
||
|
// 105306
|
||
|
|
||
|
DWORD flags =
|
||
|
NETSETUP_JOIN_DOMAIN
|
||
|
| NETSETUP_DOMAIN_JOIN_IF_JOINED
|
||
|
| NETSETUP_ACCT_DELETE;
|
||
|
|
||
|
if (deferSpn)
|
||
|
{
|
||
|
flags |= NETSETUP_DEFER_SPN_SET;
|
||
|
}
|
||
|
|
||
|
HRESULT hr = MyNetJoinDomain(domainName, username, password, flags);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
LOG(L"Retry with account create flag");
|
||
|
|
||
|
flags |= NETSETUP_ACCT_CREATE;
|
||
|
|
||
|
hr = MyNetJoinDomain(domainName, username, password, flags);
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
popup.Info(
|
||
|
dialog,
|
||
|
String::format(
|
||
|
IDS_DOMAIN_WELCOME,
|
||
|
domainName.c_str()));
|
||
|
|
||
|
HINSTANCE hNetWiz = LoadLibrary(c_szWizardFilename);
|
||
|
if (hNetWiz) {
|
||
|
HRESULT (*pfnClearAutoLogon)(VOID) =
|
||
|
(HRESULT (*)(VOID)) GetProcAddress(
|
||
|
hNetWiz,
|
||
|
"ClearAutoLogon"
|
||
|
);
|
||
|
|
||
|
if (pfnClearAutoLogon) {
|
||
|
(*pfnClearAutoLogon)();
|
||
|
}
|
||
|
|
||
|
FreeLibrary(hNetWiz);
|
||
|
}
|
||
|
}
|
||
|
else if (hr == Win32ToHresult(ERROR_DISK_FULL)) // 17367
|
||
|
{
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
String::format(IDS_DISK_FULL, domainName.c_str()));
|
||
|
}
|
||
|
// JonN 1/03/01 106601
|
||
|
// When the dns suffix checkbox is unchecked,
|
||
|
// domain join fails with a confusing message
|
||
|
else if (hr == Win32ToHresult(ERROR_DS_COULDNT_UPDATE_SPNS)) // 106601
|
||
|
{
|
||
|
bool fWarnDnsSuffix = WarnDnsSuffix(domainName);
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
String::format( (fWarnDnsSuffix)
|
||
|
? IDS_JOIN_DOMAIN_COULDNT_UPDATE_SPNS_SUFFIX
|
||
|
: IDS_JOIN_DOMAIN_COULDNT_UPDATE_SPNS,
|
||
|
domainName.c_str(),
|
||
|
computerDomainDnsName.c_str()));
|
||
|
}
|
||
|
else // any other error
|
||
|
{
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(IDS_JOIN_DOMAIN_FAILED, domainName.c_str()));
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Changes the local computer's DNS domain suffix.
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
|
||
|
HRESULT
|
||
|
SetDomainDnsName(HWND dialog)
|
||
|
{
|
||
|
LOG_FUNCTION2(SetDomainDnsName, current.ComputerDomainDnsName);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
HRESULT hr =
|
||
|
Win::SetComputerNameEx(
|
||
|
ComputerNamePhysicalDnsDomain,
|
||
|
current.ComputerDomainDnsName);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
// 335055
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(
|
||
|
IDS_SET_DOMAIN_DNS_NAME_FAILED,
|
||
|
current.ComputerDomainDnsName.c_str()));
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Changes the local netbios computer name, and, if tcp/ip is installed,
|
||
|
// the local DNS hostname.
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
|
||
|
HRESULT
|
||
|
SetShortName(HWND dialog)
|
||
|
{
|
||
|
LOG_FUNCTION2(setShortName, current.ShortComputerName);
|
||
|
ASSERT(!current.ShortComputerName.empty());
|
||
|
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
bool isTcpInstalled = networkingInstalled && IsTcpIpInstalled();
|
||
|
if (isTcpInstalled)
|
||
|
{
|
||
|
// also sets the netbios name
|
||
|
|
||
|
hr =
|
||
|
Win::SetComputerNameEx(
|
||
|
ComputerNamePhysicalDnsHostname,
|
||
|
current.ShortComputerName);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
String netbiosName =
|
||
|
Dns::HostnameToNetbiosName(current.ShortComputerName);
|
||
|
hr =
|
||
|
Win::SetComputerNameEx(ComputerNamePhysicalNetBIOS, netbiosName);
|
||
|
}
|
||
|
|
||
|
// the only reason that this is likely to fail is if the user is not
|
||
|
// a local administrator. The other cases are that the machine is
|
||
|
// in a hosed state.
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(
|
||
|
IDS_SHORT_NAME_CHANGE_FAILED,
|
||
|
current.ShortComputerName.c_str()));
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Returns true if a new netbios computer name has been saved, but the machine
|
||
|
// has not yet been rebooted. In other words, true if the netbios computer
|
||
|
// name will change on next reboot. 417570
|
||
|
|
||
|
bool
|
||
|
ShortComputerNameHasChangedSinceReboot()
|
||
|
{
|
||
|
LOG_FUNCTION(ShortComputerNameHasChangedSinceReboot());
|
||
|
|
||
|
String active = Computer::GetActivePhysicalNetbiosName();
|
||
|
String future = Computer::GetFuturePhysicalNetbiosName();
|
||
|
|
||
|
return (active != future) ? true : false;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Return true if all changes were successful, false if not. Called when a
|
||
|
// machine is to be joined to a domain, or if a machine is changing membership
|
||
|
// from one domain to another.
|
||
|
//
|
||
|
// workgroup -> domain
|
||
|
// domain A -> domain B
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
|
||
|
bool
|
||
|
State::DoSaveDomainChange(HWND dialog)
|
||
|
{
|
||
|
LOG_FUNCTION(State::DoSaveDomainChange);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
String username;
|
||
|
EncryptedString password;
|
||
|
if (!RetrieveCredentials(dialog,
|
||
|
IDS_JOIN_CREDENTIALS,
|
||
|
username,
|
||
|
password))
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
HRESULT hr = S_OK;
|
||
|
bool result = true;
|
||
|
bool joinFailed = false;
|
||
|
bool changedSyncFlag = false;
|
||
|
bool shortNameNeedsChange = false;
|
||
|
bool changedShortName = false;
|
||
|
bool dnsSuffixNeedsChange = false;
|
||
|
bool changedDnsSuffix = false;
|
||
|
bool isTcpInstalled = networkingInstalled && IsTcpIpInstalled();
|
||
|
|
||
|
do
|
||
|
{
|
||
|
//
|
||
|
// JonN 6/27/01 26151
|
||
|
// Attempting to join domain with username "someone\" gives a cryptic error
|
||
|
//
|
||
|
hr = MassageUserName(current.DomainName, username);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// update the sync dns suffix flag, if necessary. We do this before
|
||
|
// calling NetJoinDomain, so that it will see the new flag and set the
|
||
|
// DNS suffix accordingly. This means if the join fails, we need to
|
||
|
// undo the change to the flag.
|
||
|
|
||
|
if (original.SyncDNSNames != current.SyncDNSNames)
|
||
|
{
|
||
|
hr = WriteSyncFlag(dialog, current.SyncDNSNames);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
changedSyncFlag = true;
|
||
|
}
|
||
|
|
||
|
// we don't break on failure, as the flag is less consequential
|
||
|
// than the joined state and computer name.
|
||
|
}
|
||
|
|
||
|
// update NV Hostname and NV Domain, if necessary. This is required
|
||
|
// before calling NetJoinDomain in order to fix bugs 31084 and 40496.
|
||
|
// If the join fails, then we need to undo this change
|
||
|
|
||
|
if (
|
||
|
// short name changed since changes last saved in this session
|
||
|
|
||
|
(original.ShortComputerName.icompare(
|
||
|
current.ShortComputerName) != 0)
|
||
|
|
||
|
or ShortComputerNameHasChangedSinceReboot() )
|
||
|
{
|
||
|
shortNameNeedsChange = true;
|
||
|
}
|
||
|
|
||
|
if (original.ComputerDomainDnsName.icompare(
|
||
|
current.ComputerDomainDnsName) != 0 )
|
||
|
{
|
||
|
dnsSuffixNeedsChange = true;
|
||
|
}
|
||
|
|
||
|
// JonN 12/5/00 244762
|
||
|
// NV Domain only applies when TCP/IP is present.
|
||
|
if (isTcpInstalled && dnsSuffixNeedsChange)
|
||
|
{
|
||
|
RegistryKey key;
|
||
|
hr = key.Open(HKEY_LOCAL_MACHINE,
|
||
|
TCPIP_PARAMS_KEY,
|
||
|
KEY_WRITE);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
hr = key.SetValue(NEW_SUFFIX_VALUE,
|
||
|
current.ComputerDomainDnsName);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
|
||
|
changedDnsSuffix = true;
|
||
|
}
|
||
|
|
||
|
hr =
|
||
|
JoinDomain(
|
||
|
dialog,
|
||
|
current.DomainName,
|
||
|
username,
|
||
|
password,
|
||
|
current.ComputerDomainDnsName,
|
||
|
shortNameNeedsChange);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
joinFailed = true;
|
||
|
|
||
|
// JonN 12/5/00 244762
|
||
|
// If we set NW Domain before the join attempt,
|
||
|
// and the join failed, we need to undo that setting now.
|
||
|
if (isTcpInstalled && changedDnsSuffix)
|
||
|
{
|
||
|
RegistryKey key;
|
||
|
HRESULT hr2 = key.Open(HKEY_LOCAL_MACHINE,
|
||
|
TCPIP_PARAMS_KEY,
|
||
|
KEY_WRITE);
|
||
|
ASSERT(SUCCEEDED(hr2));
|
||
|
hr2 = key.SetValue(NEW_SUFFIX_VALUE,
|
||
|
original.ComputerDomainDnsName);
|
||
|
ASSERT(SUCCEEDED(hr2));
|
||
|
}
|
||
|
|
||
|
// don't attempt to save any other changes. If the machine is
|
||
|
// already joined to a domain, changing the short name will cause the
|
||
|
// netbios machine name to not match the machine account, and the
|
||
|
// user will not be able to log in with a domain account.
|
||
|
//
|
||
|
// If the machine is not already joined to the domain, then it
|
||
|
// is possible to change the short name and the dns suffix, and
|
||
|
// emit a message that those things were changed even though
|
||
|
// the join failed.
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// At this point, the machine is joined to the new domain. But, it will
|
||
|
// have joined with the old netbios computer name. So, if the user has
|
||
|
// changed the name, or the name has been changed at all since the last
|
||
|
// reboot, then we must rename the machine.
|
||
|
//
|
||
|
// ever get the feeling that NetJoinDomain is a poor API?
|
||
|
|
||
|
if (shortNameNeedsChange)
|
||
|
{
|
||
|
// short name changed.
|
||
|
|
||
|
// JonN 12/5/00 244762
|
||
|
// We don't set NV Hostname until after the join succeeds.
|
||
|
if (isTcpInstalled)
|
||
|
{
|
||
|
RegistryKey key;
|
||
|
hr = key.Open(HKEY_LOCAL_MACHINE, TCPIP_PARAMS_KEY, KEY_WRITE);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
hr = key.SetValue(NEW_HOSTNAME_VALUE, current.ShortComputerName);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
|
||
|
changedShortName = true;
|
||
|
}
|
||
|
|
||
|
bool renameFailed = false;
|
||
|
|
||
|
hr =
|
||
|
MyNetRenameMachineInDomain(
|
||
|
|
||
|
// We need to pass the hostname instead of the
|
||
|
// netbios name here in order to get the correct DNS hostname
|
||
|
// and SPN set on the computer object. See ntraid (ntbug9)
|
||
|
// #128204
|
||
|
|
||
|
current.ShortComputerName,
|
||
|
username,
|
||
|
password,
|
||
|
NETSETUP_ACCT_CREATE);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
renameFailed = true;
|
||
|
|
||
|
// JonN 12/5/00 244762
|
||
|
// If we set NV Hostname before the rename attempt,
|
||
|
// and the rename failed, we need to undo that setting now.
|
||
|
if (isTcpInstalled)
|
||
|
{
|
||
|
RegistryKey key;
|
||
|
HRESULT hr2 = key.Open(HKEY_LOCAL_MACHINE,
|
||
|
TCPIP_PARAMS_KEY,
|
||
|
KEY_WRITE);
|
||
|
ASSERT(SUCCEEDED(hr2));
|
||
|
hr2 = key.SetValue(NEW_HOSTNAME_VALUE,
|
||
|
original.ShortComputerName);
|
||
|
ASSERT(SUCCEEDED(hr2));
|
||
|
}
|
||
|
|
||
|
// don't fail the whole operation 'cause the rename failed.
|
||
|
// We make a big noise about how the join worked under the old
|
||
|
// name. We need to succeed with the operation as a whole so
|
||
|
// the change dialog will close and the joined state of the
|
||
|
// machine is refreshed. Otherwise, the change dialog stays up,
|
||
|
// the domain name has changed but we don't realize it, so if
|
||
|
// the user types a new domain name in the still open change
|
||
|
// dialog that is the same as the domain the machine was joined
|
||
|
// to (matching the stale state), we don't attempt to join
|
||
|
// again. Whew.
|
||
|
|
||
|
// JonN 1/03/01 106601
|
||
|
// When the dns suffix checkbox is unchecked,
|
||
|
// domain join fails with a confusing message
|
||
|
if (hr == Win32ToHresult(ERROR_DS_COULDNT_UPDATE_SPNS)) // 106601
|
||
|
{
|
||
|
bool fWarnDnsSuffix = WarnDnsSuffix(current.DomainName);
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
String::format( (fWarnDnsSuffix)
|
||
|
? IDS_RENAME_JOINED_WITH_OLD_NAME_COULDNT_UPDATE_SPNS_SUFFIX
|
||
|
: IDS_RENAME_JOINED_WITH_OLD_NAME_COULDNT_UPDATE_SPNS,
|
||
|
current.ShortComputerName.c_str(),
|
||
|
current.DomainName.c_str(),
|
||
|
original.ShortComputerName.c_str(),
|
||
|
current.ComputerDomainDnsName.c_str()));
|
||
|
} else {
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(
|
||
|
IDS_RENAME_FAILED_JOINED_WITH_OLD_NAME,
|
||
|
current.ShortComputerName.c_str(),
|
||
|
current.DomainName.c_str(),
|
||
|
original.ShortComputerName.c_str()));
|
||
|
}
|
||
|
|
||
|
|
||
|
hr = S_FALSE;
|
||
|
}
|
||
|
|
||
|
// don't change the hostname if the rename failed, as this will
|
||
|
// prevent the user from logging in as the new computer name will not
|
||
|
// match the sam account name.
|
||
|
|
||
|
if (!renameFailed) // 401355
|
||
|
{
|
||
|
// now set the new hostname and netbios name
|
||
|
|
||
|
hr = SetShortName(dialog);
|
||
|
|
||
|
// this had better work...
|
||
|
|
||
|
ASSERT(SUCCEEDED(hr));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// NetJoinDomain will change the DNS suffix, if it succeeded. If it
|
||
|
// failed, then we shouldn't change the suffix anyway. 421824
|
||
|
|
||
|
// this is only true if the sync flag is true: otherwise, we need to
|
||
|
// save the suffix when join succeeds.
|
||
|
|
||
|
if (
|
||
|
!current.SyncDNSNames
|
||
|
&& dnsSuffixNeedsChange
|
||
|
&& !changedDnsSuffix)
|
||
|
{
|
||
|
hr = SetDomainDnsName(dialog);
|
||
|
|
||
|
// this had better work...
|
||
|
|
||
|
ASSERT(SUCCEEDED(hr));
|
||
|
}
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
if (joinFailed)
|
||
|
{
|
||
|
HRESULT hr2 = S_OK;
|
||
|
if (changedSyncFlag)
|
||
|
{
|
||
|
// change the sync flag back to its original state.
|
||
|
|
||
|
hr2 = WriteSyncFlag(dialog, original.SyncDNSNames);
|
||
|
|
||
|
// if we can't restore the flag (unlikely), then that's just tough
|
||
|
// potatos.
|
||
|
|
||
|
ASSERT(SUCCEEDED(hr2));
|
||
|
}
|
||
|
|
||
|
// JonN 11/27/00 233783 JoinDomain reports its own errors
|
||
|
} else if (FAILED(hr))
|
||
|
{
|
||
|
popup.Error(dialog, hr, IDS_JOIN_FAILED);
|
||
|
}
|
||
|
|
||
|
return SUCCEEDED(hr) ? true : false;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Call NetUnjoinDomain, first with the account delete flag, if that fails
|
||
|
// then again without it (which almost always "works"). If the first
|
||
|
// attempt fails, but the second succeeds, raise a message to the user
|
||
|
// informing him of the orphaned computer account.
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
//
|
||
|
// domain - domain to unjoin, i.e. the domain the machine is currently
|
||
|
// a member of.
|
||
|
//
|
||
|
// username - user account to be used. If empty, the currently logged in
|
||
|
// user's context is used.
|
||
|
//
|
||
|
// password - password for the above account. May be empty.
|
||
|
|
||
|
HRESULT
|
||
|
UnjoinDomain(
|
||
|
HWND dialog,
|
||
|
const String& domain,
|
||
|
const String& username,
|
||
|
const EncryptedString& password)
|
||
|
{
|
||
|
LOG_FUNCTION(UnjoinDomain);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
ASSERT(!domain.empty());
|
||
|
|
||
|
// username and password may be empty
|
||
|
|
||
|
Win::CursorSetting cursor(IDC_WAIT);
|
||
|
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
do
|
||
|
{
|
||
|
hr =
|
||
|
MyNetUnjoinDomain(
|
||
|
username,
|
||
|
password,
|
||
|
NETSETUP_ACCT_DELETE);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// try again: not trying to delete the computer account. If the
|
||
|
// user cancelled the credential dialog from the second attempt, then
|
||
|
// this attempt will use the current context.
|
||
|
|
||
|
LOG(L"Calling NetUnjoinDomain again, w/o account delete");
|
||
|
|
||
|
hr =
|
||
|
MyNetUnjoinDomain(
|
||
|
username,
|
||
|
password,
|
||
|
0);
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
|
||
|
// if we make it here, then the attempt to unjoin and remove the
|
||
|
// account failed, but the attempt to unjoin and abandon the account
|
||
|
// succeeded. So we tell the user about the abandonment, and hope
|
||
|
// they feel really guilty about it.
|
||
|
|
||
|
// Don't hassle them. They just panic. 95386
|
||
|
|
||
|
LOG(
|
||
|
String::format(
|
||
|
IDS_COMPUTER_ACCOUNT_ORPHANED,
|
||
|
domain.c_str()));
|
||
|
|
||
|
// Win::MessageBox(
|
||
|
// dialog,
|
||
|
// String::format(
|
||
|
// IDS_COMPUTER_ACCOUNT_ORPHANED,
|
||
|
// domain.c_str()),
|
||
|
// String::load(IDS_APP_TITLE),
|
||
|
// MB_OK | MB_ICONWARNING);
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(
|
||
|
IDS_UNJOIN_FAILED,
|
||
|
domain.c_str()));
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Return true if all changes were successful, false if not. Called when the
|
||
|
// current domain membership is to be severed, or when changing from one
|
||
|
// workgroup to another.
|
||
|
//
|
||
|
// domain -> workgroup
|
||
|
// workgroup A -> workgroup B
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
|
||
|
bool
|
||
|
State::DoSaveWorkgroupChange(HWND dialog)
|
||
|
{
|
||
|
LOG_FUNCTION(State::DoSaveWorkgroupChange);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
HRESULT hr = S_OK;
|
||
|
bool result = true;
|
||
|
bool unjoinFailed = false;
|
||
|
bool changedSyncFlag = false;
|
||
|
|
||
|
do
|
||
|
{
|
||
|
// update the sync dns suffix flag, if the user changed it. Do this
|
||
|
// before calling NetUnjoinDomain, which will clear the dns suffix
|
||
|
// for us.
|
||
|
|
||
|
if (original.SyncDNSNames != current.SyncDNSNames)
|
||
|
{
|
||
|
hr = WriteSyncFlag(dialog, current.SyncDNSNames);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
changedSyncFlag = true;
|
||
|
}
|
||
|
|
||
|
// we don't break on failure, as the flag is less consequential
|
||
|
// than the joined state and computer name.
|
||
|
}
|
||
|
|
||
|
// only unjoin if we were previously joined to a domain
|
||
|
|
||
|
if (!original.JoinedToWorkgroup and networkingInstalled)
|
||
|
{
|
||
|
// get credentials for removing the computer account
|
||
|
|
||
|
String username;
|
||
|
EncryptedString password;
|
||
|
if (!RetrieveCredentials(dialog,
|
||
|
IDS_UNJOIN_CREDENTIALS,
|
||
|
username,
|
||
|
password))
|
||
|
{
|
||
|
result = false;
|
||
|
unjoinFailed = true;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// JonN 6/27/01 26151
|
||
|
// Attempting to join domain with username "someone\" gives a cryptic error
|
||
|
//
|
||
|
hr = MassageUserName(original.DomainName, username);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
hr =
|
||
|
UnjoinDomain(
|
||
|
dialog,
|
||
|
original.DomainName,
|
||
|
username,
|
||
|
password);
|
||
|
|
||
|
// Don't try to change anything else, especially the hostname. If
|
||
|
// the unjoin failed, and we change the name locally, this will
|
||
|
// prevent the user from logging in, as the new computer name will
|
||
|
// not match the computer account name in the domain.
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
unjoinFailed = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// join the workgroup
|
||
|
|
||
|
hr = MyNetJoinDomain(current.DomainName, String(), EncryptedString(), 0);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
// this is extremely unlikely to fail, and if it did, the
|
||
|
// workgroup would simply be "WORKGROUP"
|
||
|
|
||
|
result = false;
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(
|
||
|
IDS_JOIN_WORKGROUP_FAILED,
|
||
|
current.DomainName.c_str()));
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
popup.Info(
|
||
|
dialog,
|
||
|
String::format(
|
||
|
IDS_WORKGROUP_WELCOME,
|
||
|
current.DomainName.c_str()));
|
||
|
|
||
|
// change the host name, if the user has changed it.
|
||
|
|
||
|
if (
|
||
|
original.ShortComputerName.icompare(
|
||
|
current.ShortComputerName) != 0)
|
||
|
{
|
||
|
hr = SetShortName(dialog);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// change the domain name, if the user changed it.
|
||
|
|
||
|
if (
|
||
|
original.ComputerDomainDnsName.icompare(
|
||
|
current.ComputerDomainDnsName) != 0 )
|
||
|
{
|
||
|
hr = SetDomainDnsName(dialog);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
if (unjoinFailed and changedSyncFlag)
|
||
|
{
|
||
|
// change the sync flag back to its original state.
|
||
|
|
||
|
hr = WriteSyncFlag(dialog, original.SyncDNSNames);
|
||
|
|
||
|
// if we can't restore the flag (unlikely), then that's just tough
|
||
|
// potatos.
|
||
|
|
||
|
ASSERT(SUCCEEDED(hr));
|
||
|
}
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Return true if all changes succeeded, false otherwise. Called only
|
||
|
// when domain membership is not to be changed.
|
||
|
//
|
||
|
// dialog - window handle to dialog window to be used as a parent window
|
||
|
// for any child dialogs that may need to be raised.
|
||
|
|
||
|
bool
|
||
|
State::DoSaveNameChange(HWND dialog)
|
||
|
{
|
||
|
LOG_FUNCTION(State::DoSaveNameChange);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
bool result = true;
|
||
|
HRESULT hr = S_OK;
|
||
|
|
||
|
do
|
||
|
{
|
||
|
// change the hostname, if the user has made changes
|
||
|
|
||
|
if (
|
||
|
original.ShortComputerName.icompare(
|
||
|
current.ShortComputerName) != 0)
|
||
|
{
|
||
|
if (!original.JoinedToWorkgroup and networkingInstalled)
|
||
|
{
|
||
|
// machine is joined to a domain -- we need to rename the
|
||
|
// machine's domain account
|
||
|
|
||
|
String username;
|
||
|
EncryptedString password;
|
||
|
if (!RetrieveCredentials(dialog,
|
||
|
IDS_RENAME_CREDENTIALS,
|
||
|
username,
|
||
|
password))
|
||
|
{
|
||
|
result = false;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// JonN 6/27/01 26151
|
||
|
// Attempting to join domain with username "someone\" gives a cryptic error
|
||
|
//
|
||
|
hr = MassageUserName(current.DomainName, username);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
hr =
|
||
|
MyNetRenameMachineInDomain(
|
||
|
|
||
|
// We need to pass the full hostname instead of just the
|
||
|
// netbios name here in order to get the correct DNS
|
||
|
// hostname and SPN set on the computer object. See ntraid
|
||
|
// (ntbug9) #128204
|
||
|
|
||
|
current.ShortComputerName,
|
||
|
username,
|
||
|
password,
|
||
|
NETSETUP_ACCT_CREATE);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
// JonN 1/03/01 106601
|
||
|
// When the dns suffix checkbox is unchecked,
|
||
|
// domain join fails with a confusing message
|
||
|
if (hr == Win32ToHresult(ERROR_DS_COULDNT_UPDATE_SPNS)) // 106601
|
||
|
{
|
||
|
bool fWarnDnsSuffix = WarnDnsSuffix(current.DomainName);
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
String::format( (fWarnDnsSuffix)
|
||
|
? IDS_RENAME_COULDNT_UPDATE_SPNS_SUFFIX
|
||
|
: IDS_RENAME_COULDNT_UPDATE_SPNS,
|
||
|
current.ShortComputerName.c_str(),
|
||
|
current.DomainName.c_str(),
|
||
|
current.ComputerDomainDnsName.c_str()));
|
||
|
} else {
|
||
|
popup.Error(
|
||
|
dialog,
|
||
|
hr,
|
||
|
String::format(
|
||
|
IDS_RENAME_FAILED,
|
||
|
current.ShortComputerName.c_str()));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Don't try to change anything else, especially the netbios name.
|
||
|
// If the rename failed, and we change the name locally, this will
|
||
|
// prevent the user from logging in, as the new computer name will
|
||
|
// not match the computer account name in the domain.
|
||
|
|
||
|
BREAK_ON_FAILED_HRESULT(hr);
|
||
|
}
|
||
|
|
||
|
// Set the dns hostname and the netbios name. If we called
|
||
|
// NetRenameMachineInDomain, this may redundantly set netbios name
|
||
|
// (as NetRenameMachineInDomain calls SetComputerNameEx with the
|
||
|
// netbios name).
|
||
|
|
||
|
hr = SetShortName(dialog);
|
||
|
|
||
|
// Since NetRenameMachineInDomain calls SetComputerNameEx, if that
|
||
|
// failed, the rename would also have failed. So our 2nd call to
|
||
|
// SetComputerNameEx in SetShortName is almost certain to succeed.
|
||
|
// If it does fail, we're not going to attempt to roll back the
|
||
|
// rename.
|
||
|
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// update the sync dns suffix flag, if the user changed it
|
||
|
|
||
|
if (original.SyncDNSNames != current.SyncDNSNames)
|
||
|
{
|
||
|
hr = WriteSyncFlag(dialog, current.SyncDNSNames);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// change the domain name, if the user changed it.
|
||
|
|
||
|
if (
|
||
|
original.ComputerDomainDnsName.icompare(
|
||
|
current.ComputerDomainDnsName) != 0 )
|
||
|
{
|
||
|
hr = SetDomainDnsName(dialog);
|
||
|
if (FAILED(hr))
|
||
|
{
|
||
|
result = false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::SaveChanges(HWND dialog)
|
||
|
{
|
||
|
LOG_FUNCTION(State::SaveChanges);
|
||
|
ASSERT(Win::IsWindow(dialog));
|
||
|
|
||
|
// Changes to domain membership are made first, then to the computer name.
|
||
|
|
||
|
//
|
||
|
// workgroup -> domain
|
||
|
// domain A -> domain B
|
||
|
//
|
||
|
|
||
|
if (
|
||
|
(original.JoinedToWorkgroup && !current.JoinedToWorkgroup)
|
||
|
||
|
||
|
( !original.JoinedToWorkgroup
|
||
|
&& !current.JoinedToWorkgroup
|
||
|
&& original.DomainName.icompare(current.DomainName) != 0) )
|
||
|
{
|
||
|
return DoSaveDomainChange(dialog);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// domain -> workgroup
|
||
|
// workgroup A -> workgroup B
|
||
|
//
|
||
|
|
||
|
else if (
|
||
|
!original.JoinedToWorkgroup && current.JoinedToWorkgroup
|
||
|
||
|
||
|
original.JoinedToWorkgroup && current.JoinedToWorkgroup
|
||
|
&& original.DomainName.icompare(current.DomainName) != 0)
|
||
|
|
||
|
{
|
||
|
return DoSaveWorkgroupChange(dialog);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// name change only
|
||
|
//
|
||
|
|
||
|
ASSERT(original.JoinedToWorkgroup == current.JoinedToWorkgroup);
|
||
|
ASSERT(original.DomainName == original.DomainName);
|
||
|
|
||
|
return DoSaveNameChange(dialog);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void
|
||
|
State::SetChangesMadeThisSession(bool yesNo)
|
||
|
{
|
||
|
LOG_FUNCTION2(
|
||
|
State::SetChangesMadeThisSession,
|
||
|
yesNo ? L"true" : L"false");
|
||
|
|
||
|
mustReboot = yesNo;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
State::ChangesMadeThisSession() const
|
||
|
{
|
||
|
LOG_FUNCTION2(
|
||
|
State::ChangesMadeThisSession,
|
||
|
mustReboot ? L"true" : L"false");
|
||
|
|
||
|
return mustReboot;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
String
|
||
|
State::GetNetbiosComputerName() const
|
||
|
{
|
||
|
return current.NetbiosComputerName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
String
|
||
|
State::GetOriginalShortComputerName() const
|
||
|
{
|
||
|
return original.ShortComputerName;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
DSROLE_OPERATION_STATE
|
||
|
GetDsRoleChangeState()
|
||
|
{
|
||
|
LOG_FUNCTION(GetDsRoleChangeState);
|
||
|
|
||
|
DSROLE_OPERATION_STATE result = ::DsRoleOperationIdle;
|
||
|
DSROLE_OPERATION_STATE_INFO* info = 0;
|
||
|
HRESULT hr = MyDsRoleGetPrimaryDomainInformation(0, info);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
if (info)
|
||
|
{
|
||
|
result = info->OperationState;
|
||
|
::DsRoleFreeMemory(info);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
bool
|
||
|
IsUpgradingDc()
|
||
|
{
|
||
|
LOG_FUNCTION(IsUpgradingDc);
|
||
|
|
||
|
bool isUpgrade = false;
|
||
|
DSROLE_UPGRADE_STATUS_INFO* info = 0;
|
||
|
|
||
|
HRESULT hr = MyDsRoleGetPrimaryDomainInformation(0, info);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
isUpgrade = ( (info->OperationState & DSROLE_UPGRADE_IN_PROGRESS) ? true : false );
|
||
|
::DsRoleFreeMemory(info);
|
||
|
|
||
|
}
|
||
|
|
||
|
return isUpgrade;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Evaluate a list of preconditions that must be met before a name change can
|
||
|
// be committed. Return a string describing the first unmet condition, or
|
||
|
// an empty string if all conditions are met.
|
||
|
//
|
||
|
// These preconditions are a subset of those checked before enabling the
|
||
|
// button to allow the user to enter changes. The conditions not checked here
|
||
|
// are those that cannot be changed while the ui is running (logged on as
|
||
|
// local admin, machine is DC)
|
||
|
//
|
||
|
// 389646
|
||
|
|
||
|
String
|
||
|
CheckPreconditions()
|
||
|
{
|
||
|
LOG_FUNCTION(CheckPreconditions);
|
||
|
|
||
|
String result;
|
||
|
|
||
|
do
|
||
|
{
|
||
|
// could have started dcpromo after opening netid
|
||
|
|
||
|
if (IsDcpromoRunning())
|
||
|
{
|
||
|
result = String::load(IDS_PRECHK_DCPROMO_RUNNING);
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// this test is redundant if dcpromo is running, so only perform
|
||
|
// it when dcpromo is not running.
|
||
|
|
||
|
if (IsUpgradingDc())
|
||
|
{
|
||
|
result = String::load(IDS_PRECHK_MUST_COMPLETE_DCPROMO);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// could have installed cert svc after opening netid
|
||
|
|
||
|
NTService certsvc(L"CertSvc");
|
||
|
if (certsvc.IsInstalled())
|
||
|
{
|
||
|
// sorry- renaming cert issuers invalidates their certs.
|
||
|
result = String::load(IDS_PRECHK_CANT_RENAME_CERT_SVC);
|
||
|
}
|
||
|
|
||
|
// could have completed dcpromo after opening netid
|
||
|
|
||
|
switch (GetDsRoleChangeState())
|
||
|
{
|
||
|
case ::DsRoleOperationIdle:
|
||
|
{
|
||
|
// do nothing
|
||
|
break;
|
||
|
}
|
||
|
case ::DsRoleOperationActive:
|
||
|
{
|
||
|
// a role change operation is underway
|
||
|
result = String::load(IDS_PRECHK_ROLE_CHANGE_IN_PROGRESS);
|
||
|
break;
|
||
|
}
|
||
|
case ::DsRoleOperationNeedReboot:
|
||
|
{
|
||
|
// a role change has already taken place, need to reboot before
|
||
|
// attempting another.
|
||
|
result = String::load(IDS_PRECHK_ROLE_CHANGE_NEEDS_REBOOT);
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
{
|
||
|
ASSERT(false);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!result.empty())
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// could have installed/uninstalled networking after opening
|
||
|
// netid
|
||
|
|
||
|
// re-evaluate this here again, which will be globally visible.
|
||
|
|
||
|
networkingInstalled = IsNetworkingInstalled();
|
||
|
|
||
|
State& state = State::GetInstance();
|
||
|
if (!state.IsNetworkingInstalled() && !state.IsMemberOfWorkgroup())
|
||
|
{
|
||
|
// domain members need to be able to reach a dc
|
||
|
result = String::load(IDS_PRECHK_NETWORKING_NEEDED);
|
||
|
}
|
||
|
}
|
||
|
while (0);
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|