Azure SQL データ同期でスキーマ変更のレプリケートを自動化する
適用対象: Azure SQL データベース
重要
SQL データ同期 は、2027 年 9 月 30 日に廃止される予定です。 代替のデータ レプリケーション/同期ソリューションへの移行を検討してください。
SQL データ同期は、Azure SQL データベースと SQL Server インスタンスにあるデータベース間でデータを一方向または双方向に同期できるようになります。 SQL データ同期における現在の制限事項の 1 つは、スキーマ変更のレプリケートのサポートが十分ではないことです。 テーブル スキーマを変更するたびに、ハブやすべてのメンバーを含め、すべてのエンドポイントに変更を手動で適用してから、同期スキーマを更新する必要があります。
この記事では、すべての SQL データ同期エンドポイントにスキーマの変更を自動的にレプリケートするソリューションを紹介します。
- このソリューションでは DDL トリガーを使用して、スキーマの変更を追跡します。
- トリガーによって、スキーマ変更コマンドが追跡テーブルに挿入されます。
- この追跡テーブルが、データ同期サービスを使って、すべてのエンドポイントに同期されます。
- 挿入後の DML トリガーは、他のエンドポイントにスキーマの変更を適用するために使用されます。
この記事では、スキーマ変更の例として ALTER TABLE
を使用していますが、このソリューションは他の種類のスキーマ変更にも使用できます。
お使いの同期環境に自動化したスキーマ変更のレプリケートを実装し始める前に、特に「トラブルシューティング」と「その他の考慮事項」に関するセクションを中心に、この記事をよくお読みください。 一部のデータベース操作は、この記事で説明されているソリューションを損ねる可能性があります。 これらの問題をトラブルシューティングするには、SQL Server と Transact-SQL の詳細なドメインの知識が必要になる場合があります。
自動化されたスキーマ変更のレプリケートを設定する
スキーマ変更を追跡するテーブルを作成する
同期グループ内のすべてのデータベースのスキーマ変更を追跡するためのテーブルを作成します。
CREATE TABLE SchemaChanges (
ID bigint IDENTITY(1,1) PRIMARY KEY,
SqlStmt nvarchar(max),
[Description] nvarchar(max)
)
このテーブルには、スキーマ変更の順序を追跡するための ID 列が含まれます。 必要に応じて、フィールドをさらに追加して、より多くの情報を記録できます。
スキーマ変更の履歴を追跡するテーブルを作成する
すべてのエンドポイント上で、直近で適用されたスキーマ変更コマンドの ID を追跡するテーブルを作成します。
CREATE TABLE SchemaChangeHistory (
LastAppliedId bigint PRIMARY KEY
)
GO
INSERT INTO SchemaChangeHistory VALUES (0)
スキーマ変更が行われたデータベースで ALTER TABLE DDL トリガーを作成する
ALTER TABLE 操作のための DDL トリガーを作成します。 スキーマ変更が行われたデータベースに、このトリガーを作成するだけでかまいません。 競合を避けるために、同期グループごとに 1 つのデータベースでしか、スキーマ変更が許可されていません。
CREATE TRIGGER AlterTableDDLTrigger
ON DATABASE
FOR ALTER_TABLE
AS
-- You can add your own logic to filter ALTER TABLE commands instead of replicating all of them.
IF NOT (EVENTDATA().value('(/EVENT_INSTANCE/SchemaName)[1]', 'nvarchar(512)') like 'DataSync')
INSERT INTO SchemaChanges (SqlStmt, Description)
VALUES (EVENTDATA().value('(/EVENT_INSTANCE/TSQLCommand/CommandText)[1]', 'nvarchar(max)'), 'From DDL trigger')
トリガーでは、ALTER TABLE コマンドごとに、スキーマ変更追跡テーブルにレコードを挿入します。 スキーマ DataSync 下で行われたスキーマ変更は、データ同期サービスによって行われたことが十分予想できるため、この例では、これらの変更のレプリケートを防止するためにフィルターを追加しています。 特定の種類のスキーマ変更だけをレプリケートしたい場合は、さらにフィルターを追加します。
また、他の種類のスキーマ変更をレプリケートするために、さらにトリガーを追加することも可能です。 たとえば、CREATE_PROCEDURE、ALTER_PROCEDURE、および DROP_PROCEDURE トリガーを作成して、ストアド プロシージャに変更をレプリケートします。
他のエンドポイント上にトリガーを作成して、挿入中にスキーマ変更を適用する
このトリガーは、他のエンドポイントに同期されるときに、スキーマ変更のコマンドを実行します。 スキーマ変更が行われた (つまり、前の手順で DDL トリガー AlterTableDDLTrigger
が作成されたデータベース内の) エンドポイントを除くすべてのエンドポイント上に、このトリガーを作成する必要があります。
CREATE TRIGGER SchemaChangesTrigger
ON SchemaChanges
AFTER INSERT
AS
DECLARE @lastAppliedId bigint
DECLARE @id bigint
DECLARE @sqlStmt nvarchar(max)
SELECT TOP 1 @lastAppliedId=LastAppliedId FROM SchemaChangeHistory
SELECT TOP 1 @id = id, @SqlStmt = SqlStmt FROM SchemaChanges WHERE id > @lastAppliedId ORDER BY id
IF (@id = @lastAppliedId + 1)
BEGIN
EXEC sp_executesql @SqlStmt
UPDATE SchemaChangeHistory SET LastAppliedId = @id
WHILE (1 = 1)
BEGIN
SET @id = @id + 1
IF exists (SELECT id FROM SchemaChanges WHERE ID = @id)
BEGIN
SELECT @sqlStmt = SqlStmt FROM SchemaChanges WHERE ID = @id
EXEC sp_executesql @SqlStmt
UPDATE SchemaChangeHistory SET LastAppliedId = @id
END
ELSE
BREAK;
END
END
このトリガーは、挿入後に実行され、現在のコマンドを次に実行する必要があるかどうかをチェックします。 コード ロジックでは、スキーマ変更のどのステートメントもスキップされていないこと、また、挿入が順不同な場合にもすべての変更が適用されることを保証しています。
スキーマ変更追跡テーブルをすべてのエンドポイントに同期する
既存の同期グループまたは新しい同期グループを使用して、すべてのエンドポイントにスキーマ変更追跡テーブルを同期できます。 特に一方向の同期を使用している場合、追跡テーブルの変更が必ずすべてのエンドポイントに同期されるようにします。
スキーマ変更履歴テーブルは、さまざまなエンドポイント上で異なるステータスを保持するため、このテーブルを同期しないでください。
同期グループでスキーマ変更を適用する
DDL トリガーが作成されたデータベースで行われたスキーマ変更のみが、レプリケートされます。 他のデータベースで行われたスキーマ変更は、レプリケートされません。
また、スキーマ変更がすべてのエンドポイントにレプリケートされた後は、同期スキーマを更新する追加の手順を実行して、新しい列の同期を開始または停止する必要があります。
新しい列を追加する
スキーマへの変更を行います。
トリガーを作成する手順が完了するまで、新しい列が関連するすべてのデータ変更を防止します。
スキーマの変更がすべてのエンドポイントに適用されるまで待機します。
データベース スキーマを更新して、新しい列を同期スキーマに追加します。
次の同期操作の中で、新しい列のデータが同期されます。
列の削除
同期スキーマから列を削除します。 データ同期は、これらの列のデータの同期を停止します。
スキーマへの変更を行います。
データベース スキーマを更新します。
データ型の更新
スキーマへの変更を行います。
スキーマの変更がすべてのエンドポイントに適用されるまで待機します。
データベース スキーマを更新します。
新規と従来のデータ型に完全な互換性がない場合 (たとえば、
int
をbigint
に変更した場合)、トリガーを作成する手順が完了する前に同期が失敗する可能性があります。 同期は、再試行後に成功します。
列またはテーブル名の変更
列またはテーブルの名前を変更すると、データ同期は動作を停止します。 名前を変更する代わりに、新しいテーブルまたは列を作成し、データのバックフィルを行ってから、古いテーブルまたは列を削除します。
その他の種類のスキーマ変更
ストアド プロシージャの作成やインデックスの削除など、その他の種類のスキーマ変更では、同期スキーマの更新は必要ありません。
スキーマ変更のレプリケートの自動化に関するトラブルシューティング
この記事で説明したレプリケート ロジックは、Azure SQL Database でサポートされていないオンプレミス データベースでスキーマ変更を行った場合など、一定の状況下で動作を停止します。 停止した場合、スキーマ変更追跡テーブルの同期はエラーになります。 この問題は、次の手順を実行して、手動で修正する必要があります。
DDL トリガーを無効にして、問題が解決されるまで、スキーマ変更がそれ以上行われないようにします。
問題が発生しているエンドポイント データベースで、スキーマ変更を実行できないエンドポイントでの AFTER INSERT トリガーを無効にします。 このアクションによって、スキーマ変更コマンドを同期できるようになります。
同期をトリガーして、スキーマ変更追跡テーブルを同期します。
問題が発生しているエンドポイント データベースで、スキーマ変更履歴テーブルにクエリを実行し、直近で適用されたスキーマ変更コマンドの ID を取得します。
スキーマ変更追跡テーブルにクエリを実行して、前の手順で取得した ID 値よりも大きい ID 値のすべてのコマンドを一覧表示します。
a. エンドポイント データベースで実行できないコマンドを無視します。 スキーマの不一致に対応する必要があります。 不一致がアプリケーションに影響を及ぼす場合は、元のスキーマ変更を元に戻します。
b. 適用する必要があるコマンドを手動で適用します。
スキーマ変更履歴テーブルを更新して、直近で適用された ID に正しい値を設定します。
スキーマが最新かどうかを再確認します。
手順 2 で無効にした AFTER INSERT トリガーをもう一度有効にします。
手順 1 で無効にした DDL トリガーをもう一度有効にします。
スキーマ変更追跡テーブル内のレコードをクリーンアップする場合は、TRUNCATE ではなく DELETE を使用します。 絶対に DBCC CHECKIDENT を使用してスキーマ変更追跡テーブル内の ID 列を再シードしてはいけません。 再シードが必要な場合は、新しいスキーマ変更追跡テーブルを作成して、DDL トリガーの中でテーブル名を更新できます。
その他の注意事項
ハブおよびメンバー データベースを構成するデータベース ユーザーは、スキーマ変更コマンドを実行するための十分な権限を保持している必要があります。
選択したテーブルまたは操作の中でスキーマの変更をレプリケートするためだけに、DDL トリガー内でさらにフィルターを追加できます。
DDL トリガーが作成されたデータベースでスキーマ変更を行うことしかできません。
SQL Server データベースで変更を行っている場合、Azure SQL Database でスキーマの変更がサポートされていることを確認してください。
DDL トリガーが作成されたデータベース以外のデータベースでスキーマ変更が行われた場合、変更はレプリケートされません。 この問題を回避するために、他のエンドポイント上での変更をブロックする DLL トリガーを作成できます。
スキーマ変更追跡テーブルのスキーマを変更する必要がある場合は、変更を加える前に DDL トリガーを無効にしてから、手動ですべてのエンドポイントに変更を適用します。 このテーブルに AFTER INSERT トリガーでスキーマを更新しても、機能しません。
DBCC CHECKIDENT を使用して ID 列を再シードしないでください。
TRUNCATE を使用して、スキーマ変更追跡テーブルのデータをクリーンアップしないでください。
関連するコンテンツ
- Azure の SQL データ同期とは
- チュートリアル:Azure SQL Database と SQL Server のデータベース間の SQL データ同期を設定する
- PowerShell を使用して Azure SQL Database の複数データベース間でデータを同期する
- PowerShell を使用して SQL Database と SQL Server の間でデータの同期
- SQL データ同期の Data Sync Agent
- Azure SQL データ同期のベスト プラクティス
- Azure SQL データベースと Azure SQL Managed Instance での監視とパフォーマンス チューニング
- SQL データ同期に関する問題のトラブルシューティング
- PowerShell を使用して、既存の同期グループの同期スキーマを更新する