Leaked source code of windows server 2003
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.
 
 
 
 
 
 

996 lines
21 KiB

/*++
Copyright (c) 1994-2001 Microsoft Corporation
Module Name :
wservic.cpp
Abstract:
WWW Service Property Page
Author:
Ronald Meijer (ronaldm)
Sergei Antonov (sergeia)
Project:
Internet Services Manager
Revision History:
--*/
#include "stdafx.h"
#include "resource.h"
#include "common.h"
#include "inetmgrapp.h"
#include "inetprop.h"
#include "shts.h"
#include "w3sht.h"
#include "wservic.h"
#include "mmmdlg.h"
#include "iisobj.h"
#ifdef _DEBUG
#undef THIS_FILE
static char BASED_CODE THIS_FILE[] = __FILE__;
#endif
//
// Values for PWS
//
#define LIMITED_CONNECTIONS_MIN (10)
#define LIMITED_CONNECTIONS_MAX (40)
//
// Default SSL port
//
#define DEFAULT_SSL_PORT (441)
//#define ZERO_IS_A_VALID_SSL_PORT
IMPLEMENT_DYNCREATE(CW3ServicePage, CInetPropertyPage)
CW3ServicePage::CW3ServicePage(CInetPropertySheet * pSheet)
: CInetPropertyPage(CW3ServicePage::IDD, pSheet),
m_nSSLPort(DEFAULT_SSL_PORT),
m_nTCPPort(80),
m_iSSL(-1),
m_iaIpAddress(NULL_IP_ADDRESS),
m_iaIpAddressSSL(NULL_IP_ADDRESS),
m_strDomainName()
{
#ifdef _DEBUG
afxMemDF |= checkAlwaysMemDF;
#endif // _DEBUG
#if 0 // Keep Class Wizard Happy
//{{AFX_DATA_INIT(CW3ServicePage)
m_nUnlimited = RADIO_LIMITED;
m_nIpAddressSel = -1;
m_nTCPPort = 80;
m_fEnableLogging = FALSE;
m_fUseKeepAlives = FALSE;
m_strComment = _T("");
m_strDomainName = _T("");
m_nSSLPort = DEFAULT_SSL_PORT;
//}}AFX_DATA_INIT
m_iaIpAddress = (LONG)0L;
m_iaIpAddressSSL = (LONG)0L;
m_nMaxConnections = 50;
m_nVisibleMaxConnections = 50;
m_nConnectionTimeOut = 600;
m_nSSLPort = DEFAULT_SSL_PORT;
m_fUnlimitedConnections = FALSE;
#endif // 0
}
CW3ServicePage::~CW3ServicePage()
{
}
void
CW3ServicePage::GetTopBinding()
/*++
Routine Description:
Get the first binding information in the list
--*/
{
//
// Show primary values;
//
ASSERT(m_strlBindings.GetCount() > 0 || IS_MASTER_INSTANCE(QueryInstance()));
if (m_strlBindings.GetCount() > 0)
{
CString & strBinding = m_strlBindings.GetHead();
CInstanceProps::CrackBinding(strBinding, m_iaIpAddress, m_nTCPPort, m_strDomainName);
}
m_iSSL = -1;
m_nSSLPort = -1;
if (m_strlSecureBindings.GetCount() > 0)
{
CString strDomainName;
CString & strBindingSSL = m_strlSecureBindings.GetHead();
CInstanceProps::CrackBinding(strBindingSSL, m_iaIpAddressSSL, m_nSSLPort, strDomainName);
//
// Find SSL port that is bound to this IP address
//
m_iSSL = CInstanceProps::FindMatchingSecurePort(
m_strlSecureBindings, m_iaIpAddressSSL, m_nSSLPort);
if (-1 == m_iSSL)
{
m_nSSLPort = -1;
}
}
}
BOOL
CW3ServicePage::StoreTopBinding()
/*++
Routine Description:
Take values from the dialog, and put them into the top level
binding string.
Arguments:
None
Return Value:
TRUE if the values are correct, FALSE otherwise.
--*/
{
if (!FetchIpAddressFromCombo(m_combo_IpAddresses, m_oblIpAddresses, m_iaIpAddress))
{
//
// Because UpdateData() is called before this, this should NEVER fail
//
ASSERT(FALSE);
return FALSE;
}
CString strBinding;
ASSERT(m_nTCPPort > 0);
if (m_nTCPPort == m_nSSLPort)
{
//
// TCP port and SSL port cannot be the same
//
EditShowBalloon(GetDlgItem(IDC_EDIT_SSL_PORT)->m_hWnd, IDS_TCP_SSL_PART);
return FALSE;
}
CInstanceProps::BuildBinding(strBinding, m_iaIpAddress, m_nTCPPort, m_strDomainName);
//
// Check binding ok
//
if (m_strlBindings.GetCount() > 0)
{
if (!IsBindingUnique(strBinding, m_strlBindings, 0))
{
EditShowBalloon(GetDlgItem(IDC_EDIT_TCP_PORT)->m_hWnd, IDS_ERR_BINDING);
return FALSE;
}
m_strlBindings.SetAt(m_strlBindings.GetHeadPosition(), strBinding);
}
else
{
m_strlBindings.AddTail(strBinding);
}
//
// Now do the same for the SSL binding
//
// if (m_fCertInstalled)
// {
#ifdef ZERO_IS_A_VALID_SSL_PORT
if (m_nSSLPort != -1)
#else
if (m_nSSLPort > 0 && m_nSSLPort != -1)
#endif
{
CInstanceProps::BuildSecureBinding(strBinding, m_iaIpAddressSSL, m_nSSLPort);
if (m_strlSecureBindings.GetCount() > 0)
{
if (IsBindingUnique(strBinding, m_strlSecureBindings, m_iSSL))
{
//
// Find its place
//
if (m_iSSL != -1)
{
//
// Replace selected entry
//
m_strlSecureBindings.SetAt(
m_strlSecureBindings.FindIndex(m_iSSL), strBinding);
}
else
{
//
// Add to end of list
//
ASSERT(!m_strlSecureBindings.IsEmpty());
m_strlSecureBindings.AddTail(strBinding);
m_iSSL = (int)m_strlSecureBindings.GetCount() - 1;
}
}
else
{
//
// Entry already existed in the list. This is OK, just
// delete the current entry rather than bothering
// to change it.
//
ASSERT(m_iSSL != -1);
if (m_iSSL != -1)
{
m_strlSecureBindings.RemoveAt(
m_strlSecureBindings.FindIndex(m_iSSL)
);
m_iSSL = CInstanceProps::FindMatchingSecurePort(
m_strlSecureBindings, m_iaIpAddress, m_nSSLPort);
ASSERT(m_iSSL != -1);
if (-1 == m_iSSL)
{
m_nSSLPort = -1;
}
}
}
}
else
{
//
// List of secure bindings was empty, add new entry
//
m_strlSecureBindings.AddTail(strBinding);
m_iSSL = 0;
}
}
else
{
//
// Delete the secure binding if it did exist
//
if (m_iSSL != -1)
{
m_strlSecureBindings.RemoveAt(
m_strlSecureBindings.FindIndex(m_iSSL)
);
m_iSSL = -1;
}
}
// }
return TRUE;
}
void
CW3ServicePage::DoDataExchange(
IN CDataExchange * pDX
)
/*++
Routine Description:
Initialise/Store Control Data
Arguments:
CDataExchange * pDX : Data exchange object
Return Value:
None
--*/
{
CInetPropertyPage::DoDataExchange(pDX);
if (!pDX->m_bSaveAndValidate)
{
m_fEnableLogging = LoggingEnabled(m_dwLogType);
}
//{{AFX_DATA_MAP(CW3ServicePage)
DDX_Control(pDX, IDC_BUTTON_PROPERTIES, m_button_LogProperties);
DDX_Control(pDX, IDC_STATIC_LOG_PROMPT, m_static_LogPrompt);
DDX_Control(pDX, IDC_EDIT_SSL_PORT, m_edit_SSLPort);
DDX_Control(pDX, IDC_EDIT_TCP_PORT, m_edit_TCPPort);
DDX_Control(pDX, IDC_COMBO_LOG_FORMATS, m_combo_LogFormats);
DDX_Control(pDX, IDC_COMBO_IP_ADDRESS, m_combo_IpAddresses);
DDX_Check(pDX, IDC_CHECK_USE_KEEPALIVE, m_fUseKeepAlives);
DDX_Check(pDX, IDC_CHECK_ENABLE_LOGGING, m_fEnableLogging);
DDX_Text(pDX, IDC_EDIT_COMMENT, m_strComment);
DDV_MinMaxChars(pDX, m_strComment, 0, MAX_PATH);
//}}AFX_DATA_MAP
if ( pDX->m_bSaveAndValidate
&& !FetchIpAddressFromCombo(m_combo_IpAddresses, m_oblIpAddresses, m_iaIpAddress)
)
{
pDX->Fail();
}
// This Needs to come before DDX_Text which will try to put text big number into small number
DDV_MinMaxBalloon(pDX, IDC_EDIT_CONNECTION_TIMEOUT, 0, MAX_TIMEOUT);
DDX_Text(pDX, IDC_EDIT_CONNECTION_TIMEOUT, m_nConnectionTimeOut);
//
// Port DDXV must be done just prior to storetopbinding,
// so as to activate the right control in case of
// failure
//
if (!IS_MASTER_INSTANCE(QueryInstance()))
{
DDXV_UINT(pDX, IDC_EDIT_TCP_PORT, m_nTCPPort, 1, 65535, IDS_NO_PORT);
// If user will clear SSL port or set it to 0, we will remove this property
if (pDX->m_bSaveAndValidate)
{
// user is not forced to put a number in when saving port
if (GetDlgItem(IDC_EDIT_SSL_PORT)->GetWindowTextLength())
{
DDXV_UINT(pDX, IDC_EDIT_SSL_PORT, m_nSSLPort, 1, 65535, IDS_NO_PORT);
DDX_TextBalloon(pDX, IDC_EDIT_SSL_PORT, m_nSSLPort);
}
else
{
m_nSSLPort = -1;
}
}
else
{
DDXV_UINT(pDX, IDC_EDIT_SSL_PORT, m_nSSLPort, 1, 65535, IDS_NO_PORT);
if (m_nSSLPort == -1)
{
CString strTemp = _T("");
DDX_Text(pDX, IDC_EDIT_SSL_PORT, strTemp);
}
else
{
DDX_Text(pDX, IDC_EDIT_SSL_PORT, m_nSSLPort);
}
}
}
if (pDX->m_bSaveAndValidate)
{
if (!IS_MASTER_INSTANCE(QueryInstance()))
{
if (!StoreTopBinding())
{
pDX->Fail();
}
}
EnableLogging(m_dwLogType, m_fEnableLogging);
}
}
//
// Message Map
//
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
BEGIN_MESSAGE_MAP(CW3ServicePage, CInetPropertyPage)
//{{AFX_MSG_MAP(CW3ServicePage)
// ON_BN_CLICKED(IDC_RADIO_LIMITED, OnRadioLimited)
// ON_BN_CLICKED(IDC_RADIO_UNLIMITED, OnRadioUnlimited)
ON_BN_CLICKED(IDC_CHECK_ENABLE_LOGGING, OnCheckEnableLogging)
ON_BN_CLICKED(IDC_BUTTON_ADVANCED, OnButtonAdvanced)
ON_BN_CLICKED(IDC_BUTTON_PROPERTIES, OnButtonProperties)
ON_WM_DESTROY()
//}}AFX_MSG_MAP
ON_BN_CLICKED(IDC_CHECK_USE_KEEPALIVE, OnItemChanged)
ON_EN_CHANGE(IDC_EDIT_TCP_PORT, OnItemChanged)
ON_EN_CHANGE(IDC_EDIT_COMMENT, OnItemChanged)
ON_EN_CHANGE(IDC_EDIT_CONNECTION_TIMEOUT, OnItemChanged)
// ON_EN_CHANGE(IDC_EDIT_MAX_CONNECTIONS, OnItemChanged)
ON_EN_CHANGE(IDC_EDIT_IP_ADDRESS, OnItemChanged)
ON_EN_CHANGE(IDC_EDIT_SSL_PORT, OnItemChanged)
ON_EN_CHANGE(IDC_EDIT_DOMAIN_NAME, OnItemChanged)
ON_CBN_EDITCHANGE(IDC_COMBO_IP_ADDRESS, OnItemChanged)
ON_CBN_SELCHANGE(IDC_COMBO_IP_ADDRESS, OnItemChanged)
ON_CBN_SELCHANGE(IDC_COMBO_LOG_FORMATS, OnItemChanged)
END_MESSAGE_MAP()
void
CW3ServicePage::SetControlStates()
/*++
Routine Description:
Set control states depending on the currently selected items
Arguments:
None
Return Value:
None.
--*/
{
// if (m_edit_MaxConnections.m_hWnd)
// {
// m_edit_MaxConnections.EnableWindow(!m_fUnlimitedConnections);
// m_static_Connections.EnableWindow(!m_fUnlimitedConnections);
// }
}
void
CW3ServicePage::SetLogState()
/*++
Routine Description:
Enable/disable logging controls depending on whether logging
is enabled or not.
Arguments:
None
Return Value:
None
--*/
{
m_static_LogPrompt.EnableWindow(m_fEnableLogging);
m_combo_LogFormats.EnableWindow(m_fEnableLogging);
m_button_LogProperties.EnableWindow(m_fEnableLogging);
}
//
// Message Handlers
//
// <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
BOOL
CW3ServicePage::OnSetActive()
/*++
Routine Description:
Property page is getting activation notification
Arguments:
None
Return Value:
TRUE to activate the page, FALSE otherwise.
--*/
{
//
// No certificates, no SSL
//
BeginWaitCursor();
m_fCertInstalled = IsCertInstalledOnServer(QueryAuthInfo(), QueryMetaPath());
EndWaitCursor();
GetDlgItem(IDC_STATIC_SSL_PORT)->EnableWindow(
!IS_MASTER_INSTANCE(QueryInstance())
&& HasAdminAccess()
);
GetDlgItem(IDC_EDIT_SSL_PORT)->EnableWindow(
!IS_MASTER_INSTANCE(QueryInstance())
&& HasAdminAccess()
);
return CInetPropertyPage::OnSetActive();
}
BOOL
CW3ServicePage::OnInitDialog()
/*++
Routine Description:
WM_INITDIALOG handler. Initialize the dialog.
Arguments:
None.
Return Value:
TRUE if focus is to be set automatically, FALSE if the focus
is already set.
--*/
{
AFX_MANAGE_STATE(::AfxGetStaticModuleState());
CInetPropertyPage::OnInitDialog();
//
// Take our direction from a phony button
//
CRect rc(0, 0, 0, 0);
VERIFY(m_ocx_LogProperties.Create(_T("LogUI"), WS_BORDER,
rc, this, IDC_LOGUICTRL));
//
// Initialize the logging ocx
//
m_ocx_LogProperties.SetAdminTarget(QueryServerName(), QueryMetaPath());
m_ocx_LogProperties.SetUserData(QueryAuthInfo()->QueryUserName(), QueryAuthInfo()->QueryPassword());
m_ocx_LogProperties.SetComboBox(m_combo_LogFormats.m_hWnd);
//
// Disable non heritable properties for master instance
// or operator
//
if (IS_MASTER_INSTANCE(QueryInstance()) || !HasAdminAccess())
{
GetDlgItem(IDC_STATIC_IP_ADDRESS)->EnableWindow(FALSE);
GetDlgItem(IDC_COMBO_IP_ADDRESS)->EnableWindow(FALSE);
GetDlgItem(IDC_STATIC_TCP_PORT)->EnableWindow(FALSE);
GetDlgItem(IDC_EDIT_TCP_PORT)->EnableWindow(FALSE);
GetDlgItem(IDC_STATIC_SSL_PORT)->EnableWindow(FALSE);
GetDlgItem(IDC_EDIT_SSL_PORT)->EnableWindow(FALSE);
GetDlgItem(IDC_BUTTON_ADVANCED)->EnableWindow(FALSE);
GetDlgItem(IDC_STATIC_DESCRIPTION)->EnableWindow(FALSE);
GetDlgItem(IDC_EDIT_COMMENT)->EnableWindow(FALSE);
}
{
CWaitCursor wait;
PopulateComboWithKnownIpAddresses(
QueryServerName(),
m_combo_IpAddresses,
m_iaIpAddress,
m_oblIpAddresses,
m_nIpAddressSel
);
}
SetControlStates();
SetLogState();
return TRUE;
}
/* virtual */
HRESULT
CW3ServicePage::FetchLoadedValues()
/*++
Routine Description:
Move configuration data from sheet to dialog controls
Arguments:
None
Return Value:
HRESULT
--*/
{
CError err;
m_fCertInstalled = ::IsCertInstalledOnServer(QueryAuthInfo(), QueryMetaPath());
BEGIN_META_INST_READ(CW3Sheet)
FETCH_INST_DATA_FROM_SHEET(m_fUseKeepAlives);
// FETCH_INST_DATA_FROM_SHEET(m_nMaxConnections);
FETCH_INST_DATA_FROM_SHEET(m_nConnectionTimeOut);
FETCH_INST_DATA_FROM_SHEET(m_strComment);
FETCH_INST_DATA_FROM_SHEET(m_dwLogType);
FETCH_INST_DATA_FROM_SHEET(m_strlBindings);
FETCH_INST_DATA_FROM_SHEET(m_strlSecureBindings);
GetTopBinding();
#if 0
m_fUnlimitedConnections =
((ULONG)(LONG)m_nMaxConnections >= UNLIMITED_CONNECTIONS);
if (Has10ConnectionLimit())
{
m_fUnlimitedConnections = FALSE;
if ((LONG)m_nMaxConnections > LIMITED_CONNECTIONS_MAX)
{
m_nMaxConnections = LIMITED_CONNECTIONS_MAX;
}
}
//
// Set the visible max connections edit field, which
// may start out with a default value
//
m_nVisibleMaxConnections = m_fUnlimitedConnections
? INITIAL_MAX_CONNECTIONS
: m_nMaxConnections;
//
// Set radio value
//
m_nUnlimited = m_fUnlimitedConnections ? RADIO_UNLIMITED : RADIO_LIMITED;
#endif
m_nOldTCPPort = m_nTCPPort;
END_META_INST_READ(err)
return err;
}
/* virtual */
HRESULT
CW3ServicePage::SaveInfo()
/*++
Routine Description:
Save the information on this property page
Arguments:
BOOL fUpdateData : If TRUE, control data has not yet been stored. This
is the case when "apply" is pressed.
Return Value:
Error return code
--*/
{
AFX_MANAGE_STATE(::AfxGetStaticModuleState());
ASSERT(IsDirty());
TRACEEOLID("Saving W3 service page now...");
CError err;
#if 0
m_nMaxConnections = m_fUnlimitedConnections
? UNLIMITED_CONNECTIONS
: m_nVisibleMaxConnections;
//
// Check to make sure we're not violating the license
// agreement
//
if (Has10ConnectionLimit())
{
if (m_nMaxConnections > LIMITED_CONNECTIONS_MAX)
{
DoHelpMessageBox(m_hWnd,IDS_CONNECTION_LIMIT, MB_APPLMODAL | MB_OK | MB_ICONINFORMATION, 0);
m_nMaxConnections = LIMITED_CONNECTIONS_MIN;
}
else if (m_nMaxConnections > LIMITED_CONNECTIONS_MIN
&& m_nMaxConnections <= LIMITED_CONNECTIONS_MAX)
{
DoHelpMessageBox(m_hWnd,IDS_WRN_CONNECTION_LIMIT, MB_APPLMODAL | MB_OK | MB_ICONINFORMATION, 0);
}
}
#endif
m_ocx_LogProperties.ApplyLogSelection();
// BOOL fUpdateNode = FALSE;
BeginWaitCursor();
BEGIN_META_INST_WRITE(CW3Sheet)
STORE_INST_DATA_ON_SHEET(m_fUseKeepAlives);
STORE_INST_DATA_ON_SHEET(m_nConnectionTimeOut);
STORE_INST_DATA_ON_SHEET(m_strComment);
// fUpdateNode = MP_D(((CW3Sheet *)GetSheet())->GetInstanceProperties().m_strComment);
STORE_INST_DATA_ON_SHEET(m_dwLogType);
STORE_INST_DATA_ON_SHEET(m_strlBindings);
STORE_INST_DATA_ON_SHEET(m_strlSecureBindings);
END_META_INST_WRITE(err)
if (err.Succeeded()/* && fUpdateNode*/)
{
NotifyMMC(PROP_CHANGE_DISPLAY_ONLY);
}
EndWaitCursor();
return err;
}
void
CW3ServicePage::OnItemChanged()
/*++
Routine Description
All EN_CHANGE and BN_CLICKED messages map to this function
Arguments:
None
Return Value:
None
--*/
{
SetControlStates();
SetModified(TRUE);
}
#if 0
void
CW3ServicePage::OnRadioLimited()
/*++
Routine Description:
'limited' radio button handler
Arguments:
None
Return Value:
None
--*/
{
m_fUnlimitedConnections = FALSE;
SetControlStates();
m_edit_MaxConnections.SetSel(0,-1);
m_edit_MaxConnections.SetFocus();
OnItemChanged();
}
void
CW3ServicePage::OnRadioUnlimited()
/*++
Routine Description:
'unlimited' radio button handler
Arguments:
None
Return Value:
None
--*/
{
m_fUnlimitedConnections = TRUE;
OnItemChanged();
}
#endif
void
CW3ServicePage::ShowTopBinding()
/*++
Routine Description:
Put information about the top level binding in the dialog controls
Arguments:
None
Return Value:
None
--*/
{
BeginWaitCursor();
GetTopBinding();
PopulateComboWithKnownIpAddresses(
QueryServerName(),
m_combo_IpAddresses,
m_iaIpAddress,
m_oblIpAddresses,
m_nIpAddressSel
);
EndWaitCursor();
CString strTCPPort, strSSLPort;
if (m_nTCPPort)
{
strTCPPort.Format(_T("%ld"), m_nTCPPort);
}
#ifdef ZERO_IS_A_VALID_SSL_PORT
if (-1 == m_nSSLPort)
#else
if (0 == m_nSSLPort || -1 == m_nSSLPort)
#endif
{
strSSLPort = _T("");
}
else
{
strSSLPort.Format(_T("%ld"), m_nSSLPort);
}
m_edit_TCPPort.SetWindowText(strTCPPort);
m_edit_SSLPort.SetWindowText(strSSLPort);
}
void
CW3ServicePage::OnButtonAdvanced()
/*++
Routine Description:
'advanced' button handler -- bring up the bindings dialog
Arguments:
None
Return Value:
None
--*/
{
if (!UpdateData(TRUE))
{
return;
}
CMMMDlg dlg(
QueryServerName(),
QueryInstance(),
QueryAuthInfo(),
QueryMetaPath(),
m_strlBindings,
m_strlSecureBindings,
this
);
if (dlg.DoModal() == IDOK)
{
//
// Get information about the top level binding
//
m_strlBindings.RemoveAll();
m_strlSecureBindings.RemoveAll();
m_strlBindings.AddTail(&(dlg.GetBindings()));
m_strlSecureBindings.AddTail(&(dlg.GetSecureBindings()));
ShowTopBinding();
OnItemChanged();
}
}
void
CW3ServicePage::OnCheckEnableLogging()
/*++
Routine Description:
'enable logging' checkbox handler
Arguments:
None
Return Value:
None
--*/
{
m_fEnableLogging = !m_fEnableLogging;
SetLogState();
OnItemChanged();
}
void
CW3ServicePage::OnButtonProperties()
/*++
Routine Description:
Pass on "log properties" button click to the ocx.
Arguments:
None
Return Value:
None
--*/
{
AFX_MANAGE_STATE(::AfxGetStaticModuleState());
m_ocx_LogProperties.DoClick();
}
void
CW3ServicePage::OnDestroy()
/*++
Routine Description:
WM_DESTROY handler. Clean up internal data
Arguments:
None
Return Value:
None
--*/
{
CInetPropertyPage::OnDestroy();
if (m_ocx_LogProperties.m_hWnd)
{
m_ocx_LogProperties.Terminate();
}
}