-
Notifications
You must be signed in to change notification settings - Fork 3.9k
/
Copy pathfilter_opt.go
664 lines (605 loc) · 23.5 KB
/
filter_opt.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
// Copyright 2016 The Cockroach Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
// implied. See the License for the specific language governing
// permissions and limitations under the License.
//
// Author: Raphael 'kena' Poss ([email protected])
package sql
import (
"fmt"
"github.com/cockroachdb/cockroach/pkg/sql/parser"
)
// This file contains the functions that perform filter propagation.
//
// The explanation below uses the following notations:
// P, Q, R : logical predicates (formulas with boolean values)
// T : the tautology ("always true" predicate)
// A, B : data sources
// t : table
//
// Definitions:
//
// "t" is abbreviation for "triggerFilterPropagation".
// Initiates filter propagation on the given data source.
//
// t :: data source -> data source
// t( A ) = B
// where _, B := f(A, T)
//
// "w" is abbreviation for "propagateOrWrapFilters".
// Calls f and wraps the results in a filterNode if
// the remaining filter is not the tautology.
//
// w :: data source, predicate -> data source
// w( A, P ) = B if R = T
// = [filter(R) FROM B] otherwise
// where (R, B) := f(A, P)
//
// "f" is abbreviation for "propagateFilters".
// Performs filter propagation and returns the part
// of the filter that was not propagated and could
// not be attached at the current level.
//
// f :: data source, predicate -> predicate, data source
// (definition below)
//
// f (propagateFilters) knows how to attach an incoming filter to
// empty and scan nodes:
//
// f( [<empty>], P ) = T, [<empty>]
// f( [scan t // P], Q ) = T, [scan t // (P AND Q)]
//
// f combines an incoming filter with an existing filter and pushes
// forward:
//
// f( [filter(P) FROM A], Q ) = f(A, (P AND Q))
//
// Filtering is distributive over UNION:
//
// f( [union FROM A, B], P ) = T, [union FROM w(A, P), w(B, P)]
//
// f propagates filters "through" distinct and sort nodes:
//
// f( [distinct FROM A], P ) = T, [distinct FROM w(B, P) ]
//
// f( [sort FROM A], P ) = T, [sort FROM w(B, P) ]
//
// Some nodes "block" filter propagation entirely:
//
// f( [limit FROM A], P ) = P, [limit FROM t(A)]
// f( [ordinality FROM A], P ) = P, [ordinality FROM t(A)]
//
// Some nodes currently block filtering entirely, but could be
// optimized later to let some filters through (see inline comments
// below):
//
// f( [group FROM A], P ) = P, [group FROM t(A)]
// f( [window FROM A], P ) = P, [window FROM t(A)]
//
// f propagates filters through render nodes, though only for simple renders:
//
// f( [render <colname> AS x FROM A], P ) = T, [render <colname> AS x FROM w(A, [P/x/<colname>])]
// f( [render Expr(...) AS x FROM A], P ) = P, [render Expr(...) AS x FROM t(A)]
//
// (The notation [P/x/<colname>] means replace all occurrences of "x" in P by <colname>.
//
// f propagates filters through inner joins only:
//
// f( [outerjoin(P) FROM A, B], Q ) = Q, [outerjoin(P) FROM t(A), t(B)]
// f( [innerjoin(P) FROM A, B], Q ) = T, [innerjoin(combi(R)) FROM w(A, left(R)), w(B, right(R))
// where:
// R = (P AND Q)
// left(R) is the part of R that only uses columns from A
// right(R) is the part of R that only uses columns from B
// combi(R) is the part of R that uses columns from both A and B
//
// (see the RFC for filter propagation over joins.)
//
// General implementation principles:
//
// Predicates are either:
// - free, meaning they can be evaluated independently of context;
// this includes simple constants or calls to functions with only
// free arguments.
// - dependent, meaning they use references (IndexedVars) to
// values in the data source at the current level.
//
// Dependent predicates contain IndexedVars to refer to values at the
// current level.
// From the perspective of this algorithm, we can abstract and
// consider that IndexedVars are just positional indices to a virtual
// array of the values. We ignore in particular that their actual
// implementation is more complex (e.g. they carry also a
// "container"), because this complexity is fully inconsequential to
// the predicate transformations performed here.
//
// Because IndexedVars are indices, they need to be adjusted
// when propagating filters across nodes that augment or reduce
// the column set of their source. For example, take a join:
//
// n := [join FROM A, B]
//
// the virtual array of values in join results is the (virtual)
// concatenation of value arrays from both operands (we ignore for the
// sake of this explanation merged columns that result from USING or
// NATURAL). So, for example, if (say) A has 2 columns and B has 3
// columns, and we are considering a filter predicate on n which
// contains an IndexedVar @4 (using base-1 indexing), then that
// IndexedVar is really relative to B, and *when propagated in the
// context of B* must be adjusted to @2.
// propagateFilters tries to combine the extraFilter with the filter
// at the current level (if any), propagate the result, re-attach the
// remainder filter at the current level (if possible) and returns the
// remaining filter.
//
// The reason why propagateFilter returns a remainingFilter instead of
// adding a new filterNode itself, and thus the reason why
// propagateFilter and propagateOrWrapFilter are two separate
// functions, has to do with the handling of renderNodes:
//
// - if two renderNodes are stacked onto each other (e.g. SELECT *
// FROM (SELECT * FROM ...) ), we don't want to introduce a filter
// in between them, because...
// - another optimization (pending #12763) seeks to combine adjacent
// renderNodes, and...
// - we can't combine renderNodes beforehand either, because filter
// propagation may eliminate some filterNodes and that may create
// more opportunities to merge renderNodes.
//
// Perhaps the way #12763 will be addressed is to combine the two
// recursions that propagate filters and merge renderNodes, in which
// case perhaps propagateFilter and propagateOrWrapFilter can merge,
// but we are not there yet.
func (p *planner) propagateFilters(
plan planNode, info *dataSourceInfo, extraFilter parser.TypedExpr,
) (newPlan planNode, remainingFilter parser.TypedExpr, err error) {
remainingFilter = extraFilter
switch n := plan.(type) {
case *emptyNode:
if !n.results {
// There is no row (by definition), so all filters
// are "already applied". Silently absorb any extra filter.
return plan, parser.DBoolTrue, nil
}
// TODO(knz) We could evaluate the filter here and set/reset
// n.results accordingly, assuming the filter is not "row
// dependent" (cf. resolveNames()).
case *filterNode:
newFilter := mergeConj(n.filter, extraFilter)
newPlan, err = p.propagateOrWrapFilters(n.source.plan, n.source.info, newFilter)
if err != nil {
return plan, extraFilter, err
}
return newPlan, parser.DBoolTrue, nil
case *scanNode:
n.filter = mergeConj(n.filter, n.filterVars.Rebind(extraFilter, true, false))
return plan, parser.DBoolTrue, nil
case *renderNode:
return p.addRenderFilter(n, extraFilter)
case *joinNode:
return p.addJoinFilter(n, extraFilter)
case *indexJoinNode:
panic("filter optimization must occur before index selection")
case *distinctNode:
// A distinct node can propagate a filter. Source filtering
// reduces the amount of work.
n.plan, err = p.propagateOrWrapFilters(n.plan, nil, extraFilter)
if err != nil {
return plan, extraFilter, err
}
return plan, parser.DBoolTrue, nil
case *sortNode:
// A sort node can propagate a filter, and source filtering
// reduces the amount of work.
n.plan, err = p.propagateOrWrapFilters(n.plan, nil, extraFilter)
if err != nil {
return plan, extraFilter, err
}
return plan, parser.DBoolTrue, nil
case *selectTopNode:
if n.limit == nil && n.window == nil && n.group == nil {
// We only know how to propagate the filter if there is no
// intervening GROUP BY or LIMIT clause nor window functions.
// (For GROUP BY, see below for details.)
newPlan, remainingFilter, err = p.propagateFilters(n.source, nil, extraFilter)
if err != nil {
return plan, extraFilter, err
}
n.source = newPlan
} else if n.source, err = p.triggerFilterPropagation(n.source); err != nil {
return plan, extraFilter, err
}
case *unionNode:
// Filtering is distributive over set operations.
// Source filtering reduces the amount of work, so force propagation.
newLeft, err := p.propagateOrWrapFilters(n.left, nil, extraFilter)
if err != nil {
return plan, extraFilter, err
}
newRight, err := p.propagateOrWrapFilters(n.right, nil, extraFilter)
if err != nil {
return plan, extraFilter, err
}
n.left = newLeft
n.right = newRight
return plan, parser.DBoolTrue, nil
case *groupNode:
// Filters can't propagate across aggregates:
// SELECT * FROM (SELECT MAX(x) AS m FROM foo GROUP BY y) WHERE m > 123
// can only be filtered after the aggregation has occurred.
// They can however propagate across non-aggregated GROUP BY columns:
// SELECT * FROM (SELECT MAX(x) AS m, y FROM foo GROUP BY y) WHERE y > 123
// can be transformed to:
// SELECT MAX(x) AS m, y FROM (SELECT * FROM foo WHERE y > 123) GROUP BY y
// However we don't do that yet.
// For now, simply trigger optimization for the child node.
//
// TODO(knz) implement the aforementioned optimization.
//
if n.plan, err = p.triggerFilterPropagation(n.plan); err != nil {
return plan, extraFilter, err
}
case *limitNode:
if n.plan, err = p.triggerFilterPropagation(n.plan); err != nil {
return plan, extraFilter, err
}
case *windowNode:
if n.plan, err = p.triggerFilterPropagation(n.plan); err != nil {
return plan, extraFilter, err
}
case *ordinalityNode:
if n.source, err = p.triggerFilterPropagation(n.source); err != nil {
return plan, extraFilter, err
}
case *createTableNode:
if n.n.As() {
if n.sourcePlan, err = p.triggerFilterPropagation(n.sourcePlan); err != nil {
return plan, extraFilter, err
}
}
case *createViewNode:
if n.sourcePlan, err = p.triggerFilterPropagation(n.sourcePlan); err != nil {
return plan, extraFilter, err
}
case *deleteNode:
if n.run.rows, err = p.triggerFilterPropagation(n.run.rows); err != nil {
return plan, extraFilter, err
}
case *insertNode:
if n.run.rows, err = p.triggerFilterPropagation(n.run.rows); err != nil {
return plan, extraFilter, err
}
case *updateNode:
if n.run.rows, err = p.triggerFilterPropagation(n.run.rows); err != nil {
return plan, extraFilter, err
}
case *explainDebugNode:
if n.plan, err = p.triggerFilterPropagation(n.plan); err != nil {
return plan, extraFilter, err
}
case *explainPlanNode:
if n.optimized {
if n.plan, err = p.triggerFilterPropagation(n.plan); err != nil {
return plan, extraFilter, err
}
}
case *explainTraceNode:
if n.plan, err = p.triggerFilterPropagation(n.plan); err != nil {
return plan, extraFilter, err
}
case *alterTableNode:
case *copyNode:
case *createDatabaseNode:
case *createIndexNode:
case *createUserNode:
case *delayedNode:
case *dropDatabaseNode:
case *dropIndexNode:
case *dropTableNode:
case *dropViewNode:
case *hookFnNode:
case *splitNode:
case *valueGenerator:
case *valuesNode:
default:
panic(fmt.Sprintf("unhandled node type: %T", plan))
}
return plan, remainingFilter, nil
}
// triggerFilterPropagation initiates filter propagation on the given plan.
func (p *planner) triggerFilterPropagation(plan planNode) (planNode, error) {
newPlan, remainingFilter, err := p.propagateFilters(plan, nil, parser.DBoolTrue)
if err != nil {
return plan, err
}
if !isFilterTrue(remainingFilter) {
panic(fmt.Sprintf("propagateFilters on \n%s\n spilled a non-trivial remaining filter: %s",
planToString(plan), remainingFilter))
}
return newPlan, nil
}
// propagateOrWrapFilters triggers filter propagation on the given
// node, and creates a new filterNode if there is any remaining filter
// after the propagation.
func (p *planner) propagateOrWrapFilters(
plan planNode, info *dataSourceInfo, filter parser.TypedExpr,
) (planNode, error) {
newPlan, remainingFilter, err := p.propagateFilters(plan, info, filter)
if err != nil {
return plan, err
}
// If there is no remaining filter, simply return the new plan.
if isFilterTrue(remainingFilter) {
return newPlan, nil
}
// Otherwise, wrap it using a new filterNode.
if info == nil {
info = newSourceInfoForSingleTable(anonymousTable, newPlan.Columns())
}
f := &filterNode{
p: p,
source: planDataSource{plan: newPlan, info: info},
}
f.ivarHelper = parser.MakeIndexedVarHelper(f, len(info.sourceColumns))
f.filter = f.ivarHelper.Rebind(remainingFilter,
false /* helper is fresh, no reset needed */, false)
return f, nil
}
// addRenderFilter attempts to add the extraFilter to the renderNode.
// The filter is only propagated to the sub-plan if it is expressed
// using renders that are either simple datums or simple column
// references to the source.
func (p *planner) addRenderFilter(
s *renderNode, extraFilter parser.TypedExpr,
) (planNode, parser.TypedExpr, error) {
// outerFilter is the filter expressed using values rendered by this renderNode.
var outerFilter parser.TypedExpr = parser.DBoolTrue
// innerFilter is the filter on the source planNode.
var innerFilter parser.TypedExpr = parser.DBoolTrue
if !isFilterTrue(extraFilter) {
// The filter that's being added refers to the rendered
// expressions, not the render's source node.
// We propagate only those filters that use simple renders
// of columns from the source, that is, we do not replace:
//
// SELECT * FROM (SELECT a + 2 * b AS c FROM t) WHERE c > 123
// by
// SELECT a + 2 * b AS c FROM (SELECT * FROM t WHERE a + 2 * b > 123)
//
// Because that would cause the (more complex) expression `a + 2 *
// b` to be computed at both levels. This cannot be "simplified"
// further by trying to factor the common sub-expression: `SELECT
// c FROM (SELECT * FROM (SELECT a + 2 * b AS c FROM t) WHERE c >
// 123)` contains the original query.
//
// To perform this propagation, we apply the following convFunc to
// all the filter expressions. This replaces IndexedVars in the
// filter that are relative to this renderNode's render list by
// IndexedVars that are relative to the renderNode's source
// columns. It also sets complexRenderEncountered to true if any
// of the IndexedVars refers to non-trivial expressions.
//
// Note: we don't really care about the IndexedVars' container
// here, as no filter will stay at this node -- either they
// propagate down or spill upward as remaining filter.
// See also the comment for propagateFilters() above.
//
convFunc := func(v parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := v.(*parser.IndexedVar); ok {
renderExpr := s.render[iv.Idx]
if d, ok := renderExpr.(parser.Datum); ok {
// A standalone Datum is not complex, so it can be propagated further.
return true, d
}
if rv, ok := renderExpr.(*parser.IndexedVar); ok {
// A naked IndexedVar is not complex, so it can be propagated further.
return true, rv
}
}
return false, v
}
// Do the replacement proper.
innerFilter, outerFilter = splitFilter(extraFilter, convFunc)
}
// Propagate the inner filter.
newPlan, err := s.planner.propagateOrWrapFilters(
s.source.plan, s.source.info, innerFilter)
if err != nil {
return s, extraFilter, err
}
// Attach what remains as the new source.
s.source.plan = newPlan
return s, outerFilter, nil
}
// addJoinFilter propagates the given filter to a joinNode.
func (p *planner) addJoinFilter(
n *joinNode, extraFilter parser.TypedExpr,
) (planNode, parser.TypedExpr, error) {
// TODO(knz) support outer joins.
if n.joinType != joinTypeInner {
// Outer joins not supported; simply trigger filter optimization in the sub-nodes.
var err error
if n.left.plan, err = p.triggerFilterPropagation(n.left.plan); err == nil {
n.right.plan, err = p.triggerFilterPropagation(n.right.plan)
}
return n, extraFilter, err
}
// There are three steps to the transformation below:
//
// 1. if there are merged columns (i.e. USING/NATURAL), since the
// incoming extra filter may refer to them, transform the
// extra filter to only use the left and right columns.
// 2. merge the existing ON predicate with the extra filter, yielding
// an "initial predicate".
// 3. split the initial predicate into a left, right and combined parts.
// 4. propagate the left and right part to the left and right join operands.
// 5. use the combined part from step #3 to create a new ON predicate,
// possibly detecting new equality columns.
// Reminder: the layout of the virtual data values for a join is:
// [ merged columns ] [ columns from left ] [ columns from right]
//
// The merged columns result from USING or NATURAL; for example
// SELECT * FROM a JOIN b USING(x)
// has columns: x (merged), a.*, b.*
// mergedBoundary is the logical index of the first column after
// the merged columns.
mergedBoundary := n.pred.numMergedEqualityColumns
// sourceBoundary is the logical index of the first column in
// the right data source.
sourceBoundary := mergedBoundary + len(n.left.info.sourceColumns)
if mergedBoundary > 0 && !isFilterTrue(extraFilter) {
// There are merged columns, and the incoming extra filter may be
// referring to them.
// To understand what's going on below consider the query:
//
// SELECT * FROM a JOIN b USING(x) WHERE f(a,b) AND g(a) AND g(b) AND h(x)
//
// What we want at the end (below) is this:
// SELECT x, ... FROM
// (SELECT * FROM a WHERE g(a) AND h(a.x))
// JOIN (SELECT * FROM b WHERE g(b) AND h(b.x))
// ON a.x = b.x AND f(a,b)
//
// So we need to replace h(x) which refers to the merged USING
// column by h(x) on the source tables. But we can't simply
// replace it by a single reference to *either* of the two source
// tables (say, via h(a.x) in the example), because if we did then
// the filter propagation would push it towards that source table
// only (a in the example) -- and we want it propagated on both
// sides!
//
// So what we do instead:
// - we first split the expression to extract those expressions that
// refer to merged columns (notUsingMerged // perhapsUsingMerged):
// "f(a,b) AND g(a) AND g(b)" // "h(x)"
// - we replace the part that refers to merged column by an AND
// of its substitution by references to the left and righ sources:
// "h(x)" -> "h(a.x) AND h(b.x)"
// - we recombine all of them:
// f(a,b) AND g(a) AND g(b) AND h(a.x) AND h(b.x)
// Then the rest of the optimization below can go forward, and
// will take care of splitting the expressions among the left and
// right operands.
noMergedVars := func(expr parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := expr.(*parser.IndexedVar); ok && iv.Idx >= mergedBoundary {
return true, expr
}
return false, expr
}
// Note: we use a negative condition here because splitFilter()
// doesn't guarantee that its right return value doesn't contain
// sub-expressions where the conversion function returns true.
notUsingMerged, perhapsUsingMerged := splitFilter(extraFilter, noMergedVars)
leftFilter := exprConvertVars(perhapsUsingMerged,
func(expr parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := expr.(*parser.IndexedVar); ok && iv.Idx < mergedBoundary {
newIv := n.pred.iVarHelper.IndexedVar(mergedBoundary + n.pred.leftEqualityIndices[iv.Idx])
return true, newIv
}
return true, expr
})
rightFilter := exprConvertVars(perhapsUsingMerged,
func(expr parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := expr.(*parser.IndexedVar); ok && iv.Idx < mergedBoundary {
newIv := n.pred.iVarHelper.IndexedVar(sourceBoundary + n.pred.rightEqualityIndices[iv.Idx])
return true, newIv
}
return true, expr
})
extraFilter = mergeConj(notUsingMerged, mergeConj(leftFilter, rightFilter))
}
// Merge the existing ON predicate with the extra filter.
// We don't need to reset the helper here, as this will be done
// later for the final predicate below.
initialPred := mergeConj(n.pred.iVarHelper.Rebind(extraFilter, false, false), n.pred.onCond)
// Split the initial predicate into left, right and combined parts.
leftExpr, remainder := splitFilter(initialPred,
func(expr parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := expr.(*parser.IndexedVar); ok && iv.Idx < sourceBoundary {
return true, expr
}
return false, expr
})
rightExpr, combinedExpr := splitFilter(remainder,
func(expr parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := expr.(*parser.IndexedVar); ok && iv.Idx >= sourceBoundary {
return true, expr
}
return false, expr
})
// Propagate the left and right predicates to the left and right
// sides of the join. The predicates must first be "shifted"
// i.e. their IndexedVars which are relative to the join columns
// must be modified to become relative to the operand's columns.
propagate := func(pred parser.TypedExpr, offset int, side *planDataSource) error {
pred = shiftConj(n.pred.iVarHelper, pred, offset)
newPlan, err := p.propagateOrWrapFilters(side.plan, side.info, pred)
if err != nil {
return err
}
side.plan = newPlan
return nil
}
if err := propagate(leftExpr, mergedBoundary, &n.left); err != nil {
return n, extraFilter, err
}
if err := propagate(rightExpr, sourceBoundary, &n.right); err != nil {
return n, extraFilter, err
}
// Extract possibly new equality columns from the combined predicate, and
// use the rest as new ON condition.
var newCombinedExpr parser.TypedExpr = parser.DBoolTrue
for _, e := range splitAndExpr(combinedExpr, nil) {
if e == parser.DBoolTrue {
continue
}
if !n.pred.tryAddEqualityFilter(e, n.left.info, n.right.info) {
newCombinedExpr = mergeConj(newCombinedExpr, e)
}
}
combinedExpr = newCombinedExpr
n.pred.onCond = n.pred.iVarHelper.Rebind(combinedExpr, true, false)
return n, nil, nil
}
// shiftConj renumbers the IndexedVars in the given filter by shifting
// their index to the left by the given offset. For example a shift of
// "@3+@5" by 2 results in the expression "@1+@3".
func shiftConj(
ivarHelper parser.IndexedVarHelper, filter parser.TypedExpr, offset int,
) parser.TypedExpr {
if isFilterTrue(filter) {
return nil
}
shiftFunc := func(expr parser.VariableExpr) (bool, parser.Expr) {
if iv, ok := expr.(*parser.IndexedVar); ok {
return true, ivarHelper.IndexedVar(iv.Idx - offset)
}
return true, expr
}
return exprConvertVars(filter, shiftFunc)
}
// mergeConj combines two predicates.
func mergeConj(left, right parser.TypedExpr) parser.TypedExpr {
if isFilterTrue(left) {
if right == parser.DBoolTrue {
return nil
}
return right
}
if isFilterTrue(right) {
return left
}
return parser.NewTypedAndExpr(left, right)
}
func isFilterTrue(expr parser.TypedExpr) bool {
return expr == nil || expr == parser.DBoolTrue
}