MS SQL监控数据库的DDL操作

简介: 前言: 有时候,一个数据库有多个帐号,包括数据库管理员,开发人员,运维支撑人员等,可能有很多帐号都有比较大的权限,例如DDL操作权限(创建,修改,删除存储过程,创建,修改,删除表等),账户多了,管理起来就会相当麻烦,容易产生混乱,如果数据库管理员不监控数据库架构变更的话,就不知道谁对数据库架...

    前言: 有时候,一个数据库有多个帐号,包括数据库管理员,开发人员,运维支撑人员等,可能有很多帐号都有比较大的权限,例如DDL操作权限(创建,修改,删除存储过程,创建,修改,删除表等),账户多了,管理起来就会相当麻烦,容易产生混乱,如果数据库管理员不监控数据库架构变更的话,就不知道谁对数据库架构做了啥改动(此处改动仅仅只DDL操作),尤其有时候,有些开发人员可能不按规章制度办事,绕过或忘了通知发布人员或DBA,直接去生产机做一些DDL操作,那么我们就需要对数据库架构某些更改的事件进行监控,如果能够监控并留下证据,这样既可以让DBA或相关管理人员知晓这些变更,有效管理数据库,也可以避免出现问题,出现扯皮现象,最后DBA成了背黑锅的。

  下面就是一个解决上述问题的方案,我们通过创建一个表DatabaseLog和DDL触发器来解决问题,首先在msdb数据库里面新建一个表DatabaseLog,用来保存DDL触发器获取的信息。其中DDL触发器主要通过EVENTDATA()函数返回有关服务器或数据库事件的信息。

