Colocation Join
Colocation Joinは、一部のJoinクエリに対してローカル最適化を提供し、ノード間のデータ転送時間を削減してクエリ実行を加速します。
注意: このプロパティはCCRによって同期されません。このTableがCCRによってコピーされる場合、つまりPROPERTIESにis_being_synced = trueが含まれている場合、このプロパティはこのTableで消去されます。
用語の解釈
- FE: Frontend、Dorisのフロントエンドノード。メタデータ管理とリクエストアクセスを担当します。
- BE: Backend、Dorisのバックエンドノード。クエリ実行とデータストレージを担当します。
- Colocation Group (CG): CGは1つ以上のTableを含みます。同一グループ内のTableは同じColocation Group Schemaと同じデータ断片分散を持ちます。
- Colocation Group Schema (CGS): CG内のTableとColocationに関連する一般的なスキーマ情報を記述するために使用されます。バケット列タイプ、バケット数、コピー数を含みます。
原理
Colocation Join機能は、同じCGSを持つTableのセットのCGを作成することです。これらのTableの対応するデータ断片が同じBEノード上に配置されることを保証します。CG内のTableがバケット列でJoin操作を実行する場合、ローカルデータJoinを直接実行してノード間のデータ転送時間を削減できます。
Tableのデータは、バケット列値のHashとバケット数のモデリングに従って、最終的にバケットに配置されます。Tableのバケット数が8であると仮定すると、8つのバケット[0, 1, 2, 3, 4, 5, 6, 7]があります。このようなシーケンスをBuckets Sequenceと呼びます。各Bucketには1つ以上のTabletがあります。Tableが単一パーティションTableの場合、Bucket内にはTabletが1つだけあります。マルチパーティションTableの場合は、複数存在します。
Tableが同じデータ分散を持つために、同じCG内のTableは以下の属性が同じであることを保証する必要があります:
- バケット列とバケット数
バケット列とは、Table作成文のDISTRIBUTED BY HASH (col1, col2,...)で指定された列です。バケット列は、Tableからのデータを異なるTabletにHashするために使用される列値を決定します。同じCG内のTableは、バケット列のタイプと数が同一であり、バケット数が同一であることを保証する必要があります。これにより、複数のTableのデータ断片を一対一で制御できます。
- コピー数
同じCG内のすべてのTableのすべてのパーティションのコピー数は同じでなければなりません。一致しない場合、Tabletのコピーがある一方で、同じBE上に他のTable断片の対応するコピーが存在しない可能性があります。
同じCG内のTableは、パーティション列の数、範囲、タイプにおいて一致している必要はありません。
バケット列とバケット数を固定した後、同じCG内のTableは同じBuckets Sequenceを持ちます。レプリカ数は各バケット内のTabletのレプリカ数と、それらがどのBEに保存されるかを決定します。Buckets Sequenceが[0, 1, 2, 3, 4, 5, 6, 7]で、BEノードが[A, B, C, D]の4つであると仮定します。データの分散の可能性は以下の通りです:
+---+ +---+ +---+ +---+ +---+ +---+ +---+ +---+
| 0 | | 1 | | 2 | | 3 | | 4 | | 5 | | 6 | | 7 |
+---+ +---+ +---+ +---+ +---+ +---+ +---+ +---+
| A | | B | | C | | D | | A | | B | | C | | D |
| | | | | | | | | | | | | | | |
| B | | C | | D | | A | | B | | C | | D | | A |
| | | | | | | | | | | | | | | |
| C | | D | | A | | B | | C | | D | | A | | B |
+---+ +---+ +---+ +---+ +---+ +---+ +---+ +---+
CGの全Tableのデータは上記のルールに従って均一に分散され、同じbarrel column値を持つデータが同一のBEノード上に配置されることを保証し、ローカルデータJoinを実行できます。
Usage
Tableの作成
Table作成時に、PROPERTIESで"colocate_with"="group_name"属性を指定できます。これは、そのTableがColocation JoinTableであり、指定されたColocation Groupに属することを意味します。
例:
CREATE TABLE tbl (k1 int, v1 int sum)
DISTRIBUTED BY HASH(k1)
BUCKETS 8
PROPERTIES(
"colocate_with" = "group1"
);
指定されたグループが存在しない場合、Dorisは現在のTableのみを含むグループを自動的に作成します。Groupが既に存在する場合、Dorisは現在のTableがColocation Group Schemaを満たすかどうかをチェックします。満たしている場合、Tableが作成されGroupに追加されます。同時に、TableはGroups内の既存のデータ分散ルールに基づいてフラグメントとレプリカを作成します。
Groupはデータベースに属し、その名前はデータベース内で一意です。内部ストレージはGroupの完全名dbId_groupNameですが、ユーザーはgroupNameのみを認識します。
バージョン2.0では、DorisはクロスDatabaseGroupをサポートします。Tableを作成する際は、Group名のプレフィックスとしてキーワード__global__を使用する必要があります。例:
CREATE TABLE tbl (k1 int, v1 int sum)
DISTRIBUTED BY HASH(k1)
BUCKETS 8
PROPERTIES(
"colocate_with" = "__global__group1"
);
__global__接頭辞が付いたGroupはもはやDatabaseに属さず、その名前もグローバルに一意となります。
Global Groupを作成することで、Cross-Database Colocate Joinを実現できます。
Tableの削除
Group内の最後のTableが完全に削除された場合(完全な削除とは、ごみ箱からの削除を意味します)。通常、TableがDROP TABLEコマンドで削除された場合、デフォルトの1日間ごみ箱に留まった後に削除され、グループは自動的に削除されます。
Groupの表示
以下のコマンドにより、クラスター内の既存のGroup情報を表示できます。
SHOW PROC '/colocation_group';
+-------------+--------------+--------------+------------+----------------+----------+----------+
| GroupId | GroupName | TableIds | BucketsNum | ReplicationNum | DistCols | IsStable |
+-------------+--------------+--------------+------------+----------------+----------+----------+
| 10005.10008 | 10005_group1 | 10007, 10040 | 10 | 3 | int(11) | true |
+-------------+--------------+--------------+------------+----------------+----------+----------+
- GroupId: グループのクラスター全体の一意の識別子で、前半にDB ID、後半にgroup IDが含まれます。
- GroupName: Groupのフルネームです。
- TabletIds: グループに含まれるTablesのIDのリストです。
- Buckets Num: バケットの数です。
- Replication Num: コピーの数です。
- DistCols: 分散カラムです。
- IsStable: グループが安定しているかどうかです(安定性の定義については、セクション
Collocation replica balancing and repairを参照してください)。
以下のコマンドでグループのデータ分散をさらに詳しく確認できます:
SHOW PROC '/colocation_group/10005.10008';
+-------------+---------------------+
| BucketIndex | BackendIds |
+-------------+---------------------+
| 0 | 10004, 10002, 10001 |
| 1 | 10003, 10002, 10004 |
| 2 | 10002, 10004, 10001 |
| 3 | 10003, 10002, 10004 |
| 4 | 10002, 10004, 10003 |
| 5 | 10003, 10002, 10001 |
| 6 | 10003, 10004, 10001 |
| 7 | 10003, 10004, 10002 |
+-------------+---------------------+
- BucketIndex: バケットシーケンスに対するサブスクリプト。
- Backend Ids: データフラグメントがバケット内に配置されているBEノードIDのリスト。
上記のコマンドはADMIN権限が必要です。通常ユーザーのビューは現在サポートされていません。
Colocate Groupの修正
作成済みのTableのColocation Groupプロパティを修正できます。例:
ALTER TABLE tbl SET ("colocate_with" = "group2");
- Tableが以前にGroupを指定していない場合、コマンドはSchemaをチェックし、TableをGroupに追加します(Groupが存在しない場合は作成されます)。
- Tableに他のグループが以前に指定されている場合、コマンドは最初にTableを元のグループから削除し、新しいグループを追加します(グループが存在しない場合は作成されます)。
以下のコマンドでTableのColocation属性を削除することもできます:
ALTER TABLE tbl SET ("colocate_with" = "");
その他の関連操作
Colocation属性を持つTableにADD PARTITIONが追加され、コピー数が修正された場合、Dorisは修正がColocation Group Schemaに違反するかどうかをチェックし、違反する場合はそれを拒否します。
Colocation Duplicate BalancingとRepair
ColocationTableのコピーディストリビューションはGroupで指定されたディストリビューションに従う必要があるため、レプリカの修復とバランシングにおいて一般的なフラグメンテーションとは異なります。
Group自体にはStable属性があり、Stableがtrueの場合、現在のGroup内のTableのすべてのフラグメントが変更されておらず、Colocation機能が正常に使用できることを示します。Stableがfalseの場合、Group内の一部のTableが修復または移行中であることを示します。この時、関連TableのColocation Joinは通常のJoinに退化します。
Replica Repair
コピーは指定されたBEノードにのみ保存できます。そのため、BE が利用できない(ダウンタイム、Decommissionなど)場合、それを置き換える新しいBEが必要です。Dorisは最初に最も負荷の低いBEを探してそれを置き換えます。置き換え後、Bucket内の古いBE上のすべてのデータフラグメントが修復されます。移行プロセス中、Groupは不安定とマークされます。
Replica Balancing
DorisはCollocationTableのフラグメントをすべてのBEノードに均等に分散しようとします。一般的なTableのレプリカバランシングでは、粒度は単一レプリカであり、つまり、各レプリカ単独で低負荷のBEノードを見つけるだけで十分です。ColocationTableの均衡はBucketレベルで、Bucket内のすべてのレプリカが一緒に移行されます。私たちは簡単な均等化アルゴリズムを採用しており、レプリカの実際のサイズに関係なく、レプリカの数のみに従って、すべてのBE上にBuckets Sequenceを均等に分散します。具体的なアルゴリズムについては、ColocateTableBalancer.javaのコード注釈を参照してください。
注1:現在のColocationレプリカバランシングおよび修復アルゴリズムは、異種デプロイされたDorisクラスターではうまく動作しない可能性があります。いわゆる異種デプロイとは、BEノードのディスク容量、数、ディスクタイプ(SSDとHDD)が一致しないことです。異種デプロイの場合、小さなBEノードと大きなBEノードが同じ数のレプリカを保存する可能性があります。
注2:グループが不安定状態にある時、その中のTableのJoinは通常のJoinに退化します。この時、クラスターのクエリパフォーマンスが大幅に低下する可能性があります。システムが自動的にバランスを取ることを望まない場合は、FE設定項目
disable_colocate_balanceを設定して自動バランシングを禁止できます。その後、適切な時期にそれを開くことができます。(詳細はAdvanced 運用セクションを参照)
Query
ColocationTableは通常のTableと同じ方法でクエリされ、ユーザーはColocation属性を認識する必要はありません。ColocationTableが配置されているGroupが不安定状態にある場合、自動的に通常のJoinに退化します。
例を挙げて説明します:
Table 1:
CREATE TABLE `tbl1` (
`k1` date NOT NULL COMMENT "",
`k2` int(11) NOT NULL COMMENT "",
`v1` int(11) SUM NOT NULL COMMENT ""
) ENGINE=OLAP
AGGREGATE KEY(`k1`, `k2`)
PARTITION BY RANGE(`k1`)
(
PARTITION p1 VALUES LESS THAN ('2019-05-31'),
PARTITION p2 VALUES LESS THAN ('2019-06-30')
)
DISTRIBUTED BY HASH(`k2`) BUCKETS 8
PROPERTIES (
"colocate_with" = "group1"
);
表2:
CREATE TABLE `tbl2` (
`k1` datetime NOT NULL COMMENT "",
`k2` int(11) NOT NULL COMMENT "",
`v1` double SUM NOT NULL COMMENT ""
) ENGINE=OLAP
AGGREGATE KEY(`k1`, `k2`)
DISTRIBUTED BY HASH(`k2`) BUCKETS 8
PROPERTIES (
"colocate_with" = "group1"
);
クエリプランを表示する:
DESC SELECT * FROM tbl1 INNER JOIN tbl2 ON (tbl1.k2 = tbl2.k2);
+----------------------------------------------------+
| Explain String |
+----------------------------------------------------+
| PLAN FRAGMENT 0 |
| OUTPUT EXPRS:`tbl1`.`k1` | |
| PARTITION: RANDOM |
| |
| RESULT SINK |
| |
| 2:HASH JOIN |
| | join op: INNER JOIN |
| | hash predicates: |
| | colocate: true |
| | `tbl1`.`k2` = `tbl2`.`k2` |
| | tuple ids: 0 1 |
| | |
| |----1:OlapScanNode |
| | TABLE: tbl2 |
| | PREAGGREGATION: OFF. Reason: null |
| | partitions=0/1 |
| | rollup: null |
| | buckets=0/0 |
| | cardinality=-1 |
| | avgRowSize=0.0 |
| | numNodes=0 |
| | tuple ids: 1 |
| | |
| 0:OlapScanNode |
| TABLE: tbl1 |
| PREAGGREGATION: OFF. Reason: No AggregateInfo |
| partitions=0/2 |
| rollup: null |
| buckets=0/0 |
| cardinality=-1 |
| avgRowSize=0.0 |
| numNodes=0 |
| tuple ids: 0 |
+----------------------------------------------------+
Colocation Joinが動作する場合、Hash Join Nodeはcolocate: trueを表示します。
そうでない場合、クエリプランは以下のようになります:
+----------------------------------------------------+
| Explain String |
+----------------------------------------------------+
| PLAN FRAGMENT 0 |
| OUTPUT EXPRS:`tbl1`.`k1` | |
| PARTITION: RANDOM |
| |
| RESULT SINK |
| |
| 2:HASH JOIN |
| | join op: INNER JOIN (BROADCAST) |
| | hash predicates: |
| | colocate: false, reason: group is not stable |
| | `tbl1`.`k2` = `tbl2`.`k2` |
| | tuple ids: 0 1 |
| | |
| |----3:EXCHANGE |
| | tuple ids: 1 |
| | |
| 0:OlapScanNode |
| TABLE: tbl1 |
| PREAGGREGATION: OFF. Reason: No AggregateInfo |
| partitions=0/2 |
| rollup: null |
| buckets=0/0 |
| cardinality=-1 |
| avgRowSize=0.0 |
| numNodes=0 |
| tuple ids: 0 |
| |
| PLAN FRAGMENT 1 |
| OUTPUT EXPRS: |
| PARTITION: RANDOM |
| |
| STREAM DATA SINK |
| EXCHANGE ID: 03 |
| UNPARTITIONED |
| |
| 1:OlapScanNode |
| TABLE: tbl2 |
| PREAGGREGATION: OFF. Reason: null |
| partitions=0/1 |
| rollup: null |
| buckets=0/0 |
| cardinality=-1 |
| avgRowSize=0.0 |
| numNodes=0 |
| tuple ids: 1 |
+----------------------------------------------------+
HASH JOINノードは対応する理由を表示します:colocate: false, reason: group is not stable。同時に、EXCHANGEノードが生成されます。
高度な操作
FE設定項目
- disable_colocate_relocate
DorisのColocationレプリカ自動修復を無効にするかどうか。デフォルトはfalse、つまり無効化されません。このパラメータはColocationTableのレプリカ修復にのみ影響し、通常のTableには影響しません。
- disable_colocate_balance
DorisのColocationレプリカ自動バランシングを無効にするかどうか。デフォルトはfalse、つまり無効化されません。このパラメータはCollocationTableのレプリカバランスにのみ影響し、一般的なTableには影響しません。
ユーザーは実行時にこれらの設定を変更できます。HELP ADMIN SHOW CONFIG;とHELP ADMIN SET CONFIG;を参照してください。
- disable_colocate_join
Colocation Join機能を無効にするかどうか。0.10以前のバージョンでは、デフォルトはtrue、つまり無効です。後のバージョンでは、デフォルトはfalse、つまり有効になります。
- use_new_tablet_scheduler
0.10以前のバージョンでは、新しいレプリカスケジューリングロジックはColocation Join機能と互換性がないため、0.10以前のバージョンでdisable_colocate_join = falseの場合、use_new_tablet_scheduler = falseを設定する必要があります、つまり新しいレプリカスケジューラを無効にします。後のバージョンでは、use_new_tablet_schedulerはtrueになります。
HTTP RESTful API
DorisはColocation Groupの表示と変更のためのColocation Joinに関連するいくつかのHTTP RESTful APIを提供しています。
APIはFE側で実装され、fe_host: fe_http_portを使用してアクセスします。ADMIN権限が必要です。
-
クラスタのすべてのColocation情報を表示
GET /api/colocate
Return the internal Colocation info in JSON format:
{
"msg": "success",
"code": 0,
"data": {
"infos": [
["10003.12002", "10003_group1", "10037, 10043", "1", "1", "int(11)", "true"]
],
"unstableGroupIds": [],
"allGroupIds": [{
"dbId": 10003,
"grpId": 12002
}]
},
"count": 0
} -
グループをStableまたはUnstableとしてマークする
-
Stableとしてマークする
```
DELETE /api/colocate/group_stable?db_id=10005&group_id=10008
Returns: 200
``` -
Unstableとしてマークする
```
POST /api/colocate/group_stable?db_id=10005&group_id=10008
Returns: 200
```
- グループのデータ分散の設定
インターフェースは、グループのバケットシーケンス分散を強制できます。
```
POST /api/colocate/bucketseq?db_id=10005&group_id=10008
Body:
[[10004,10002],[10003,10002],[10002,10004],[10003,10002],[10002,10004],[10003,10002],[10003,10004],[10003,10004],[10003,10004],[10002,10004]]
Returns: 200
```
Bodyは、ネストした配列で表現されるBuckets Sequenceと、各Bucketでフラグメントが分散されるBEのIDです。
このコマンドを使用する場合、FE設定のdisable_colocate_relocateとdisable_colocate_balanceをtrueに設定する必要がある可能性があることに注意してください。これにより、システムがColocationレプリカを自動的に修復またはバランシングすることを防ぎます。そうしないと、変更後にシステムによって自動的にリセットされる可能性があります。