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분
정성태

... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13161정성태11/14/20225922.NET Framework: 2068. C# - PublishSingleFile로 배포한 이미지의 역어셈블 가능 여부 (난독화 필요성) [4]
13160정성태11/11/20225837.NET Framework: 2067. C# - PublishSingleFile 적용 시 native/managed 모듈 통합 옵션
13159정성태11/10/20229104.NET Framework: 2066. C# - PublishSingleFile과 관련된 옵션 [3]
13158정성태11/9/20225291오류 유형: 826. Workload definition 'wasm-tools' in manifest 'microsoft.net.workload.mono.toolchain' [...] conflicts with manifest 'microsoft.net.workload.mono.toolchain.net7'
13157정성태11/8/20225964.NET Framework: 2065. C# - Mutex의 비동기 버전파일 다운로드1
13156정성태11/7/20226862.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점파일 다운로드1
13155정성태11/4/20226366디버깅 기술: 183. TCP 동시 접속 (연결이 아닌) 시도를 1개로 제한한 서버
13154정성태11/3/20225843.NET Framework: 2063. .NET 5+부터 지원되는 GC.GetGCMemoryInfo파일 다운로드1
13153정성태11/2/20227147.NET Framework: 2062. C# - 코드로 재현하는 소켓 상태(SYN_SENT, SYN_RECV)
13152정성태11/1/20225746.NET Framework: 2061. ASP.NET Core - DI로 추가한 클래스의 초기화 방법 [1]
13151정성태10/31/20225877C/C++: 161. Windows 11 환경에서 raw socket 테스트하는 방법파일 다운로드1
13150정성태10/30/20225922C/C++: 160. Visual Studio 2022로 빌드한 C++ 프로그램을 위한 다른 PC에서 실행하는 방법
13149정성태10/27/20225849오류 유형: 825. C# - CLR ETW 이벤트 수신이 GCHeapStats_V1/V2에 대해 안 되는 문제파일 다운로드1
13148정성태10/26/20225817오류 유형: 824. msbuild 에러 - error NETSDK1005: Assets file '...\project.assets.json' doesn't have a target for 'net5.0'. Ensure that restore has run and that you have included 'net5.0' in the TargetFramew
13147정성태10/25/20224910오류 유형: 823. Visual Studio 2022 - Unable to attach to CoreCLR. The debugger's protocol is incompatible with the debuggee.
13146정성태10/24/20225759.NET Framework: 2060. C# - Java의 Xmx와 유사한 힙 메모리 최댓값 제어 옵션 HeapHardLimit
13145정성태10/21/20226037오류 유형: 822. db2 - Password validation for user db2inst1 failed with rc = -2146500508
13144정성태10/20/20225869.NET Framework: 2059. ClrMD를 이용해 윈도우 환경의 메모리 덤프로부터 닷넷 모듈을 추출하는 방법파일 다운로드1
13143정성태10/19/20226406오류 유형: 821. windbg/sos - Error code - 0x000021BE
13142정성태10/18/20225368도서: 시작하세요! C# 12 프로그래밍
13141정성태10/17/20226906.NET Framework: 2058. [in,out] 배열을 C#에서 C/C++로 넘기는 방법 - 세 번째 이야기파일 다운로드1
13140정성태10/11/20226280C/C++: 159. C/C++ - 리눅스 환경에서 u16string 문자열을 출력하는 방법 [2]
13139정성태10/9/20226086.NET Framework: 2057. 리눅스 환경의 .NET Core 3/5+ 메모리 덤프로부터 모든 닷넷 모듈을 추출하는 방법파일 다운로드1
13138정성태10/8/20227387.NET Framework: 2056. C# - await 비동기 호출을 기대한 메서드가 동기로 호출되었을 때의 부작용 [1]
13137정성태10/8/20225749.NET Framework: 2055. 리눅스 환경의 .NET Core 3/5+ 메모리 덤프로부터 닷넷 모듈을 추출하는 방법
13136정성태10/7/20226322.NET Framework: 2054. .NET Core/5+ SDK 설치 없이 dotnet-dump 사용하는 방법
... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...