Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

SQL 서버 - DB 테이블의 데이터 변경에 대한 알림 처리

마침, 아래와 같이 질문해 주신 분이 있군요. ^^

주식형태의 프로그램 처럼 SQL서버의 특정 필드 데이터의 변화가 있을때 재 클라이언트가 정보를 갱신 할 수 있게 하는 방법은 없을까요?
; https://www.sysnet.pe.kr/3/0/981

덕분에, Notification Services가 2008부터 누락되었다는 것도 알게 되었습니다. 아래의 글에 따르면, 누락은 되었지만 SQL Server 2005 SP3에 포함된 Notification Services와 연동이 되는 것은 가능하다고 합니다.

SQL Server Notification Services
; http://en.wikipedia.org/wiki/SQL_Server_Notification_Services

어찌되었든, 저도 Notification Services를 사용해 본 적은 없으므로 더 이상 언급할 만한 가치는 없겠고.

이제 다른 방법을 찾아봐야 하는데, 다행히 검색을 해보면 WMI 이벤트로 알림 기능이 제공된다는 글이 있습니다.

wmi Sql table notification
; http://www.ureader.com/msg/14861679.aspx

아하, AUDIT_SCHEMA_OBJECT_ACCESS_EVENT가 있다는군요. 사용 방법에 관한 구체적인 코드는 다음과 같이 하면 되겠습니다.

Sample: Using the WMI Event Provider with the .NET Framework
; https://docs.microsoft.com/en-us/sql/relational-databases/wmi-provider-server-events/sample-using-the-wmi-event-provider-with-the-net-framework




실제로 한번 테스트를 해보았습니다. 우선, 다음과 같이 간단한 DB 및 테이블을 만들고,

sql_wmi_event_1.png

이를 기반으로 WMI Query를 구성하는데, DB 이름으로 'TestDB' 및 테이블 이름으로 'TestTable'로 제한하고 수행되는 쿼리에 "INSERT INTO", "UPDATE" 문자열을 담고 있으면 알림을 받도록 했습니다.

string query = @"Select * From AUDIT_SCHEMA_OBJECT_ACCESS_EVENT Where ObjectName = 
'TestTable' And DatabaseName = 'TestDB' and (TextData Like '%INSERT INTO%' or TextData Like '%UPDATE %')";
            
// Default namespace for default instance of SQL Server 
string managementPath = @"\\.\root\Microsoft\SqlServer\ServerEvents\MSSQLSERVER";
ManagementEventWatcher watcher = new ManagementEventWatcher(new WqlEventQuery(query));
ManagementScope scope = new ManagementScope(managementPath);
scope.Connect();
watcher.Scope = scope;
Console.WriteLine("Watching...");
while (true)
{
    ManagementBaseObject obj = watcher.WaitForNextEvent();
    foreach (PropertyData data in obj.Properties)
    {
        Console.Write("{0}:", data.Name);
        if (data.Value == null)
        {
            Console.WriteLine("<null>");
        }
        else
        {
            Console.WriteLine(data.Value.ToString());
        }
    }
}

아쉽게도, 위의 WMI 코드를 실행하려면 '관리자 권한'이 필요하다는 정도겠군요. 동작이 잘 되는지 아래와 같이 INSERT 쿼리를 실행해 보면,

INSER INTO TestTable(id) VALUES(1)

이벤트를 받게 되고, PropertyData 내용은 다음과 같이 출력됩니다.

ApplicationName:Microsoft SQL Server Management Studio - Query
BinaryData:System.Byte[]
ClientProcessID:12312
ColumnPermissions:0
ComputerName:TESTPC
DatabaseID:7
DatabaseName:TestDB
DBUserName:dbo
EventSequence:744
EventSubClass:0
HostName:TESTPC
IsSystem:0
LineNumber:1
LoginName:TESTPC\TestUser
LoginSid:System.Byte[]
NestLevel:0
NTDomainName:TESTPC
NTUserName:TestUser
ObjectName:TestTable
ObjectType:8277
OwnerName:dbo
ParentName:dbo
Permissions:8
PostTime:20110711175034.000470+000
RequestID:0
SECURITY_DESCRIPTOR:<null>
ServerName:TESTPC
SessionLoginName:TESTPC\TestUser
SPID:53
SQLInstance:MSSQLSERVER
StartTime:20110711175034.000470+000
Success:1
TextData:INSERT INTO [TestTable]([Id]) values(@1)
TIME_CREATED:129548478344755975
TransactionID:95564
XactSequence:227633266689

UPDATE의 경우에도,

UPDATE TestTable SET Id = 4

알림으로 받게되는 속성은 다음과 같습니다.

ApplicationName:Microsoft SQL Server Management Studio
BinaryData:System.Byte[]
ClientProcessID:12312
ColumnPermissions:1
ComputerName:TESTPC
DatabaseID:7
DatabaseName:TestDB
DBUserName:dbo
EventSequence:1180
EventSubClass:0
HostName:TESTPC
IsSystem:0
LineNumber:1
LoginName:TESTPC\TestUser
LoginSid:System.Byte[]
NestLevel:0
NTDomainName:TESTPC
NTUserName:TestUser
ObjectName:TestTable
ObjectType:8277
OwnerName:dbo
ParentName:dbo
Permissions:2
PostTime:20110711175410.000623+000
RequestID:0
SECURITY_DESCRIPTOR:<null>
ServerName:TESTPC
SessionLoginName:TESTPC\TestUser
SPID:58
SQLInstance:MSSQLSERVER
StartTime:20110711175410.000623+000
Success:1
TextData:UPDATE [TestTable] set [Id] = @1
TIME_CREATED:129548480515150114
TransactionID:98062
XactSequence:249108103169

