MS SQL监控数据库的DDL操作

GPS平台、网站建设、软件开发、系统运维,找森大网络科技!
https://cnsendnet.taobao.com
来自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=1778

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

  1. USE msdb;
  2. GO
  3. CREATE TABLE [dbo].[DatabaseLog]
  4. (
  5. [DatabaseLogID] [int] IDENTITY(1,1) NOT NULL,
  6. [PostTime] [datetime] NOT NULL,
  7. [DatabaseUser] [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL,
  8. [LoginName] [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  9. [ClientHost] [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  10. [Event] [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL,
  11. [Schema] [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  12. [Object] [sysname] COLLATE SQL_Latin1_General_CP1_CI_AS NULL,
  13. [TSQL] nvarchar COLLATE SQL_Latin1_General_CP1_CI_AS NOT NULL,
  14. [XmlEvent] [xml] NOT NULL,
  15. CONSTRAINT [PK_DatabaseLog_DatabaseLogID] PRIMARY KEY NONCLUSTERED
  16. (
  17. [DatabaseLogID] ASC
  18. )WITH (PAD_INDEX= OFF, IGNORE_DUP_KEY = OFF) ON [PRIMARY]
  19. ) ON [PRIMARY]
  20. GO
  21. 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’
  22. GO
  23. 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’
  24. GO
  25. 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’
  26. GO
  27. 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’
  28. GO
  29. 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’
  30. GO
  31. 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’
  32. GO
  33. 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’
  34. GO
  35. 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’
  36. GO
  37. 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’
  38. GO
  39. 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’
  40. GO
  41. 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’
  42. GO
  43. 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’
  44. GO
    例如,我要监控数据库MyAssistant的DDL操作,那么我们首先在“数据库邮件”里面创建一个配置名为“ DataBase_DDL_Event”的配置文件(profile name),这个就不多讲了,不知道配置的,自己先练练手把,假如我需要让数据库把监控到DDL操作变动相信信息发送到我的邮箱 **@.com(用你自己的邮箱替代),那么只需要修改下面代码的邮箱和profile_name即可。
    SQL Code 2
  45. USE MyAssistant;
  46. GO
  47. CREATE TRIGGER [DTG_DatabaseDdlTriggerLog]
  48. ON DATABASE
  49. FOR DDL_DATABASE_LEVEL_EVENTS
  50. AS
  51. BEGIN
  52. SET NOCOUNT ON;
  53. DECLARE @data XML;
  54. DECLARE @schema sysname;
  55. DECLARE @object sysname;
  56. DECLARE @eventType sysname;
  57. DECLARE @tableHTML NVARCHAR(MAX) ;
  58. SET @data = EVENTDATA();
  59. SET @eventType = @data.value(’(/EVENT_INSTANCE/EventType)[1]’, ‘sysname’);
  60. SET @schema = @data.value(’(/EVENT_INSTANCE/SchemaName)[1]’, ‘sysname’);
  61. SET @object = @data.value(’(/EVENT_INSTANCE/ObjectName)[1]’, ‘sysname’)
  62. IF @object IS NOT NULL
  63. PRINT ’ ’ + @eventType + ’ - ’ + @schema + ‘.’ + @object;
  64. ELSE
  65. PRINT ’ ’ + @eventType + ’ - ’ + @schema;
  66. IF @eventType IS NULL
  67. PRINT CONVERT(nvarchar(max), @data);
  68. INSERT [msdb].[dbo].[DatabaseLog]
  69. (
  70. [PostTime],
  71. [DatabaseUser],
  72. [LoginName],
  73. [ClientHost],
  74. [Event],
  75. [Schema],
  76. [Object],
  77. [TSQL],
  78. [XmlEvent]
  79. )
  80. VALUES
  81. (
  82. GETDATE(),
  83. CONVERT(sysname, CURRENT_USER),
  84. @data.value(’(/EVENT_INSTANCE/LoginName)[1]’, ‘nvarchar(max)’),
  85. CONVERT(sysname, HOST_NAME()),
  86. @eventType,
  87. CONVERT(sysname, @schema),
  88. CONVERT(sysname, @object),
  89. @data.value(’(/EVENT_INSTANCE/TSQLCommand)[1]’, ‘nvarchar(max)’),
  90. @data
  91. );
  92. SET @tableHTML =
  93. N’

    DDL Event

    ’ +
  94. N’’ +
  95. N’Post TimeUserLoginClientHost’ +
  96. N’TSQL’ +
  97. CAST(( SELECT
  98. td = PostTime, ‘’,
  99. td = DatabaseUser, ‘’,
  100. td = LoginName, ‘’,
  101. td = ClientHost, ‘’,
  102. td = TSQL, ‘’
  103. FROM msdb.dbo.DatabaseLog
  104. WHERE DatabaseLogID =(select max(DatabaseLogID) from msdb.dbo.DatabaseLog)
  105. FOR XML PATH(‘tr’), TYPE ) AS NVARCHAR(MAX) ) + N’’ ;
  106. EXEC msdb.dbo.sp_send_dbmail
  107. @profile_name = ‘DataBase_DDL_Event’,
  108. @recipients=’***@***.com’,
  109. @subject = ‘DDL Event - DataBase MyAssistant’,
  110. @body = @tableHTML,
  111. @body_format = ‘HTML’ ;
  112. END;
  113. GO
    接下来我们来测试一下,假如一个用户Test登录数据库,一不小心删除了一个Test的表,如下图一所示,那么我将收到一封邮件,提示我用户Test在那台客户端主机执行了啥DDL操作(如下图二所示),当然邮件的样式、排版有兴趣的可以去美化一下。
    在这里插入图片描述
    在这里插入图片描述

GPS平台、网站建设、软件开发、系统运维,找森大网络科技!
https://cnsendnet.taobao.com
来自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=1778

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值