From c22904e20f2337fb61a7d11e61cedf9bdafb3d2b Mon Sep 17 00:00:00 2001 From: Mattias Jonsson Date: Fri, 16 Dec 2022 18:07:53 +0100 Subject: [PATCH 1/4] Fix issue 39999, used wrong column id list for checking partitions --- executor/builder.go | 23 +++++-------- executor/partition_table_test.go | 56 ++++++++++++++++++++++++++++++++ table/table.go | 1 + table/tables/partition.go | 23 +++++++++++++ 4 files changed, 88 insertions(+), 15 deletions(-) diff --git a/executor/builder.go b/executor/builder.go index b33c57d3de234..1fecec7753d62 100644 --- a/executor/builder.go +++ b/executor/builder.go @@ -3511,13 +3511,10 @@ func buildIndexRangeForEachPartition(ctx sessionctx.Context, usedPartitions []ta return nextRange, nil } -func keyColumnsIncludeAllPartitionColumns(keyColumns []int, pe *tables.PartitionExpr) bool { - tmp := make(map[int]struct{}, len(keyColumns)) - for _, offset := range keyColumns { - tmp[offset] = struct{}{} - } - for _, offset := range pe.ColumnOffset { - if _, ok := tmp[offset]; !ok { +func keyColumnsIncludeAllPartitionColumns(keyColumnIDs []int64, pt table.PartitionedTable) bool { + partColIDs := pt.GetPartitionColumnIDs() + for _, id := range keyColumnIDs { + if _, ok := partColIDs[id]; !ok { return false } } @@ -4149,12 +4146,6 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte } tbl, _ := builder.is.TableByID(tbInfo.ID) pt := tbl.(table.PartitionedTable) - pe, err := tbl.(interface { - PartitionExpr() (*tables.PartitionExpr, error) - }).PartitionExpr() - if err != nil { - return nil, err - } partitionInfo := &v.PartitionInfo usedPartitionList, err := builder.partitionPruning(pt, partitionInfo.PruningConds, partitionInfo.PartitionNames, partitionInfo.Columns, partitionInfo.ColumnNames) if err != nil { @@ -4166,13 +4157,14 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte } var kvRanges []kv.KeyRange if v.IsCommonHandle { - if len(lookUpContents) > 0 && keyColumnsIncludeAllPartitionColumns(lookUpContents[0].keyCols, pe) { + if len(lookUpContents) > 0 && keyColumnsIncludeAllPartitionColumns(lookUpContents[0].keyColIDs, pt) { locateKey := make([]types.Datum, e.Schema().Len()) kvRanges = make([]kv.KeyRange, 0, len(lookUpContents)) // lookUpContentsByPID groups lookUpContents by pid(partition) so that kv ranges for same partition can be merged. lookUpContentsByPID := make(map[int64][]*indexJoinLookUpContent) for _, content := range lookUpContents { for i, date := range content.keys { + // TODO: Add a test where partition column is not a prefix or out of order with source joined table locateKey[content.keyCols[i]] = date } p, err := pt.GetPartitionByRow(e.ctx, locateKey) @@ -4212,11 +4204,12 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte handles, lookUpContents := dedupHandles(lookUpContents) - if len(lookUpContents) > 0 && keyColumnsIncludeAllPartitionColumns(lookUpContents[0].keyCols, pe) { + if len(lookUpContents) > 0 && keyColumnsIncludeAllPartitionColumns(lookUpContents[0].keyColIDs, pt) { locateKey := make([]types.Datum, e.Schema().Len()) kvRanges = make([]kv.KeyRange, 0, len(lookUpContents)) for _, content := range lookUpContents { for i, date := range content.keys { + // TODO: Add test to see if keyColIDs should be used instead? locateKey[content.keyCols[i]] = date } p, err := pt.GetPartitionByRow(e.ctx, locateKey) diff --git a/executor/partition_table_test.go b/executor/partition_table_test.go index 15d2c2872ca9c..49b8d7bb137a8 100644 --- a/executor/partition_table_test.go +++ b/executor/partition_table_test.go @@ -3831,3 +3831,59 @@ func TestIssue21732(t *testing.T) { }) } } + +func TestIssue39999(t *testing.T) { + store := testkit.CreateMockStore(t) + + tk := testkit.NewTestKit(t, store) + + tk.MustExec(`create schema test39999`) + tk.MustExec(`use test39999`) + tk.MustExec(`drop table if exists c, t`) + tk.MustExec("CREATE TABLE `c` (" + + "`serial_id` varchar(24)," + + "`txt_account_id` varchar(24)," + + "`capital_sub_class` varchar(10)," + + "`occur_trade_date` date," + + "`occur_amount` decimal(16,2)," + + "`broker` varchar(10)," + + "PRIMARY KEY (`txt_account_id`,`occur_trade_date`,`serial_id`) /*T![clustered_index] CLUSTERED */," + + "KEY `idx_serial_id` (`serial_id`)" + + ") ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci " + + "PARTITION BY RANGE COLUMNS(`serial_id`) (" + + "PARTITION `p202209` VALUES LESS THAN ('20221001')," + + "PARTITION `p202210` VALUES LESS THAN ('20221101')," + + "PARTITION `p202211` VALUES LESS THAN ('20221201')" + + ")") + + tk.MustExec("CREATE TABLE `t` ( " + + "`txn_account_id` varchar(24), " + + "`account_id` varchar(32), " + + "`broker` varchar(10), " + + "PRIMARY KEY (`txn_account_id`) /*T![clustered_index] CLUSTERED */ " + + ") ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci") + + tk.MustExec("INSERT INTO `c` VALUES ('2022111700196920','04482786','CUST','2022-11-17',-2.01,'0009')") + tk.MustExec("INSERT INTO `t` VALUES ('04482786','1142927','0009')") + + tk.MustExec(`set tidb_partition_prune_mode='dynamic'`) + tk.MustExec(`analyze table c`) + tk.MustExec(`analyze table t`) + query := `select + /*+ inl_join(c) */ + c.occur_amount +from + c + join t on c.txt_account_id = t.txn_account_id + and t.broker = '0009' + and c.occur_trade_date = '2022-11-17'` + tk.MustQuery("explain " + query).Check(testkit.Rows(""+ + "IndexJoin_22 1.00 root inner join, inner:TableReader_21, outer key:test39999.t.txn_account_id, inner key:test39999.c.txt_account_id, equal cond:eq(test39999.t.txn_account_id, test39999.c.txt_account_id)", + "├─TableReader_27(Build) 1.00 root data:Selection_26", + "│ └─Selection_26 1.00 cop[tikv] eq(test39999.t.broker, \"0009\")", + "│ └─TableFullScan_25 1.00 cop[tikv] table:t keep order:false", + "└─TableReader_21(Probe) 1.00 root partition:all data:Selection_20", + " └─Selection_20 1.00 cop[tikv] eq(test39999.c.occur_trade_date, 2022-11-17 00:00:00.000000)", + " └─TableRangeScan_19 1.00 cop[tikv] table:c range: decided by [eq(test39999.c.txt_account_id, test39999.t.txn_account_id) eq(test39999.c.occur_trade_date, 2022-11-17 00:00:00.000000)], keep order:false")) + tk.MustQuery(query).Check(testkit.Rows("-2.01")) +} diff --git a/table/table.go b/table/table.go index 813131df90896..396c2a77d75ca 100644 --- a/table/table.go +++ b/table/table.go @@ -246,6 +246,7 @@ type PartitionedTable interface { GetPartitionByRow(sessionctx.Context, []types.Datum) (PhysicalTable, error) GetAllPartitionIDs() []int64 GetPartitionColumnNames() []model.CIStr + GetPartitionColumnIDs() map[int64]struct{} CheckForExchangePartition(ctx sessionctx.Context, pi *model.PartitionInfo, r []types.Datum, pid int64) error } diff --git a/table/tables/partition.go b/table/tables/partition.go index 6a0b315b856e9..8e197087d3aad 100644 --- a/table/tables/partition.go +++ b/table/tables/partition.go @@ -962,6 +962,29 @@ func (t *partitionedTable) GetPartitionColumnNames() []model.CIStr { return colNames } +// GetPartitionColumnIDs returns the column IDs from the partition expression +// TODO: refactor and have the column ids or a hash on PartitionInfo instead +func (t *partitionedTable) GetPartitionColumnIDs() map[int64]struct{} { + var ids map[int64]struct{} + meta := t.Meta() + pi := meta.Partition + if len(pi.Columns) > 0 { + ids = make(map[int64]struct{}, len(pi.Columns)) + for _, name := range pi.Columns { + col := table.FindColLowerCase(t.Cols(), name.L) + ids[col.ID] = struct{}{} + } + return ids + } + + partitionCols := expression.ExtractColumns(t.partitionExpr.Expr) + ids = make(map[int64]struct{}, len(partitionCols)) + for _, col := range partitionCols { + ids[col.ID] = struct{}{} + } + return ids +} + // PartitionRecordKey is exported for test. func PartitionRecordKey(pid int64, handle int64) kv.Key { recordPrefix := tablecodec.GenTableRecordPrefix(pid) From 10cf1b308b1e06bd6b3a1b8a1932d5648d202781 Mon Sep 17 00:00:00 2001 From: Mattias Jonsson Date: Tue, 20 Dec 2022 00:17:23 +0000 Subject: [PATCH 2/4] Less intrusive refactoring (due to only a bug fix). Refactoring should still be done I think, but in a separate PR. --- executor/builder.go | 77 +++++++++++++++++++-------------------- table/table.go | 1 - table/tables/partition.go | 23 ------------ 3 files changed, 37 insertions(+), 64 deletions(-) diff --git a/executor/builder.go b/executor/builder.go index 1fecec7753d62..1f215aaa40931 100644 --- a/executor/builder.go +++ b/executor/builder.go @@ -3511,14 +3511,39 @@ func buildIndexRangeForEachPartition(ctx sessionctx.Context, usedPartitions []ta return nextRange, nil } -func keyColumnsIncludeAllPartitionColumns(keyColumnIDs []int64, pt table.PartitionedTable) bool { - partColIDs := pt.GetPartitionColumnIDs() - for _, id := range keyColumnIDs { - if _, ok := partColIDs[id]; !ok { - return false +func getPartitionKeyColOffsets(keyColIDs []int64, pt table.PartitionedTable) []int { + keyColOffsets := make([]int, len(keyColIDs)) + for i, colID := range keyColIDs { + offset := -1 + for j, col := range pt.Cols() { + if colID == col.ID { + offset = j + break + } + } + if offset == -1 { + return nil } + keyColOffsets[i] = offset } - return true + + pe, err := pt.(interface { + PartitionExpr() (*tables.PartitionExpr, error) + }).PartitionExpr() + if err != nil { + return nil + } + + offsetMap := make(map[int]struct{}) + for _, offset := range keyColOffsets { + offsetMap[offset] = struct{}{} + } + for _, offset := range pe.ColumnOffset { + if _, ok := offsetMap[offset]; !ok { + return nil + } + } + return keyColOffsets } func (builder *dataReaderBuilder) prunePartitionForInnerExecutor(tbl table.Table, schema *expression.Schema, partitionInfo *plannercore.PartitionInfo, @@ -3533,15 +3558,6 @@ func (builder *dataReaderBuilder) prunePartitionForInnerExecutor(tbl table.Table return nil, false, nil, err } - // check whether can runtime prune. - type partitionExpr interface { - PartitionExpr() (*tables.PartitionExpr, error) - } - pe, err := tbl.(partitionExpr).PartitionExpr() - if err != nil { - return nil, false, nil, err - } - // recalculate key column offsets if len(lookUpContent) == 0 { return nil, false, nil, nil @@ -3549,29 +3565,9 @@ func (builder *dataReaderBuilder) prunePartitionForInnerExecutor(tbl table.Table if lookUpContent[0].keyColIDs == nil { return nil, false, nil, plannercore.ErrInternal.GenWithStack("cannot get column IDs when dynamic pruning") } - keyColOffsets := make([]int, len(lookUpContent[0].keyColIDs)) - for i, colID := range lookUpContent[0].keyColIDs { - offset := -1 - for j, col := range partitionTbl.Cols() { - if colID == col.ID { - offset = j - break - } - } - if offset == -1 { - return nil, false, nil, plannercore.ErrInternal.GenWithStack("invalid column offset when dynamic pruning") - } - keyColOffsets[i] = offset - } - - offsetMap := make(map[int]bool) - for _, offset := range keyColOffsets { - offsetMap[offset] = true - } - for _, offset := range pe.ColumnOffset { - if _, ok := offsetMap[offset]; !ok { - return condPruneResult, false, nil, nil - } + keyColOffsets := getPartitionKeyColOffsets(lookUpContent[0].keyColIDs, partitionTbl) + if len(keyColOffsets) == 0 { + return condPruneResult, false, nil, nil } locateKey := make([]types.Datum, len(partitionTbl.Cols())) @@ -4156,8 +4152,9 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte usedPartitions[p.GetPhysicalID()] = p } var kvRanges []kv.KeyRange + keyColOffsets := getPartitionKeyColOffsets(lookUpContents[0].keyColIDs, pt) if v.IsCommonHandle { - if len(lookUpContents) > 0 && keyColumnsIncludeAllPartitionColumns(lookUpContents[0].keyColIDs, pt) { + if len(lookUpContents) > 0 && len(keyColOffsets) > 0 { locateKey := make([]types.Datum, e.Schema().Len()) kvRanges = make([]kv.KeyRange, 0, len(lookUpContents)) // lookUpContentsByPID groups lookUpContents by pid(partition) so that kv ranges for same partition can be merged. @@ -4204,7 +4201,7 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte handles, lookUpContents := dedupHandles(lookUpContents) - if len(lookUpContents) > 0 && keyColumnsIncludeAllPartitionColumns(lookUpContents[0].keyColIDs, pt) { + if len(lookUpContents) > 0 && len(keyColOffsets) > 0 { locateKey := make([]types.Datum, e.Schema().Len()) kvRanges = make([]kv.KeyRange, 0, len(lookUpContents)) for _, content := range lookUpContents { diff --git a/table/table.go b/table/table.go index 396c2a77d75ca..813131df90896 100644 --- a/table/table.go +++ b/table/table.go @@ -246,7 +246,6 @@ type PartitionedTable interface { GetPartitionByRow(sessionctx.Context, []types.Datum) (PhysicalTable, error) GetAllPartitionIDs() []int64 GetPartitionColumnNames() []model.CIStr - GetPartitionColumnIDs() map[int64]struct{} CheckForExchangePartition(ctx sessionctx.Context, pi *model.PartitionInfo, r []types.Datum, pid int64) error } diff --git a/table/tables/partition.go b/table/tables/partition.go index 8e197087d3aad..6a0b315b856e9 100644 --- a/table/tables/partition.go +++ b/table/tables/partition.go @@ -962,29 +962,6 @@ func (t *partitionedTable) GetPartitionColumnNames() []model.CIStr { return colNames } -// GetPartitionColumnIDs returns the column IDs from the partition expression -// TODO: refactor and have the column ids or a hash on PartitionInfo instead -func (t *partitionedTable) GetPartitionColumnIDs() map[int64]struct{} { - var ids map[int64]struct{} - meta := t.Meta() - pi := meta.Partition - if len(pi.Columns) > 0 { - ids = make(map[int64]struct{}, len(pi.Columns)) - for _, name := range pi.Columns { - col := table.FindColLowerCase(t.Cols(), name.L) - ids[col.ID] = struct{}{} - } - return ids - } - - partitionCols := expression.ExtractColumns(t.partitionExpr.Expr) - ids = make(map[int64]struct{}, len(partitionCols)) - for _, col := range partitionCols { - ids[col.ID] = struct{}{} - } - return ids -} - // PartitionRecordKey is exported for test. func PartitionRecordKey(pid int64, handle int64) kv.Key { recordPrefix := tablecodec.GenTableRecordPrefix(pid) From 32bc21bfdcccd301e055bd659e0a08f4c6711bef Mon Sep 17 00:00:00 2001 From: Mattias Jonsson Date: Tue, 20 Dec 2022 01:06:53 +0000 Subject: [PATCH 3/4] Accidentally opend a bug I closed a year ago :D --- executor/builder.go | 10 ++++++---- executor/index_lookup_join_test.go | 20 +++++++++++--------- executor/partition_table_test.go | 17 +++++++++++++++-- 3 files changed, 32 insertions(+), 15 deletions(-) diff --git a/executor/builder.go b/executor/builder.go index 1f215aaa40931..fe8220891da2c 100644 --- a/executor/builder.go +++ b/executor/builder.go @@ -4152,9 +4152,12 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte usedPartitions[p.GetPhysicalID()] = p } var kvRanges []kv.KeyRange - keyColOffsets := getPartitionKeyColOffsets(lookUpContents[0].keyColIDs, pt) + var keyColOffsets []int + if len(lookUpContents) > 0 { + keyColOffsets = getPartitionKeyColOffsets(lookUpContents[0].keyColIDs, pt) + } if v.IsCommonHandle { - if len(lookUpContents) > 0 && len(keyColOffsets) > 0 { + if len(keyColOffsets) > 0 { locateKey := make([]types.Datum, e.Schema().Len()) kvRanges = make([]kv.KeyRange, 0, len(lookUpContents)) // lookUpContentsByPID groups lookUpContents by pid(partition) so that kv ranges for same partition can be merged. @@ -4201,12 +4204,11 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte handles, lookUpContents := dedupHandles(lookUpContents) - if len(lookUpContents) > 0 && len(keyColOffsets) > 0 { + if len(keyColOffsets) > 0 { locateKey := make([]types.Datum, e.Schema().Len()) kvRanges = make([]kv.KeyRange, 0, len(lookUpContents)) for _, content := range lookUpContents { for i, date := range content.keys { - // TODO: Add test to see if keyColIDs should be used instead? locateKey[content.keyCols[i]] = date } p, err := pt.GetPartitionByRow(e.ctx, locateKey) diff --git a/executor/index_lookup_join_test.go b/executor/index_lookup_join_test.go index 9a021568b20ee..600f052b1225e 100644 --- a/executor/index_lookup_join_test.go +++ b/executor/index_lookup_join_test.go @@ -428,17 +428,19 @@ PARTITIONS 1`) // Why does the t2.prefiller need be at least 2^32 ? If smaller the bug will not appear!?! tk.MustExec("insert into t2 values ( pow(2,32), 1, 1), ( pow(2,32)+1, 2, 0)") + tk.MustExec(`analyze table t1`) + tk.MustExec(`analyze table t2`) // Why must it be = 1 and not 2? - tk.MustQuery("explain select /* +INL_JOIN(t1,t2) */ t1.id, t1.pc from t1 where id in ( select prefiller from t2 where t2.postfiller = 1 )").Check(testkit.Rows("" + - "IndexJoin_15 10.00 root inner join, inner:TableReader_14, outer key:test.t2.prefiller, inner key:test.t1.id, equal cond:eq(test.t2.prefiller, test.t1.id)]\n" + - "[├─HashAgg_25(Build) 8.00 root group by:test.t2.prefiller, funcs:firstrow(test.t2.prefiller)->test.t2.prefiller]\n" + - "[│ └─TableReader_26 8.00 root data:HashAgg_20]\n" + - "[│ └─HashAgg_20 8.00 cop[tikv] group by:test.t2.prefiller, ]\n" + - "[│ └─Selection_24 10.00 cop[tikv] eq(test.t2.postfiller, 1)]\n" + - "[│ └─TableFullScan_23 10000.00 cop[tikv] table:t2 keep order:false, stats:pseudo]\n" + - "[└─TableReader_14(Probe) 8.00 root partition:all data:TableRangeScan_13]\n" + - "[ └─TableRangeScan_13 8.00 cop[tikv] table:t1 range: decided by [eq(test.t1.id, test.t2.prefiller)], keep order:false, stats:pseudo")) + tk.MustQuery("explain format='brief' select /* +INL_JOIN(t1,t2) */ t1.id, t1.pc from t1 where id in ( select prefiller from t2 where t2.postfiller = 1 )").Check(testkit.Rows(""+ + `IndexJoin 1.25 root inner join, inner:TableReader, outer key:test.t2.prefiller, inner key:test.t1.id, equal cond:eq(test.t2.prefiller, test.t1.id)`, + `├─HashAgg(Build) 1.00 root group by:test.t2.prefiller, funcs:firstrow(test.t2.prefiller)->test.t2.prefiller`, + `│ └─TableReader 1.00 root data:HashAgg`, + `│ └─HashAgg 1.00 cop[tikv] group by:test.t2.prefiller, `, + `│ └─Selection 1.00 cop[tikv] eq(test.t2.postfiller, 1)`, + `│ └─TableFullScan 2.00 cop[tikv] table:t2 keep order:false`, + `└─TableReader(Probe) 1.00 root partition:all data:TableRangeScan`, + ` └─TableRangeScan 1.00 cop[tikv] table:t1 range: decided by [eq(test.t1.id, test.t2.prefiller)], keep order:false, stats:pseudo`)) tk.MustQuery("show warnings").Check(testkit.Rows()) // without fix it fails with: "runtime error: index out of range [0] with length 0" tk.MustQuery("select /* +INL_JOIN(t1,t2) */ t1.id, t1.pc from t1 where id in ( select prefiller from t2 where t2.postfiller = 1 )").Check(testkit.Rows()) diff --git a/executor/partition_table_test.go b/executor/partition_table_test.go index 49b8d7bb137a8..5696b56f6f730 100644 --- a/executor/partition_table_test.go +++ b/executor/partition_table_test.go @@ -3842,9 +3842,9 @@ func TestIssue39999(t *testing.T) { tk.MustExec(`drop table if exists c, t`) tk.MustExec("CREATE TABLE `c` (" + "`serial_id` varchar(24)," + + "`occur_trade_date` date," + "`txt_account_id` varchar(24)," + "`capital_sub_class` varchar(10)," + - "`occur_trade_date` date," + "`occur_amount` decimal(16,2)," + "`broker` varchar(10)," + "PRIMARY KEY (`txt_account_id`,`occur_trade_date`,`serial_id`) /*T![clustered_index] CLUSTERED */," + @@ -3863,7 +3863,7 @@ func TestIssue39999(t *testing.T) { "PRIMARY KEY (`txn_account_id`) /*T![clustered_index] CLUSTERED */ " + ") ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci") - tk.MustExec("INSERT INTO `c` VALUES ('2022111700196920','04482786','CUST','2022-11-17',-2.01,'0009')") + tk.MustExec("INSERT INTO `c` (serial_id, txt_account_id, capital_sub_class, occur_trade_date, occur_amount, broker) VALUES ('2022111700196920','04482786','CUST','2022-11-17',-2.01,'0009')") tk.MustExec("INSERT INTO `t` VALUES ('04482786','1142927','0009')") tk.MustExec(`set tidb_partition_prune_mode='dynamic'`) @@ -3886,4 +3886,17 @@ from " └─Selection_20 1.00 cop[tikv] eq(test39999.c.occur_trade_date, 2022-11-17 00:00:00.000000)", " └─TableRangeScan_19 1.00 cop[tikv] table:c range: decided by [eq(test39999.c.txt_account_id, test39999.t.txn_account_id) eq(test39999.c.occur_trade_date, 2022-11-17 00:00:00.000000)], keep order:false")) tk.MustQuery(query).Check(testkit.Rows("-2.01")) + + // Add the missing partition key part. + tk.MustExec(`alter table t add column serial_id varchar(24) default '2022111700196920'`) + query += ` and c.serial_id = t.serial_id` + tk.MustQuery(query).Check(testkit.Rows("-2.01")) + tk.MustQuery("explain " + query).Check(testkit.Rows(""+ + `IndexJoin_20 0.80 root inner join, inner:TableReader_19, outer key:test39999.t.txn_account_id, test39999.t.serial_id, inner key:test39999.c.txt_account_id, test39999.c.serial_id, equal cond:eq(test39999.t.serial_id, test39999.c.serial_id), eq(test39999.t.txn_account_id, test39999.c.txt_account_id)`, + `├─TableReader_25(Build) 0.80 root data:Selection_24`, + `│ └─Selection_24 0.80 cop[tikv] eq(test39999.t.broker, "0009"), not(isnull(test39999.t.serial_id))`, + `│ └─TableFullScan_23 1.00 cop[tikv] table:t keep order:false`, + `└─TableReader_19(Probe) 0.80 root partition:all data:Selection_18`, + ` └─Selection_18 0.80 cop[tikv] eq(test39999.c.occur_trade_date, 2022-11-17 00:00:00.000000)`, + ` └─TableRangeScan_17 0.80 cop[tikv] table:c range: decided by [eq(test39999.c.txt_account_id, test39999.t.txn_account_id) eq(test39999.c.serial_id, test39999.t.serial_id) eq(test39999.c.occur_trade_date, 2022-11-17 00:00:00.000000)], keep order:false`)) } From 14da74a7345621b1c8fe06f8814a4245172d2207 Mon Sep 17 00:00:00 2001 From: Mattias Jonsson Date: Tue, 20 Dec 2022 01:38:20 +0000 Subject: [PATCH 4/4] Removed a comment after adding a test --- executor/builder.go | 1 - 1 file changed, 1 deletion(-) diff --git a/executor/builder.go b/executor/builder.go index fe8220891da2c..d4270397eecd0 100644 --- a/executor/builder.go +++ b/executor/builder.go @@ -4164,7 +4164,6 @@ func (builder *dataReaderBuilder) buildTableReaderForIndexJoin(ctx context.Conte lookUpContentsByPID := make(map[int64][]*indexJoinLookUpContent) for _, content := range lookUpContents { for i, date := range content.keys { - // TODO: Add a test where partition column is not a prefix or out of order with source joined table locateKey[content.keyCols[i]] = date } p, err := pt.GetPartitionByRow(e.ctx, locateKey)