データの変更の準備ができているかどうかを判断する
適用対象: SQL Server Azure Data Factory の SSIS 統合ランタイム
変更データの増分読み込みを実行する Integration Services パッケージの制御フローにおいて、2 番目のタスクは、選択した間隔の変更データが準備できていることを確認することです。 選択したエンドポイントまでの変更が非同期キャプチャ プロセスでまだ一部処理されていない可能性があるため、この手順が必要となります。
注意
制御フローの最初のタスクは、変更間隔のエンドポイントを計算することです。 このタスクに関する詳細については、「 変更データの間隔を指定する」を参照してください。 制御フローをデザインするプロセス全体の説明については、「変更データ キャプチャ (SSIS)」を参照してください。
ソリューションのコンポーネントについて
このトピックで説明するソリューションでは、次の 4 つの Integration Services コンポーネントを使用します。
SQL 実行タスクの出力を繰り返し評価する For ループ コンテナー。
変更データ キャプチャ プロセスによって保持されている特殊なテーブルに対してクエリを実行し、この情報を使用してデータが準備できているかどうかを判断する SQL 実行タスク。
データが準備できていない場合に処理の遅延を実装するコンポーネント。 このコンポーネントは、スクリプト タスクまたは SQL 実行タスクのいずれかになります。
(省略可) SQL 実行タスクによってエラーまたはタイムアウト状態を示す値が返されたときにエラーまたはタイムアウトを報告するコンポーネント。
これらのコンポーネントによって、いくつかのパッケージ変数の値が設定されるか読み取られ、ループ内およびパッケージの後続の実行フローが制御されます。
パッケージ変数を設定するには
SQL Server Data Tools (SSDT)の [変数] ウィンドウで、次の変数を作成します。
SQL 実行タスクによって返される状態値を格納する整数データ型の変数を作成します。
この例では、初期値が 0 である DataReady という名前の変数を使用します。
データが準備できていない場合の遅延時間を格納する変数を作成します。 スクリプト タスクを使用して遅延を実装する場合、変数には整数データ型が格納されます。 WAITFOR ステートメントを実行する SQL 実行タスクを使用する場合、変数には文字列データ型が格納され、"00:00:10" などの値を設定できます。
この例では、初期値が 10 である DelaySeconds という名前の変数を使用します。
ループの現在の反復を格納する整数データ型の変数を作成します。
この例では、初期値が 0 である TimeoutCount という名前の変数を使用します。
タイムアウト状態を報告する前にループでデータをテストする回数を指定する整数データ型の変数を作成します。
この例では、初期値が 20 である TimeoutCeiling という名前の変数を使用します。
(省略可) 変更データの最初の読み込みを示すために使用できる整数データ型の変数を作成します。
この例では、IntervalID という名前の変数を使用し、値が最初の読み込みを示す 0 であることだけをチェックします。
For ループ コンテナーの構成
変数を設定したら、まず For ループ コンテナーを追加します。
変更データが準備できるまで待機するように For ループ コンテナーを構成するには
デザイナーの [制御フロー] SSIS タブで、For ループ コンテナーを制御フローに追加します。
間隔のエンドポイントを計算する SQL 実行タスクを For ループ コンテナーに連結します。
[For ループ エディター]で、次のオプションを選択します。
[InitExpression]に「
@DataReady = 0
」と入力します。この式によって、ループ変数の初期値が設定されます。
[EvalExpression]に「
@DataReady == 0
」と入力します。この式が Falseと評価されると、実行がループの外に移り、増分読み込みが開始されます。
変更データのクエリを実行する SQL 実行タスクの構成
For ループ コンテナー内に SQL 実行タスクを追加します。 このタスクでは、変更データ キャプチャ プロセスがデータベースに保持しているテーブルに対してクエリを実行します。 このクエリの結果は、変更データが準備できているかどうかを示す状態値です。
次の表の 1 列目は、サンプルの Transact-SQL クエリによって SQL 実行タスクから返される値を示しています。 2 列目は、その他のコンポーネントがこれらの値に応答する方法を示しています。
戻り値 | 説明 | Response |
---|---|---|
0 | 変更データが準備できていないことを示します。 選択した間隔の終了時点より後に変更データ キャプチャ レコードがありません。 |
遅延を実装するコンポーネントから実行が継続されます。 その後、制御が For ループ コンテナーに戻り、返される値が 0 である限り引き続きコンテナーによって SQL 実行タスクがチェックされます。 |
1 | 間隔全体にわたって変更データがキャプチャされていないか、変更データが削除されていることを示します。 これは、エラー状態として扱われます。 選択した間隔の開始時点より前に変更データ キャプチャ レコードがありません。 |
エラーをログに記録するオプションのコンポーネントから実行が継続されます。 |
2 | データが準備できていることを示します。 選択した間隔の開始時点より前にも終了時点より後にも変更データ キャプチャ レコードがあります。 |
実行が For ループ コンテナーの外に移り、増分読み込みが開始されます。 |
3 | 使用可能なすべての変更データの最初の読み込みを示します。 この値は、このためだけに使用される特殊なパッケージ変数から条件ロジックによって取得されます。 |
実行が For ループ コンテナーの外に移り、増分読み込みが開始されます。 |
5 | TimeoutCeiling に達したことを示します。 指定された回数だけループでデータがテストされましたが、データはまだ使用できません。 このテストまたは同様のテストを実行しないと、パッケージが無期限に実行される可能性があります。 |
タイムアウトをログに記録するオプションのコンポーネントから実行が継続されます。 |
変更データが準備できているかどうかをクエリするように SQL 実行タスクを構成するには
For ループ コンテナー内に SQL 実行タスクを追加します。
[SQL 実行タスク エディター]の [全般] ページで、次のオプションを選択します。
[ResultSet]で [単一行]を選択します。
ソース データベースへの有効な接続を構成します。
[SQLSourceType]で [直接入力]を選択します。
[SQLStatement]に、次の SQL ステートメントを入力します。
declare @DataReady int, @TimeoutCount int if not exists (select tran_end_time from cdc.lsn_time_mapping where tran_end_time > ? ) select @DataReady = 0 else if ? = 0 select @DataReady = 3 else if not exists (select tran_end_time from cdc.lsn_time_mapping where tran_end_time <= ? ) select @DataReady = 1 else select @DataReady = 2 select @TimeoutCount = ? if (@DataReady = 0) select @TimeoutCount = @TimeoutCount + 1 else select @TimeoutCount = 0 if (@TimeoutCount > ?) select @DataReady = 5 select @DataReady as DataReady, @TimeoutCount as TimeoutCount
[SQL 実行タスク エディター] の [パラメーター マッピング]ページで、次のマッピングを行います。
ExtractEndTime 変数をパラメーター 0 にマップします。
IntervalID 変数をパラメーター 1 にマップします。
ExtractStartTime 変数をパラメーター 2 にマップします。
TimeoutCount 変数をパラメーター 3 にマップします。
TimeoutCeiling 変数をパラメーター 4 にマップします。
[SQL 実行タスク エディター] の [結果セット]ページで、DataReady の結果を DataReady 変数に、TimeoutCount の結果を TimeoutCount 変数にマップします。
変更データが準備できるまでの待機
いくつかある方法のうちいずれかを使用して、変更データが準備できていない場合に遅延を実装することができます。 次の 2 つの手順は、スクリプト タスクまたは SQL 実行タスクを使用して遅延を実装する方法を示しています。
注意
発生するオーバーヘッドは、SQL 実行タスクよりも事前コンパイルしたスクリプトの方が少なくなります。
スクリプト タスクを使用して遅延を実装するには
For ループ コンテナー内にスクリプト タスクを追加します。
変更データが準備できているかどうかを判断するためにクエリを実行する SQL 実行タスクを新しいスクリプト タスクに連結します。
SQL 実行タスクをスクリプト タスクに連結する優先順位制約のために、 [優先順位制約エディター] を開いて次のオプションを選択します。
[評価操作]で [式と制約]を選択します。
[値]で [成功]を選択します。
制約値 [成功] は、前のタスクの成功を表します。 この場合は、SQL 実行タスクの成功を表します。
[式]に「
@DataReady == 0 && @TimeoutCount <= @TimeoutCeiling
」と入力します。[論理 AND (すべての制約が True と評価される必要があります)] が選択されていない場合は、選択します。
[スクリプト タスク エディター]の [スクリプト] ページの [ReadOnlyVariables]で、 [User::DelaySeconds] 整数変数を一覧から選択します。
[スクリプト タスク エディター]の [スクリプト] ページで、 [スクリプトの編集] をクリックしてスクリプト開発環境を開きます。
Main プロシージャに、次のいずれかのコード行を入力します。
C# でプログラミングしている場合は、次のコード行を入力します。
System.Threading.Thread.Sleep((int)Dts.Variables["DelaySeconds"].Value * 1000);
- または -
Visual Basicでプログラミングしている場合は、次のコード行を入力します。
System.Threading.Thread.Sleep(Ctype(Dts.Variables("DelaySeconds").Value, Integer) * 1000)
注意
Thread.Sleep メソッドは、ミリ秒単位で指定される引数を想定しています。
スクリプトの実行から DtsExecResult.Success を返す既定のコード行はそのまま使用します。
スクリプト開発環境と [スクリプト タスク エディター]を閉じます。
SQL 実行タスクを使用して遅延を実装するには
For ループ コンテナー内に SQL 実行タスクを追加します。
変更データが準備できているかどうかを判断するためにクエリを実行する SQL 実行タスクを新しい SQL 実行タスクに連結します。
2 つの SQL 実行タスクを連結する優先順位制約のために、 [優先順位制約エディター] を開いて次のオプションを選択します。
[評価操作]で [式と制約]を選択します。
[値]で [成功]を選択します。
制約値 [成功] は、前の SQL 実行タスクの成功を表します。
[式]に「
@DataReady == 0
」と入力します。[論理 AND (すべての制約が True と評価される必要があります)] が選択されていない場合は、選択します。
この選択により、制約と式の両方の条件が True であることが必要になります。
[SQL 実行タスク エディター]の [全般] ページで、次のオプションを選択します。
[ResultSet]で [単一行]を選択します。
ソース データベースへの有効な接続を構成します。
[SQLSourceType]で [直接入力]を選択します。
[SQLStatement]に、次の SQL ステートメントを入力します。
WAITFOR DELAY ?
エディターの [パラメーター マッピング] ページで、DelaySeconds 文字列変数をパラメーター 0 にマップします。
エラー状態の処理
必要に応じて、エラーまたはタイムアウト状態をログに記録するようにループ内に追加のコンポーネントを構成することができます。
このコンポーネントでは、DataReady 変数の値が 1 の場合にエラー状態をログに記録できます。 この値は、選択した間隔の開始前に使用可能な変更データがないことを示します。
このコンポーネントでは、TimeoutCeiling 変数の値に達した場合にタイムアウト状態をログに記録することもできます。 この値は、指定された回数だけループでデータがテストされたが、データがまだ使用できないことを示します。 このテストまたは同様のテストを実行しないと、パッケージが無期限に実行される可能性があります。
エラー状態をログに記録するようにオプションのスクリプト タスクを構成するには
メッセージをログに書き込んでエラーまたはタイムアウトを報告する場合は、パッケージのログ記録を構成します。 詳細については、「 SQL Server Data Tools でパッケージのログ記録を有効にする」を参照してください。
For ループ コンテナー内にスクリプト タスクを追加します。
変更データが準備できているかどうかを判断するためにクエリを実行する SQL 実行タスクを新しいスクリプト タスクに連結します。
SQL 実行タスクをスクリプト タスクに連結する優先順位制約のために、 [優先順位制約エディター] を開いて次のオプションを選択します。
[評価操作]で [式と制約]を選択します。
[値]で [成功]を選択します。
制約値 [成功] は、前のタスクの成功を表します。 この場合は、SQL 実行タスクの成功を表します。
[式]に「
@DataReady == 1 || @DataReady == 5
」と入力します。[論理 AND (すべての制約が True と評価される必要があります)] が選択されていない場合は、選択します。
この選択により、制約と式の両方の条件が True であることが必要になります。
[スクリプト タスク エディター]の [スクリプト] ページの [ReadOnlyVariables]で、 [User::DataReady] および [User::ExtractStartTime] を一覧から選択し、それらの値をスクリプトに使用できるようにします。
特定のシステム変数 (System::PackageName など) の情報をログに書き込む情報に含める場合は、それらの変数も選択します。
[スクリプト タスク エディター]の [スクリプト] ページで、 [スクリプトの編集] をクリックしてスクリプト開発環境を開きます。
Main プロシージャに、 Dts.Log メソッドを呼び出してエラーをログに記録するコードか、 Dts.Events インターフェイスのいずれかのメソッドを呼び出してイベントを発生させるコードを入力します。
Dts.TaskResult = Dts.Results.Failure
を返すことによってエラーをパッケージに通知します。次の例は、メッセージをログに書き込む方法を示しています。 詳細については、「 スクリプト タスクでのログ記録」、「 スクリプト タスクでのイベントの発生」、「 スクリプト タスクから結果を返す」を参照してください。
' User variables. Dim dataReady As Integer = _ CType(Dts.Variables("DataReady").Value, Integer) Dim extractStartTime As Date = _ CType(Dts.Variables("ExtractStartTime").Value, DateTime) ' System variables. Dim packageName As String = _ Dts.Variables("PackageName").Value.ToString() Dim executionStartTime As Date = _ CType(Dts.Variables("StartTime").Value, DateTime) Dim eventMessage As New System.Text.StringBuilder() If dataReady = 1 OrElse dataReady = 5 Then If dataReady = 1 Then eventMessage.AppendLine("Start Time Error") Else eventMessage.AppendLine("Timeout Error") End If With eventMessage .Append("The package ") .Append(packageName) .Append(" started at ") .Append(executionStartTime.ToString()) .Append(" and ended at ") .AppendLine(DateTime.Now().ToString()) If dataReady = 1 Then .Append("The specified ExtractStartTime was ") .AppendLine(extractStartTime.ToString()) End If End With System.Windows.Forms.MessageBox.Show(eventMessage.ToString()) Dts.Log(eventMessage.ToString(), 0, Nothing) Dts.TaskResult = Dts.Results.Failure Else Dts.TaskResult = Dts.Results.Success End If
スクリプト開発環境と [スクリプト タスク エディター]を閉じます。
次の手順
変更データが準備できていると判断したら、次に変更データのクエリを準備します。
次のトピック: 変更データのクエリを準備する