メインコンテンツへスキップ

問題

ClickHouse Cloud は、従来の RDBMS における意味でのマルチステートメントトランザクションをサポートしていません。 そのため、よくある課題が 2 つあります。
  1. バルクロード時の単一テーブルのアトミック性: 一般的には、まず一時的な部分キーに insert し、その後レコードを実際のキーにコピーして一時レコードを削除します。ただし、この方法はパフォーマンスが低く、特に削除ステップが処理全体の 90% 以上を占めることがあります。
  2. 複数テーブル間の整合性: パイプラインが Table A のロードには成功しても、Table B で失敗した場合、Table A はすでにコミット済みのためロールバックできません。その結果、両方のテーブルをまたいでクエリするアナリストには、同期の取れていないデータが見えてしまいます。

背景

ClickHouse は、単一の insert・単一のパーティションという粒度でアトミック性を保証します。INSERT が成功した場合、そのブロック内のすべての行が可視になります。失敗した場合は、1 行も可視になりません。ただし、複数の insert や複数のテーブルにまたがるデータをアトミックにコミットするための組み込みの仕組みはありません。 パーティション操作コマンド (MOVE PARTITION TO TABLEREPLACE PARTITIONATTACH PARTITION FROM) は、ソーステーブルと宛先テーブルが同じストレージポリシーを共有している場合、メタデータレベルで動作します。 つまり、これらはデータサイズに関係なくほぼ瞬時に実行されるため、アトミックなスワップパターンの理想的な構成要素となります。 障害発生時のクリーンアップを前提に本番テーブルへ直接 insert するのではなく、受け入れ先として専用のステージングテーブルを使用してください。データを検証した後、パーティションレベルの操作を使って、データをアトミックに本番へ移行します。

単一テーブルでアトミック性を実現する手順

1

ステージングテーブルを作成する

本番テーブルと同じスキーマ、パーティションキー、ORDER BY、ストレージポリシーを使用します。
CREATE TABLE my_table_staging AS my_table_prod;
2

ステージングテーブルにデータを挿入する

本番テーブルではなく、ステージングテーブルに対して insert を実行します。
INSERT INTO my_table_staging SELECT ... FROM source;
3

insert が失敗した場合は、TRUNCATE して再試行する

ステージングテーブルを TRUNCATE し、再度ロードを実行します。本番データには影響ありません。
TRUNCATE TABLE my_table_staging;
4

insert が成功した場合は、パーティションを本番環境に移動する

データを本番環境に移動し、ステージングテーブルから取り除くには、MOVE PARTITION を使用します。
ALTER TABLE my_table_staging MOVE PARTITION <partition_expr> TO TABLE my_table_prod;
または、ATTACH PARTITION を使用して、本番環境の既存パーティションにデータをコピーすることもできます。
ALTER TABLE my_table_prod ATTACH PARTITION tuple() FROM my_table_staging;
どちらの操作も同じストレージポリシー上でのメタデータレベルの変更であり、ほぼ瞬時に完了します。
5

ステージングテーブルをクリーンアップする

すべてのパーティションの移動が完了したら、次のロードに備えて空の状態にしておくため、ステージングテーブルを TRUNCATE します。
TRUNCATE TABLE my_table_staging;

複数テーブル間の整合性

同じパターンは、2 つ以上のテーブルをすべて読み込み終えるまで、どのテーブルもアナリストに見える状態にしてはいけないパイプラインにも有効です。各テーブルのデータをそれぞれ専用のステージングテーブルに読み込み、すべてを検証してから、パーティションの移動をまとめて実行します。各移動はほぼ瞬時に完了するメタデータ操作であるため、テーブル間で不整合が生じる期間は、読み込み全体にかかる時間から、パーティションの入れ替えに要する時間まで短縮されます。

要件と制約

MOVE PARTITION TO TABLEREPLACE PARTITIONATTACH PARTITION FROM では、転送元テーブルと転送先テーブルが次の条件を満たしている必要があります。
  • 同一のカラム構造
  • 同一のパーティションキー、ORDER BY キー、および主キー
  • 同一のストレージポリシー
  • 転送先テーブルには、転送元テーブルのすべてのインデックスとプロジェクションが含まれている必要があります

以下の例は、fiddle 上で対話的に実行できます。
CREATE TABLE prod
(
  uid Int16,
  name String,
  age Int16
)
ENGINE=MergeTree
ORDER BY ();

CREATE TABLE staging
(
  uid Int16,
  name String,
  age Int16
)
ENGINE=MergeTree
ORDER BY ();

-- 初期データ
INSERT INTO prod VALUES (123, 'John', 33);
INSERT INTO prod VALUES (456, 'Ksenia', 48);
-- データの読み込み
INSERT INTO staging VALUES (8811, 'Alice', 50);
INSERT INTO staging VALUES (8812, 'Bob', 23);

-- インポートの検証
SELECT 'Staging count:', COUNT() FROM staging;
-- パーティションの移動
ALTER TABLE staging MOVE PARTITION tuple() TO TABLE prod; -- アトミック操作

-- データの確認
SELECT 'Prod count:', COUNT() FROM prod;
SELECT * FROM prod;

参考資料

最終更新日 2026年6月10日