|
|
//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1997.
//
// File: ncident.cpp
//
// Contents: Implementation of CNetCfgIdentification.
//
// Notes:
//
// History: 21 Mar 1997 danielwe Created
//
//----------------------------------------------------------------------------
#include "pch.h"
#pragma hdrstop
#include <nceh.h>
#include "ncfgval.h"
#include "ncident.h"
#include "ncmisc.h"
#include "ncreg.h"
#include "nsbase.h"
#include "nccom.h"
#include "ncerror.h"
//+---------------------------------------------------------------------------
//
// Function: DeleteStringAndSetNull
//
// Purpose: Frees the given string with delete, and sets it to
// NULL before exiting.
//
// Arguments:
// pszw [in, out] Pointer to string to be freed. The pointer is set to
// NULL before the function exits.
//
// Returns: Nothing.
//
// Author: danielwe 1 Apr 1997
//
// Notes:
//
inline VOID DeleteStringAndSetNull(PWSTR *pszw) { AssertSz(pszw, "Param is NULL!");
delete *pszw; *pszw = NULL; }
inline HRESULT HrNetValidateName(IN PCWSTR lpMachine, IN PCWSTR lpName, IN PCWSTR lpAccount, IN PCWSTR lpPassword, IN NETSETUP_NAME_TYPE NameType) { NET_API_STATUS nerr;
nerr = NetValidateName(const_cast<PWSTR>(lpMachine), const_cast<PWSTR>(lpName), const_cast<PWSTR>(lpAccount), const_cast<PWSTR>(lpPassword), NameType);
TraceError("NetValidateName", HRESULT_FROM_WIN32(nerr)); return HrFromNerr(nerr); }
inline HRESULT HrNetJoinDomain(IN PWSTR lpMachine, IN PWSTR lpMachineObjectOU, IN PWSTR lpDomain, IN PWSTR lpAccount, IN PWSTR lpPassword, IN DWORD fJoinOptions) { NET_API_STATUS nerr; HRESULT hr;
if ( fJoinOptions & NETSETUP_JOIN_DOMAIN ) { hr = HrNetValidateName( lpMachine, lpDomain, lpAccount, lpPassword, NetSetupDomain ); } else { hr = HrNetValidateName( lpMachine, lpDomain, lpAccount, lpPassword, NetSetupWorkgroup ); }
if (SUCCEEDED(hr)) { nerr = NetJoinDomain(lpMachine, lpDomain, lpMachineObjectOU, lpAccount, lpPassword, fJoinOptions);
TraceError("NetJoinDomain", HRESULT_FROM_WIN32(nerr));
hr = HrFromNerr(nerr); }
return hr; }
inline HRESULT HrNetRenameInDomain(IN PWSTR lpMachine, IN PWSTR lpNewMachine, IN PWSTR lpAccount, IN PWSTR lpPassword, IN DWORD fJoinOptions) { NET_API_STATUS nerr;
nerr = NetRenameMachineInDomain(lpMachine, lpNewMachine, lpAccount, lpPassword, fJoinOptions); TraceError("NetRenameMachineInDomain", HRESULT_FROM_WIN32(nerr)); return HrFromNerr(nerr); }
inline HRESULT HrNetUnjoinDomain(IN PWSTR lpAccount, IN PWSTR lpPassword, IN DWORD fJoinOptions) { NET_API_STATUS nerr;
nerr = NetUnjoinDomain(NULL,lpAccount, lpPassword, fJoinOptions);
TraceError("NetUnjoinDomain", HRESULT_FROM_WIN32(nerr)); return HrFromNerr(nerr); }
inline HRESULT HrNetGetJoinInformation(IN PWSTR lpNameBuffer, OUT LPDWORD lpNameBufferSize, OUT PNETSETUP_JOIN_STATUS BufferType) { NET_API_STATUS nerr; PWSTR JoinBuff = NULL;
nerr = NetGetJoinInformation(NULL, &JoinBuff, BufferType);
if ( nerr == NERR_Success ) {
if ( *BufferType == NetSetupUnjoined ) {
*lpNameBufferSize = 0; *lpNameBuffer = UNICODE_NULL;
} else {
if ( *lpNameBufferSize >= ( wcslen( JoinBuff ) +1 ) * sizeof( WCHAR ) ) {
wcscpy( lpNameBuffer, JoinBuff );
}
*lpNameBufferSize = wcslen( JoinBuff ) +1;
NetApiBufferFree( JoinBuff );
}
}
TraceError("NetGetJoinInformation", HRESULT_FROM_WIN32(nerr)); return HrFromNerr(nerr); }
#ifdef DBG
BOOL CNetCfgIdentification::FIsJoinedToDomain() { HRESULT hr = S_OK; NETSETUP_JOIN_STATUS js; WCHAR wszBuffer[256]; DWORD cchBuffer = celems(wszBuffer);
hr = HrNetGetJoinInformation(wszBuffer, &cchBuffer, &js); if (SUCCEEDED(hr)) { if (js == NetSetupUnjoined) { // If we're as yet unjoined, only make sure that we marked our
// internal state as being joined to a workgroup called "WORKGROUP"
AssertSz(m_jsCur == NetSetupWorkgroupName, "We're unjoined but not " "joined to a workgroup!"); AssertSz(m_szwCurDWName, "No current domain or " "workgroup name?"); AssertSz(!lstrcmpiW(m_szwCurDWName, SzLoadIds(IDS_WORKGROUP)), "Workgroup name is not generic!"); } else { AssertSz(js == m_jsCur, "Join status is not what we think it is!!"); } }
TraceError("CNetCfgIdentification::FIsJoinedToDomain - " "HrNetGetJoinInformation", hr);
return (m_jsCur == NetSetupDomainName); } #endif
//+---------------------------------------------------------------------------
//
// Function: HrFromNerr
//
// Purpose: Converts a NET_API_STATUS code into a NETCFG_E_* HRESULT
// value.
//
// Arguments:
// nerr [in] Status code to convert.
//
// Returns: HRESULT, Converted HRESULT value.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
HRESULT HrFromNerr(NET_API_STATUS nerr) { HRESULT hr;
switch (nerr) { case NERR_Success: hr = S_OK; break; case NERR_SetupAlreadyJoined: hr = NETCFG_E_ALREADY_JOINED; break; case ERROR_DUP_NAME: hr = NETCFG_E_NAME_IN_USE; break; case NERR_SetupNotJoined: hr = NETCFG_E_NOT_JOINED; break; // case NERR_SetupIsDC:
// hr = NETCFG_E_MACHINE_IS_DC;
// break;
// case NERR_SetupNotAServer:
// hr = NETCFG_E_NOT_A_SERVER;
// break;
// case NERR_SetupImproperRole:
// hr = NETCFG_E_INVALID_ROLE;
// break;
case ERROR_INVALID_PARAMETER: hr = E_INVALIDARG; break; case ERROR_ACCESS_DENIED: hr = E_ACCESSDENIED; break; case NERR_InvalidComputer: case ERROR_NO_SUCH_DOMAIN: hr = NETCFG_E_INVALID_DOMAIN; break; default: // Generic INetCfgIdentification error
//$ REVIEW (danielwe) 24 Jun 1997: What if this isn't a Win32 error?
hr = HRESULT_FROM_WIN32(nerr); break; }
return hr; }
//
// INetCfgIdentification implementation
//
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrEnsureCurrentComputerName
//
// Purpose: Ensures that the current computer name exists to act on.
//
// Arguments:
// (none)
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes: Sets the m_szwCurComputerName variable.
//
HRESULT CNetCfgIdentification::HrEnsureCurrentComputerName() { HRESULT hr = S_OK;
if (!m_szwCurComputerName) { PWSTR pszwComputer;
// Go get the current computer name because we don't know it yet.
hr = HrGetCurrentComputerName(&pszwComputer); if (SUCCEEDED(hr)) { // m_szwCurComputerName is now set as a side effect
CoTaskMemFree(pszwComputer); } }
AssertSz(FImplies(SUCCEEDED(hr), m_szwCurComputerName), "I MUST have a name here!");
TraceError("CNetCfgIdentification::HrEnsureCurrentComputerName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrGetNewestComputerName
//
// Purpose: Places the most recently referenced computer name into the
// output parameter.
//
// Arguments:
// pwszName [out] Most recently referenced computer name.
//
// Returns: Possible Win32 error code.
//
// Author: danielwe 24 Mar 1997
//
// Notes: If the SetComputerName() method was never called, this sets
// the m_szwCurComputerName variable and returns a pointer to it.
// Otherwise, it will return a pointer to the computer name
// given in the SetComputerName() call.
//
HRESULT CNetCfgIdentification::HrGetNewestComputerName(PCWSTR *pwszName) { HRESULT hr = S_OK;
AssertSz(pwszName, "NULL out param!");
*pwszName = NULL;
// New computer name is absent, use current computer name.
hr = HrEnsureCurrentComputerName(); if (FAILED(hr)) goto err;
*pwszName = m_szwCurComputerName;
err: TraceError("CNetCfgIdentification::HrGetNewestComputerName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrValidateMachineName
//
// Purpose: Validates the given machine name.
//
// Arguments:
// szwName [in] Machine name to validate.
//
// Returns: S_OK if machine name is valid, NETCFG_E_NAME_IN_USE if machine
// name is in use.
//
// Author: danielwe 24 Mar 1997
//
// Notes:
//
HRESULT CNetCfgIdentification::HrValidateMachineName(PCWSTR szwName) { HRESULT hr = S_OK;
// Only validate if networking is installed
hr = HrIsNetworkingInstalled(); if (hr == S_OK) { // Current computer name is unused for validation of machine name.
hr = HrNetValidateName(NULL, szwName, NULL, NULL, NetSetupMachine); if (FAILED(hr)) { //$REVIEW(danielwe): What error code to return here?
TraceError("NetValidateName - Machine Name", hr); } } else if (hr == S_FALSE) { // no networking installed. We're fine.
hr = S_OK; }
TraceError("CNetCfgIdentification::HrValidateMachineName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrValidateWorkgroupName
//
// Purpose: Validates the given workgroup name.
//
// Arguments:
// szwName [in] Workgroup name to validate.
//
// Returns: S_OK if machine name is valid, NETCFG_E_NAME_IN_USE if machine
// name is in use.
//
// Author: danielwe 24 Mar 1997
//
// Notes:
//
HRESULT CNetCfgIdentification::HrValidateWorkgroupName(PCWSTR szwName) { HRESULT hr = S_OK; PCWSTR wszComputerName = NULL;
// If the user has changed the computer name, use it, otherwise get the
// current computer name and use that.
hr = HrGetNewestComputerName(&wszComputerName); if (FAILED(hr)) goto err;
AssertSz(wszComputerName, "We don't have a computer name!");
hr = HrNetValidateName(const_cast<PWSTR>(wszComputerName), szwName, NULL, NULL, NetSetupWorkgroup); if (FAILED(hr)) { //$REVIEW(danielwe): What error code to return here?
TraceError("NetValidateName - Workgroup Name", hr); goto err; } err: TraceError("CNetCfgIdentification::HrValidateWorkgroupName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrValidateDomainName
//
// Purpose: Validates the given domain name.
//
// Arguments:
// szwName [in] Name of domain to validate.
// szwUserName [in] Username for authorization purposes.
// szwPassword [in] Password for authorization purposes.
//
// Returns: S_OK if machine name is valid, NETCFG_E_INVALID_DOMAIN if
// domain name is invalid (or non-existent).
//
// Author: danielwe 24 Mar 1997
//
// Notes:
//
HRESULT CNetCfgIdentification::HrValidateDomainName(PCWSTR szwName, PCWSTR szwUserName, PCWSTR szwPassword) { HRESULT hr = S_OK;
// NetValidateName does not use the machine name in validating the
// domain name. So it is NULL here.
hr = HrNetValidateName(NULL, szwName, szwUserName, szwPassword, NetSetupDomain); if (FAILED(hr)) { //$REVIEW(danielwe): What error code to return here?
TraceError("NetValidateName - Domain Name", hr); goto err; } err: TraceError("CNetCfgIdentification::HrValidateDomainName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::Validate
//
// Purpose: Implements COM function to validate current set of values
// used during the lifetime of this object.
//
// Arguments:
// (none)
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
STDMETHODIMP CNetCfgIdentification::Validate() { HRESULT hr = S_OK;
Validate_INetCfgIdentification_Validate();
COM_PROTECT_TRY { if (m_szwNewDWName) { if (GetNewJoinStatus() == NetSetupWorkgroupName) { // Validate workgroup name
hr = HrValidateWorkgroupName(m_szwNewDWName); if (FAILED(hr)) goto err; } else if (GetNewJoinStatus() == NetSetupDomainName) { // Validate domain name
hr = HrValidateDomainName(m_szwNewDWName, m_szwUserName, m_szwPassword); if (FAILED(hr)) goto err; } #ifdef DBG
else { AssertSz(FALSE, "Invalid join status!"); } #endif
} } COM_PROTECT_CATCH;
err: // Translate all errors to S_FALSE.
if (SUCCEEDED(hr)) { m_fValid = TRUE; } else { // spew out trace *before* the assignment so we know what the *real*
// error code was.
TraceError("CNetCfgIdentification::Validate (before S_FALSE)", hr); hr = S_FALSE; } return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::Cancel
//
// Purpose: Cancels any changes made during the lifetime of the object.
//
// Arguments:
// (none)
//
// Returns:
//
// Author: danielwe 25 Mar 1997
//
// Notes: Resets state information and frees any memory previously
// allocted.
//
STDMETHODIMP CNetCfgIdentification::Cancel() { HRESULT hr = S_OK;
Validate_INetCfgIdentification_Cancel();
COM_PROTECT_TRY { DeleteStringAndSetNull(&m_szwNewDWName); DeleteStringAndSetNull(&m_szwPassword); DeleteStringAndSetNull(&m_szwUserName); DeleteStringAndSetNull(&m_szwCurComputerName); DeleteStringAndSetNull(&m_szwCurDWName);
m_dwJoinFlags = 0; m_dwCreateFlags = 0; m_fValid = FALSE; m_jsNew = NetSetupUnjoined; }
COM_PROTECT_CATCH;
return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::Apply
//
// Purpose: Implements COM function to apply changes that were made
// during the lifetime of this object.
//
// Arguments:
// (none)
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
STDMETHODIMP CNetCfgIdentification::Apply() { HRESULT hr = S_OK;
Validate_INetCfgIdentification_Apply();
COM_PROTECT_TRY { // Has data been validated?
if (!m_fValid) { hr = E_UNEXPECTED; goto err; }
if (m_szwNewDWName) { if (GetNewJoinStatus() == NetSetupWorkgroupName) { // The user specified a workgroup name. This means they want
// to join a workgroup.
hr = HrJoinWorkgroup(); if (FAILED(hr)) goto err; } else if (GetNewJoinStatus() == NetSetupDomainName) { // The user specified a domain name. This means they want to
// join a domain.
hr = HrJoinDomain(); if (FAILED(hr)) goto err; } #ifdef DBG
else { AssertSz(FALSE, "Invalid join status!"); } #endif
} } COM_PROTECT_CATCH;
err: // Regardless of result, set valid flag to false again to require
// Validate() to be called again before calling Apply().
// $REVIEW (danielwe): Is this how we want to do it?
m_fValid = FALSE;
TraceError("CNetCfgIdentification::Apply", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrGetCurrentComputerName
//
// Purpose: Calls the Win32 GetComputerName API to get the current
// computer name.
//
// Arguments:
// ppszwComputer [out] Returned computer name.
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes: Makes a private copy of the computer name if obtained from the
// system (for further use).
//
HRESULT CNetCfgIdentification::HrGetCurrentComputerName(PWSTR* ppszwComputer) { HRESULT hr = S_OK; WCHAR szBuffer[MAX_COMPUTERNAME_LENGTH + 1]; DWORD cchBuffer = celems(szBuffer);
if (::GetComputerName(szBuffer, &cchBuffer)) { // Make a copy for the out param.
hr = HrCoTaskMemAllocAndDupSz ( szBuffer, ppszwComputer); if (SUCCEEDED(hr)) { // Make another copy for our own use.
DeleteStringAndSetNull(&m_szwCurComputerName); m_szwCurComputerName = SzDupSz(szBuffer);
AssertSz((DWORD)lstrlenW(*ppszwComputer) == cchBuffer, "This is not how big the string is!"); } } else { TraceLastWin32Error("::GetComputerName"); hr = HrFromLastWin32Error(); }
TraceError("CNetCfgIdentification::HrGetCurrentComputerName", hr); return hr; }
static const WCHAR c_szRegKeyComputerName[] = L"System\\CurrentControlSet\\Control\\ComputerName\\ComputerName"; static const WCHAR c_szRegValueComputerName[] = L"ComputerName";
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrGetNewComputerName
//
// Purpose: Helper function to retreive the new computer name from the
// registry. This will be the same as the active computer name
// unless the user has changed the computer name since booting.
//
// Arguments:
// ppszwComputer [out] Returns new computer name.
//
// Returns: S_OK if successful, Win32 error code otherwise.
//
// Author: danielwe 21 May 1997
//
// Notes:
//
HRESULT CNetCfgIdentification::HrGetNewComputerName(PWSTR* ppszwComputer) { HRESULT hr = S_OK; WCHAR szBuffer[MAX_COMPUTERNAME_LENGTH + 1]; DWORD cbBuffer = sizeof(szBuffer); HKEY hkeyComputerName;
hr = HrRegOpenKeyEx(HKEY_LOCAL_MACHINE, c_szRegKeyComputerName, KEY_READ, &hkeyComputerName); if (SUCCEEDED(hr)) { hr = HrRegQuerySzBuffer(hkeyComputerName, c_szRegValueComputerName, szBuffer, &cbBuffer); if (SUCCEEDED(hr)) { // Make a copy for the out param.
hr = HrCoTaskMemAllocAndDupSz ( szBuffer, ppszwComputer);
AssertSz(FImplies(SUCCEEDED(hr), (lstrlenW(*ppszwComputer) + 1) * sizeof(WCHAR) == cbBuffer), "This is not how big the string is!"); }
RegCloseKey(hkeyComputerName); }
TraceError("CNetCfgIdentification::HrGetCurrentComputerName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrEnsureCurrentDomainOrWorkgroupName
//
// Purpose: Obtains the current domain or workgroup to which this machine
// belongs.
//
// Arguments:
// (none)
//
// Returns: S_OK if success, error code otherwise.
//
// Author: danielwe 26 Mar 1997
//
// Notes: A machine can be joined to either or workgroup or a domain. It
// must be joined to one or the other. If it is not, we'll use a
// more or less hardcoded string and "fake it" as if the machine
// was joined to a workgroup. The member variables:
// m_jsCur and m_szwCurDWName are set by this function.
//
// This call only does work once. Subsequent calls do nothing.
//
HRESULT CNetCfgIdentification::HrEnsureCurrentDomainOrWorkgroupName() { HRESULT hr = S_OK;
if (!m_szwCurDWName) { NETSETUP_JOIN_STATUS js; WCHAR wszBuffer[256]; PCWSTR wszName; DWORD cchBuffer = celems(wszBuffer);
hr = HrNetGetJoinInformation(wszBuffer, &cchBuffer, &js); if (FAILED(hr)) goto err;
AssertSz(FIff(*wszBuffer, cchBuffer), "Buffer size inconsistency!");
if (js == NetSetupUnjoined) { // Uh oh. Machine is not joined to workgroup OR domain. Set
// default workgroup name and proceed as if joined to a workgroup.
js = NetSetupWorkgroupName;
// Use default name since HrNetGetJoinInformation() will return
// an empty string which is useless.
wszName = SzLoadIds(IDS_WORKGROUP); } else { // Use string returned from HrNetGetJoinInformation().
wszName = wszBuffer; }
m_szwCurDWName = SzDupSz(wszName);
m_jsCur = js;
AssertSz(GetCurrentJoinStatus() == NetSetupWorkgroupName || GetCurrentJoinStatus() == NetSetupDomainName, "Invalid join status flag!"); }
err: TraceError("CNetCfgIdentification::HrEnsureCurrentDomainOrWorkgroupName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrGetNewestDomainOrWorkgroupName
//
// Purpose: Returns the domain or workgroup name that was most recently
// referenced.
//
// Arguments:
// js [in] Tells whether the domain or workgroup name is wanted.
// pwszName [out] Domain or workgroup name.
//
// Returns: S_OK if success, E_OUTOFMEMORY if no memory.
//
// Author: danielwe 26 Mar 1997
//
// Notes: If the requested name has not been set by the user in a prior
// call, the current name is returned. Otherwise the name the
// user chose previously is returned.
//
HRESULT CNetCfgIdentification::HrGetNewestDomainOrWorkgroupName( NETSETUP_JOIN_STATUS js, PCWSTR *pwszName) { HRESULT hr = S_OK; PWSTR szwOut = NULL;
Assert(pwszName);
if (m_szwNewDWName && (GetNewJoinStatus() == js)) { // Give them back a copy of the domain or workgroup name they
// previously gave us.
szwOut = m_szwNewDWName; } else { // Get the current workgroup or domain for this machine. It has
// to be in one or the other.
hr = HrEnsureCurrentDomainOrWorkgroupName(); if (FAILED(hr)) goto err;
if (GetCurrentJoinStatus() == js) { // Use the current name.
szwOut = m_szwCurDWName; } }
Assert(SUCCEEDED(hr));
*pwszName = szwOut;
err: TraceError("CNetCfgIdentification::HrGetNewestDomainOrWorkgroupName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::GetWorkgroupName
//
// Purpose: Implements COM function to get the current workgroup name.
//
// Arguments:
// ppszwWorkgroup [out] Returns name of current workgroup.
//
// Returns: S_OK if succeeded, S_FALSE if machine is not joined to a
// workgroup, error code otherwise.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
STDMETHODIMP CNetCfgIdentification::GetWorkgroupName(PWSTR* ppszwWorkgroup) { HRESULT hr = S_OK; PCWSTR szwWorkgroup = NULL;
Validate_INetCfgIdentification_GetWorkgroupName(ppszwWorkgroup);
COM_PROTECT_TRY { *ppszwWorkgroup = NULL;
hr = HrGetNewestDomainOrWorkgroupName(NetSetupWorkgroupName, &szwWorkgroup); if (FAILED(hr)) goto err;
if (szwWorkgroup) { hr = HrCoTaskMemAllocAndDupSz ( szwWorkgroup, ppszwWorkgroup);
AssertSz(FImplies(SUCCEEDED(hr), lstrlenW(*ppszwWorkgroup) > 0), "Why is *ppszwWorkgroup empty?"); } else { // Not joined to a workgroup
hr = S_FALSE; } } COM_PROTECT_CATCH;
err: TraceError("CNetCfgIdentification::GetWorkgroupName", (hr == S_FALSE) ? S_OK : hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::GetDomainName
//
// Purpose: Implements COM function to get the current domain name.
//
// Arguments:
// ppszwDomain [out] Returns name of domain to which this computer
// currently belongs.
//
// Returns: S_OK if succeeded, S_FALSE if machine is not joined to a
// domain, error code otherwise.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
STDMETHODIMP CNetCfgIdentification::GetDomainName(PWSTR* ppszwDomain) { HRESULT hr = S_OK; PCWSTR szwDomain = NULL;
Validate_INetCfgIdentification_GetDomainName(ppszwDomain);
COM_PROTECT_TRY { *ppszwDomain = NULL;
hr = HrGetNewestDomainOrWorkgroupName(NetSetupDomainName, &szwDomain); if (FAILED(hr)) goto err;
if (szwDomain) { hr = HrCoTaskMemAllocAndDupSz ( szwDomain, ppszwDomain);
AssertSz(FImplies(SUCCEEDED(hr), lstrlenW(*ppszwDomain) > 0), "Why is *ppszwDomain empty?"); } else { // Not joined to a domain
hr = S_FALSE; } } COM_PROTECT_CATCH;
err: TraceError("CNetCfgIdentification::GetDomainName", (hr == S_FALSE) ? S_OK : hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrJoinWorkgroup
//
// Purpose: Actually performs the JoinWorkgroup function.
//
// Arguments:
// (none)
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
HRESULT CNetCfgIdentification::HrJoinWorkgroup() { HRESULT hr = S_OK; PCWSTR wszComputerName = NULL;
AssertSz(m_szwNewDWName && GetNewJoinStatus() == NetSetupWorkgroupName, "If there was no workgroup name, why'd you call me?!");
// If the user has changed the computer name, use it, otherwise get the
// current computer name and use that.
hr = HrGetNewestComputerName(&wszComputerName); if (FAILED(hr)) goto err;
AssertSz(wszComputerName, "We don't have a computer name!");
hr = HrEnsureCurrentDomainOrWorkgroupName(); if (FAILED(hr)) goto err;
if (FIsJoinedToDomain()) { // Must unjoin from domain if currently joined.
// If currently joined to a workgroup, this is not necessary.
hr = HrNetUnjoinDomain(m_szwUserName, m_szwPassword, 0); if (FAILED(hr)) goto err;
// Free username and password
DeleteStringAndSetNull(&m_szwPassword); DeleteStringAndSetNull(&m_szwUserName); }
// Go ahead and join the workgroup
hr = HrNetJoinDomain(const_cast<PWSTR>(wszComputerName), m_szMachineObjectOU, m_szwNewDWName, NULL, NULL, 0); if (FAILED(hr)) goto err;
// Make the current workgroup name the new one since the join on the
// new workgroup has succeeded
hr = HrEstablishNewDomainOrWorkgroupName(NetSetupWorkgroupName); if (FAILED(hr)) goto err;
err: TraceError("CNetCfgIdentification::HrJoinWorkgroup", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::JoinWorkgroup
//
// Purpose: Implements COM interface to join this computer to a new
// workgroup.
//
// Arguments:
// szwWorkgroup [in] Name of new workgroup to join.
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes: Validates, but does not actually join the workgroup. Only holds
// onto the information until Apply() is called.
//
STDMETHODIMP CNetCfgIdentification::JoinWorkgroup(PCWSTR szwWorkgroup) { HRESULT hr = S_OK;
Validate_INetCfgIdentification_JoinWorkgroup(szwWorkgroup);
COM_PROTECT_TRY { hr = HrValidateWorkgroupName(szwWorkgroup); if (FAILED(hr)) goto err;
// Free domain and password given if JoinDomain was called previously
DeleteStringAndSetNull(&m_szwPassword); DeleteStringAndSetNull(&m_szwUserName);
// Assign in new workgroup name.
m_szwNewDWName = SzDupSz(szwWorkgroup);
m_jsNew = NetSetupWorkgroupName; } COM_PROTECT_CATCH;
err: TraceError("CNetCfgIdentification::JoinWorkgroup", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrEstablishNewDomainOrWorkgroupName
//
// Purpose: When the computer is joined to a new domain or workgroup, this
// function is called to set the correct member variables and
// free the old ones.
//
// Arguments:
// js [in] Indicates whether the computer is joined to a domain or
// workgroup.
//
// Returns: S_OK, or E_OUTOFMEMORY.
//
// Author: danielwe 1 Apr 1997
//
// Notes: Replaces the m_szwCurDWName variable with the new one
// (m_szwNewDWName).
//
HRESULT CNetCfgIdentification::HrEstablishNewDomainOrWorkgroupName( NETSETUP_JOIN_STATUS js) { HRESULT hr = S_OK;
// Make the current domain or workgroup name the new one.
DeleteStringAndSetNull(&m_szwCurDWName); m_szwCurDWName = SzDupSz(m_szwNewDWName);
m_jsCur = js;
AssertSz(GetCurrentJoinStatus() == NetSetupWorkgroupName || GetCurrentJoinStatus() == NetSetupDomainName, "Invalid join status flag!");
// Free "new" name
DeleteStringAndSetNull(&m_szwNewDWName);
// Also make sure that we don't have a "new" join status either
m_jsNew = NetSetupUnjoined;
TraceError("CNetCfgIdentification::HrEstablishNewDomainOrWorkgroupName", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::HrJoinDomain
//
// Purpose: Actually performs the JoinDomain function.
//
// Arguments:
// (none)
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes:
//
HRESULT CNetCfgIdentification::HrJoinDomain() { HRESULT hr = S_OK; PCWSTR wszComputerName = NULL; DWORD dwJoinOption = 0; BOOL fIsRename = FALSE; BOOL fUseNulls = FALSE;
AssertSz(m_szwNewDWName && m_jsNew == NetSetupDomainName, "If there was no domain name, why'd you call me?!"); AssertSz(FImplies(m_szwPassword, m_szwUserName), "Password without username!!");
// If the user has changed the computer name, use it, otherwise get the
// current computer name and use that.
hr = HrGetNewestComputerName(&wszComputerName); if (FAILED(hr)) goto err; { AssertSz(wszComputerName == m_szwCurComputerName, "If I don't have a " "new computer name, this better be the original one!"); dwJoinOption |= NETSETUP_JOIN_DOMAIN; }
AssertSz(wszComputerName, "We don't have a computer name!"); AssertSz(dwJoinOption, "No option was set??");
AssertSz(FImplies(m_szwPassword, m_szwUserName), "Password without username!");
// Create a machine account if so asked.
if (m_dwJoinFlags & JDF_CREATE_ACCOUNT) { dwJoinOption |= NETSETUP_ACCT_CREATE; }
if (m_dwJoinFlags & JDF_WIN9x_UPGRADE) { dwJoinOption |= NETSETUP_WIN9X_UPGRADE; }
if (m_dwJoinFlags & JDF_JOIN_UNSECURE) { dwJoinOption |= NETSETUP_JOIN_UNSECURE; }
#if defined(REMOTE_BOOT)
// TEMP: On a remote boot machine, prevent machine password change
if (HrIsRemoteBootMachine() == S_OK) { TraceTag (ttidNetcfgBase, "Machine is remote boot, specifying WIN9X_UPGRADE flag to JoinDomain."); dwJoinOption |= NETSETUP_WIN9X_UPGRADE; } #endif // defined(REMOTE_BOOT)
//$ REVIEW (danielwe) 2 Apr 1997: If new domain is same as old, unjoin
// then rejoin??
if (!(fIsRename) && FIsJoinedToDomain()) { // Must unjoin from domain if currently joined.
// If currently joined to a workgroup, this is not necessary.
// Also we don't unjoin if we are renaming the machine in the domain.
hr = HrNetUnjoinDomain(m_szwUserName, m_szwPassword, 0); if (FAILED(hr)) goto err; }
if (FInSystemSetup()) { // During system setup, need to pass special flag that tells join code
// to not do certain operations because SAM is not initialized yet.
dwJoinOption |= NETSETUP_INSTALL_INVOCATION; }
// If the supplied username has astring length of zero, then the join
// API's should be called with NULL's.
//
if ((NULL == m_szwUserName) || (0 == wcslen(m_szwUserName))) { fUseNulls = TRUE; }
// Go ahead and join the domain
if( fIsRename) {
hr = HrNetRenameInDomain(const_cast<PWSTR>(wszComputerName), m_szwNewDWName, (fUseNulls ? NULL : m_szwUserName), (fUseNulls ? NULL : m_szwPassword), dwJoinOption);
} else {
hr = HrNetJoinDomain(const_cast<PWSTR>(wszComputerName), m_szMachineObjectOU, m_szwNewDWName, (fUseNulls ? NULL : m_szwUserName), (fUseNulls ? NULL : m_szwPassword), dwJoinOption); }
if (FAILED(hr)) { // Note: (danielwe) Making assumption that failure to join a domain puts us in
// a workgroup. MacM owns the code responsible for this.
m_jsCur = NetSetupWorkgroupName; goto err; }
// Make the current domain name the new one since the join on the
// new domain has succeeded
hr = HrEstablishNewDomainOrWorkgroupName(NetSetupDomainName); if (FAILED(hr)) goto err;
err: // Free username and password
DeleteStringAndSetNull(&m_szwPassword); DeleteStringAndSetNull(&m_szwUserName);
TraceError("CNetCfgIdentification::HrJoinDomain", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::JoinDomain
//
// Purpose: Implements COM interface to join this computer to a new
// domain.
//
// Arguments:
// szwDomain [in] New domain name.
// szMachineObjectOU [in] Machine object OU (optional)
// szwUserName [in] User name to use in validation.
// szwPassword [in] Password to use in validation.
// dwJoinFlags [in] Currently can be 0 or JDF_CREATE_ACCOUNT.
//
// Returns: HRESULT, Error code.
//
// Author: danielwe 21 Mar 1997
//
// Notes: Validates, but does not actually join the domain. Only holds
// onto the information until Apply() is called.
//
STDMETHODIMP CNetCfgIdentification::JoinDomain(PCWSTR szwDomain, PCWSTR szMachineObjectOU, PCWSTR szwUserName, PCWSTR szwPassword, DWORD dwJoinFlags) { HRESULT hr = S_OK; static const WCHAR c_wszBackslash[] = L"\\"; static const WCHAR c_wszAt[] = L"@";
COM_PROTECT_TRY { Validate_INetCfgIdentification_JoinDomain(szwDomain, szwUserName, szwPassword);
#if defined(REMOTE_BOOT)
if (HrIsRemoteBootMachine() == S_FALSE) #endif // defined(REMOTE_BOOT)
{ // look for non-empty password and empty username or username
// consisting of only the backslash character
if (!FIsStrEmpty(szwPassword) && FIsStrEmpty(szwUserName) || !lstrcmpW(szwUserName, c_wszBackslash)) { // Password without username is invalid.
hr = E_INVALIDARG; goto err; }
PWSTR wszNewUserName; INT cchNewUserName;
// Check if username that was passed in has a backslash in it or
// an '@', or if it is empty.
if (FIsStrEmpty(szwUserName) || wcschr(szwUserName, c_wszBackslash[0]) || wcschr(szwUserName, c_wszAt[0])) { // if so, don't do anything extra
wszNewUserName = NULL; } else { // if not, we have to append the domain name to the username
cchNewUserName = lstrlenW(szwUserName) + // original username
lstrlenW(szwDomain) + // domain name
1 + // backslash character
1; // terminating NULL
wszNewUserName = new WCHAR[cchNewUserName];
if(wszNewUserName) { // Turn username into domain\username format
lstrcpyW(wszNewUserName, szwDomain); lstrcatW(wszNewUserName, c_wszBackslash); lstrcatW(wszNewUserName, szwUserName);
AssertSz(lstrlenW(wszNewUserName) + 1 == cchNewUserName, "Possible memory overwrite in username!"); } }
// Use wszNewUserName if non-NULL, otherwise use szwUserName
PCWSTR szwUserNameToCopy;
szwUserNameToCopy = wszNewUserName ? wszNewUserName : szwUserName; m_szwUserName = SzDupSz(szwUserNameToCopy); m_szwPassword = SzDupSz(szwPassword);
delete [] wszNewUserName; }
AssertSz(FImplies(m_szwPassword, m_szwUserName), "Password without username!");
hr = HrValidateDomainName(szwDomain, m_szwUserName, m_szwPassword); if (FAILED(hr)) goto err;
// Assign in new strings
m_szwNewDWName = SzDupSz(szwDomain); if (szMachineObjectOU) { m_szMachineObjectOU = SzDupSz(szMachineObjectOU); }
m_dwJoinFlags = dwJoinFlags; m_jsNew = NetSetupDomainName; err: // suppress compiler error
; } COM_PROTECT_CATCH;
TraceError("CNetCfgIdentification::JoinDomain", hr); return hr; }
//+---------------------------------------------------------------------------
//
// Member: CNetCfgIdentification::GetComputerRole
//
// Purpose: Returns the current role of the computer.
//
// Arguments:
// pdwRoleFlags [out] Returns value which determines the role of this
// computer.
//
// Returns: S_OK if success, error code otherwise.
//
// Author: danielwe 26 Mar 1997
//
// Notes: Returned role can be one of:
// SERVER_STANDALONE - The machine is part of a workgroup.
// SERVER_MEMBER - The machine is joined to the domain.
// SERVER_PDC - The machine is a primary domain controller.
// SERVER_BDC - The machine is a backup domain controller.
//
STDMETHODIMP CNetCfgIdentification::GetComputerRole(DWORD* pdwRoleFlags) { HRESULT hr = S_OK;
Validate_INetCfgIdentification_GetComputerRole(pdwRoleFlags);
COM_PROTECT_TRY { *pdwRoleFlags = 0;
hr = HrEnsureCurrentDomainOrWorkgroupName(); if (SUCCEEDED(hr)) { if (m_jsNew == NetSetupUnjoined) { // The workgroup or domain has not been changed since this
// object was instantiated
if (GetCurrentJoinStatus() == NetSetupDomainName) { *pdwRoleFlags = GCR_MEMBER; } else if (GetCurrentJoinStatus() == NetSetupWorkgroupName) { *pdwRoleFlags = GCR_STANDALONE; } #ifdef DBG
else { AssertSz(FALSE, "Invalid join status flag!"); } #endif
} else { // This means the workgroup or domain name has been changed
// since this object was instantiated
if (GetNewJoinStatus() == NetSetupDomainName) { *pdwRoleFlags = GCR_MEMBER; } else if (GetNewJoinStatus() == NetSetupWorkgroupName) { *pdwRoleFlags = GCR_STANDALONE; } #ifdef DBG
else { AssertSz(FALSE, "Invalid join status flag!"); } #endif
} } } COM_PROTECT_CATCH;
TraceError("CNetCfgIdentification::GetComputerRole", hr); return hr; }
|