SQL Code 1
  1. USE msdb;
  2.  
  3. GO
  4.  
  5.  
  6. CREATE TABLE [dbo].[DatabaseLog]
  7.  
  8. (
  9.  
  10.     [DatabaseLogID]   [int]    IDENTITY(1,1) NOT NULL,
  11.  
  12.     [PostTime]        [datetime] NOT NULL,
  13.  
  14.     [DatabaseUser]    [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL,
  15.  
  16.     [LoginName]       [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  17.  
  18.     [ClientHost]      [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  19.  
  20.     [Event]           [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL,
  21.  
  22.     [Schema]          [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  23.  
  24.     [Object]          [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  25.  
  26.     [TSQL]            [nvarchar](max) COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL,
  27.  
  28.     [XmlEvent]        [xml] NOT NULL,
  29.  
  30. CONSTRAINT [PK_DatabaseLog_DatabaseLogID] PRIMARY KEY NONCLUSTERED
  31.  
  32. (
  33.  
  34.     [DatabaseLogID] ASC
  35.  
  36.   )WITH (PAD_INDEX= OFF, IGNORE_DUP_KEY = OFF) ON [PRIMARY]
  37.  
  38. ) ON [PRIMARY]
  39.  
  40.  
  41. GO
  42.  
  43. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'Primary key for DatabaseLog records.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'DatabaseLogID'
  44.  
  45. GO
  46.  
  47. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The date and time the DDL change occurred.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'PostTime'
  48.  
  49. GO
  50.  
  51. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The user who implemented the DDL change.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'DatabaseUser'
  52.  
  53. GO
  54.  
  55. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The login which implemented the DDL change.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'LoginName'
  56.  
  57. GO
  58.  
  59. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The client machine on which implemented the DDL change.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'ClientHost'
  60.  
  61. GO
  62.  
  63. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The type of DDL statement that was executed.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'Event'
  64.  
  65. GO
  66.  
  67. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The schema to which the changed object belongs.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'Schema'
  68.  
  69. GO
  70.  
  71. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The object that was changed by the DDL statment.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'Object'
  72.  
  73. GO
  74.  
  75. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The exact Transact-SQL statement that was executed.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'TSQL'
  76.  
  77. GO
  78.  
  79. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'The raw XML data generated by database trigger.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'COLUMN',@level2name=N'XmlEvent'
  80.  
  81. GO
  82.  
  83. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'Audit table tracking all DDL changes made to the database. Data is captured by the database trigger ddlDatabaseTriggerLog.' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog'
  84.  
  85. GO
  86.  
  87. EXEC sys.sp_addextendedproperty@name=N'MS_Description', @value=N'Primary key (nonclustered) constraint' , @level0type=N'SCHEMA',@level0name=N'dbo', @level1type=N'TABLE',@level1name=N'DatabaseLog', @level2type=N'CONSTRAINT',@level2name=N'PK_DatabaseLog_DatabaseLogID'
  88.  
  89.  
  90. GO

例如,我要监控数据库MyAssistant的DDL操作,那么我们首先在“数据库邮件”里面创建一个配置名为“ DataBase_DDL_Event”的配置文件(profile name),这个就不多讲了,不知道配置的,自己先练练手把,假如我需要让数据库把监控到DDL操作变动相信信息发送到我的邮箱 *****@***.com(用你自己的邮箱替代),那么只需要修改下面代码的邮箱和profile_name即可。

SQL Code 2
  1. USE MyAssistant;
  2. GO
  3.  
  4. CREATE TRIGGER [DTG_DatabaseDdlTriggerLog]
  5. ON DATABASE
  6. FOR DDL_DATABASE_LEVEL_EVENTS
  7. AS
  8. BEGIN
  9.     SET NOCOUNT ON;
  10.  
  11.     DECLARE @data XML;
  12.     DECLARE @schema sysname;
  13.     DECLARE @object sysname;
  14.     DECLARE @eventType sysname;
  15.     DECLARE @tableHTML  NVARCHAR(MAX) ;
  16.     
  17.     SET @data = EVENTDATA();
  18.     SET @eventType = @data.value('(/EVENT_INSTANCE/EventType)[1]', 'sysname');
  19.     SET @schema = @data.value('(/EVENT_INSTANCE/SchemaName)[1]', 'sysname');
  20.     SET @object = @data.value('(/EVENT_INSTANCE/ObjectName)[1]', 'sysname')
  21.  
  22.     IF @object IS NOT NULL
  23.         PRINT '  ' + @eventType + ' - ' + @schema + '.' + @object;
  24.     ELSE
  25.         PRINT '  ' + @eventType + ' - ' + @schema;
  26.  
  27.     IF @eventType IS NULL
  28.         PRINT CONVERT(nvarchar(max), @data);
  29.  
  30.     INSERT [msdb].[dbo].[DatabaseLog]
  31.         (
  32.         [PostTime],
  33.         [DatabaseUser],
  34.         [LoginName],
  35.         [ClientHost],
  36.         [Event],
  37.         [Schema],
  38.         [Object],
  39.         [TSQL],
  40.         [XmlEvent]
  41.         )
  42.     VALUES
  43.         (
  44.         GETDATE(),
  45.         CONVERT(sysname, CURRENT_USER),
  46.         @data.value('(/EVENT_INSTANCE/LoginName)[1]', 'nvarchar(max)'),
  47.         CONVERT(sysname, HOST_NAME()),
  48.         @eventType,
  49.         CONVERT(sysname, @schema),
  50.         CONVERT(sysname, @object),
  51.         @data.value('(/EVENT_INSTANCE/TSQLCommand)[1]', 'nvarchar(max)'),
  52.         @data
  53.         );
  54.  
  55.     SET @tableHTML =   
  56.     N'<H1>DDL Event</H1>' +     
  57.     N'<table border="0">' +     
  58.     N'<tr><th>Post Time</th><th>User</th><th>Login</th><th>ClientHost</th>' +     
  59.     N'<th>TSQL</th><th></tr>' +     
  60.     CAST(( SELECT
  61.     td = PostTime,       '',                     
  62.     td = DatabaseUser, '',        
  63.     td = LoginName, '',     
  64.     td = ClientHost, '',         
  65.     td = TSQL, ''               
  66.     FROM msdb.dbo.DatabaseLog               
  67.     WHERE DatabaseLogID =(select max(DatabaseLogID) from msdb.dbo.DatabaseLog)               
  68.     FOR XML PATH('tr'), TYPE     ) AS NVARCHAR(MAX) ) +     N'</table>' ;
  69.  
  70.     EXEC msdb.dbo.sp_send_dbmail     
  71.              @profile_name = 'DataBase_DDL_Event',
  72.         @recipients='***@***.com',     
  73.         @subject = 'DDL Event - DataBase MyAssistant',     
  74.         @body = @tableHTML,   
  75.         @body_format = 'HTML' ;
  76. END;
  77.  
  78.  
  79.  
  80. GO

接下来我们来测试一下,假如一个用户Test登录数据库,一不小心删除了一个Test的表,如下图一所示,那么我将收到一封邮件,提示我用户Test在那台客户端主机执行了啥DDL操作(如下图二所示),当然邮件的样式、排版有兴趣的可以去美化一下。

 

clipboard

clipboard[1]

相关文章
|
4月前
|
SQL 机器学习/深度学习 人工智能
从“写SQL”到“聊数据”:NL2SQL如何用自然语言解锁数据库?
本文系统性地阐述了自然语言转SQL(NL2SQL) 技术如何让非技术背景的业务分析师实现数据自助查询,从而提升数据驱动决策的效率与准确性。
从“写SQL”到“聊数据”:NL2SQL如何用自然语言解锁数据库?
|
3月前
|
SQL 人工智能 Linux
SQL Server 2025 RC1 发布 - 从本地到云端的 AI 就绪企业数据库
SQL Server 2025 RC1 发布 - 从本地到云端的 AI 就绪企业数据库
396 5
SQL Server 2025 RC1 发布 - 从本地到云端的 AI 就绪企业数据库
|
2月前
|
SQL 存储 监控
SQL日志优化策略:提升数据库日志记录效率
通过以上方法结合起来运行调整方案, 可以显著地提升SQL环境下面向各种搜索引擎服务平台所需要满足标准条件下之数据库登记作业流程综合表现; 同时还能确保系统稳健运行并满越用户体验预期目标.
204 6
|
3月前
|
关系型数据库 MySQL 数据库
阿里云数据库RDS费用价格:MySQL、SQL Server、PostgreSQL和MariaDB引擎收费标准
阿里云RDS数据库支持MySQL、SQL Server、PostgreSQL、MariaDB,多种引擎优惠上线!MySQL倚天版88元/年,SQL Server 2核4G仅299元/年,PostgreSQL 227元/年起。高可用、可弹性伸缩,安全稳定。详情见官网活动页。
|
3月前
|
关系型数据库 分布式数据库 数据库
阿里云数据库收费价格:MySQL、PostgreSQL、SQL Server和MariaDB引擎费用整理
阿里云数据库提供多种类型,包括关系型与NoSQL,主流如PolarDB、RDS MySQL/PostgreSQL、Redis等。价格低至21元/月起,支持按需付费与优惠套餐,适用于各类应用场景。
|
3月前
|
SQL Oracle 关系型数据库
Oracle数据库创建表空间和索引的SQL语法示例
以上SQL语法提供了一种标准方式去组织Oracle数据库内部结构,并且通过合理使用可以显著改善查询速度及整体性能。需要注意,在实际应用过程当中应该根据具体业务需求、系统资源状况以及预期目标去合理规划并调整参数设置以达到最佳效果。
309 8
|
4月前
|
SQL 人工智能 Java
用 LangChain4j+Ollama 打造 Text-to-SQL AI Agent,数据库想问就问
本文介绍了如何利用AI技术简化SQL查询操作,让不懂技术的用户也能轻松从数据库中获取信息。通过本地部署PostgreSQL数据库和Ollama模型,结合Java代码,实现将自然语言问题自动转换为SQL查询,并将结果以易懂的方式呈现。整个流程简单直观,适合初学者动手实践,同时也展示了AI在数据查询中的潜力与局限。
483 8
|
4月前
|
SQL 人工智能 Linux
SQL Server 2025 RC0 发布 - 从本地到云端的 AI 就绪企业数据库
SQL Server 2025 RC0 发布 - 从本地到云端的 AI 就绪企业数据库
277 5
|
6月前
|
SQL 关系型数据库 MySQL
Go语言数据库编程:使用 `database/sql` 与 MySQL/PostgreSQL
Go语言通过`database/sql`标准库提供统一数据库操作接口,支持MySQL、PostgreSQL等多种数据库。本文介绍了驱动安装、连接数据库、基本增删改查操作、预处理语句、事务处理及错误管理等内容,涵盖实际开发中常用的技巧与注意事项,适合快速掌握Go语言数据库编程基础。
489 62
|
3月前
|
关系型数据库 MySQL 数据库
阿里云数据库RDS支持MySQL、SQL Server、PostgreSQL和MariaDB引擎
阿里云数据库RDS支持MySQL、SQL Server、PostgreSQL和MariaDB引擎,提供高性价比、稳定安全的云数据库服务,适用于多种行业与业务场景。

热门文章

最新文章