/**********************************************************************/ /** Microsoft Windows/NT **/ /** Copyright(c) Microsoft Corporation, 1997 - 1999 **/ /**********************************************************************/ /* ServStat.cpp The server statistics dialog FILE HISTORY: */ #include #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); } } if (hmutStatistics != NULL) CloseHandle(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++; } }