아... 아깝군요. 실행 시간 정보까지만 나왔어도 제니퍼 닷넷에 기능 추가를 해보는 것도 좋을 텐데. ^^

첨부된 파일은 위의 코드를 포함한 예제 프로젝트입니다.





[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/17/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2019-09-04 04시59분
[JangHun] WMI 코드를 실행하려면 관리자 권한이 필요하다고 말씀해주셨는데
코드 실행을 관리자 권한으로 하려면 어떻게 해야하나요?
[guest]
2019-09-04 05시44분
[JangHun] SQL SERVER 2017을 설치하고, 데이터베이스와 테이블을 만들어놓은 상태에서

string query = @"Select * From AUDIT_SCHEMA_OBJECT_ACCESS_EVENT Where ObjectName =
'TestTable' And DatabaseName = 'TestDB' and (TextData Like '%INSERT INTO%' or TextData Like '%UPDATE %')";
            
// Default namespace for default instance of SQL Server
string managementPath = @"\\.\root\Microsoft\SqlServer\ServerEvents\MSSQLSERVER";
.
.

위 코드를 따라서 작성하는데, Access-Denied가 됩니다.. 혹시 SQL SERVER버전이 달라져서 뭔가 달라진것일까요..?
[guest]
2019-09-05 08시42분
"코드"만을 관리자 권한으로 실행할 수 있는 방법은 없고, 실행 프로세스 자체를 애당초 관리자 권한으로 실행해야 합니다. 관련해서는 다음의 글을 참고하시고.

ClickOnce - 관리자 권한 상승하는 방법
; http://www.sysnet.pe.kr/2/0/950

관리자 권한이 필요한 작업을 COM+ 에 대행
; http://www.sysnet.pe.kr/2/0/1290

그다음 문제인, Access-Denied가 발생한 것은 버전과는 무관할 것으로 보입니다. 아마도 관리자 권한으로 실행하지 않아서 발생하는 것 같은데, 위의 문제를 해결하면 자연스럽게 없어질 것입니다.
정성태
2022-01-18 01시16분
정성태

... 166  167  168  169  170  171  172  173  174  175  176  177  [178]  179  180  ...
NoWriterDateCnt.TitleFile(s)
637정성태1/1/200930651기타: 26. 2008년 인기 순위 정리
636정성태12/31/200825886.NET Framework: 118. 2진 검색을 이용한 리스트 정렬 삽입파일 다운로드1
635정성태12/29/200828742오류 유형: 66. 파일 암호화 오류 - Recovery policy configured for this system contains invalid recovery certificate
634정성태12/29/200843052기타: 25. 가상 키보드 관련 정리 [4]
633정성태12/20/200828326기타: 24. RMClock for x64 [2]
632정성태12/19/200836787기타: 23. D820 - 배터리 없이 바이오스 업데이트 방법 [2]파일 다운로드1
631정성태12/10/200845916VC++: 36. Detours 라이브러리를 이용한 Win32 API - Sleep 호출 가로채기 [3]
630정성태12/9/200826482.NET Framework: 117. WPF - TreeView에서 항목이 펼쳐질 때 Cursors.Wait 사용파일 다운로드1
629정성태12/7/200836131.NET Framework: 116. 소켓 연결 시간 제한
628정성태12/6/200824128.NET Framework: 115. Marshal 타입 관련 2가지 자원 해제 메서드파일 다운로드1
627정성태12/6/200827002VS.NET IDE: 58. VS.NET IDE 팁 - 커서 위치 이동 [1]
626정성태12/6/200826834오류 유형: 65. TF53018: The application tier XXXXXXX is attempting to connect to a data tier with an incompatible version
625정성태12/6/200827276오류 유형: 64. TFS 2008 SP1 설치 - MsiApplyMultiplePatches returned 0x643
624정성태12/5/200828273.NET Framework: 114. WPF 이벤트에 속한 핸들러 확인 [2]파일 다운로드1
623정성태12/4/200832830디버깅 기술: 22. VS.NET SP1 + .NET Framework 소스 코드 디버깅 [2]파일 다운로드1
622정성태12/1/200835073오류 유형: 63. WPF - XamlParseException 대응 방법 [2]
621정성태11/30/200826972Team Foundation Server: 27. TeamBuild + VDPROJ 셋업 프로젝트 [1]
620정성태11/30/200825770디버깅 기술: 21. 올바른 이벤트 예외 정보 출력
619정성태11/30/200825889디버깅 기술: 20. 예외 처리를 방해하는 WPF Modal 대화창파일 다운로드1
618정성태11/29/200826239.NET Framework: 113. 이벤트에 속한 이벤트 핸들러 확인파일 다운로드1
617정성태11/26/200833155.NET Framework: 112. How to Interop DISPPARAMS [2]파일 다운로드2
616정성태11/26/200825768디버깅 기술: 19. C++/CLI - F11 디버깅 시의 변수 초기화파일 다운로드1
615정성태11/9/200835207.NET Framework: 111. WPF - Window, UserControl 클래스 상속 [1]
614정성태11/9/200835104.NET Framework: 110. WPF - 전역 예외 처리 [4]파일 다운로드1
613정성태11/8/200824507.NET Framework: 109. WPF - SystemColors 색상표파일 다운로드1
612정성태11/1/200830067.NET Framework: 108. WPF + WCF 환경에서는 DataContract를 권장 [1]
... 166  167  168  169  170  171  172  173  174  175  176  177  [178]  179  180  ...