mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1120 lines
42 KiB
1120 lines
42 KiB
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright(C) 1999-2000 Microsoft Corporation all rights reserved.
|
|
//
|
|
// Module: migratemdb.cpp
|
|
//
|
|
// Project: Windows 2000 IAS
|
|
//
|
|
// Description: IAS NT 4 MDB to IAS W2K MDB Migration Logic
|
|
//
|
|
// Author: TLP 1/13/1999
|
|
//
|
|
//
|
|
// Revision 02/24/2000 Moved to a separate dll
|
|
// 03/15/2000 Almost completely rewritten
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
#include "stdafx.h"
|
|
|
|
#include "Attributes.h"
|
|
#include "Clients.h"
|
|
#include "DefaultProvider.h"
|
|
#include "GlobalData.h"
|
|
#include "migratemdb.h"
|
|
#include "migrateregistry.h"
|
|
#include "Objects.h"
|
|
#include "Policy.h"
|
|
#include "Properties.h"
|
|
#include "Profiles.h"
|
|
#include "profileattributelist.h"
|
|
#include "Providers.h"
|
|
#include "proxyservergroupcollection.h"
|
|
#include "RadiusAttributeValues.h"
|
|
#include "Realms.h"
|
|
#include "RemoteRadiusServers.h"
|
|
#include "ServiceConfiguration.h"
|
|
#include "Version.h"
|
|
#include "updatemschap.h"
|
|
|
|
// To remember:
|
|
// IAS_MAX_VSA_LENGTH = (253 * 2);
|
|
// 1.0 Format Offsets
|
|
//VSA_OFFSET = 0;
|
|
//VSA_OFFSET_ID = 0;
|
|
//VSA_OFFSET_TYPE = 8;
|
|
//VSA_OFFSET_LENGTH = 10;
|
|
//VSA_OFFSET_VALUE_RFC = 12;
|
|
//VSA_OFFSET_VALUE_NONRFC = 8;
|
|
|
|
// 2.0 Format Offsets
|
|
//VSA_OFFSET_NEW = 2;
|
|
//VSA_OFFSET_ID_NEW = 2;
|
|
//VSA_OFFSET_TYPE_NEW = 10;
|
|
//VSA_OFFSET_LENGTH_NEW = 12;
|
|
//VSA_OFFSET_VALUE_NONRFC_NEW = 10;
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// Helper Functions
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
/////// From inet.c in ias util directory /////////////
|
|
//////////
|
|
// Macro to test if a character is a digit.
|
|
//////////
|
|
#define IASIsDigit(p) ((_TUCHAR)(p - _T('0')) <= 9)
|
|
|
|
//////////
|
|
// Macro to strip one byte of an IP address from a character string.
|
|
// 'p' pointer to the string to be parsed
|
|
// 'ul' unsigned long that will receive the result.
|
|
//////////
|
|
#define STRIP_BYTE(p,ul) { \
|
|
if (!IASIsDigit(*p)) goto error; \
|
|
ul = *p++ - _T('0'); \
|
|
if (IASIsDigit(*p)) { \
|
|
ul *= 10; ul += *p++ - _T('0'); \
|
|
if (IASIsDigit(*p)) { \
|
|
ul *= 10; ul += *p++ - _T('0'); \
|
|
} \
|
|
} \
|
|
if (ul > 0xff) goto error; \
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// FUNCTION
|
|
//
|
|
// ias_inet_addr
|
|
//
|
|
// DESCRIPTION
|
|
//
|
|
// This function is similar to the WinSock inet_addr function (q.v.) except
|
|
// it returns the address in host order and it can operate on both ANSI
|
|
// and UNICODE strings.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
unsigned long __stdcall ias_inet_addr(const WCHAR* cp)
|
|
{
|
|
unsigned long token;
|
|
unsigned long addr;
|
|
|
|
STRIP_BYTE(cp,addr);
|
|
if (*cp++ != _T('.')) goto error;
|
|
|
|
STRIP_BYTE(cp,token);
|
|
if (*cp++ != _T('.')) goto error;
|
|
addr <<= 8;
|
|
addr |= token;
|
|
|
|
STRIP_BYTE(cp,token);
|
|
if (*cp++ != _T('.')) goto error;
|
|
addr <<= 8;
|
|
addr |= token;
|
|
|
|
STRIP_BYTE(cp,token);
|
|
addr <<= 8;
|
|
return addr | token;
|
|
|
|
error:
|
|
return 0xffffffff;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// ConvertVSA
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::ConvertVSA(
|
|
/*[in]*/ LPCWSTR pAttributeValueName,
|
|
/*[in]*/ LPCWSTR pAttrValue,
|
|
_bstr_t& NewString
|
|
)
|
|
{
|
|
const long IAS_MAX_VSA_LENGTH = (253 * 2);
|
|
const byte VSA_OFFSET = 0;
|
|
const byte VSA_OFFSET_NEW = 2;
|
|
const byte VSA_OFFSET_VALUE_RFC_NEW = 14;
|
|
|
|
wchar_t szNewValue[IAS_MAX_VSA_LENGTH + 1];
|
|
|
|
szNewValue[0] = '\0';
|
|
|
|
// RFC compliant integer
|
|
if ( ! lstrcmp(pAttributeValueName, L"URDecimal or Hexadecimal (0x.. "
|
|
L"format) Integer") )
|
|
{
|
|
// Add the "02" as the type to the head of the string
|
|
lstrcat(szNewValue,L"02");
|
|
// Copy the old value to the new
|
|
lstrcpy(szNewValue + VSA_OFFSET_NEW, pAttrValue + VSA_OFFSET);
|
|
// Strip of the "0x" if necessary and convert to decimal
|
|
if ( ! wcsncmp(&szNewValue[VSA_OFFSET_VALUE_RFC_NEW], L"0x", 2) )
|
|
{
|
|
lstrcpy(&szNewValue[VSA_OFFSET_VALUE_RFC_NEW],
|
|
&szNewValue[VSA_OFFSET_VALUE_RFC_NEW + 2] );
|
|
}
|
|
}
|
|
// RFC compliant string
|
|
else if ( ! lstrcmp(pAttributeValueName, L"URString") )
|
|
{
|
|
// Set the new string type to "01"
|
|
lstrcat(szNewValue,L"01");
|
|
// Copy the old string to the new
|
|
lstrcpy(szNewValue + VSA_OFFSET_NEW, pAttrValue + VSA_OFFSET);
|
|
// Convert the old hex formatted string to a BSTR (in place)
|
|
wchar_t wcSaved;
|
|
wchar_t* pXlatPos = &szNewValue[VSA_OFFSET_VALUE_RFC_NEW];
|
|
wchar_t* pNewCharPos = pXlatPos;
|
|
wchar_t* pEnd;
|
|
while ( *pXlatPos != '\0' )
|
|
{
|
|
wcSaved = *(pXlatPos + 2);
|
|
*(pXlatPos + 2) = '\0';
|
|
*pNewCharPos = (wchar_t) wcstol(pXlatPos, &pEnd, 16);
|
|
*(pXlatPos + 2) = wcSaved;
|
|
pXlatPos += 2;
|
|
++pNewCharPos;
|
|
}
|
|
*pNewCharPos = '\0';
|
|
}
|
|
// RFC compliant hex
|
|
else if ( ! lstrcmp(pAttributeValueName, L"URHexadecimal") )
|
|
{
|
|
// Set the new type and copy the old string to the new
|
|
lstrcat(szNewValue,L"03");
|
|
lstrcpy(szNewValue + VSA_OFFSET_NEW, pAttrValue + VSA_OFFSET);
|
|
}
|
|
// Non RFC compliant (always hex)
|
|
else if ( ! lstrcmp(pAttributeValueName, L"UHHexadecimal") )
|
|
{
|
|
// Set the new type and copy the old string to the new
|
|
lstrcat(szNewValue,L"00");
|
|
lstrcpy(szNewValue + VSA_OFFSET_NEW, pAttrValue + VSA_OFFSET);
|
|
}
|
|
// Error
|
|
else
|
|
{
|
|
_ASSERT(FALSE);
|
|
}
|
|
|
|
// Return the new string
|
|
NewString = szNewValue;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// MigrateProxyServers
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::MigrateProxyServers()
|
|
{
|
|
const long DEFAULT_PRIORITY = 1;
|
|
const long DEFAULT_WEIGHT = 50;
|
|
|
|
// If there isn't any servers.
|
|
if ( m_GlobalData.m_pRadiusServers->IsEmpty() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
CProxyServerGroupCollection& ServerCollection
|
|
= CProxyServerGroupCollection::Instance();
|
|
|
|
// Do a loop on the RadiusServers sorted by server group
|
|
_bstr_t CurrentGroupName;
|
|
|
|
CProxyServersGroupHelper* pCurrentServerGroup = NULL; //avoid warning
|
|
|
|
// CurrentGroupName will never match a name received therefore
|
|
// pCurrentServerGroup will always be initialized before being used
|
|
HRESULT hr;
|
|
do
|
|
{
|
|
_bstr_t GroupName = m_GlobalData.m_pRadiusServers->GetGroupName();
|
|
|
|
if ( CurrentGroupName != GroupName )
|
|
{
|
|
CProxyServersGroupHelper ServerGroup(m_GlobalData);
|
|
CurrentGroupName = GroupName;
|
|
|
|
ServerGroup.SetName(GroupName);
|
|
|
|
// Add a server to the collection
|
|
pCurrentServerGroup = ServerCollection.Add(ServerGroup);
|
|
}
|
|
|
|
if ( !pCurrentServerGroup )
|
|
{
|
|
_com_issue_error(E_FAIL);
|
|
}
|
|
|
|
CProxyServerHelper Server(m_GlobalData);
|
|
|
|
_bstr_t ServerName = m_GlobalData.m_pRadiusServers->
|
|
GetProxyServerName();
|
|
Server.SetAddress(ServerName);
|
|
Server.SetAuthenticationPort(
|
|
m_GlobalData.m_pRadiusServers->GetAuthenticationPortNumber()
|
|
);
|
|
Server.SetAccountingPort(
|
|
m_GlobalData.m_pRadiusServers->GetAccountingPortNumber()
|
|
);
|
|
_bstr_t Secret = m_GlobalData.m_pRadiusServers->GetSharedSecret();
|
|
Server.SetAuthenticationSecret(Secret);
|
|
Server.SetPriority(DEFAULT_PRIORITY);
|
|
Server.SetWeight(DEFAULT_WEIGHT);
|
|
|
|
pCurrentServerGroup->Add(Server); // cannot be NULL pointer
|
|
|
|
hr = m_GlobalData.m_pRadiusServers->GetNext();
|
|
}
|
|
while (hr == S_OK);
|
|
// persist everything in the database
|
|
ServerCollection.Persist();
|
|
|
|
_com_util::CheckError(hr);
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// NewMigrateClients
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::NewMigrateClients()
|
|
{
|
|
// If there isn't any client, return
|
|
if ( m_GlobalData.m_pClients->IsEmpty() )
|
|
{
|
|
return;
|
|
}
|
|
|
|
// for each client in the client table, add it (blindly) to the dest table
|
|
// i.e. walkpath to find the clients container
|
|
// create the properties for that containes (clients).
|
|
const WCHAR ClientContainerPath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"Protocols\0"
|
|
L"Microsoft Radius Protocol\0"
|
|
L"Clients\0";
|
|
|
|
LONG ClientContainerIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(
|
|
ClientContainerPath,
|
|
ClientContainerIdentity
|
|
);
|
|
|
|
HRESULT hr;
|
|
do
|
|
{
|
|
_bstr_t ClientName = m_GlobalData.m_pClients->GetHostName();
|
|
_bstr_t ClientSecret = m_GlobalData.m_pClients->GetSecret();
|
|
|
|
LONG ClientIdentity;
|
|
m_GlobalData.m_pObjects->InsertObject(
|
|
ClientName,
|
|
ClientContainerIdentity,
|
|
ClientIdentity
|
|
);
|
|
|
|
// Now insert the properties:
|
|
// IP Address
|
|
_bstr_t PropertyName = L"IP Address";
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
ClientIdentity,
|
|
PropertyName,
|
|
VT_BSTR,
|
|
ClientName
|
|
);
|
|
// NAS Manufacturer
|
|
PropertyName = L"NAS Manufacturer";
|
|
_bstr_t StrValZero = L"0"; // RADIUS Standard
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
ClientIdentity,
|
|
PropertyName,
|
|
VT_I4,
|
|
StrValZero
|
|
);
|
|
// Require Signature
|
|
PropertyName = L"Require Signature";
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
ClientIdentity,
|
|
PropertyName,
|
|
VT_BOOL,
|
|
StrValZero
|
|
);
|
|
// Shared Secret
|
|
PropertyName = L"Shared Secret";
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
ClientIdentity,
|
|
PropertyName,
|
|
VT_BSTR,
|
|
ClientSecret
|
|
);
|
|
hr = m_GlobalData.m_pClients->GetNext();
|
|
}
|
|
while ( hr == S_OK );
|
|
|
|
_com_util::CheckError(hr);
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// ConvertAttribute
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::ConvertAttribute(
|
|
const _bstr_t& Value,
|
|
LONG Syntax,
|
|
LONG& Type,
|
|
bstr_t& StrVal
|
|
)
|
|
{
|
|
const size_t SIZE_LONG_MAX = 14;
|
|
switch (Syntax)
|
|
{
|
|
case IAS_SYNTAX_OCTETSTRING:
|
|
{
|
|
// abinary => OctetString
|
|
Type = VT_BSTR;
|
|
StrVal = Value;
|
|
break;
|
|
}
|
|
|
|
case IAS_SYNTAX_STRING:
|
|
case IAS_SYNTAX_UTCTIME:
|
|
case IAS_SYNTAX_PROVIDERSPECIFIC:
|
|
{
|
|
Type = VT_BSTR;
|
|
StrVal = Value;
|
|
break;
|
|
}
|
|
case IAS_SYNTAX_INETADDR:
|
|
{
|
|
unsigned long ulValue = ias_inet_addr(Value);
|
|
_ASSERT( ulValue != 0xffffffff );
|
|
Type = VT_I4;
|
|
WCHAR TempString[SIZE_LONG_MAX];
|
|
StrVal = _ultow(ulValue, TempString, 10);
|
|
break;
|
|
}
|
|
case IAS_SYNTAX_BOOLEAN:
|
|
{
|
|
LONG lValue = _wtol(Value);
|
|
Type = VT_BOOL;
|
|
StrVal = lValue? L"-1":L"0";
|
|
break;
|
|
}
|
|
case IAS_SYNTAX_INTEGER:
|
|
case IAS_SYNTAX_UNSIGNEDINTEGER:
|
|
case IAS_SYNTAX_ENUMERATOR:
|
|
{
|
|
Type = VT_I4;
|
|
StrVal = Value;
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
_com_issue_error(E_INVALIDARG);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// MigrateAttribute
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::MigrateAttribute(
|
|
const _bstr_t& Attribute,
|
|
LONG AttributeNumber,
|
|
const _bstr_t& AttributeValueName,
|
|
const _bstr_t& StringValue,
|
|
LONG RASProfileIdentity
|
|
)
|
|
{
|
|
// Note: Order might not be needed if the previous DB is always sorted
|
|
const size_t SIZE_LONG_MAX = 14;
|
|
_bstr_t LDAPName, StrVal;
|
|
LONG Syntax, Type; // Type: VT_BSTR, VT_I4, VT_BOOL
|
|
BOOL IsMultiValued;
|
|
HRESULT hr = m_GlobalData.m_pAttributes->GetAttribute(
|
|
AttributeNumber,
|
|
LDAPName,
|
|
Syntax,
|
|
IsMultiValued
|
|
);
|
|
if ( FAILED(hr) )
|
|
{
|
|
// Attribute unknown in the new dictionary
|
|
// that should never happen
|
|
return;
|
|
}
|
|
|
|
const LONG VSA = 26; //Vendor-Specific-Attribute
|
|
if ( StringValue.length() && ( AttributeNumber != VSA) )
|
|
{
|
|
// ordinary attribute, not an enumerator
|
|
ConvertAttribute(
|
|
StringValue,
|
|
Syntax,
|
|
Type,
|
|
StrVal
|
|
);
|
|
if ( IsMultiValued )
|
|
{
|
|
// If the attribute is multivalued then we need to add the value
|
|
// Otherwise we just update the attribute value
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
RASProfileIdentity,
|
|
LDAPName,
|
|
Type,
|
|
StrVal
|
|
);
|
|
}
|
|
else
|
|
{
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
RASProfileIdentity,
|
|
LDAPName,
|
|
Type,
|
|
StrVal
|
|
);
|
|
}
|
|
}
|
|
else if ( StringValue.length() && ( AttributeNumber == VSA) )
|
|
{
|
|
// VSA Attribute (convert...)
|
|
if ( !AttributeValueName )
|
|
{
|
|
_com_issue_error(E_INVALIDARG);
|
|
}
|
|
ConvertVSA(AttributeValueName, StringValue, StrVal);
|
|
Type = VT_BSTR;
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
RASProfileIdentity,
|
|
LDAPName,
|
|
Type,
|
|
StrVal
|
|
);
|
|
}
|
|
else if ( !StringValue.length() )
|
|
{
|
|
// Multivalued attribute.
|
|
Type = VT_I4;
|
|
// Get the number associated with it from the RadiusAttributeValues
|
|
// table
|
|
LONG Number = m_GlobalData.m_pRADIUSAttributeValues->GetAttributeNumber
|
|
(
|
|
Attribute,
|
|
AttributeValueName
|
|
);
|
|
|
|
WCHAR TempString[SIZE_LONG_MAX ];
|
|
StrVal = _ltow(Number, TempString, 10);
|
|
|
|
// The attribute should be multivalued
|
|
_ASSERTE(IsMultiValued);
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
RASProfileIdentity,
|
|
LDAPName,
|
|
Type,
|
|
StrVal
|
|
);
|
|
|
|
}
|
|
else
|
|
{
|
|
// other (unknown)
|
|
_com_issue_error(E_FAIL);
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// MigrateOtherProfile
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::MigrateOtherProfile(
|
|
const _bstr_t& ProfileName,
|
|
LONG ProfileIdentity
|
|
)
|
|
{
|
|
_bstr_t Attribute, AttributeValueName, StringValue;
|
|
LONG Order, AttributeNumber;
|
|
|
|
// Now add the attributes from the NT4 file
|
|
HRESULT hr = m_GlobalData.m_pProfileAttributeList->GetAttribute(
|
|
ProfileName,
|
|
Attribute,
|
|
AttributeNumber,
|
|
AttributeValueName,
|
|
StringValue,
|
|
Order
|
|
);
|
|
LONG IndexAttribute = 1;
|
|
// For each attribute in the Profile Attribute List
|
|
// with szProfile = ProfileName
|
|
while ( SUCCEEDED(hr) )
|
|
{
|
|
// migrate it to the default RAS profile in IAS.mdb
|
|
MigrateAttribute(
|
|
Attribute,
|
|
AttributeNumber,
|
|
AttributeValueName,
|
|
StringValue,
|
|
ProfileIdentity
|
|
);
|
|
|
|
hr = m_GlobalData.m_pProfileAttributeList->GetAttribute(
|
|
ProfileName,
|
|
Attribute,
|
|
AttributeNumber,
|
|
AttributeValueName,
|
|
StringValue,
|
|
Order,
|
|
IndexAttribute
|
|
);
|
|
++IndexAttribute;
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// MigrateCorpProfile
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::MigrateCorpProfile(
|
|
const _bstr_t& ProfileName,
|
|
const _bstr_t& Description
|
|
)
|
|
{
|
|
_bstr_t Attribute, AttributeValueName, StringValue;
|
|
LONG Order, AttributeNumber;
|
|
|
|
// empty the default profiles attributes
|
|
const WCHAR RASProfilePath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"RadiusProfiles\0";
|
|
|
|
LONG RASProfileIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(RASProfilePath, RASProfileIdentity);
|
|
|
|
// Now get the first profile: that's the default (localized) RAS profile
|
|
_bstr_t DefaultProfileName;
|
|
LONG DefaultProfileIdentity;
|
|
m_GlobalData.m_pObjects->GetObject(
|
|
DefaultProfileName,
|
|
DefaultProfileIdentity,
|
|
RASProfileIdentity
|
|
);
|
|
|
|
// Clean the default attributes
|
|
m_GlobalData.m_pProperties->DeleteProperty(
|
|
DefaultProfileIdentity,
|
|
L"msRADIUSServiceType"
|
|
);
|
|
m_GlobalData.m_pProperties->DeleteProperty(
|
|
DefaultProfileIdentity,
|
|
L"msRADIUSFramedProtocol"
|
|
);
|
|
|
|
// Now add the attributes from the NT4 file
|
|
HRESULT hr = m_GlobalData.m_pProfileAttributeList->GetAttribute(
|
|
ProfileName,
|
|
Attribute,
|
|
AttributeNumber,
|
|
AttributeValueName,
|
|
StringValue,
|
|
Order
|
|
);
|
|
LONG IndexAttribute = 1;
|
|
// For each attribute in the Profile Attribute List
|
|
// with szProfile = ProfileName
|
|
while ( SUCCEEDED(hr) )
|
|
{
|
|
// migrate it to the default RAS profile in IAS.mdb
|
|
MigrateAttribute(
|
|
Attribute,
|
|
AttributeNumber,
|
|
AttributeValueName,
|
|
StringValue,
|
|
DefaultProfileIdentity
|
|
);
|
|
|
|
hr = m_GlobalData.m_pProfileAttributeList->GetAttribute(
|
|
ProfileName,
|
|
Attribute,
|
|
AttributeNumber,
|
|
AttributeValueName,
|
|
StringValue,
|
|
Order,
|
|
IndexAttribute
|
|
);
|
|
++IndexAttribute;
|
|
}
|
|
|
|
// now not matter what, if Description == ODBC,
|
|
// Then the policy should have a condition to never match.
|
|
// (update msNPConstraint
|
|
const _bstr_t BadProvider = L"ODBC";
|
|
if ( Description == BadProvider ) //safe compare
|
|
{
|
|
// Get the Policy container
|
|
const WCHAR RASPolicyPath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"NetworkPolicy\0";
|
|
// Get its (unique) child
|
|
LONG RASPolicyIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(RASPolicyPath, RASPolicyIdentity);
|
|
|
|
// Now get the first policy: that's the default (localized) RAS policy
|
|
_bstr_t DefaultPolicyName;
|
|
LONG DefaultPolicyIdentity;
|
|
m_GlobalData.m_pObjects->GetObject(
|
|
DefaultPolicyName,
|
|
DefaultPolicyIdentity,
|
|
RASPolicyIdentity
|
|
);
|
|
// delete the msNPConstraint (s)
|
|
const _bstr_t Constraint = L"msNPConstraint";
|
|
m_GlobalData.m_pProperties->DeleteProperty(
|
|
DefaultPolicyIdentity,
|
|
Constraint
|
|
);
|
|
// add a TIMEOFDAY that never matches
|
|
const _bstr_t DumbTime = L"TIMEOFDAY(\"\")";
|
|
m_GlobalData.m_pProperties->InsertProperty(
|
|
RASPolicyIdentity,
|
|
Constraint,
|
|
VT_BSTR,
|
|
DumbTime
|
|
);
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// NewMigrateProfiles
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::NewMigrateProfiles()
|
|
{
|
|
const LONG AUTH_PROVIDER_WINDOWS = 1;
|
|
const LONG AUTH_PROVIDER_RADIUS_PROXY = 2;
|
|
const LONG ACCT_PROVIDER_RADIUS_PROXY = 2;
|
|
const _bstr_t RemoteRADIUSServers = L"Remote RADIUS Servers";
|
|
const _bstr_t MCIS = L"MCIS";
|
|
const _bstr_t MCISv2 = L"MCIS version 2.0";
|
|
const _bstr_t ODBC = L"ODBC";
|
|
const _bstr_t WindowsNT = L"Windows NT";
|
|
const _bstr_t MatchAll = L"TIMEOFDAY(\"0 00:00-24:00; 1 00:00"
|
|
L"-24:00; 2 00:00-24:00; 3 00:00-24:00; 4 00:00-24:00; 5 00:"
|
|
L"00-24:00; 6 00:00-24:00\")";
|
|
|
|
// Get the Default Provider's data
|
|
_bstr_t DPUserDefinedName, DPProfile;
|
|
VARIANT_BOOL DPForwardAccounting, DPSupressAccounting
|
|
, DPLogoutAccounting;
|
|
|
|
m_GlobalData.m_pDefaultProvider->GetDefaultProvider(
|
|
DPUserDefinedName,
|
|
DPProfile,
|
|
DPForwardAccounting,
|
|
DPSupressAccounting,
|
|
DPLogoutAccounting
|
|
);
|
|
_bstr_t ProfileName = m_GlobalData.m_pProfiles->GetProfileName();
|
|
|
|
// Do the NT4 CORP migration first if needed
|
|
if ( m_Utils.IsNT4Corp() )
|
|
{
|
|
// This is NT4 CORP. migrate the default profile into
|
|
// the default policy/profile (not the proxy default)
|
|
_bstr_t Description = m_GlobalData.m_pProviders->
|
|
GetProviderDescription(DPUserDefinedName);
|
|
|
|
MigrateCorpProfile(ProfileName, Description);
|
|
|
|
// stop here
|
|
return;
|
|
}
|
|
// Now that is not a NT4 CORP migration
|
|
|
|
// Delete the default Proxy Policy / profile
|
|
const WCHAR ProxyPoliciesPath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"Proxy Policies\0";
|
|
|
|
LONG ProxyPolicyIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(ProxyPoliciesPath, ProxyPolicyIdentity);
|
|
|
|
// Now get the first profile: that's the default (localized) RAS policy
|
|
_bstr_t DefaultPolicyName;
|
|
LONG DefaultPolicyIdentity;
|
|
m_GlobalData.m_pObjects->GetObject(
|
|
DefaultPolicyName,
|
|
DefaultPolicyIdentity,
|
|
ProxyPolicyIdentity
|
|
);
|
|
m_GlobalData.m_pObjects->DeleteObject(DefaultPolicyIdentity);
|
|
// From now on the default proxy policy / profile is deleted
|
|
|
|
// Now empty the default RAS profiles attributes
|
|
const WCHAR RASProfilePath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"RadiusProfiles\0";
|
|
|
|
LONG RASProfileIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(RASProfilePath, RASProfileIdentity);
|
|
|
|
// Now get the first profile: that's the default (localized) RAS profile
|
|
_bstr_t DefaultProfileName;
|
|
LONG DefaultProfileIdentity;
|
|
m_GlobalData.m_pObjects->GetObject(
|
|
DefaultProfileName,
|
|
DefaultProfileIdentity,
|
|
RASProfileIdentity
|
|
);
|
|
|
|
// Clean the default attributes
|
|
m_GlobalData.m_pProperties->DeleteProperty(
|
|
DefaultProfileIdentity,
|
|
L"msRADIUSServiceType"
|
|
);
|
|
m_GlobalData.m_pProperties->DeleteProperty(
|
|
DefaultProfileIdentity,
|
|
L"msRADIUSFramedProtocol"
|
|
);
|
|
|
|
// from now on the default RAS profile has its default attributes (the one
|
|
// in the Advanced tab in the UI) deleted.
|
|
|
|
HRESULT hr;
|
|
LONG Sequence = 1;
|
|
// Get the list of the profiles
|
|
do
|
|
{
|
|
LONG RealmIndex = 0;
|
|
ProfileName = m_GlobalData.m_pProfiles->GetProfileName();
|
|
|
|
// Note: hr should be set only by GetRealmIndex
|
|
do
|
|
{
|
|
// Get the realms associated with that profile
|
|
CPolicy TempPolicy;
|
|
hr = m_GlobalData.m_pRealms->GetRealmIndex(ProfileName,RealmIndex);
|
|
if ( hr != S_OK )
|
|
{
|
|
// exit that internal do / while to get the next profile
|
|
break;
|
|
}
|
|
_bstr_t RealmName = m_GlobalData.m_pRealms->GetRealmName();
|
|
TempPolicy.SetmsNPAction(RealmName);
|
|
|
|
++RealmIndex;
|
|
|
|
// That will set the realm part of the profile based on the
|
|
// values in NT4 as well as the reg keys values
|
|
m_GlobalData.m_pRealms->SetRealmDetails(
|
|
TempPolicy,
|
|
m_Utils
|
|
);
|
|
|
|
|
|
_bstr_t UserDefinedName = m_GlobalData.m_pRealms
|
|
->GetUserDefinedName();
|
|
// Look up the provider in the providers table. Note: Assume
|
|
// the proxy servers (and groups) are already migrated.
|
|
_bstr_t ProviderDescription = m_GlobalData.m_pProviders
|
|
->GetProviderDescription(UserDefinedName);
|
|
|
|
// Set the sequence order
|
|
TempPolicy.SetmsNPSequence(Sequence);
|
|
|
|
// Now set the authentication provider
|
|
if ( ProviderDescription == RemoteRADIUSServers )
|
|
{
|
|
TempPolicy.SetmsAuthProviderType(
|
|
AUTH_PROVIDER_RADIUS_PROXY,
|
|
UserDefinedName
|
|
);
|
|
}
|
|
else if ( ( ProviderDescription == MCIS ) ||
|
|
( ProviderDescription == MCISv2 ) ||
|
|
( ProviderDescription == WindowsNT ) )
|
|
{
|
|
TempPolicy.SetmsAuthProviderType(AUTH_PROVIDER_WINDOWS);
|
|
}
|
|
else if ( ProviderDescription == ODBC )
|
|
{
|
|
// If ODBC is the authentication provider,
|
|
// then convert that realm into a policy that would never match.
|
|
// Authentication provider should be NT Domain.
|
|
TempPolicy.SetmsAuthProviderType(AUTH_PROVIDER_WINDOWS);
|
|
const _bstr_t MatchNothing = L"TIMEOFDAY(\"\")";
|
|
TempPolicy.SetmsNPConstraint(MatchNothing);
|
|
}
|
|
else
|
|
{
|
|
_com_issue_error(E_INVALIDARG);
|
|
}
|
|
|
|
// persist the policy
|
|
LONG ProfileIdentity = TempPolicy.Persist(m_GlobalData);
|
|
|
|
// migrate the profile associated with that policy
|
|
MigrateOtherProfile(ProfileName, ProfileIdentity);
|
|
|
|
++Sequence;
|
|
} while (hr == S_OK);
|
|
|
|
hr = m_GlobalData.m_pProfiles->GetNext();
|
|
} while ( hr == S_OK );
|
|
|
|
if ( DPUserDefinedName.length() )
|
|
{
|
|
// there is a default provider: a default policy needs to be created
|
|
// same logic as above (mostly)
|
|
CPolicy DefaultPolicy;
|
|
DefaultPolicy.SetmsNPAction(DPProfile);
|
|
|
|
_bstr_t ProviderDescription = m_GlobalData.m_pProviders
|
|
->GetProviderDescription(DPUserDefinedName);
|
|
|
|
if ( ProviderDescription == RemoteRADIUSServers )
|
|
{
|
|
DefaultPolicy.SetmsAuthProviderType(
|
|
AUTH_PROVIDER_RADIUS_PROXY,
|
|
DPUserDefinedName
|
|
);
|
|
DefaultPolicy.SetmsNPConstraint(MatchAll);
|
|
}
|
|
else if ( ( ProviderDescription == MCIS ) ||
|
|
( ProviderDescription == MCISv2 ) ||
|
|
( ProviderDescription == WindowsNT ) )
|
|
{
|
|
DefaultPolicy.SetmsNPConstraint(MatchAll);
|
|
DefaultPolicy.SetmsAuthProviderType(AUTH_PROVIDER_WINDOWS);
|
|
}
|
|
else if ( ProviderDescription == ODBC )
|
|
{
|
|
// If ODBC is the authentication provider,
|
|
// then convert that realm into a policy that would never match.
|
|
// Authentication provider should be NT Domain.
|
|
DefaultPolicy.SetmsAuthProviderType(AUTH_PROVIDER_WINDOWS);
|
|
const _bstr_t MatchNothing = L"TIMEOFDAY(\"\")";
|
|
DefaultPolicy.SetmsNPConstraint(MatchNothing);
|
|
}
|
|
else
|
|
{
|
|
_com_issue_error(E_INVALIDARG);
|
|
}
|
|
|
|
DefaultPolicy.SetmsNPSequence(Sequence);
|
|
if ( DPForwardAccounting )
|
|
{
|
|
DefaultPolicy.SetmsAcctProviderType(ACCT_PROVIDER_RADIUS_PROXY);
|
|
}
|
|
|
|
LONG ProfileIdentity = DefaultPolicy.Persist(m_GlobalData);
|
|
|
|
MigrateOtherProfile(DPProfile, ProfileIdentity);
|
|
}
|
|
// else no default provider: no default policy
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// NewMigrateAccounting
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::NewMigrateAccounting()
|
|
{
|
|
const WCHAR AccountingPath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"RequestHandlers\0"
|
|
L"Microsoft Accounting\0";
|
|
|
|
LONG AccountingIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(AccountingPath, AccountingIdentity);
|
|
|
|
_bstr_t MaxLogSize = m_GlobalData.m_pServiceConfiguration->GetMaxLogSize();
|
|
|
|
_bstr_t LogFrequency = m_GlobalData.m_pServiceConfiguration->
|
|
GetLogFrequency();
|
|
|
|
|
|
_bstr_t PropertyName = L"New Log Frequency";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
AccountingIdentity,
|
|
PropertyName,
|
|
VT_I4,
|
|
LogFrequency
|
|
);
|
|
|
|
PropertyName = L"New Log Size";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
AccountingIdentity,
|
|
PropertyName,
|
|
VT_I4,
|
|
MaxLogSize
|
|
);
|
|
DWORD Value;
|
|
m_Utils.NewGetAuthSrvParameter(L"LogAuthentications", Value);
|
|
|
|
_bstr_t LogAuth;
|
|
Value ? LogAuth = L"-1": LogAuth = L"0";
|
|
|
|
PropertyName = L"Log Authentication Packets";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
AccountingIdentity,
|
|
PropertyName,
|
|
VT_BOOL,
|
|
LogAuth
|
|
);
|
|
|
|
m_Utils.NewGetAuthSrvParameter(L"LogAccounting", Value);
|
|
|
|
_bstr_t LogAcct;
|
|
Value ? LogAcct = L"-1": LogAcct = L"0";
|
|
|
|
PropertyName = L"Log Accounting Packets";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
AccountingIdentity,
|
|
PropertyName,
|
|
VT_BOOL,
|
|
LogAcct
|
|
);
|
|
|
|
_bstr_t FormatIAS1 = L"0";
|
|
PropertyName = L"Log Format";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
AccountingIdentity,
|
|
PropertyName,
|
|
VT_I4,
|
|
FormatIAS1
|
|
);
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// NewMigrateEventLog
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::NewMigrateEventLog()
|
|
{
|
|
const WCHAR EventLogPath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"Auditors\0"
|
|
L"Microsoft NT Event Log Auditor\0";
|
|
|
|
LONG EventLogIdentity;
|
|
m_GlobalData.m_pObjects->WalkPath(EventLogPath, EventLogIdentity);
|
|
|
|
DWORD Value;
|
|
m_Utils.NewGetAuthSrvParameter(L"LogData", Value);
|
|
|
|
_bstr_t LogData;
|
|
Value ? LogData = L"-1": LogData = L"0";
|
|
|
|
_bstr_t PropertyName = L"Log Verbose";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
EventLogIdentity,
|
|
PropertyName,
|
|
VT_BOOL,
|
|
LogData
|
|
);
|
|
|
|
m_Utils.NewGetAuthSrvParameter(L"LogBogus", Value);
|
|
|
|
_bstr_t LogBogus;
|
|
Value ? LogBogus = L"-1": LogBogus = L"0";
|
|
|
|
PropertyName = L"Log Malformed Packets";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
EventLogIdentity,
|
|
PropertyName,
|
|
VT_BOOL,
|
|
LogBogus
|
|
);
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// NewMigrateService
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::NewMigrateService()
|
|
{
|
|
const LONG PORT_SIZE_MAX = 34;
|
|
const WCHAR ServicePath[] =
|
|
L"Root\0"
|
|
L"Microsoft Internet Authentication Service\0"
|
|
L"Protocols\0"
|
|
L"Microsoft Radius Protocol\0";
|
|
|
|
LONG ServiceIdentity;
|
|
|
|
m_GlobalData.m_pObjects->WalkPath(ServicePath, ServiceIdentity);
|
|
|
|
|
|
DWORD Value;
|
|
m_Utils.NewGetAuthSrvParameter(L"RadiusPort", Value);
|
|
|
|
WCHAR TempString[PORT_SIZE_MAX];
|
|
_bstr_t RadiusPort = _ultow(Value, TempString, 10);
|
|
|
|
|
|
_bstr_t PropertyName = L"Authentication Port";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
ServiceIdentity,
|
|
PropertyName,
|
|
VT_BSTR,
|
|
RadiusPort
|
|
);
|
|
|
|
m_Utils.NewGetAuthSrvParameter(L"AcctPort", Value);
|
|
|
|
_bstr_t AcctPort = _ltow(Value, TempString, 10);
|
|
|
|
|
|
PropertyName = L"Accounting Port";
|
|
m_GlobalData.m_pProperties->UpdateProperty(
|
|
ServiceIdentity,
|
|
PropertyName,
|
|
VT_BSTR,
|
|
AcctPort
|
|
);
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
// NewMigrate
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
void CMigrateMdb::NewMigrate()
|
|
{
|
|
NewMigrateClients();
|
|
|
|
if ( !m_Utils.IsNT4Corp() ) // it's either Win2k or NT4 ISP
|
|
{
|
|
// the proxy servers must be migrated before the policies and
|
|
// profiles
|
|
MigrateProxyServers();
|
|
}
|
|
|
|
NewMigrateProfiles();
|
|
NewMigrateAccounting();
|
|
NewMigrateEventLog();
|
|
NewMigrateService();
|
|
|
|
/////////////////////////////
|
|
// Migrate the Registry Keys
|
|
/////////////////////////////
|
|
CMigrateRegistry MigrateRegistry(m_Utils);
|
|
MigrateRegistry.MigrateProviders();
|
|
|
|
//////////////////////////////////////////////////////
|
|
// Update the MSChap Authentication types (password)
|
|
//////////////////////////////////////////////////////
|
|
CUpdateMSCHAP UpdateMSCHAP(m_GlobalData);
|
|
UpdateMSCHAP.Execute();
|
|
}
|
|
|