windows-nt/Source/XPSP1/NT/net/mmc/wins/svrstats.cpp
2020-09-26 16:20:57 +08:00

751 lines
21 KiB
C++

/**********************************************************************/
/** Microsoft Windows/NT **/
/** Copyright(c) Microsoft Corporation, 1997 - 1999 **/
/**********************************************************************/
/*
ServStat.cpp
The server statistics dialog
FILE HISTORY:
*/
#include <afx.h>
#include "dbgutil.h"
#include "stdafx.h"
#include "winssnap.h"
#include "server.h"
#include "resource.h"
#include "svrstats.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
#define INTLTIMESTR(time) (((CIntlTime)(time)).CIntlTime::operator const CString())
#define TMST1(x) INTLTIMESTR((m_wrResults).WinsStat.TimeStamps.x)
#define INTLNUMSTR(num) (((CIntlNumber)(num)).CIntlNumber::operator const CString())
#define TCTR(x) INTLNUMSTR((m_wrResults).WinsStat.Counters.x)
#define STATMUTEXNAME _T("WINSADMNGETSTATISTICS")
#define WM_UPDATE_STATS WM_USER + 1099
enum
{
SERVER_STAT_START_TIME = 0,
SERVER_STAT_DB_INIT_TIME,
SERVER_STAT_STAT_CLEAR_TIME,
SERVER_STAT_BLANK0,
SERVER_STAT_LAST_PERIODIC_REP,
SERVER_STAT_LAST_ADMIN_REP,
SERVER_STAT_LAST_NET_UPDATE_REP,
SERVER_STAT_LAST_ADDRESS_CHANGE_REP,
SERVER_STAT_BLANK1,
SERVER_STAT_TOTAL_QUERIES,
SERVER_STAT_SUCCESSFUL_QUERRIES,
SERVER_STAT_FAILED_QUERRIES,
SERVER_STAT_BLANK2,
SERVER_STAT_TOTAL_RELEASES,
SERVER_STAT_SUCCESSFUL_RELEASES,
SERVER_STAT_FAILED_RELEASES,
SERVER_STAT_BLANK3,
SERVER_STAT_UNIQUE_REGISTRATIONS,
SERVER_STAT_UNIQUE_CONFLICTS,
SERVER_STAT_UNIQUE_RENEWALS,
SERVER_STAT_BLANK4,
SERVER_STAT_GROUP_REGISTRATIONS,
SERVER_STAT_GROUP_CONFLICTS,
SERVER_STAT_GROUP_RENEWALS,
SERVER_STAT_BLANK5,
SERVER_STAT_TOTAL_REG,
SERVER_STAT_BLANK6,
SERVER_STAT_LAST_PERIODIC_SCAV,
SERVER_STAT_LAST_ADMIN_SCAV,
SERVER_STAT_LAST_EXTINCTION_SCAV,
SERVER_STAT_LAST_VERIFICATION_SCAV,
SERVER_STAT_BLANK7,
SERVER_STAT_PARTNERS_HEADER,
SERVER_STAT_MAX
};
/*---------------------------------------------------------------------------
CServerStatsFrame implementation
---------------------------------------------------------------------------*/
const ContainerColumnInfo s_rgServerStatsColumnInfo[] =
{
{ IDS_SERVER_STATS_START_TIME, 0, TRUE },
{ IDS_SERVER_STATS_DB_INIT_TIME, 0, TRUE },
{ IDS_SERVER_STATS_LAST_CLEAR_TIME, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_LAST_PREP, 0, TRUE },
{ IDS_SERVER_STATS_LAST_AREP, 0, TRUE },
{ IDS_SERVER_STATS_LAST_NREP, 0, TRUE },
{ IDS_SERVER_STATS_LAST_ACREP, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_TOTAL_QUERRIES, 0, TRUE },
{ IDS_SERVER_STATS_SUCCESSFUL, 0, TRUE },
{ IDS_SERVER_STATS_FAILED, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_TOTAL_RELEASES, 0, TRUE },
{ IDS_SERVER_STATS_SUCCESSFUL, 0, TRUE },
{ IDS_SERVER_STATS_FAILED, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_UNIQUE_REGISTRATIONS, 0, TRUE },
{ IDS_SERVER_STATS_CONFLICTS, 0, TRUE },
{ IDS_SERVER_STATS_RENEWALS, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_GROUP_REGISTRATIONS, 0, TRUE },
{ IDS_SERVER_STATS_CONFLICTS, 0, TRUE },
{ IDS_SERVER_STATS_RENEWALS, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_TOTAL_REGISTRATIONS, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_LAST_PSCAV, 0, TRUE },
{ IDS_SERVER_STATS_LAST_ASCAV, 0, TRUE },
{ IDS_SERVER_STATS_LAST_ESCAV, 0, TRUE },
{ IDS_SERVER_STATS_LAST_VSCAV, 0, TRUE },
{ IDS_BLANK, 0, TRUE },
{ IDS_SERVER_STATS_PARTNERS_HEADER, 0, TRUE },
};
UINT _cdecl
RefreshStatsThread(LPVOID pParam)
{
DWORD dw;
ThreadInfo * pInfo = (ThreadInfo *)(pParam);
HANDLE hmutStatistics = NULL;
HRESULT hr = hrOK;
COM_PROTECT_TRY
{
// Open the existing mutex
while (hmutStatistics == NULL)
{
if ((hmutStatistics = ::OpenMutex(SYNCHRONIZE, FALSE, STATMUTEXNAME)) == NULL)
{
::Sleep(2000L);
}
}
// This is where the work gets done
for (;;)
{
pInfo->dwInterval = pInfo->pDlg->GetRefreshInterval();
//::Sleep((pInfo->dwInterval)*1000);
if (::WaitForSingleObject(pInfo->pDlg->m_hAbortEvent, (pInfo->dwInterval) * 1000) == WAIT_OBJECT_0)
{
// we're going away, lets get outta here
break;
}
// Wait until we get the go ahead
dw = ::WaitForSingleObject(hmutStatistics, INFINITE);
if (dw == WAIT_OBJECT_0)
{
dw = pInfo->pDlg->GetStats();
if (dw == ERROR_SUCCESS)
{
PostMessage(pInfo->pDlg->GetSafeHwnd(), WM_UPDATE_STATS, 0, 0);
}
::ReleaseMutex(hmutStatistics);
}
}
delete pInfo;
}
COM_PROTECT_CATCH
return 0;
}
/*---------------------------------------------------------------------------
CServerStatsFrame::CServerStatsFrame()
Constructor
---------------------------------------------------------------------------*/
CServerStatsFrame::CServerStatsFrame()
: StatsDialog(STATSDLG_VERTICAL|STATSDLG_CLEAR)
{
SetColumnInfo(s_rgServerStatsColumnInfo,
DimensionOf(s_rgServerStatsColumnInfo));
m_hmutStatistics = NULL;
m_hAbortEvent = NULL;
m_pRefreshThread = NULL;
ZeroMemory(&m_wrResults, sizeof(m_wrResults));
}
/*---------------------------------------------------------------------------
CServerStatsFrame::CServerStatsFrame()
Destructor
---------------------------------------------------------------------------*/
CServerStatsFrame::~CServerStatsFrame()
{
}
BEGIN_MESSAGE_MAP(CServerStatsFrame, StatsDialog)
//{{AFX_MSG_MAP(CServerStatistics)
//}}AFX_MSG_MAP
ON_MESSAGE(WM_NEW_STATS_AVAILABLE, OnNewStatsAvailable)
ON_MESSAGE(WM_UPDATE_STATS, OnUpdateStats)
ON_BN_CLICKED(IDC_STATSDLG_BTN_CLEAR, OnClear)
ON_WM_DESTROY()
END_MESSAGE_MAP()
/*---------------------------------------------------------------------------
CServerStatsFrame::CServerStatsFrame()
Gets the statistics from the server handler and updates the
internal variable m_wrResults
---------------------------------------------------------------------------*/
HRESULT CServerStatsFrame::RefreshData(BOOL fGrabNewData)
{
if (fGrabNewData)
{
DWORD dwError = 0;
BEGIN_WAIT_CURSOR;
dwError = GetStats();
END_WAIT_CURSOR;
if (dwError != ERROR_SUCCESS)
{
WinsMessageBox(dwError);
}
else
{
UpdateWindow(&m_wrResults);
}
}
return hrOK;
}
DWORD CServerStatsFrame::GetStats()
{
DWORD dwError = 0;
SPITFSNode spNode;
PWINSINTF_RESULTS_T pResults = NULL;
CWinsServerHandler * pServer;
pServer = GETHANDLER(CWinsServerHandler, m_spNode);
dwError = pServer->GetStatistics(m_spNode, &pResults);
if (dwError == ERROR_SUCCESS)
{
m_wrResults = *(pResults);
}
return dwError;
}
BOOL CServerStatsFrame::OnInitDialog()
{
CString st;
BOOL bRet;
AfxFormatString1(st, IDS_SERVER_STATS_TITLE, m_strServerAddress);
SetWindowText((LPCTSTR) st);
bRet = StatsDialog::OnInitDialog();
// add the blank column because the replication partner stats need them
m_listCtrl.InsertColumn(2, _T(" "), LVCFMT_LEFT, 100);
// Set the default column widths to the width of the widest column
SetColumnWidths(2 /* Number of Columns */);
UpdateWindow(&m_wrResults);
// set the default window size
RECT rect;
GetWindowRect(&rect);
int nWidth = 0;
for (int i = 0; i < 3; i++)
{
nWidth += m_listCtrl.GetColumnWidth(i);
}
SetWindowPos(NULL, rect.left, rect.top, nWidth + 20, SERVER_STATS_DEFAULT_HEIGHT, SWP_SHOWWINDOW);
if ((m_hmutStatistics = ::CreateMutex(NULL, FALSE, STATMUTEXNAME)) == NULL)
{
Trace1("CServerStatsFrame::OnInitDialog() - CreateMutex failed! %lx\n", GetLastError());
return FALSE;
}
if ((m_hAbortEvent = ::CreateEvent(NULL, FALSE, FALSE, NULL)) == NULL)
{
Trace1("CServerStatsFrame::OnInitDialog() - CreateEvent failed! %lx\n", GetLastError());
return FALSE;
}
StartRefresherThread();
return bRet;
}
void CServerStatsFrame::Sort(UINT nColumnId)
{
// we don't sort any of our stats
}
/*---------------------------------------------------------------------------
CServerStatsFrame::OnNewStatsAvailable(UINT wParam, LONG lParam)
called in response to the message ON_NEW_STATS_AVAILABLE
Author: EricDav
---------------------------------------------------------------------------*/
afx_msg long
CServerStatsFrame::OnNewStatsAvailable(UINT wParam, LONG lParam)
{
DWORD dwErr = GetStats();
if (dwErr == ERROR_SUCCESS)
{
UpdateWindow(&m_wrResults);
}
else
{
WinsMessageBox(dwErr);
}
return 0;
}
/*---------------------------------------------------------------------------
CServerStatsFrame::OnNewStatsAvailable(UINT wParam, LONG lParam)
called in response to the message WM_UPDATE_STATS
The background thread updates the stats, now we need to update
the UI on the correct thread.
Author: EricDav
---------------------------------------------------------------------------*/
afx_msg long
CServerStatsFrame::OnUpdateStats(UINT wParam, LONG lParam)
{
UpdateWindow(&m_wrResults);
return 0;
}
/*---------------------------------------------------------------------------
CServerStatsFrame::UpdateWindow(PWINSINTF_RESULTS_T pwrResults)
Updates the contents of the dialog
---------------------------------------------------------------------------*/
void
CServerStatsFrame::UpdateWindow(PWINSINTF_RESULTS_T pwrResults)
{
SPITFSNode spNode;
CWinsServerHandler * pServer;
CString str;
UINT i;
int nTotalAddresses = 0, nTotalInUse = 0, nTotalAvailable = 0;
// now fill in the data
for (i = 0; i < SERVER_STAT_MAX; i++)
{
if (!pwrResults)
str = _T("---");
else
{
switch (i)
{
// server stsrt time
case SERVER_STAT_START_TIME:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.WinsStartTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.WinsStartTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
// database initialized time
case SERVER_STAT_DB_INIT_TIME:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastInitDbTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastInitDbTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
// statistics last cleared time
case SERVER_STAT_STAT_CLEAR_TIME:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.CounterResetTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.CounterResetTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
// some blank lines in between
case SERVER_STAT_BLANK0:
case SERVER_STAT_BLANK1:
case SERVER_STAT_BLANK2:
case SERVER_STAT_BLANK3:
case SERVER_STAT_BLANK4:
case SERVER_STAT_BLANK5:
case SERVER_STAT_BLANK6:
case SERVER_STAT_BLANK7:
str = _T("");
break;
case SERVER_STAT_LAST_PERIODIC_REP:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastPRplTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastPRplTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
case SERVER_STAT_LAST_ADMIN_REP:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastATRplTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastATRplTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
case SERVER_STAT_LAST_NET_UPDATE_REP:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastNTRplTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastNTRplTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
// last address changed time
case SERVER_STAT_LAST_ADDRESS_CHANGE_REP:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastACTRplTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastACTRplTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
// total queries received
case SERVER_STAT_TOTAL_QUERIES:
str = TCTR(NoOfQueries);
break;
// successful queries
case SERVER_STAT_SUCCESSFUL_QUERRIES:
str = TCTR(NoOfSuccQueries);
break;
// Queries that failed
case SERVER_STAT_FAILED_QUERRIES:
str = TCTR(NoOfFailQueries);
break;
case SERVER_STAT_TOTAL_RELEASES:
str = TCTR(NoOfRel);
break;
case SERVER_STAT_SUCCESSFUL_RELEASES:
str = TCTR(NoOfSuccRel);
break;
case SERVER_STAT_FAILED_RELEASES:
str = TCTR(NoOfFailRel);
break;
// unique registrations
case SERVER_STAT_UNIQUE_REGISTRATIONS:
str = TCTR(NoOfUniqueReg);
break;
case SERVER_STAT_UNIQUE_CONFLICTS:
str = TCTR(NoOfUniqueCnf);
break;
case SERVER_STAT_UNIQUE_RENEWALS:
str = TCTR(NoOfUniqueRef);
break;
// group registrations
case SERVER_STAT_GROUP_REGISTRATIONS:
str = TCTR(NoOfGroupReg);
break;
case SERVER_STAT_GROUP_CONFLICTS:
str = TCTR(NoOfGroupCnf);
break;
case SERVER_STAT_GROUP_RENEWALS:
str = TCTR(NoOfGroupRef);
break;
// total registrations
case SERVER_STAT_TOTAL_REG:
str = INTLNUMSTR(m_wrResults.WinsStat.Counters.NoOfGroupReg +
m_wrResults.WinsStat.Counters.NoOfUniqueReg);
break;
case SERVER_STAT_LAST_PERIODIC_SCAV:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastPScvTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastPScvTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
case SERVER_STAT_LAST_ADMIN_SCAV:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastATScvTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastATScvTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
case SERVER_STAT_LAST_EXTINCTION_SCAV:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastTombScvTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastTombScvTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
case SERVER_STAT_LAST_VERIFICATION_SCAV:
{
CTime timeStart(m_wrResults.WinsStat.TimeStamps.LastVerifyScvTime);
if(timeStart != 0)
FormatDateTime(str, &m_wrResults.WinsStat.TimeStamps.LastVerifyScvTime);
else
str.LoadString(IDS_INVALID_TIME);
}
break;
case SERVER_STAT_PARTNERS_HEADER:
{
str.LoadString(IDS_SERVER_STATS_NO_OF_FAILED);
m_listCtrl.SetItemText(i, 2, (LPCTSTR) str);
str.LoadString(IDS_SERVER_STATS_NO_OF_REPLS);
}
break;
default:
Assert("Invalid server statistic type!");
break;
}
}
// now the string is set, display in the dlg
m_listCtrl.SetItemText(i, 1, (LPCTSTR) str);
}
UpdatePartnerStats();
}
/*---------------------------------------------------------------------------
CServerStatsFrame::StartRefresherThread()
Starts the refresher thread, called as soon as the dilog is
brought up
---------------------------------------------------------------------------*/
void
CServerStatsFrame::StartRefresherThread()
{
ThreadInfo *info = new ThreadInfo;
CWinsServerHandler * pServer;
pServer = GETHANDLER(CWinsServerHandler, m_spNode);
info->dwInterval = pServer->m_dwRefreshInterval;
info->pDlg = this;
m_pRefreshThread = ::AfxBeginThread(RefreshStatsThread, info);
if (m_pRefreshThread == NULL)
{
Trace0("Failed to create thread\n");
m_pRefreshThread = NULL;
return;
}
Trace0("Auto refresh thread succesfully started\n");
}
/*---------------------------------------------------------------------------
CServerStatsFrame::ReInitRefresherThread()
If the refresher thread was running, re-start it. It is needed
in order to pick on the fly the new refresh intervals
---------------------------------------------------------------------------*/
void
CServerStatsFrame::ReInitRefresherThread()
{
// if there is a refresher thread, just restart it
if (m_pRefreshThread != NULL)
{
KillRefresherThread();
StartRefresherThread();
}
}
/*---------------------------------------------------------------------------
CServerStatsFrame::KillRefresherThread()
Kills the refresh data thread, caled when the dlg is destroyed
---------------------------------------------------------------------------*/
void
CServerStatsFrame::KillRefresherThread()
{
//
// Kill refresher thread if necessary.
//
if (m_pRefreshThread == NULL)
{
//
// No thread running
//
return;
}
//::TerminateThread(m_pRefreshThread->m_hThread, 0);
//::CloseHandle(m_pRefreshThread->m_hThread);
::SetEvent(m_hAbortEvent);
::WaitForSingleObject(m_pRefreshThread->m_hThread, 5000);
m_pRefreshThread = NULL;
}
/*---------------------------------------------------------------------------
CServerStatsFrame::OnDestroy( )
Message Handler
---------------------------------------------------------------------------*/
void
CServerStatsFrame::OnDestroy( )
{
KillRefresherThread();
if (m_hmutStatistics)
{
::CloseHandle(m_hmutStatistics);
m_hmutStatistics = NULL;
}
if (m_hAbortEvent)
{
::CloseHandle(m_hAbortEvent);
m_hAbortEvent = NULL;
}
}
/*---------------------------------------------------------------------------
CServerStatsFrame::GetRefreshInterval()
Returns the refresh interval, stored in the server handler
---------------------------------------------------------------------------*/
DWORD
CServerStatsFrame::GetRefreshInterval()
{
CWinsServerHandler * pServer;
pServer = GETHANDLER(CWinsServerHandler, m_spNode);
return pServer->m_dwRefreshInterval;
}
/*---------------------------------------------------------------------------
CServerStatsFrame::OnClear()
Calls the wins api to reset counters and updates data, in response to
click of the clear button
---------------------------------------------------------------------------*/
void
CServerStatsFrame::OnClear()
{
DWORD dwErr = ERROR_SUCCESS;
CWinsServerHandler * pServer = GETHANDLER(CWinsServerHandler, m_spNode);
// clear the ststistics
BEGIN_WAIT_CURSOR
dwErr = pServer->ClearStatistics(m_spNode);
END_WAIT_CURSOR
if (dwErr == ERROR_SUCCESS)
{
// refresh the data now
RefreshData(TRUE);
UpdateWindow(&m_wrResults);
}
else
{
WinsMessageBox(dwErr);
}
}
/*---------------------------------------------------------------------------
CServerStatsFrame::UpdatePartnerStats()
---------------------------------------------------------------------------*/
void
CServerStatsFrame::UpdatePartnerStats()
{
UINT i, uCount;
CString strText;
// first remove all old partner info
uCount = m_listCtrl.GetItemCount();
for (i = 0; i < (uCount - SERVER_STAT_MAX); i++)
{
m_listCtrl.DeleteItem(SERVER_STAT_MAX);
}
int nPartner = 0;
// now add all of the partner information in
for (i = SERVER_STAT_MAX; i < SERVER_STAT_MAX + m_wrResults.WinsStat.NoOfPnrs; i++)
{
m_listCtrl.InsertItem(i, _T(""));
// ip address
::MakeIPAddress(m_wrResults.WinsStat.pRplPnrs[nPartner].Add.IPAdd, strText);
m_listCtrl.SetItemText(i, 0, strText);
// replication count
strText.Format(_T("%d"), m_wrResults.WinsStat.pRplPnrs[nPartner].NoOfRpls);
m_listCtrl.SetItemText(i, 1, strText);
// failed count
strText.Format(_T("%d"), m_wrResults.WinsStat.pRplPnrs[nPartner].NoOfCommFails);
m_listCtrl.SetItemText(i, 2, strText);
nPartner++;
}
}