INF: Sample of SQL-DMO Connection Point Interface (156100)
The information in this article applies to:
This article was previously published under Q156100 SUMMARY
The sample code in this article shows you how to:
- Create a connection point interface.
- Display detailed error information.
- Access Severity 0 to 10 error messages returned from SQL
Server.
It also contains a workaround to the Microsoft Knowledge Base article
Q152621, "BUG: SQL-DMO: GetMemUsage Method Returns Empty String."
Before proceeding, make sure you take note of the following:
- You must implement all methods exposed by the Sink object. These are
documented in the "What's New in SQL Server 6.5" portion of Books
Online.
- You must appropriately handle the IUnknown interface methods. This
sample uses the default implementations of the IUnknown interface where
the last Release deletes the object.
MORE INFORMATIONMAIN.CPP
//
// The following code is an example of how to implement the SQL-DMO
// Server connection point to trap all error messages returned from the
// SQL Server.
//
//
// Includes:
//
#include "afx.h"
#include "afxole.h" // OLE
#include "stdio.h" // Standard I/O header
#include "initguid.h" // Only included once to define the GUID and UUID's properly
#include "sqloleid.h" // SQL-DMO headers
#include "sqlole.h"
#include "ServerConnectionPoint.h" // Server connection point implementation
//
// Defines:
//
#define _SERVER ""
#define _USER "sa"
#define _PWD ""
#define _BAD_COOKIE 99999
//
// Function declarations:
//
BOOL bInitialize(void);
void vUninitialize(void);
HRESULT hrDisplayError(HRESULT hRes);
BOOL bInstallConnectionPointHandler(void);
void vDoSomeStuff(void);
//
// Local variables:
//
BOOL bCoInit = FALSE;
LPSQLOLESERVER iSQLServer = NULL;
LPCONNECTIONPOINT iCP = NULL;
CSQLServerSink * pServerSink = NULL;
DWORD dwCookie = _BAD_COOKIE;
//
// MAIN ROUTINE
// ------------
void main(void)
{
if(bInitialize())
{
if(bInstallConnectionPointHandler())
{
vDoSomeStuff();
}
}
//
// Perform cleanup operations:
//
vUninitialize();
printf("\n\nSample run completed...\n\n");
}
//
// Initialize - perform all init operations:
//
BOOL bInitialize(void)
{
BOOL bRC = FALSE;
printf("\n...Initializing OLE...");
if(SUCCEEDED(CoInitialize(NULL)))
{
bCoInit = TRUE;
//
// Create a Server object:
//
printf("\n...Creating SQL Server Object...");
if(SUCCEEDED(hrDisplayError(CoCreateInstance(CLSID_SQLOLEServer,
NULL,
CLSCTX_INPROC_SERVER,
IID_ISQLOLEServer,
(LPVOID *)&iSQLServer))))
{
bRC = TRUE;
}
}
else
{
printf("\nCoInitialize failed.");
bRC = bCoInit = FALSE;
}
return bRC;
}
//
// Uninitialize - perform all shutdown operations:
//
void vUninitialize(void)
{
printf("\n...Performing the Unadvice...");
if((dwCookie != _BAD_COOKIE) && (iCP))
iCP->Unadvise(dwCookie);
printf("\n...Cleaning up object memory...");
//
// Not part of above sequence in case Advise failed and dwCookie was
// (0)
//
if(iCP)
iCP->Release();
//
// Clean up the server object:
//
if(iSQLServer)
iSQLServer->Release();
//
// Shut down OLE:
//
if(bCoInit)
CoUninitialize();
}
//
// Install the connection point handler:
//
BOOL bInstallConnectionPointHandler(void)
{
BOOL bRC = FALSE;
LPCONNECTIONPOINTCONTAINER iCPContainer = NULL;
//
// Create an instance of the Server Sink object:
//
pServerSink = new CSQLServerSink;
if(pServerSink)
{
printf("\n...Creating Connection Point Container...");
if(SUCCEEDED(hrDisplayError(iSQLServer->QueryInterface(IID_IConnectionPointContainer,
(LPVOID *) &iCPContainer))))
{
printf("\n...Finding the SQL Server Sink connection point...");
if(SUCCEEDED(hrDisplayError(iCPContainer->FindConnectionPoint
(IID_ISQLOLEServerSink, &iCP))))
{
printf("\n...Advising connection point that we are available to
receive events...");
if(SUCCEEDED(hrDisplayError(iCP->Advise(pServerSink, &dwCookie))))
{
bRC = TRUE;
}
else
dwCookie = _BAD_COOKIE;
}
iCPContainer->Release();
}
}
else
{
printf("\nAttempt to create server sink object failed due to a memory error.");
}
//
// Free memory if necessary:
//
if((pServerSink) && (FALSE == bRC))
{
delete pServerSink;
pServerSink = NULL;
}
return bRC;
}
//
// Do some things to cause the Sink events to fire:
//
void vDoSomeStuff(void)
{
printf("\n...Setting connection options...");
if(SUCCEEDED(hrDisplayError(iSQLServer->SetLoginTimeout(5))))
{
if(SUCCEEDED(hrDisplayError(iSQLServer->SetApplicationName("Server Sink"))))
{
if(SUCCEEDED(hrDisplayError(iSQLServer->SetHostName("SQL-DMO"))))
{
if(SUCCEEDED(hrDisplayError(iSQLServer->SetNetPacketSize(4096))))
{
printf("\n...Attempting to connect...");
if(SUCCEEDED(hrDisplayError(iSQLServer->Connect(_SERVER,
_USER, _PWD))))
{
//
// Cause a meesage to be fired from SQL Server.
//
// This is one workaround for Q152621 (Bug #15500.)
// You could also use ExecuteWithResultsAndMessages.
//
// You may want to use Sink to capute information
// output from any DBCC command because the standard
// implementation of Database.CheckAllocations uses
// 'WITH NO_INFOMSGS'
//
hrDisplayError(iSQLServer->ExecuteImmediate("dbcc memusage"));
//
// This will fire the Sink event.
//
hrDisplayError(iSQLServer->ExecuteImmediate("raiserror(1204, 1, 1)"));
//
// This will NOT fire the Sink event due to Sev level.
//
hrDisplayError(iSQLServer->ExecuteImmediate("raiserror(1204, 11, 1)"));
//
// Close connection.
//
hrDisplayError(iSQLServer->DisConnect());
}
}
}
}
}
}
//
// Display Error Information as a result of an OLE call
//
HRESULT hrDisplayError(HRESULT hRes)
{
LPERRORINFO lpErrorInfo = NULL;
BSTR bstrDesc;
BSTR bstrSource;
if(FAILED(hRes))
{
if(SUCCEEDED(GetErrorInfo(0, &lpErrorInfo)))
{
lpErrorInfo->GetDescription(&bstrDesc);
lpErrorInfo->GetSource(&bstrSource);
printf("\n\nhrDisplayError: %S\n%S\n", bstrSource, bstrDesc);
lpErrorInfo->Release();
SysFreeString(bstrDesc);
SysFreeString(bstrSource);
}
else
printf("\n\nUnable to obtain detailed error information.");
}
return hRes;
}
ServerConnectionPoint.h
//
// SQL-DMO Server Connection Point Interface Declaration and
// Implementation
//
#include "windows.h"
#include "stdio.h"
#include "sqloleid.h" // SQL-DMO headers
#include "sqlole.h"
class CSQLServerSink : public ISQLOLEServerSink
{
public:
CSQLServerSink()
{
m_uiRefCount = 0;
}
~CSQLServerSink()
{
}
//
// IUnknown Interface
//
STDMETHOD(QueryInterface) (THIS_ REFIID riid, LPVOID *ppvObj)
{
if((riid == IID_IUnknown) || (riid == IID_IASQLOLEServerSink))
{
AddRef();
*ppvObj = this;
return NOERROR;
}
else
return E_NOINTERFACE;
}
STDMETHOD_(ULONG,AddRef) (THIS)
{
return ++m_uiRefCount;
}
STDMETHOD_(ULONG,Release) (THIS)
{
--m_uiRefCount;
if(0 == m_uiRefCount)
delete this;
return m_uiRefCount;
}
//
// Sink properties and methods:
//
STDMETHOD(QueryTimeout)(THIS_ SQLOLE_LPCSTR strMessage, LPBOOL pbContinue)
{
printf("\nSINK QueryTimeout - %s", strMessage);
*pbContinue = FALSE;
return NOERROR;
}
//
// Only for designated error messages Sev 10 or less.
//
STDMETHOD(ServerMessage)(THIS_ long lMessageSeverity, long lMessageNumber,
long lMessageState, SQLOLE_LPCSTR strMessage)
{
printf( "\nSINK ServerMessage"
"\nError: %ld"
"\nSev: %ld"
"\nState: %ld"
"\n%s\n",
lMessageNumber, lMessageSeverity, lMessageState, strMessage);
return NOERROR;
}
STDMETHOD(ConnectionBroken)(THIS_ SQLOLE_LPCSTR strMessage, LPBOOL pbRetry)
{
printf("\nSINK ConnectionBroken - %s", strMessage);
*pbRetry = TRUE;
return NOERROR;
}
STDMETHOD(RemoteLoginFailed)(THIS_ long lMessageSeverity, long lMessageNumber,
long lMessageState, SQLOLE_LPCSTR strMessage)
{
printf( "\nSINK RemoteLoginFailed"
"\nError: %ld"
"\nSev: %ld"
"\nState: %ld"
"\n%s\n",
lMessageNumber, lMessageSeverity, lMessageState, strMessage);
return NOERROR;
}
//
// Shows actual commands begin sent from SQL-DMO to SQL Server.
//
STDMETHOD(CommandSent)(THIS_ SQLOLE_LPCSTR strSQL)
{
printf("\nSINK CommandSent - %s", strSQL);
return NOERROR;
}
private:
UINT m_uiRefCount;
};
Modification Type: | Major | Last Reviewed: | 10/3/2003 |
---|
Keywords: | kbnetwork KB156100 |
---|
|