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

... 31  32  33  34  35  36  37  38  39  40  41  42  43  [44]  45  ...
NoWriterDateCnt.TitleFile(s)
12872정성태12/12/202116793개발 환경 구성: 615. GoLang과 Python 빌드가 모두 가능한 docker 이미지 만들기
12871정성태12/12/202115699오류 유형: 771. docker: Error response from daemon: OCI runtime create failed
12870정성태12/9/202115295개발 환경 구성: 614. 파이썬 - PyPI 패키지 만들기 (4) package_data 옵션
12869정성태12/8/202117832개발 환경 구성: 613. git clone 실행 시 fingerprint 묻는 단계를 생략하는 방법
12868정성태12/7/202116514오류 유형: 770. twine 업로드 시 "HTTPError: 400 Bad Request ..." 오류 [1]
12867정성태12/7/202116185개발 환경 구성: 612. 파이썬 - PyPI 패키지 만들기 (3) entry_points 옵션
12866정성태12/7/202123120오류 유형: 769. "docker build ..." 시 "failed to solve with frontend dockerfile.v0: failed to read dockerfile ..." 오류
12865정성태12/6/202116488개발 환경 구성: 611. 파이썬 - PyPI 패키지 만들기 (2) long_description, cmdclass 옵션
12864정성태12/6/202114158Linux: 46. WSL 환경에서 find 명령을 사용해 파일을 찾는 방법
12863정성태12/4/202116197개발 환경 구성: 610. 파이썬 - PyPI 패키지 만들기
12862정성태12/3/202113902오류 유형: 768. Golang - 빌드 시 "cmd/go: unsupported GOOS/GOARCH pair linux /amd64" 오류
12861정성태12/3/202117900개발 환경 구성: 609. 파이썬 - "Windows embeddable package"로 개발 환경 구성하는 방법 [1]
12860정성태12/1/202113933오류 유형: 767. SQL Server - 127.0.0.1로 접속하는 경우 "Access is denied"가 발생한다면?
12859정성태12/1/202122315개발 환경 구성: 608. Hyper-V 가상 머신에 Console 모드로 로그인하는 방법
12858정성태11/30/202119627개발 환경 구성: 607. 로컬의 USB 장치를 원격 머신에 제공하는 방법 - usbip-win
12857정성태11/24/202116257개발 환경 구성: 606. WSL Ubuntu 20.04에서 파이썬을 위한 uwsgi 설치 방법
12856정성태11/23/202119009.NET Framework: 1121. C# - 동일한 IP:Port로 바인딩 가능한 서버 소켓 [2]
12855정성태11/13/202114257개발 환경 구성: 605. Azure App Service - Kudu SSH 환경에서 FTP를 이용한 파일 전송
12854정성태11/13/202116379개발 환경 구성: 604. Azure - 윈도우 VM에서 FTP 여는 방법
12853정성태11/10/202114833오류 유형: 766. Azure App Service - JBoss 호스팅 생성 시 "This region has quota of 0 PremiumV3 instances for your subscription. Try selecting different region or SKU."
12851정성태11/1/202116527스크립트: 34. 파이썬 - MySQLdb 기본 예제 코드
12850정성태10/27/202118322오류 유형: 765. 우분투에서 pip install mysqlclient 실행 시 "OSError: mysql_config not found" 오류
12849정성태10/17/202117884스크립트: 33. JavaScript와 C#의 시간 변환 [1]
12848정성태10/17/202117029스크립트: 32. 파이썬 - sqlite3 기본 예제 코드 [1]
12847정성태10/14/202117732스크립트: 31. 파이썬 gunicorn - WORKER TIMEOUT 오류 발생
12846정성태10/7/202118773스크립트: 30. 파이썬 __debug__ 플래그 변수에 따른 코드 실행 제어
... 31  32  33  34  35  36  37  38  39  40  41  42  43  [44]  45  ...