EVENTDATA 関数の使用
DDL トリガーを起動するイベントに関する情報は、EVENTDATA 関数を使用してキャプチャされます。 この関数は、xml
値を返します。 XML スキーマには、次の項目に関する情報が含まれています。
イベントの時刻。
トリガーが実行されたときの接続のシステム プロセス ID (SPID)。
トリガーを起動したイベントの種類。
その後、イベントの種類に応じて、スキーマには、イベントが発生したデータベース、イベントが発生したオブジェクト、イベントの Transact-SQL ステートメントなどの追加情報が含まれます。 詳細については、「 DDL トリガー」を参照してください。
たとえば、次の DDL トリガーが AdventureWorks2012 サンプル データベースに作成されたとします。
CREATE TRIGGER safety
ON DATABASE
FOR CREATE_TABLE
AS
PRINT 'CREATE TABLE Issued.'
SELECT EVENTDATA().value('(/EVENT_INSTANCE/TSQLCommand/CommandText)[1]','nvarchar(max)')
RAISERROR ('New tables cannot be created in this database.', 16, 1)
ROLLBACK
;
次に、以下の CREATE TABLE
ステートメントが実行されます。
CREATE TABLE NewTable (Column1 int);
DDL トリガーの EVENTDATA()
ステートメントにより、 CREATE TABLE
ステートメントでは許可されないテキストがキャプチャされます。 これは、EVENTDATA によって生成されたデータに対して XQuery ステートメントをxml
使用し、CommandText> 要素を取得することによって実現されます<。 詳細については、「XQuery 言語リファレンス (SQL Server)」を参照してください。
注意事項
EVENTDATA は、CREATE_SCHEMAイベントのデータと <、対応する CREATE SCHEMA 定義のschema_element> (存在する場合) をキャプチャします。 さらに、EVENTDATA は、schema_element>定義を<別のイベントとして認識します。 したがって、CREATE_SCHEMA イベントと CREATE SCHEMA 定義のschema_element>で<表されるイベントの両方で作成された DDL トリガーは、データなどの同じイベント データを TSQLCommand
2 回返す場合があります。 たとえば、CREATE_SCHEMA イベントと CREATE_TABLE イベントの両方で DDL トリガーが作成され、次のバッチを実行するとします。
CREATE SCHEMA s
CREATE TABLE t1 (col1 int)
アプリケーションで CREATE_TABLE イベントの TSQLCommand
データを取得する場合は、このデータが 2 回発生する可能性があることに注意してください。つまり、CREATE_SCHEMA イベントの発生時と、CREATE_TABLE イベントの発生時です。 CREATE_SCHEMA イベントと <対応する CREATE SCHEMA 定義のschema_element> テキストの両方に DDL トリガーを作成したり、同じイベントが 2 回処理されないようにロジックをアプリケーションに組み込んだりしないでください。
ALTER TABLE イベントと ALTER DATABASE イベント
ALTER_TABLE および ALTER_DATABASE イベントのイベント データには、DDL ステートメントの影響を受けた他のオブジェクトの名前と種類、およびそれらのオブジェクトで実行されたアクションも含まれます。 ALTER_TABLE イベント データには、ALTER TABLE ステートメントの影響を受けた列、制約、またはトリガーの名前と、それらのオブジェクトで実行されたアクション (作成、変更、削除、有効化、または無効化) が含まれます。 ALTER_DATABASE イベント データには、ALTER DATABASE ステートメントの影響を受けたファイルまたはファイル グループの名前と、それらのオブジェクトで実行されたアクション (作成、変更、または削除) が含まれます。
たとえば、次の DDL トリガーを AdventureWorks サンプル データベースに作成します。
CREATE TRIGGER ColumnChanges
ON DATABASE
FOR ALTER_TABLE
AS
-- Detect whether a column was created/altered/dropped.
SELECT EVENTDATA().value('(/EVENT_INSTANCE/TSQLCommand/CommandText)[1]', 'nvarchar(max)')
RAISERROR ('Table schema cannot be modified in this database.', 16, 1);
ROLLBACK;
次に、制約に違反する次の ALTER TABLE ステートメントを実行します。
ALTER TABLE Person.Address ALTER COLUMN ModifiedDate date;
DDL トリガーの EVENTDATA() ステートメントにより、 ALTER TABLE
ステートメントでは許可されないテキストがキャプチャされます。
例
EVENTDATA 関数を使用して、イベントのログを作成できます。 次の例では、イベント情報を格納するためのテーブルが作成されます。 次に、現在のデータベースに DDL トリガーが作成されます。この DDL トリガーにより、データベース レベルの DDL イベントが発生するたびに、次の情報がテーブルに設定されます。
イベントの時刻 (GETDATE 関数を使用)。
イベントが発生したセッションのデータベース ユーザー (CURRENT_USER 関数を使用)。
イベントの種類。
イベントを構成する Transact-SQL ステートメント。
最後の 2 つの項目は、EVENTDATA によって生成された xml
データに対して XQuery を使用することによってキャプチャされます。
USE AdventureWorks2012;
GO
CREATE TABLE ddl_log (PostTime datetime, DB_User nvarchar(100), Event nvarchar(100), TSQL nvarchar(2000));
GO
CREATE TRIGGER log
ON DATABASE
FOR DDL_DATABASE_LEVEL_EVENTS
AS
DECLARE @data XML
SET @data = EVENTDATA()
INSERT ddl_log
(PostTime, DB_User, Event, TSQL)
VALUES
(GETDATE(),
CONVERT(nvarchar(100), CURRENT_USER),
@data.value('(/EVENT_INSTANCE/EventType)[1]', 'nvarchar(100)'),
@data.value('(/EVENT_INSTANCE/TSQLCommand)[1]', 'nvarchar(2000)') ) ;
GO
--Test the trigger
CREATE TABLE TestTable (a int)
DROP TABLE TestTable ;
GO
SELECT * FROM ddl_log ;
GO
注意
イベント データを返す場合は、value()
メソッドの代わりに XQuery の query()
メソッドを使用してください。 query()
メソッドでは、出力として XML の他に、アンパサンド記号でエスケープされたキャリッジ リターンとライン フィード (CRLF) の組み合わせが返されます。それに対して value()
メソッドの出力には、CRLF の組み合わせが表示されません。
同様の DDL トリガーの例を、 AdventureWorks2012 サンプル データベースで提供しています。 この例を入手するには、 SQL Server Management Studioを使用して Database Triggers フォルダーを探します。 このフォルダーは データベースの [プログラミング] AdventureWorks2012 フォルダーにあります。 ddlDatabaseTriggerLog を右クリックし、 [データベース トリガーをスクリプト化] を選択します。 既定では、DDL トリガー ddlDatabaseTriggerLog は無効になっています。