mirror of https://github.com/lianthony/NT4.0
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.
582 lines
14 KiB
582 lines
14 KiB
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// File: tmscfg.cpp
|
|
// Purpose: API Entry points for service ui
|
|
//
|
|
|
|
#include "stdafx.h"
|
|
|
|
#define DLL_BASED __declspec(dllexport)
|
|
|
|
#include "tmscfg.h"
|
|
#include "tmservic.h"
|
|
#include "tmsessio.h"
|
|
|
|
// ===========================================================================
|
|
// Standard configuration Information
|
|
// ===========================================================================
|
|
|
|
//
|
|
// Is this server discoverable by INETSLOC?
|
|
//
|
|
#define INETSLOC_DISCOVERY FALSE
|
|
|
|
#if (INETSLOC_DISCOVERY) && !defined(_SVCLOC_)
|
|
#error You must include svcloc.h.
|
|
#endif
|
|
|
|
//
|
|
// If INETSLOC_DISCOVERY == TRUE, define the discovery MASK here.
|
|
//
|
|
#if (INETSLOC_DISCOVERY)
|
|
#define INETSLOC_MASK
|
|
#else // (!INETSLOC_DISCOVERY)
|
|
#define INETSLOC_MASK (ULONGLONG)(0x00000000)
|
|
#endif // (INETSLOC_DISCOVERY)
|
|
|
|
//
|
|
// Do we use the service control manager to control
|
|
// the service we administer?
|
|
//
|
|
#define USE_SC_MANAGER FALSE
|
|
|
|
//
|
|
// Can we change the service state (start/pause/continue)?
|
|
//
|
|
#define CAN_CHANGE_SERVICE_STATE FALSE
|
|
|
|
//
|
|
// Can we pause this service?
|
|
//
|
|
#define CAN_PAUSE_SERVICE FALSE
|
|
|
|
#if (USE_SC_MANAGER)
|
|
//
|
|
// Name used for this service by the service controller manager.
|
|
//
|
|
#define SERVICE_SC_NAME _T("TEMPLATE")
|
|
#endif // USE_SC_MANAGER
|
|
|
|
//
|
|
// Short descriptive name of the service. This
|
|
// is what will show up as the name of the service
|
|
// in the internet manager tool.
|
|
//
|
|
// Issue: I'm assuming here that this name does NOT
|
|
// require localisation.
|
|
//
|
|
#define SERVICE_SHORT_NAME _T("Template")
|
|
|
|
//
|
|
// Longer name. This is the text that shows up in
|
|
// the tooltips text on the internet manager
|
|
// tool. This probably should be localised.
|
|
//
|
|
|
|
#define SERVICE_LONG_NAME _T("Template Service")
|
|
|
|
//
|
|
// Use normal colour mapping.
|
|
//
|
|
#define NORMAL_TB_MAPPING TRUE
|
|
|
|
//
|
|
// Toolbar button background mask. This is
|
|
// the colour that gets masked out in
|
|
// the bitmap file and replaced with the
|
|
// actual button background. This setting
|
|
// is automatically assumed to be lt. gray
|
|
// if NORMAL_TB_MAPPING (above) is TRUE
|
|
//
|
|
#define BUTTON_BMP_BACKGROUND RGB(192, 192, 192) // Lt. Gray
|
|
|
|
//
|
|
// Resource ID of the toolbar button bitmap.
|
|
//
|
|
// The bitmap must be 17x17
|
|
//
|
|
#define BUTTON_BMP_ID IDB_TEMPLATE
|
|
|
|
//
|
|
// Similar to BUTTON_BMP_BACKGROUND, this is the
|
|
// background mask for the service ID
|
|
//
|
|
#define SERVICE_BMP_BACKGROUND BUTTON_BMP_BACKGROUND
|
|
|
|
//
|
|
// Bitmap id which is used in the service view
|
|
// of the service manager.
|
|
//
|
|
// The bitmap must be 17x17, and can be the
|
|
// same as BUTTON_BMP_ID bitmap.
|
|
//
|
|
#define SERVICE_BMP_ID BUTTON_BMP_ID
|
|
|
|
// ===========================================================================
|
|
// End Of Standard configuration Information
|
|
// ===========================================================================
|
|
|
|
//
|
|
// Global DLL instance
|
|
//
|
|
HINSTANCE hInstance;
|
|
|
|
// ============================================================================
|
|
// ISM API Functions
|
|
// ============================================================================
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Return service-specific information back to
|
|
// to the application. This function is called
|
|
// by the service manager immediately after
|
|
// LoadLibary(); The size element must be
|
|
// set prior to calling this API.
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMQueryServiceInfo(
|
|
ISMSERVICEINFO * psi // Service information returned.
|
|
)
|
|
{
|
|
if ( psi == NULL
|
|
|| psi->dwSize < ISMSERVICEINFO_SIZE
|
|
)
|
|
{
|
|
TRACEEOLID(_T("ISMQueryServiceInfo: ISMSERVICEINFO invalid"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
|
|
if (psi->dwSize != ISMSERVICEINFO_SIZE)
|
|
{
|
|
TRACEEOLID(_T("Warning: internet manager is newer than DLL"));
|
|
}
|
|
|
|
#endif // _DEBUG
|
|
|
|
psi->dwSize = ISMSERVICEINFO_SIZE;
|
|
psi->dwVersion = ISM_VERSION;
|
|
|
|
psi->flServiceInfoFlags = 0
|
|
#if (INETSLOC_DISCOVERY)
|
|
| ISMI_INETSLOCDISCOVER
|
|
#endif
|
|
#if (CAN_CHANGE_SERVICE_STATE)
|
|
| ISMI_CANCONTROLSERVICE
|
|
#endif
|
|
#if (CAN_PAUSE_SERVICE)
|
|
| ISMI_CANPAUSESERVICE
|
|
#endif
|
|
#if (NORMAL_TB_MAPPING)
|
|
| ISMI_NORMALTBMAPPING
|
|
#endif
|
|
; /**/
|
|
|
|
ASSERT(::lstrlen(SERVICE_LONG_NAME) <= MAX_LNLEN);
|
|
ASSERT(::lstrlen(SERVICE_SHORT_NAME) <= MAX_SNLEN);
|
|
|
|
psi->ullDiscoveryMask = INETSLOC_MASK;
|
|
psi->rgbButtonBkMask = BUTTON_BMP_BACKGROUND;
|
|
psi->nButtonBitmapID = BUTTON_BMP_ID;
|
|
psi->rgbServiceBkMask = SERVICE_BMP_BACKGROUND;
|
|
psi->nServiceBitmapID = SERVICE_BMP_ID;
|
|
::lstrcpy(psi->atchShortName, SERVICE_SHORT_NAME);
|
|
::lstrcpy(psi->atchLongName, SERVICE_LONG_NAME);
|
|
|
|
return ERROR_SUCCESS;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Discover machines running this service. This is
|
|
// only necessary for services not discovered with
|
|
// inetscloc (which don't give a mask)
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMDiscoverServers(
|
|
ISMSERVERINFO * psi, // Server info buffer.
|
|
DWORD * pdwBufferSize, // Size required/available.
|
|
int * cServers // Number of servers in buffer.
|
|
)
|
|
{
|
|
#if (INETSLOC_DISCOVERY)
|
|
*cServers = 0;
|
|
*pdwBufferSize = 0L;
|
|
|
|
TRACEEOLID(_T("Warning: service manager called bogus ISMDiscoverServers"));
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
#else
|
|
|
|
//#error Service specific discovery function must be provided.
|
|
|
|
//
|
|
// Dummy example, which discovers 2 servers running our service
|
|
//
|
|
int cServersFound = 2;
|
|
DWORD dwRequiredSize = sizeof(ISMSERVERINFO) * cServersFound;
|
|
|
|
if (*pdwBufferSize < dwRequiredSize || psi == NULL)
|
|
{
|
|
*pdwBufferSize = dwRequiredSize;
|
|
*cServers = cServersFound;
|
|
|
|
return ERROR_MORE_DATA;
|
|
}
|
|
|
|
psi[0].dwSize = ISMSERVERINFO_SIZE;
|
|
::lstrcpy(psi[0].atchServerName, _T("\\FOO"));
|
|
::lstrcpy(psi[0].atchComment, _T("Dummy Entry 1"));
|
|
psi[0].nState = INetServiceUnknown;
|
|
|
|
psi[1].dwSize = ISMSERVERINFO_SIZE;
|
|
::lstrcpy(psi[1].atchServerName, _T("\\BAR"));
|
|
::lstrcpy(psi[1].atchComment, _T("Dummy Entry 2"));
|
|
psi[1].nState = INetServiceUnknown;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
#endif // (INETSLOC_DISCOVERY)
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Get information about a specific server with
|
|
// regards to this service. Usually this uses
|
|
// the service control manager
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMQueryServerInfo(
|
|
LPCTSTR lpstrServerName, // Name of server.
|
|
ISMSERVERINFO * psi // Server information returned.
|
|
)
|
|
{
|
|
if ( psi == NULL
|
|
|| psi->dwSize < ISMSERVERINFO_SIZE
|
|
|| ::lstrlen(lpstrServerName) > MAX_SERVERNAME_LEN
|
|
)
|
|
{
|
|
TRACEEOLID(_T("ISMQueryServerInfo: bad parameters"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
|
|
if (psi->dwSize != ISMSERVERINFO_SIZE)
|
|
{
|
|
TRACEEOLID(_T("Warning internet manager is newer than DLL"));
|
|
}
|
|
|
|
#endif // _DEBUG
|
|
|
|
psi->dwSize = ISMSERVERINFO_SIZE;
|
|
::lstrcpy(psi->atchServerName, lpstrServerName);
|
|
|
|
#if (USE_SC_MANAGER)
|
|
SC_HANDLE hScManager;
|
|
|
|
hScManager = ::OpenSCManager(lpstrServerName, NULL, SC_MANAGER_ENUMERATE_SERVICE);
|
|
|
|
if (hScManager == NULL)
|
|
{
|
|
return ::GetLastError();
|
|
}
|
|
|
|
DWORD err = ERROR_SUCCESS;
|
|
|
|
SC_HANDLE hService = ::OpenService(hScManager,
|
|
SERVICE_SC_NAME, SERVICE_QUERY_STATUS);
|
|
|
|
if (hService == NULL)
|
|
{
|
|
err = ::GetLastError();
|
|
}
|
|
else
|
|
{
|
|
SERVICE_STATUS ss;
|
|
|
|
VERIFY(::QueryServiceStatus(hService, &ss));
|
|
switch(ss.dwCurrentState)
|
|
{
|
|
case SERVICE_STOPPED:
|
|
case SERVICE_STOP_PENDING:
|
|
psi->nState = INetServiceStopped;
|
|
break;
|
|
|
|
case SERVICE_RUNNING:
|
|
case SERVICE_START_PENDING:
|
|
case SERVICE_CONTINUE_PENDING:
|
|
psi->nState = INetServiceRunning;
|
|
break;
|
|
|
|
case SERVICE_PAUSE_PENDING:
|
|
case SERVICE_PAUSED:
|
|
psi->nState = INetServicePaused;
|
|
break;
|
|
|
|
default:
|
|
psi->nState = INetServiceUnknown;
|
|
}
|
|
|
|
::CloseServiceHandle(hService);
|
|
}
|
|
|
|
::CloseServiceHandle(hScManager);
|
|
|
|
return err;
|
|
|
|
#else
|
|
|
|
//#error Non SC QueryServerInfo must be defined!
|
|
|
|
::lstrcpy(psi->atchComment, _T("Dummy Entry"));
|
|
psi->nState = INetServiceUnknown;
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
#endif // USE_SC_MANAGER
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Change the service state of the servers (to paused/continue, started,
|
|
// stopped, etc)
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMChangeServiceState(
|
|
int nNewState, // INetService* definition.
|
|
int * pnCurrentState, // Ptr to current state (will be changed
|
|
DWORD dwReserved, // Reserved: must be 0
|
|
LPCTSTR lpstrServers // Double NULL terminated list of servers.
|
|
)
|
|
{
|
|
if ( dwReserved != 0L
|
|
|| nNewState < INetServiceStopped
|
|
|| nNewState > INetServicePaused
|
|
)
|
|
{
|
|
TRACEEOLID(_T("ISMChangeServiceState: Invalid information passed"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
#if (USE_SC_MANAGER)
|
|
//
|
|
// BUGBUG: SINGLE SELECTION ONLY!
|
|
//
|
|
SC_HANDLE hService;
|
|
SC_HANDLE hScManager;
|
|
hScManager = ::OpenSCManager(lpstrServers, NULL, SC_MANAGER_ALL_ACCESS);
|
|
|
|
if (hScManager == NULL)
|
|
{
|
|
return ::GetLastError();
|
|
}
|
|
hService = OpenService(hScManager, SERVICE_SC_NAME, SERVICE_ALL_ACCESS);
|
|
if (hService == NULL)
|
|
{
|
|
return ::GetLastError();
|
|
}
|
|
|
|
SERVICE_STATUS ss;
|
|
BOOL fSuccess;
|
|
|
|
switch(nNewState)
|
|
{
|
|
case INetServiceStopped:
|
|
fSuccess = ::ControlService(hService, SERVICE_CONTROL_STOP, &ss);
|
|
break;
|
|
|
|
case INetServiceRunning:
|
|
if (*pnCurrentState == INetServicePaused)
|
|
{
|
|
fSuccess = ::ControlService(hService, SERVICE_CONTROL_CONTINUE, &ss);
|
|
}
|
|
else
|
|
{
|
|
fSuccess = ::StartService(hService, 0, NULL);
|
|
}
|
|
break;
|
|
case INetServicePaused:
|
|
fSuccess = ::ControlService(hService, SERVICE_CONTROL_PAUSE, &ss);
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Update state information
|
|
//
|
|
VERIFY(::QueryServiceStatus(hService, &ss));
|
|
switch(ss.dwCurrentState)
|
|
{
|
|
case SERVICE_STOPPED:
|
|
case SERVICE_STOP_PENDING:
|
|
*pnCurrentState = INetServiceStopped;
|
|
break;
|
|
|
|
case SERVICE_RUNNING:
|
|
case SERVICE_START_PENDING:
|
|
case SERVICE_CONTINUE_PENDING:
|
|
*pnCurrentState = INetServiceRunning;
|
|
break;
|
|
|
|
case SERVICE_PAUSE_PENDING:
|
|
case SERVICE_PAUSED:
|
|
*pnCurrentState = INetServicePaused;
|
|
break;
|
|
|
|
default:
|
|
*pnCurrentState = INetServiceUnknown;
|
|
}
|
|
|
|
::CloseServiceHandle(hService);
|
|
::CloseServiceHandle(hScManager);
|
|
|
|
if (!fSuccess)
|
|
{
|
|
return ::GetLastError();
|
|
}
|
|
|
|
return ERROR_SUCCESS;
|
|
|
|
#else
|
|
|
|
//#error Non-SC ISMChangeServiceState must be defined
|
|
|
|
return ERROR_SERVICE_DOES_NOT_EXIST;
|
|
|
|
#endif // USE_SC_MANAGER
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Display configuration property sheet.
|
|
//
|
|
DLL_BASED DWORD APIENTRY
|
|
ISMConfigureServers(
|
|
HWND hWnd, // Main app window handle
|
|
DWORD dwReserved, // Reserved: must be 0
|
|
LPCTSTR lpstrServers // Double NULL terminated list of servers
|
|
)
|
|
{
|
|
DWORD err;
|
|
|
|
if ( dwReserved != 0L )
|
|
{
|
|
TRACEEOLID(_T("ISMConfigureServers: Invalid information passed"));
|
|
ASSERT(0);
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// convert the list of servers to a
|
|
// more manageable CStringList.
|
|
//
|
|
LPCTSTR pBuf = lpstrServers;
|
|
CStringList strlServers;
|
|
while ( *pBuf != TCHAR('\0' ))
|
|
{
|
|
CString strTmp = pBuf;
|
|
strlServers.AddTail( strTmp );
|
|
pBuf += strTmp.GetLength() + 1;
|
|
}
|
|
|
|
if ( strlServers.GetCount() == 0 )
|
|
{
|
|
TRACEEOLID(_T("Error: strlServers Count == 0."));
|
|
return ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
//
|
|
// Save the resource handle.
|
|
//
|
|
HINSTANCE hOldInstance = ::AfxGetResourceHandle();
|
|
ASSERT( hInstance != NULL );
|
|
|
|
//
|
|
// Load resources out of our own resource segment
|
|
//
|
|
::AfxSetResourceHandle( hInstance );
|
|
|
|
CString strCaption;
|
|
|
|
if (strlServers.GetCount() == 1)
|
|
{
|
|
CString str;
|
|
|
|
VERIFY(str.LoadString(IDS_CAPTION));
|
|
strCaption.Format(str, INETSVC_DISPLAYNAME(lpstrServers));
|
|
}
|
|
else // Multiple server caption
|
|
{
|
|
VERIFY(strCaption.LoadString(IDS_CAPTION_MULTIPLE));
|
|
}
|
|
|
|
::AfxMessageBox(_T("Initialisation happens here"));
|
|
|
|
//
|
|
// Didn't do anything, but what we did, we did
|
|
// perfectly.
|
|
//
|
|
err = ERROR_SUCCESS;
|
|
|
|
//
|
|
// reset the resource handle to the application's
|
|
// resource segment.
|
|
//
|
|
::AfxSetResourceHandle( hOldInstance );
|
|
|
|
return err;
|
|
}
|
|
|
|
// ============================================================================
|
|
// End of ISM API Functions
|
|
// ============================================================================
|
|
|
|
//
|
|
// Perform additional initialisation as necessary
|
|
//
|
|
void
|
|
InitializeDLL()
|
|
{
|
|
#ifdef _DEBUG
|
|
afxMemDF |= checkAlwaysMemDF;
|
|
#endif // _DEBUG
|
|
}
|
|
|
|
//
|
|
// DLL Main entry point
|
|
//
|
|
DLL_BASED BOOL WINAPI
|
|
LibMain(
|
|
HINSTANCE hDll,
|
|
DWORD dwReason,
|
|
LPVOID lpReserved
|
|
)
|
|
{
|
|
BOOL bResult = TRUE ;
|
|
|
|
switch ( dwReason )
|
|
{
|
|
case DLL_PROCESS_ATTACH:
|
|
case DLL_THREAD_ATTACH:
|
|
ASSERT( hDll != NULL );
|
|
hInstance = hDll;
|
|
InitializeDLL();
|
|
break ;
|
|
|
|
case DLL_THREAD_DETACH:
|
|
case DLL_PROCESS_DETACH:
|
|
//
|
|
// termination
|
|
//
|
|
break ;
|
|
}
|
|
|
|
ASSERT( hInstance != NULL );
|
|
|
|
return bResult ;
|
|
}
|