mirror of https://github.com/apache/druid.git
Create a FilterBundle.Builder class and use it to construct FilterBundle. (#17055)
This commit is contained in:
parent
64a4d115c5
commit
88c3c20ab6
|
@ -63,6 +63,7 @@ public class QueryContexts
|
||||||
public static final String REWRITE_JOIN_TO_FILTER_ENABLE_KEY = "enableRewriteJoinToFilter";
|
public static final String REWRITE_JOIN_TO_FILTER_ENABLE_KEY = "enableRewriteJoinToFilter";
|
||||||
public static final String JOIN_FILTER_REWRITE_MAX_SIZE_KEY = "joinFilterRewriteMaxSize";
|
public static final String JOIN_FILTER_REWRITE_MAX_SIZE_KEY = "joinFilterRewriteMaxSize";
|
||||||
public static final String MAX_NUMERIC_IN_FILTERS = "maxNumericInFilters";
|
public static final String MAX_NUMERIC_IN_FILTERS = "maxNumericInFilters";
|
||||||
|
public static final String CURSOR_AUTO_ARRANGE_FILTERS = "cursorAutoArrangeFilters";
|
||||||
// This flag controls whether a SQL join query with left scan should be attempted to be run as direct table access
|
// This flag controls whether a SQL join query with left scan should be attempted to be run as direct table access
|
||||||
// instead of being wrapped inside a query. With direct table access enabled, Druid can push down the join operation to
|
// instead of being wrapped inside a query. With direct table access enabled, Druid can push down the join operation to
|
||||||
// data servers.
|
// data servers.
|
||||||
|
|
|
@ -47,27 +47,26 @@ public interface Filter
|
||||||
* cursor. If both are set, the cursor will effectively perform a logical AND to combine them.
|
* cursor. If both are set, the cursor will effectively perform a logical AND to combine them.
|
||||||
* See {@link FilterBundle} for additional details.
|
* See {@link FilterBundle} for additional details.
|
||||||
*
|
*
|
||||||
* @param columnIndexSelector - provides {@link org.apache.druid.segment.column.ColumnIndexSupplier} to fetch column
|
* @param filterBundleBuilder contains {@link BitmapColumnIndex} and {@link ColumnIndexSelector}, and some additional
|
||||||
* indexes and {@link org.apache.druid.collections.bitmap.BitmapFactory} to manipulate
|
* info needed.
|
||||||
* them
|
* @param bitmapResultFactory wrapper for {@link ImmutableBitmap} operations to tie into
|
||||||
* @param bitmapResultFactory - wrapper for {@link ImmutableBitmap} operations to tie into
|
|
||||||
* {@link org.apache.druid.query.QueryMetrics} and build the output indexes
|
* {@link org.apache.druid.query.QueryMetrics} and build the output indexes
|
||||||
* @param applyRowCount - upper bound on number of rows this filter would be applied to, after removing rows
|
* @param applyRowCount upper bound on number of rows this filter would be applied to, after removing rows
|
||||||
* short-circuited by prior bundle operations. For example, given "x AND y", if "x" is
|
* short-circuited by prior bundle operations. For example, given "x AND y", if "x" is
|
||||||
* resolved using an index, then "y" will receive the number of rows that matched
|
* resolved using an index, then "y" will receive the number of rows that matched
|
||||||
* the filter "x". As another example, given "x OR y", if "x" is resolved using an
|
* the filter "x". As another example, given "x OR y", if "x" is resolved using an
|
||||||
* index, then "y" will receive the number of rows that did *not* match the filter "x".
|
* index, then "y" will receive the number of rows that did *not* match the filter "x".
|
||||||
* @param totalRowCount - total number of rows to be scanned if no indexes are applied
|
* @param totalRowCount total number of rows to be scanned if no indexes are applied
|
||||||
* @param includeUnknown - mapping for Druid native two state logic system into SQL three-state logic system. If
|
* @param includeUnknown mapping for Druid native two state logic system into SQL three-state logic system. If
|
||||||
* set to true, bitmaps returned by this method should include true bits for any rows
|
* set to true, bitmaps returned by this method should include true bits for any rows
|
||||||
* where the matching result is 'unknown', such as from the input being null valued.
|
* where the matching result is 'unknown', such as from the input being null valued.
|
||||||
* See {@link NullHandling#useThreeValueLogic()}
|
* See {@link NullHandling#useThreeValueLogic()}
|
||||||
* @return - {@link FilterBundle} containing any indexes and/or matchers that are needed to build
|
* @param <T> type of {@link BitmapResultFactory} results, {@link ImmutableBitmap} by default
|
||||||
|
* @return {@link FilterBundle} containing any indexes and/or matchers that are needed to build
|
||||||
* a cursor
|
* a cursor
|
||||||
* @param <T> - Type of {@link BitmapResultFactory} results, {@link ImmutableBitmap} by default
|
|
||||||
*/
|
*/
|
||||||
default <T> FilterBundle makeFilterBundle(
|
default <T> FilterBundle makeFilterBundle(
|
||||||
ColumnIndexSelector columnIndexSelector,
|
FilterBundle.Builder filterBundleBuilder,
|
||||||
BitmapResultFactory<T> bitmapResultFactory,
|
BitmapResultFactory<T> bitmapResultFactory,
|
||||||
int applyRowCount,
|
int applyRowCount,
|
||||||
int totalRowCount,
|
int totalRowCount,
|
||||||
|
@ -76,7 +75,7 @@ public interface Filter
|
||||||
{
|
{
|
||||||
final FilterBundle.IndexBundle indexBundle;
|
final FilterBundle.IndexBundle indexBundle;
|
||||||
final boolean needMatcher;
|
final boolean needMatcher;
|
||||||
final BitmapColumnIndex columnIndex = getBitmapColumnIndex(columnIndexSelector);
|
final BitmapColumnIndex columnIndex = filterBundleBuilder.getBitmapColumnIndex();
|
||||||
if (columnIndex != null) {
|
if (columnIndex != null) {
|
||||||
final long bitmapConstructionStartNs = System.nanoTime();
|
final long bitmapConstructionStartNs = System.nanoTime();
|
||||||
final T result = columnIndex.computeBitmapResult(
|
final T result = columnIndex.computeBitmapResult(
|
||||||
|
@ -107,7 +106,7 @@ public interface Filter
|
||||||
new FilterBundle.MatcherBundleInfo(this::toString, null, null),
|
new FilterBundle.MatcherBundleInfo(this::toString, null, null),
|
||||||
this::makeMatcher,
|
this::makeMatcher,
|
||||||
this::makeVectorMatcher,
|
this::makeVectorMatcher,
|
||||||
this.canVectorizeMatcher(columnIndexSelector)
|
this.canVectorizeMatcher(filterBundleBuilder.getColumnIndexSelector())
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
matcherBundle = null;
|
matcherBundle = null;
|
||||||
|
@ -122,7 +121,6 @@ public interface Filter
|
||||||
* examine details about the index prior to computing it, via {@link BitmapColumnIndex#getIndexCapabilities()}.
|
* examine details about the index prior to computing it, via {@link BitmapColumnIndex#getIndexCapabilities()}.
|
||||||
*
|
*
|
||||||
* @param selector Object used to create BitmapColumnIndex
|
* @param selector Object used to create BitmapColumnIndex
|
||||||
*
|
|
||||||
* @return BitmapColumnIndex that can build ImmutableBitmap of matched row numbers
|
* @return BitmapColumnIndex that can build ImmutableBitmap of matched row numbers
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
|
@ -132,7 +130,6 @@ public interface Filter
|
||||||
* Get a {@link ValueMatcher} that applies this filter to row values.
|
* Get a {@link ValueMatcher} that applies this filter to row values.
|
||||||
*
|
*
|
||||||
* @param factory Object used to create ValueMatchers
|
* @param factory Object used to create ValueMatchers
|
||||||
*
|
|
||||||
* @return ValueMatcher that applies this filter to row values.
|
* @return ValueMatcher that applies this filter to row values.
|
||||||
*/
|
*/
|
||||||
ValueMatcher makeMatcher(ColumnSelectorFactory factory);
|
ValueMatcher makeMatcher(ColumnSelectorFactory factory);
|
||||||
|
@ -141,7 +138,6 @@ public interface Filter
|
||||||
* Get a {@link VectorValueMatcher} that applies this filter to row vectors.
|
* Get a {@link VectorValueMatcher} that applies this filter to row vectors.
|
||||||
*
|
*
|
||||||
* @param factory Object used to create ValueMatchers
|
* @param factory Object used to create ValueMatchers
|
||||||
*
|
|
||||||
* @return VectorValueMatcher that applies this filter to row vectors.
|
* @return VectorValueMatcher that applies this filter to row vectors.
|
||||||
*/
|
*/
|
||||||
default VectorValueMatcher makeVectorMatcher(VectorColumnSelectorFactory factory)
|
default VectorValueMatcher makeVectorMatcher(VectorColumnSelectorFactory factory)
|
||||||
|
@ -151,6 +147,7 @@ public interface Filter
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if this filter can produce a vectorized matcher from its "makeVectorMatcher" method.
|
* Returns true if this filter can produce a vectorized matcher from its "makeVectorMatcher" method.
|
||||||
|
*
|
||||||
* @param inspector Supplies type information for the selectors this filter will match against
|
* @param inspector Supplies type information for the selectors this filter will match against
|
||||||
*/
|
*/
|
||||||
default boolean canVectorizeMatcher(ColumnInspector inspector)
|
default boolean canVectorizeMatcher(ColumnInspector inspector)
|
||||||
|
@ -176,7 +173,7 @@ public interface Filter
|
||||||
* Return a copy of this filter that is identical to the this filter except that it operates on different columns,
|
* Return a copy of this filter that is identical to the this filter except that it operates on different columns,
|
||||||
* based on a renaming map where the key is the column to be renamed in the filter, and the value is the new
|
* based on a renaming map where the key is the column to be renamed in the filter, and the value is the new
|
||||||
* column name.
|
* column name.
|
||||||
*
|
* <p>
|
||||||
* For example, if I have a filter (A = hello), and I have a renaming map (A -> B),
|
* For example, if I have a filter (A = hello), and I have a renaming map (A -> B),
|
||||||
* this should return the filter (B = hello)
|
* this should return the filter (B = hello)
|
||||||
*
|
*
|
||||||
|
|
|
@ -24,16 +24,21 @@ import com.fasterxml.jackson.annotation.JsonInclude;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
import org.apache.druid.collections.bitmap.ImmutableBitmap;
|
import org.apache.druid.collections.bitmap.ImmutableBitmap;
|
||||||
|
import org.apache.druid.query.BitmapResultFactory;
|
||||||
import org.apache.druid.query.filter.vector.VectorValueMatcher;
|
import org.apache.druid.query.filter.vector.VectorValueMatcher;
|
||||||
import org.apache.druid.segment.ColumnSelectorFactory;
|
import org.apache.druid.segment.ColumnSelectorFactory;
|
||||||
import org.apache.druid.segment.column.ColumnIndexCapabilities;
|
import org.apache.druid.segment.column.ColumnIndexCapabilities;
|
||||||
import org.apache.druid.segment.column.SimpleColumnIndexCapabilities;
|
import org.apache.druid.segment.column.SimpleColumnIndexCapabilities;
|
||||||
import org.apache.druid.segment.data.Offset;
|
import org.apache.druid.segment.data.Offset;
|
||||||
import org.apache.druid.segment.filter.FalseFilter;
|
import org.apache.druid.segment.filter.FalseFilter;
|
||||||
|
import org.apache.druid.segment.index.BitmapColumnIndex;
|
||||||
import org.apache.druid.segment.vector.ReadableVectorOffset;
|
import org.apache.druid.segment.vector.ReadableVectorOffset;
|
||||||
import org.apache.druid.segment.vector.VectorColumnSelectorFactory;
|
import org.apache.druid.segment.vector.VectorColumnSelectorFactory;
|
||||||
|
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Comparator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
|
@ -57,6 +62,21 @@ import java.util.regex.Pattern;
|
||||||
*/
|
*/
|
||||||
public class FilterBundle
|
public class FilterBundle
|
||||||
{
|
{
|
||||||
|
@Nullable
|
||||||
|
private final IndexBundle indexBundle;
|
||||||
|
@Nullable
|
||||||
|
private final MatcherBundle matcherBundle;
|
||||||
|
|
||||||
|
public FilterBundle(@Nullable IndexBundle index, @Nullable MatcherBundle matcherBundle)
|
||||||
|
{
|
||||||
|
Preconditions.checkArgument(
|
||||||
|
index != null || matcherBundle != null,
|
||||||
|
"At least one of index or matcher must be not null"
|
||||||
|
);
|
||||||
|
this.indexBundle = index;
|
||||||
|
this.matcherBundle = matcherBundle;
|
||||||
|
}
|
||||||
|
|
||||||
public static FilterBundle allFalse(long constructionTime, ImmutableBitmap emptyBitmap)
|
public static FilterBundle allFalse(long constructionTime, ImmutableBitmap emptyBitmap)
|
||||||
{
|
{
|
||||||
return new FilterBundle(
|
return new FilterBundle(
|
||||||
|
@ -69,25 +89,6 @@ public class FilterBundle
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
|
||||||
private final IndexBundle indexBundle;
|
|
||||||
@Nullable
|
|
||||||
private final MatcherBundle matcherBundle;
|
|
||||||
|
|
||||||
public FilterBundle(
|
|
||||||
@Nullable IndexBundle index,
|
|
||||||
@Nullable MatcherBundle matcherBundle
|
|
||||||
)
|
|
||||||
{
|
|
||||||
Preconditions.checkArgument(
|
|
||||||
index != null || matcherBundle != null,
|
|
||||||
"At least one of index or matcher must be not null"
|
|
||||||
);
|
|
||||||
this.indexBundle = index;
|
|
||||||
this.matcherBundle = matcherBundle;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
public IndexBundle getIndex()
|
public IndexBundle getIndex()
|
||||||
{
|
{
|
||||||
|
@ -151,6 +152,95 @@ public class FilterBundle
|
||||||
boolean canVectorize();
|
boolean canVectorize();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wraps info needed to build a {@link FilterBundle}, and provides an estimated compute cost for
|
||||||
|
* {@link BitmapColumnIndex#computeBitmapResult}.
|
||||||
|
*/
|
||||||
|
public static class Builder
|
||||||
|
{
|
||||||
|
private final Filter filter;
|
||||||
|
private final ColumnIndexSelector columnIndexSelector;
|
||||||
|
@Nullable
|
||||||
|
private final BitmapColumnIndex bitmapColumnIndex;
|
||||||
|
private final List<FilterBundle.Builder> childBuilders;
|
||||||
|
private final int estimatedIndexComputeCost;
|
||||||
|
|
||||||
|
public Builder(Filter filter, ColumnIndexSelector columnIndexSelector, boolean cursorAutoArrangeFilters)
|
||||||
|
{
|
||||||
|
this.filter = filter;
|
||||||
|
this.columnIndexSelector = columnIndexSelector;
|
||||||
|
this.bitmapColumnIndex = filter.getBitmapColumnIndex(columnIndexSelector);
|
||||||
|
// Construct Builder instances for all child filters recursively.
|
||||||
|
if (filter instanceof BooleanFilter) {
|
||||||
|
Collection<Filter> childFilters = ((BooleanFilter) filter).getFilters();
|
||||||
|
this.childBuilders = new ArrayList<>(childFilters.size());
|
||||||
|
for (Filter childFilter : childFilters) {
|
||||||
|
this.childBuilders.add(new FilterBundle.Builder(childFilter, columnIndexSelector, cursorAutoArrangeFilters));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.childBuilders = new ArrayList<>(0);
|
||||||
|
}
|
||||||
|
if (cursorAutoArrangeFilters) {
|
||||||
|
// Sort child builders by cost in ASCENDING order, should be stable by default.
|
||||||
|
this.childBuilders.sort(Comparator.comparingInt(FilterBundle.Builder::getEstimatedIndexComputeCost));
|
||||||
|
this.estimatedIndexComputeCost = calculateEstimatedIndexComputeCost();
|
||||||
|
} else {
|
||||||
|
this.estimatedIndexComputeCost = Integer.MAX_VALUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private int calculateEstimatedIndexComputeCost()
|
||||||
|
{
|
||||||
|
if (this.bitmapColumnIndex == null) {
|
||||||
|
return Integer.MAX_VALUE;
|
||||||
|
}
|
||||||
|
int cost = this.bitmapColumnIndex.estimatedComputeCost();
|
||||||
|
if (cost == Integer.MAX_VALUE) {
|
||||||
|
return Integer.MAX_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (FilterBundle.Builder childBuilder : childBuilders) {
|
||||||
|
int childCost = childBuilder.getEstimatedIndexComputeCost();
|
||||||
|
if (childCost >= Integer.MAX_VALUE - cost) {
|
||||||
|
return Integer.MAX_VALUE;
|
||||||
|
}
|
||||||
|
cost += childCost;
|
||||||
|
}
|
||||||
|
return cost;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ColumnIndexSelector getColumnIndexSelector()
|
||||||
|
{
|
||||||
|
return columnIndexSelector;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
public BitmapColumnIndex getBitmapColumnIndex()
|
||||||
|
{
|
||||||
|
return bitmapColumnIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<FilterBundle.Builder> getChildBuilders()
|
||||||
|
{
|
||||||
|
return childBuilders;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getEstimatedIndexComputeCost()
|
||||||
|
{
|
||||||
|
return estimatedIndexComputeCost;
|
||||||
|
}
|
||||||
|
|
||||||
|
public <T> FilterBundle build(
|
||||||
|
BitmapResultFactory<T> bitmapResultFactory,
|
||||||
|
int applyRowCount,
|
||||||
|
int totalRowCount,
|
||||||
|
boolean includeUnknown
|
||||||
|
)
|
||||||
|
{
|
||||||
|
return filter.makeFilterBundle(this, bitmapResultFactory, applyRowCount, totalRowCount, includeUnknown);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static class SimpleIndexBundle implements IndexBundle
|
public static class SimpleIndexBundle implements IndexBundle
|
||||||
{
|
{
|
||||||
private final IndexBundleInfo info;
|
private final IndexBundleInfo info;
|
||||||
|
@ -211,11 +301,7 @@ public class FilterBundle
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ValueMatcher valueMatcher(
|
public ValueMatcher valueMatcher(ColumnSelectorFactory selectorFactory, Offset baseOffset, boolean descending)
|
||||||
ColumnSelectorFactory selectorFactory,
|
|
||||||
Offset baseOffset,
|
|
||||||
boolean descending
|
|
||||||
)
|
|
||||||
{
|
{
|
||||||
return matcherFn.apply(selectorFactory);
|
return matcherFn.apply(selectorFactory);
|
||||||
}
|
}
|
||||||
|
@ -339,8 +425,7 @@ public class FilterBundle
|
||||||
*/
|
*/
|
||||||
public String describe()
|
public String describe()
|
||||||
{
|
{
|
||||||
final StringBuilder sb = new StringBuilder()
|
final StringBuilder sb = new StringBuilder().append("index: ")
|
||||||
.append("index: ")
|
|
||||||
.append(filter.get())
|
.append(filter.get())
|
||||||
.append(" (selectionSize = ")
|
.append(" (selectionSize = ")
|
||||||
.append(selectionSize)
|
.append(selectionSize)
|
||||||
|
@ -358,23 +443,26 @@ public class FilterBundle
|
||||||
@Override
|
@Override
|
||||||
public String toString()
|
public String toString()
|
||||||
{
|
{
|
||||||
return "{" +
|
return "{"
|
||||||
"filter=\"" + filter.get() + '\"' +
|
+ "filter=\""
|
||||||
", selectionSize=" + selectionSize +
|
+ filter.get()
|
||||||
", buildTime=" + TimeUnit.NANOSECONDS.toMicros(buildTimeNs) + "μs" +
|
+ '\"'
|
||||||
(indexes != null ? ", indexes=" + indexes : "") +
|
+ ", selectionSize="
|
||||||
'}';
|
+ selectionSize
|
||||||
|
+ ", buildTime="
|
||||||
|
+ TimeUnit.NANOSECONDS.toMicros(buildTimeNs)
|
||||||
|
+ "μs"
|
||||||
|
+ (indexes != null ? ", indexes=" + indexes : "")
|
||||||
|
+ '}';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class MatcherBundleInfo
|
public static class MatcherBundleInfo
|
||||||
{
|
{
|
||||||
private static final Pattern PATTERN_LINE_START = Pattern.compile("(?m)^");
|
private static final Pattern PATTERN_LINE_START = Pattern.compile("(?m)^");
|
||||||
|
|
||||||
private final Supplier<String> filter;
|
|
||||||
@Nullable
|
@Nullable
|
||||||
final List<MatcherBundleInfo> matchers;
|
final List<MatcherBundleInfo> matchers;
|
||||||
|
private final Supplier<String> filter;
|
||||||
@Nullable
|
@Nullable
|
||||||
private final IndexBundleInfo partialIndex;
|
private final IndexBundleInfo partialIndex;
|
||||||
|
|
||||||
|
@ -415,10 +503,7 @@ public class FilterBundle
|
||||||
*/
|
*/
|
||||||
public String describe()
|
public String describe()
|
||||||
{
|
{
|
||||||
final StringBuilder sb = new StringBuilder()
|
final StringBuilder sb = new StringBuilder().append("matcher: ").append(filter.get()).append("\n");
|
||||||
.append("matcher: ")
|
|
||||||
.append(filter.get())
|
|
||||||
.append("\n");
|
|
||||||
|
|
||||||
if (partialIndex != null) {
|
if (partialIndex != null) {
|
||||||
sb.append(" with partial ")
|
sb.append(" with partial ")
|
||||||
|
@ -437,11 +522,13 @@ public class FilterBundle
|
||||||
@Override
|
@Override
|
||||||
public String toString()
|
public String toString()
|
||||||
{
|
{
|
||||||
return "{" +
|
return "{"
|
||||||
"filter=\"" + filter.get() + '\"' +
|
+ "filter=\""
|
||||||
(partialIndex != null ? ", partialIndex=" + partialIndex : "") +
|
+ filter.get()
|
||||||
(matchers != null ? ", matchers=" + matchers : "") +
|
+ '\"'
|
||||||
'}';
|
+ (partialIndex != null ? ", partialIndex=" + partialIndex : "")
|
||||||
|
+ (matchers != null ? ", matchers=" + matchers : "")
|
||||||
|
+ '}';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@ import java.util.Objects;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class provides a mechanism to influence whether or not indexes are used for a {@link Filter} during processing
|
* This class provides a mechanism to influence whether or not indexes are used for a {@link Filter} during processing
|
||||||
* by {@link Filter#makeFilterBundle(ColumnIndexSelector, BitmapResultFactory, int, int, boolean)}
|
* by {@link Filter#makeFilterBundle(FilterBundle.Builder, BitmapResultFactory, int, int, boolean)}
|
||||||
* (i.e. will a {@link Filter} be a "pre" filter in which we union indexes for all values that match the filter to
|
* (i.e. will a {@link Filter} be a "pre" filter in which we union indexes for all values that match the filter to
|
||||||
* create a {@link org.apache.druid.segment.BitmapOffset}/{@link org.apache.druid.segment.vector.BitmapVectorOffset},
|
* create a {@link org.apache.druid.segment.BitmapOffset}/{@link org.apache.druid.segment.vector.BitmapVectorOffset},
|
||||||
* or will it be used as a "post" filter and evaluated while scanning row values from the
|
* or will it be used as a "post" filter and evaluated while scanning row values from the
|
||||||
|
|
|
@ -34,6 +34,7 @@ import org.apache.druid.query.Order;
|
||||||
import org.apache.druid.query.OrderBy;
|
import org.apache.druid.query.OrderBy;
|
||||||
import org.apache.druid.query.Query;
|
import org.apache.druid.query.Query;
|
||||||
import org.apache.druid.query.QueryContext;
|
import org.apache.druid.query.QueryContext;
|
||||||
|
import org.apache.druid.query.QueryContexts;
|
||||||
import org.apache.druid.query.QueryMetrics;
|
import org.apache.druid.query.QueryMetrics;
|
||||||
import org.apache.druid.query.aggregation.AggregatorFactory;
|
import org.apache.druid.query.aggregation.AggregatorFactory;
|
||||||
import org.apache.druid.query.filter.Filter;
|
import org.apache.druid.query.filter.Filter;
|
||||||
|
@ -112,6 +113,7 @@ public class QueryableIndexCursorHolder implements CursorHolder
|
||||||
Cursors.getTimeOrdering(ordering),
|
Cursors.getTimeOrdering(ordering),
|
||||||
interval,
|
interval,
|
||||||
filter,
|
filter,
|
||||||
|
cursorBuildSpec.getQueryContext().getBoolean(QueryContexts.CURSOR_AUTO_ARRANGE_FILTERS, false),
|
||||||
metrics
|
metrics
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@ -346,7 +348,6 @@ public class QueryableIndexCursorHolder implements CursorHolder
|
||||||
* @param timestamp the timestamp to search for
|
* @param timestamp the timestamp to search for
|
||||||
* @param startIndex first index to search, inclusive
|
* @param startIndex first index to search, inclusive
|
||||||
* @param endIndex last index to search, exclusive
|
* @param endIndex last index to search, exclusive
|
||||||
*
|
|
||||||
* @return first index that has a timestamp equal to, or greater, than "timestamp"
|
* @return first index that has a timestamp equal to, or greater, than "timestamp"
|
||||||
*/
|
*/
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
|
@ -665,6 +666,7 @@ public class QueryableIndexCursorHolder implements CursorHolder
|
||||||
Order timeOrder,
|
Order timeOrder,
|
||||||
Interval interval,
|
Interval interval,
|
||||||
@Nullable Filter filter,
|
@Nullable Filter filter,
|
||||||
|
boolean cursorAutoArrangeFilters,
|
||||||
@Nullable QueryMetrics<? extends Query<?>> metrics
|
@Nullable QueryMetrics<? extends Query<?>> metrics
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
@ -688,6 +690,7 @@ public class QueryableIndexCursorHolder implements CursorHolder
|
||||||
interval,
|
interval,
|
||||||
filter
|
filter
|
||||||
),
|
),
|
||||||
|
cursorAutoArrangeFilters,
|
||||||
bitmapIndexSelector,
|
bitmapIndexSelector,
|
||||||
numRows,
|
numRows,
|
||||||
metrics
|
metrics
|
||||||
|
@ -708,13 +711,14 @@ public class QueryableIndexCursorHolder implements CursorHolder
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a {@link FilterBundle} for a cursor hold instance.
|
* Create a {@link FilterBundle} for a cursor hold instance.
|
||||||
*
|
* <p>
|
||||||
* The provided filter must include the query-level interface if needed. To compute this properly, use
|
* The provided filter must include the query-level interface if needed. To compute this properly, use
|
||||||
* {@link #computeFilterWithIntervalIfNeeded}.
|
* {@link #computeFilterWithIntervalIfNeeded}.
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
private static FilterBundle makeFilterBundle(
|
private static FilterBundle makeFilterBundle(
|
||||||
@Nullable final Filter filter,
|
@Nullable final Filter filter,
|
||||||
|
boolean cursorAutoArrangeFilters,
|
||||||
final ColumnSelectorColumnIndexSelector bitmapIndexSelector,
|
final ColumnSelectorColumnIndexSelector bitmapIndexSelector,
|
||||||
final int numRows,
|
final int numRows,
|
||||||
@Nullable final QueryMetrics<?> metrics
|
@Nullable final QueryMetrics<?> metrics
|
||||||
|
@ -732,8 +736,11 @@ public class QueryableIndexCursorHolder implements CursorHolder
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
final long bitmapConstructionStartNs = System.nanoTime();
|
final long bitmapConstructionStartNs = System.nanoTime();
|
||||||
final FilterBundle filterBundle = filter.makeFilterBundle(
|
final FilterBundle filterBundle = new FilterBundle.Builder(
|
||||||
|
filter,
|
||||||
bitmapIndexSelector,
|
bitmapIndexSelector,
|
||||||
|
cursorAutoArrangeFilters
|
||||||
|
).build(
|
||||||
bitmapResultFactory,
|
bitmapResultFactory,
|
||||||
numRows,
|
numRows,
|
||||||
numRows,
|
numRows,
|
||||||
|
|
|
@ -72,9 +72,68 @@ public class AndFilter implements BooleanFilter
|
||||||
this(new LinkedHashSet<>(filters));
|
this(new LinkedHashSet<>(filters));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static ValueMatcher makeMatcher(final ValueMatcher[] baseMatchers)
|
||||||
|
{
|
||||||
|
Preconditions.checkState(baseMatchers.length > 0);
|
||||||
|
if (baseMatchers.length == 1) {
|
||||||
|
return baseMatchers[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
return new ValueMatcher()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public boolean matches(boolean includeUnknown)
|
||||||
|
{
|
||||||
|
for (ValueMatcher matcher : baseMatchers) {
|
||||||
|
if (!matcher.matches(includeUnknown)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void inspectRuntimeShape(RuntimeShapeInspector inspector)
|
||||||
|
{
|
||||||
|
inspector.visit("firstBaseMatcher", baseMatchers[0]);
|
||||||
|
inspector.visit("secondBaseMatcher", baseMatchers[1]);
|
||||||
|
// Don't inspect the 3rd and all consequent baseMatchers, cut runtime shape combinations at this point.
|
||||||
|
// Anyway if the filter is so complex, Hotspot won't inline all calls because of the inline limit.
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static VectorValueMatcher makeVectorMatcher(final VectorValueMatcher[] baseMatchers)
|
||||||
|
{
|
||||||
|
Preconditions.checkState(baseMatchers.length > 0);
|
||||||
|
if (baseMatchers.length == 1) {
|
||||||
|
return baseMatchers[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
return new BaseVectorValueMatcher(baseMatchers[0])
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public ReadableVectorMatch match(final ReadableVectorMatch mask, boolean includeUnknown)
|
||||||
|
{
|
||||||
|
ReadableVectorMatch match = mask;
|
||||||
|
|
||||||
|
for (VectorValueMatcher matcher : baseMatchers) {
|
||||||
|
if (match.isAllFalse()) {
|
||||||
|
// Short-circuit if the entire vector is false.
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
match = matcher.match(match, includeUnknown);
|
||||||
|
}
|
||||||
|
|
||||||
|
assert match.isValid(mask);
|
||||||
|
return match;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public <T> FilterBundle makeFilterBundle(
|
public <T> FilterBundle makeFilterBundle(
|
||||||
ColumnIndexSelector columnIndexSelector,
|
FilterBundle.Builder filterBundleBuilder,
|
||||||
BitmapResultFactory<T> bitmapResultFactory,
|
BitmapResultFactory<T> bitmapResultFactory,
|
||||||
int applyRowCount,
|
int applyRowCount,
|
||||||
int totalRowCount,
|
int totalRowCount,
|
||||||
|
@ -97,20 +156,21 @@ public class AndFilter implements BooleanFilter
|
||||||
// a nested AND filter might also partition itself into indexes and bundles, and since it is part of a logical AND
|
// a nested AND filter might also partition itself into indexes and bundles, and since it is part of a logical AND
|
||||||
// operation, this is valid (and even preferable).
|
// operation, this is valid (and even preferable).
|
||||||
final long bitmapConstructionStartNs = System.nanoTime();
|
final long bitmapConstructionStartNs = System.nanoTime();
|
||||||
for (Filter subfilter : filters) {
|
for (FilterBundle.Builder subFilterBundleBuilder : filterBundleBuilder.getChildBuilders()) {
|
||||||
final FilterBundle subBundle = subfilter.makeFilterBundle(
|
final FilterBundle subBundle = subFilterBundleBuilder.build(
|
||||||
columnIndexSelector,
|
|
||||||
bitmapResultFactory,
|
bitmapResultFactory,
|
||||||
Math.min(applyRowCount, indexIntersectionSize),
|
Math.min(applyRowCount, indexIntersectionSize),
|
||||||
totalRowCount,
|
totalRowCount,
|
||||||
includeUnknown
|
includeUnknown
|
||||||
);
|
);
|
||||||
if (subBundle.getIndex() != null) {
|
if (subBundle.hasIndex()) {
|
||||||
if (subBundle.getIndex().getBitmap().isEmpty()) {
|
if (subBundle.getIndex().getBitmap().isEmpty()) {
|
||||||
// if nothing matches for any sub filter, short-circuit, because nothing can possibly match
|
// if nothing matches for any sub filter, short-circuit, because nothing can possibly match
|
||||||
return FilterBundle.allFalse(
|
return FilterBundle.allFalse(
|
||||||
System.nanoTime() - bitmapConstructionStartNs,
|
System.nanoTime() - bitmapConstructionStartNs,
|
||||||
columnIndexSelector.getBitmapFactory().makeEmptyImmutableBitmap()
|
subFilterBundleBuilder.getColumnIndexSelector()
|
||||||
|
.getBitmapFactory()
|
||||||
|
.makeEmptyImmutableBitmap()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
merged = merged.merge(subBundle.getIndex().getIndexCapabilities());
|
merged = merged.merge(subBundle.getIndex().getIndexCapabilities());
|
||||||
|
@ -122,7 +182,7 @@ public class AndFilter implements BooleanFilter
|
||||||
}
|
}
|
||||||
indexIntersectionSize = index.size();
|
indexIntersectionSize = index.size();
|
||||||
}
|
}
|
||||||
if (subBundle.getMatcherBundle() != null) {
|
if (subBundle.hasMatcher()) {
|
||||||
matcherBundles.add(subBundle.getMatcherBundle());
|
matcherBundles.add(subBundle.getMatcherBundle());
|
||||||
matcherBundleInfos.add(subBundle.getMatcherBundle().getMatcherInfo());
|
matcherBundleInfos.add(subBundle.getMatcherBundle().getMatcherInfo());
|
||||||
}
|
}
|
||||||
|
@ -131,11 +191,7 @@ public class AndFilter implements BooleanFilter
|
||||||
final FilterBundle.IndexBundle indexBundle;
|
final FilterBundle.IndexBundle indexBundle;
|
||||||
if (index != null) {
|
if (index != null) {
|
||||||
if (indexBundleInfos.size() == 1) {
|
if (indexBundleInfos.size() == 1) {
|
||||||
indexBundle = new FilterBundle.SimpleIndexBundle(
|
indexBundle = new FilterBundle.SimpleIndexBundle(indexBundleInfos.get(0), index, merged);
|
||||||
indexBundleInfos.get(0),
|
|
||||||
index,
|
|
||||||
merged
|
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
indexBundle = new FilterBundle.SimpleIndexBundle(
|
indexBundle = new FilterBundle.SimpleIndexBundle(
|
||||||
new FilterBundle.IndexBundleInfo(
|
new FilterBundle.IndexBundleInfo(
|
||||||
|
@ -162,11 +218,7 @@ public class AndFilter implements BooleanFilter
|
||||||
if (matcherBundles.size() == 1) {
|
if (matcherBundles.size() == 1) {
|
||||||
return matcherBundleInfos.get(0);
|
return matcherBundleInfos.get(0);
|
||||||
}
|
}
|
||||||
return new FilterBundle.MatcherBundleInfo(
|
return new FilterBundle.MatcherBundleInfo(() -> "AND", null, matcherBundleInfos);
|
||||||
() -> "AND",
|
|
||||||
null,
|
|
||||||
matcherBundleInfos
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -180,7 +232,10 @@ public class AndFilter implements BooleanFilter
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public VectorValueMatcher vectorMatcher(VectorColumnSelectorFactory selectorFactory, ReadableVectorOffset baseOffset)
|
public VectorValueMatcher vectorMatcher(
|
||||||
|
VectorColumnSelectorFactory selectorFactory,
|
||||||
|
ReadableVectorOffset baseOffset
|
||||||
|
)
|
||||||
{
|
{
|
||||||
final VectorValueMatcher[] vectorMatchers = new VectorValueMatcher[matcherBundles.size()];
|
final VectorValueMatcher[] vectorMatchers = new VectorValueMatcher[matcherBundles.size()];
|
||||||
for (int i = 0; i < matcherBundles.size(); i++) {
|
for (int i = 0; i < matcherBundles.size(); i++) {
|
||||||
|
@ -204,10 +259,7 @@ public class AndFilter implements BooleanFilter
|
||||||
matcherBundle = null;
|
matcherBundle = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return new FilterBundle(
|
return new FilterBundle(indexBundle, matcherBundle);
|
||||||
indexBundle,
|
|
||||||
matcherBundle
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
|
@ -239,6 +291,13 @@ public class AndFilter implements BooleanFilter
|
||||||
return finalMerged;
|
return finalMerged;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int estimatedComputeCost()
|
||||||
|
{
|
||||||
|
// There's no additional cost on AND filter, cost in child filters would be summed.
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public <T> T computeBitmapResult(BitmapResultFactory<T> bitmapResultFactory, boolean includeUnknown)
|
public <T> T computeBitmapResult(BitmapResultFactory<T> bitmapResultFactory, boolean includeUnknown)
|
||||||
{
|
{
|
||||||
|
@ -350,65 +409,6 @@ public class AndFilter implements BooleanFilter
|
||||||
return StringUtils.format("(%s)", AND_JOINER.join(filters));
|
return StringUtils.format("(%s)", AND_JOINER.join(filters));
|
||||||
}
|
}
|
||||||
|
|
||||||
public static ValueMatcher makeMatcher(final ValueMatcher[] baseMatchers)
|
|
||||||
{
|
|
||||||
Preconditions.checkState(baseMatchers.length > 0);
|
|
||||||
if (baseMatchers.length == 1) {
|
|
||||||
return baseMatchers[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
return new ValueMatcher()
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public boolean matches(boolean includeUnknown)
|
|
||||||
{
|
|
||||||
for (ValueMatcher matcher : baseMatchers) {
|
|
||||||
if (!matcher.matches(includeUnknown)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void inspectRuntimeShape(RuntimeShapeInspector inspector)
|
|
||||||
{
|
|
||||||
inspector.visit("firstBaseMatcher", baseMatchers[0]);
|
|
||||||
inspector.visit("secondBaseMatcher", baseMatchers[1]);
|
|
||||||
// Don't inspect the 3rd and all consequent baseMatchers, cut runtime shape combinations at this point.
|
|
||||||
// Anyway if the filter is so complex, Hotspot won't inline all calls because of the inline limit.
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public static VectorValueMatcher makeVectorMatcher(final VectorValueMatcher[] baseMatchers)
|
|
||||||
{
|
|
||||||
Preconditions.checkState(baseMatchers.length > 0);
|
|
||||||
if (baseMatchers.length == 1) {
|
|
||||||
return baseMatchers[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
return new BaseVectorValueMatcher(baseMatchers[0])
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public ReadableVectorMatch match(final ReadableVectorMatch mask, boolean includeUnknown)
|
|
||||||
{
|
|
||||||
ReadableVectorMatch match = mask;
|
|
||||||
|
|
||||||
for (VectorValueMatcher matcher : baseMatchers) {
|
|
||||||
if (match.isAllFalse()) {
|
|
||||||
// Short-circuit if the entire vector is false.
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
match = matcher.match(match, includeUnknown);
|
|
||||||
}
|
|
||||||
|
|
||||||
assert match.isValid(mask);
|
|
||||||
return match;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o)
|
public boolean equals(Object o)
|
||||||
{
|
{
|
||||||
|
|
|
@ -78,330 +78,6 @@ public class OrFilter implements BooleanFilter
|
||||||
this(new LinkedHashSet<>(filters));
|
this(new LinkedHashSet<>(filters));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public <T> FilterBundle makeFilterBundle(
|
|
||||||
ColumnIndexSelector columnIndexSelector,
|
|
||||||
BitmapResultFactory<T> bitmapResultFactory,
|
|
||||||
int applyRowCount,
|
|
||||||
int totalRowCount,
|
|
||||||
boolean includeUnknown
|
|
||||||
)
|
|
||||||
{
|
|
||||||
// for OR filters, we have a few possible outcomes:
|
|
||||||
// 1 - all clauses are index only bundles. in this case we union the bitmaps together and make an index only bundle
|
|
||||||
// 2 - some clauses support indexes. in this case, we union the bitmaps of any index only bundles together to form a
|
|
||||||
// partial index which is constructed into a matcher bundle with convertIndexToMatcherBundle. We translate any
|
|
||||||
// index AND matcher bundles into a matcher only bundle with convertBundleToMatcherOnlyBundle. Finally, we
|
|
||||||
// combine these with the remaining matcher only bundles to with makeMatcher/makeVectorMatcher to make a matcher
|
|
||||||
// only bundle
|
|
||||||
// 3 - no clauses support indexes. in this case, we make a matcher only bundle using makeMatcher/makeVectorMatcher
|
|
||||||
|
|
||||||
final List<FilterBundle.IndexBundle> indexOnlyBundles = new ArrayList<>();
|
|
||||||
final List<FilterBundle.IndexBundleInfo> indexOnlyBundlesInfo = new ArrayList<>();
|
|
||||||
final List<FilterBundle.MatcherBundle> partialIndexBundles = new ArrayList<>();
|
|
||||||
final List<FilterBundle.MatcherBundle> matcherOnlyBundles = new ArrayList<>();
|
|
||||||
|
|
||||||
int indexUnionSize = 0;
|
|
||||||
ImmutableBitmap index = null;
|
|
||||||
ColumnIndexCapabilities merged = new SimpleColumnIndexCapabilities(true, true);
|
|
||||||
int emptyCount = 0;
|
|
||||||
|
|
||||||
final long bitmapConstructionStartNs = System.nanoTime();
|
|
||||||
|
|
||||||
for (Filter subfilter : filters) {
|
|
||||||
final FilterBundle bundle = subfilter.makeFilterBundle(
|
|
||||||
columnIndexSelector,
|
|
||||||
bitmapResultFactory,
|
|
||||||
Math.min(applyRowCount, totalRowCount - indexUnionSize),
|
|
||||||
totalRowCount,
|
|
||||||
includeUnknown
|
|
||||||
);
|
|
||||||
if (bundle.hasIndex()) {
|
|
||||||
final ImmutableBitmap bundleIndex = bundle.getIndex().getBitmap();
|
|
||||||
if (bundleIndex.isEmpty()) {
|
|
||||||
// we leave any indexes which are empty out of index, indexOnlyBundles, and partialIndexBundles
|
|
||||||
// even though we skip them, we still keep track of them to check for the case when we can build the OR into
|
|
||||||
// an index only bundle. We can count index and matcher bundles here too because the AND operation means that
|
|
||||||
// an empty index means the matcher can be skipped
|
|
||||||
emptyCount++;
|
|
||||||
} else {
|
|
||||||
if (bundle.hasMatcher()) {
|
|
||||||
// index and matcher bundles must be handled separately, they will need to be a single value matcher built
|
|
||||||
// by doing an AND operation between the index and the value matcher
|
|
||||||
// (a bundle is basically an AND operation between the index and matcher if the matcher is present)
|
|
||||||
partialIndexBundles.add(convertBundleToMatcherOnlyBundle(bundle, bundleIndex));
|
|
||||||
} else {
|
|
||||||
indexOnlyBundles.add(bundle.getIndex());
|
|
||||||
indexOnlyBundlesInfo.add(bundle.getIndex().getIndexInfo());
|
|
||||||
merged.merge(bundle.getIndex().getIndexCapabilities());
|
|
||||||
// union index only bitmaps together; if all sub-filters are 'index only' bundles we will make an index only
|
|
||||||
// bundle ourselves, else we will use this index as a single value matcher
|
|
||||||
if (index == null) {
|
|
||||||
index = bundle.getIndex().getBitmap();
|
|
||||||
} else {
|
|
||||||
index = index.union(bundle.getIndex().getBitmap());
|
|
||||||
}
|
|
||||||
indexUnionSize = index.size();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
matcherOnlyBundles.add(bundle.getMatcherBundle());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
final long totalBitmapConstructTimeNs = System.nanoTime() - bitmapConstructionStartNs;
|
|
||||||
|
|
||||||
|
|
||||||
// if all the filters are 'index only', we can make an index only bundle
|
|
||||||
if (indexOnlyBundles.size() + emptyCount == filters.size()) {
|
|
||||||
if (index == null || index.isEmpty()) {
|
|
||||||
return FilterBundle.allFalse(
|
|
||||||
totalBitmapConstructTimeNs,
|
|
||||||
columnIndexSelector.getBitmapFactory().makeEmptyImmutableBitmap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (indexOnlyBundles.size() == 1) {
|
|
||||||
return new FilterBundle(
|
|
||||||
indexOnlyBundles.get(0),
|
|
||||||
null
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return new FilterBundle(
|
|
||||||
new FilterBundle.SimpleIndexBundle(
|
|
||||||
new FilterBundle.IndexBundleInfo(
|
|
||||||
() -> "OR",
|
|
||||||
applyRowCount,
|
|
||||||
totalBitmapConstructTimeNs,
|
|
||||||
indexOnlyBundlesInfo
|
|
||||||
),
|
|
||||||
index,
|
|
||||||
merged
|
|
||||||
),
|
|
||||||
null
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// if not the index only outcome, we build a matcher only bundle from all the matchers
|
|
||||||
final int estimatedSize = (indexOnlyBundles.isEmpty() ? 0 : 1)
|
|
||||||
+ partialIndexBundles.size()
|
|
||||||
+ matcherOnlyBundles.size();
|
|
||||||
final List<FilterBundle.MatcherBundle> allMatcherBundles = Lists.newArrayListWithCapacity(estimatedSize);
|
|
||||||
final List<FilterBundle.MatcherBundleInfo> allMatcherBundlesInfo = Lists.newArrayListWithCapacity(estimatedSize);
|
|
||||||
if (!indexOnlyBundles.isEmpty()) {
|
|
||||||
// translate the indexOnly bundles into a single matcher
|
|
||||||
final FilterBundle.MatcherBundle matcherBundle = convertIndexToMatcherBundle(
|
|
||||||
applyRowCount,
|
|
||||||
indexOnlyBundles,
|
|
||||||
indexOnlyBundlesInfo,
|
|
||||||
totalBitmapConstructTimeNs,
|
|
||||||
index
|
|
||||||
);
|
|
||||||
allMatcherBundles.add(matcherBundle);
|
|
||||||
allMatcherBundlesInfo.add(matcherBundle.getMatcherInfo());
|
|
||||||
}
|
|
||||||
for (FilterBundle.MatcherBundle bundle : partialIndexBundles) {
|
|
||||||
allMatcherBundles.add(bundle);
|
|
||||||
allMatcherBundlesInfo.add(bundle.getMatcherInfo());
|
|
||||||
}
|
|
||||||
for (FilterBundle.MatcherBundle bundle : matcherOnlyBundles) {
|
|
||||||
allMatcherBundles.add(bundle);
|
|
||||||
allMatcherBundlesInfo.add(bundle.getMatcherInfo());
|
|
||||||
}
|
|
||||||
|
|
||||||
return new FilterBundle(
|
|
||||||
null,
|
|
||||||
new FilterBundle.MatcherBundle()
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public FilterBundle.MatcherBundleInfo getMatcherInfo()
|
|
||||||
{
|
|
||||||
return new FilterBundle.MatcherBundleInfo(
|
|
||||||
() -> "OR",
|
|
||||||
null,
|
|
||||||
allMatcherBundlesInfo
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public ValueMatcher valueMatcher(ColumnSelectorFactory selectorFactory, Offset baseOffset, boolean descending)
|
|
||||||
{
|
|
||||||
final ValueMatcher[] matchers = new ValueMatcher[allMatcherBundles.size()];
|
|
||||||
for (int i = 0; i < allMatcherBundles.size(); i++) {
|
|
||||||
matchers[i] = allMatcherBundles.get(i).valueMatcher(selectorFactory, baseOffset, descending);
|
|
||||||
}
|
|
||||||
return makeMatcher(matchers);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public VectorValueMatcher vectorMatcher(
|
|
||||||
VectorColumnSelectorFactory selectorFactory,
|
|
||||||
ReadableVectorOffset baseOffset
|
|
||||||
)
|
|
||||||
{
|
|
||||||
final VectorValueMatcher[] matchers = new VectorValueMatcher[allMatcherBundles.size()];
|
|
||||||
for (int i = 0; i < allMatcherBundles.size(); i++) {
|
|
||||||
matchers[i] = allMatcherBundles.get(i).vectorMatcher(selectorFactory, baseOffset);
|
|
||||||
}
|
|
||||||
return makeVectorMatcher(matchers);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean canVectorize()
|
|
||||||
{
|
|
||||||
for (FilterBundle.MatcherBundle bundle : allMatcherBundles) {
|
|
||||||
if (!bundle.canVectorize()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
@Override
|
|
||||||
public BitmapColumnIndex getBitmapColumnIndex(ColumnIndexSelector selector)
|
|
||||||
{
|
|
||||||
if (filters.size() == 1) {
|
|
||||||
return Iterables.getOnlyElement(filters).getBitmapColumnIndex(selector);
|
|
||||||
}
|
|
||||||
|
|
||||||
List<BitmapColumnIndex> bitmapColumnIndices = new ArrayList<>(filters.size());
|
|
||||||
ColumnIndexCapabilities merged = new SimpleColumnIndexCapabilities(true, true);
|
|
||||||
for (Filter filter : filters) {
|
|
||||||
BitmapColumnIndex index = filter.getBitmapColumnIndex(selector);
|
|
||||||
if (index == null) {
|
|
||||||
// all or nothing
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
merged = merged.merge(index.getIndexCapabilities());
|
|
||||||
bitmapColumnIndices.add(index);
|
|
||||||
}
|
|
||||||
|
|
||||||
final ColumnIndexCapabilities finalMerged = merged;
|
|
||||||
return new BitmapColumnIndex()
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public ColumnIndexCapabilities getIndexCapabilities()
|
|
||||||
{
|
|
||||||
return finalMerged;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public <T> T computeBitmapResult(BitmapResultFactory<T> bitmapResultFactory, boolean includeUnknown)
|
|
||||||
{
|
|
||||||
return bitmapResultFactory.union(
|
|
||||||
() -> bitmapColumnIndices.stream().map(x -> x.computeBitmapResult(bitmapResultFactory, includeUnknown)).iterator()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
@Override
|
|
||||||
public <T> T computeBitmapResult(
|
|
||||||
BitmapResultFactory<T> bitmapResultFactory,
|
|
||||||
int applyRowCount,
|
|
||||||
int totalRowCount,
|
|
||||||
boolean includeUnknown
|
|
||||||
)
|
|
||||||
{
|
|
||||||
List<T> results = Lists.newArrayListWithCapacity(bitmapColumnIndices.size());
|
|
||||||
for (BitmapColumnIndex index : bitmapColumnIndices) {
|
|
||||||
final T r = index.computeBitmapResult(bitmapResultFactory, applyRowCount, totalRowCount, includeUnknown);
|
|
||||||
if (r == null) {
|
|
||||||
// all or nothing
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
results.add(r);
|
|
||||||
}
|
|
||||||
return bitmapResultFactory.union(results);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public ValueMatcher makeMatcher(ColumnSelectorFactory factory)
|
|
||||||
{
|
|
||||||
final ValueMatcher[] matchers = new ValueMatcher[filters.size()];
|
|
||||||
|
|
||||||
int i = 0;
|
|
||||||
for (Filter filter : filters) {
|
|
||||||
matchers[i++] = filter.makeMatcher(factory);
|
|
||||||
}
|
|
||||||
return makeMatcher(matchers);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public VectorValueMatcher makeVectorMatcher(final VectorColumnSelectorFactory factory)
|
|
||||||
{
|
|
||||||
final VectorValueMatcher[] matchers = new VectorValueMatcher[filters.size()];
|
|
||||||
|
|
||||||
int i = 0;
|
|
||||||
for (Filter filter : filters) {
|
|
||||||
matchers[i++] = filter.makeVectorMatcher(factory);
|
|
||||||
}
|
|
||||||
return makeVectorMatcher(matchers);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean canVectorizeMatcher(ColumnInspector inspector)
|
|
||||||
{
|
|
||||||
return filters.stream().allMatch(filter -> filter.canVectorizeMatcher(inspector));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public LinkedHashSet<Filter> getFilters()
|
|
||||||
{
|
|
||||||
return filters;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean supportsRequiredColumnRewrite()
|
|
||||||
{
|
|
||||||
for (Filter filter : filters) {
|
|
||||||
if (!filter.supportsRequiredColumnRewrite()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Filter rewriteRequiredColumns(Map<String, String> columnRewrites)
|
|
||||||
{
|
|
||||||
final List<Filter> newFilters = new ArrayList<>(filters.size());
|
|
||||||
for (Filter filter : filters) {
|
|
||||||
newFilters.add(filter.rewriteRequiredColumns(columnRewrites));
|
|
||||||
}
|
|
||||||
return new OrFilter(newFilters);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String toString()
|
|
||||||
{
|
|
||||||
return StringUtils.format("(%s)", OR_JOINER.join(filters));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean equals(Object o)
|
|
||||||
{
|
|
||||||
if (this == o) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (o == null || getClass() != o.getClass()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
OrFilter orFilter = (OrFilter) o;
|
|
||||||
return Objects.equals(getFilters(), orFilter.getFilters());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int hashCode()
|
|
||||||
{
|
|
||||||
return Objects.hash(getFilters());
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
private static ValueMatcher makeMatcher(final ValueMatcher[] baseMatchers)
|
private static ValueMatcher makeMatcher(final ValueMatcher[] baseMatchers)
|
||||||
{
|
{
|
||||||
Preconditions.checkState(baseMatchers.length > 0);
|
Preconditions.checkState(baseMatchers.length > 0);
|
||||||
|
@ -693,6 +369,7 @@ public class OrFilter implements BooleanFilter
|
||||||
{
|
{
|
||||||
final VectorMatch match = VectorMatch.wrap(new int[vectorOffset.getMaxVectorSize()]);
|
final VectorMatch match = VectorMatch.wrap(new int[vectorOffset.getMaxVectorSize()]);
|
||||||
int iterOffset = -1;
|
int iterOffset = -1;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ReadableVectorMatch match(ReadableVectorMatch mask, boolean includeUnknown)
|
public ReadableVectorMatch match(ReadableVectorMatch mask, boolean includeUnknown)
|
||||||
{
|
{
|
||||||
|
@ -744,4 +421,334 @@ public class OrFilter implements BooleanFilter
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public <T> FilterBundle makeFilterBundle(
|
||||||
|
FilterBundle.Builder filterBundleBuilder,
|
||||||
|
BitmapResultFactory<T> bitmapResultFactory,
|
||||||
|
int applyRowCount,
|
||||||
|
int totalRowCount,
|
||||||
|
boolean includeUnknown
|
||||||
|
)
|
||||||
|
{
|
||||||
|
// for OR filters, we have a few possible outcomes:
|
||||||
|
// 1 - all clauses are index only bundles. in this case we union the bitmaps together and make an index only bundle
|
||||||
|
// 2 - some clauses support indexes. in this case, we union the bitmaps of any index only bundles together to form a
|
||||||
|
// partial index which is constructed into a matcher bundle with convertIndexToMatcherBundle. We translate any
|
||||||
|
// index AND matcher bundles into a matcher only bundle with convertBundleToMatcherOnlyBundle. Finally, we
|
||||||
|
// combine these with the remaining matcher only bundles to with makeMatcher/makeVectorMatcher to make a matcher
|
||||||
|
// only bundle
|
||||||
|
// 3 - no clauses support indexes. in this case, we make a matcher only bundle using makeMatcher/makeVectorMatcher
|
||||||
|
|
||||||
|
final List<FilterBundle.IndexBundle> indexOnlyBundles = new ArrayList<>();
|
||||||
|
final List<FilterBundle.IndexBundleInfo> indexOnlyBundlesInfo = new ArrayList<>();
|
||||||
|
final List<FilterBundle.MatcherBundle> partialIndexBundles = new ArrayList<>();
|
||||||
|
final List<FilterBundle.MatcherBundle> matcherOnlyBundles = new ArrayList<>();
|
||||||
|
|
||||||
|
int indexUnionSize = 0;
|
||||||
|
ImmutableBitmap index = null;
|
||||||
|
ColumnIndexCapabilities merged = new SimpleColumnIndexCapabilities(true, true);
|
||||||
|
int emptyCount = 0;
|
||||||
|
|
||||||
|
final long bitmapConstructionStartNs = System.nanoTime();
|
||||||
|
for (FilterBundle.Builder subFilterBundleBuilder : filterBundleBuilder.getChildBuilders()) {
|
||||||
|
final FilterBundle bundle = subFilterBundleBuilder.build(
|
||||||
|
bitmapResultFactory,
|
||||||
|
Math.min(applyRowCount, totalRowCount - indexUnionSize),
|
||||||
|
totalRowCount,
|
||||||
|
includeUnknown
|
||||||
|
);
|
||||||
|
if (bundle.hasIndex()) {
|
||||||
|
final ImmutableBitmap bundleIndex = bundle.getIndex().getBitmap();
|
||||||
|
if (bundleIndex.isEmpty()) {
|
||||||
|
// we leave any indexes which are empty out of index, indexOnlyBundles, and partialIndexBundles
|
||||||
|
// even though we skip them, we still keep track of them to check for the case when we can build the OR into
|
||||||
|
// an index only bundle. We can count index and matcher bundles here too because the AND operation means that
|
||||||
|
// an empty index means the matcher can be skipped
|
||||||
|
emptyCount++;
|
||||||
|
} else {
|
||||||
|
if (bundle.hasMatcher()) {
|
||||||
|
// index and matcher bundles must be handled separately, they will need to be a single value matcher built
|
||||||
|
// by doing an AND operation between the index and the value matcher
|
||||||
|
// (a bundle is basically an AND operation between the index and matcher if the matcher is present)
|
||||||
|
partialIndexBundles.add(convertBundleToMatcherOnlyBundle(bundle, bundleIndex));
|
||||||
|
} else {
|
||||||
|
indexOnlyBundles.add(bundle.getIndex());
|
||||||
|
indexOnlyBundlesInfo.add(bundle.getIndex().getIndexInfo());
|
||||||
|
merged.merge(bundle.getIndex().getIndexCapabilities());
|
||||||
|
// union index only bitmaps together; if all sub-filters are 'index only' bundles we will make an index only
|
||||||
|
// bundle ourselves, else we will use this index as a single value matcher
|
||||||
|
if (index == null) {
|
||||||
|
index = bundle.getIndex().getBitmap();
|
||||||
|
} else {
|
||||||
|
index = index.union(bundle.getIndex().getBitmap());
|
||||||
|
}
|
||||||
|
indexUnionSize = index.size();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
matcherOnlyBundles.add(bundle.getMatcherBundle());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
final long totalBitmapConstructTimeNs = System.nanoTime() - bitmapConstructionStartNs;
|
||||||
|
|
||||||
|
|
||||||
|
// if all the filters are 'index only', we can make an index only bundle
|
||||||
|
if (indexOnlyBundles.size() + emptyCount == filters.size()) {
|
||||||
|
if (index == null || index.isEmpty()) {
|
||||||
|
return FilterBundle.allFalse(
|
||||||
|
totalBitmapConstructTimeNs,
|
||||||
|
filterBundleBuilder.getColumnIndexSelector().getBitmapFactory().makeEmptyImmutableBitmap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (indexOnlyBundles.size() == 1) {
|
||||||
|
return new FilterBundle(
|
||||||
|
indexOnlyBundles.get(0),
|
||||||
|
null
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return new FilterBundle(
|
||||||
|
new FilterBundle.SimpleIndexBundle(
|
||||||
|
new FilterBundle.IndexBundleInfo(
|
||||||
|
() -> "OR",
|
||||||
|
applyRowCount,
|
||||||
|
totalBitmapConstructTimeNs,
|
||||||
|
indexOnlyBundlesInfo
|
||||||
|
),
|
||||||
|
index,
|
||||||
|
merged
|
||||||
|
),
|
||||||
|
null
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// if not the index only outcome, we build a matcher only bundle from all the matchers
|
||||||
|
final int estimatedSize = (indexOnlyBundles.isEmpty() ? 0 : 1)
|
||||||
|
+ partialIndexBundles.size()
|
||||||
|
+ matcherOnlyBundles.size();
|
||||||
|
final List<FilterBundle.MatcherBundle> allMatcherBundles = Lists.newArrayListWithCapacity(estimatedSize);
|
||||||
|
final List<FilterBundle.MatcherBundleInfo> allMatcherBundlesInfo = Lists.newArrayListWithCapacity(estimatedSize);
|
||||||
|
if (!indexOnlyBundles.isEmpty()) {
|
||||||
|
// translate the indexOnly bundles into a single matcher
|
||||||
|
final FilterBundle.MatcherBundle matcherBundle = convertIndexToMatcherBundle(
|
||||||
|
applyRowCount,
|
||||||
|
indexOnlyBundles,
|
||||||
|
indexOnlyBundlesInfo,
|
||||||
|
totalBitmapConstructTimeNs,
|
||||||
|
index
|
||||||
|
);
|
||||||
|
allMatcherBundles.add(matcherBundle);
|
||||||
|
allMatcherBundlesInfo.add(matcherBundle.getMatcherInfo());
|
||||||
|
}
|
||||||
|
for (FilterBundle.MatcherBundle bundle : partialIndexBundles) {
|
||||||
|
allMatcherBundles.add(bundle);
|
||||||
|
allMatcherBundlesInfo.add(bundle.getMatcherInfo());
|
||||||
|
}
|
||||||
|
for (FilterBundle.MatcherBundle bundle : matcherOnlyBundles) {
|
||||||
|
allMatcherBundles.add(bundle);
|
||||||
|
allMatcherBundlesInfo.add(bundle.getMatcherInfo());
|
||||||
|
}
|
||||||
|
|
||||||
|
return new FilterBundle(
|
||||||
|
null,
|
||||||
|
new FilterBundle.MatcherBundle()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public FilterBundle.MatcherBundleInfo getMatcherInfo()
|
||||||
|
{
|
||||||
|
return new FilterBundle.MatcherBundleInfo(
|
||||||
|
() -> "OR",
|
||||||
|
null,
|
||||||
|
allMatcherBundlesInfo
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ValueMatcher valueMatcher(ColumnSelectorFactory selectorFactory, Offset baseOffset, boolean descending)
|
||||||
|
{
|
||||||
|
final ValueMatcher[] matchers = new ValueMatcher[allMatcherBundles.size()];
|
||||||
|
for (int i = 0; i < allMatcherBundles.size(); i++) {
|
||||||
|
matchers[i] = allMatcherBundles.get(i).valueMatcher(selectorFactory, baseOffset, descending);
|
||||||
|
}
|
||||||
|
return makeMatcher(matchers);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public VectorValueMatcher vectorMatcher(
|
||||||
|
VectorColumnSelectorFactory selectorFactory,
|
||||||
|
ReadableVectorOffset baseOffset
|
||||||
|
)
|
||||||
|
{
|
||||||
|
final VectorValueMatcher[] matchers = new VectorValueMatcher[allMatcherBundles.size()];
|
||||||
|
for (int i = 0; i < allMatcherBundles.size(); i++) {
|
||||||
|
matchers[i] = allMatcherBundles.get(i).vectorMatcher(selectorFactory, baseOffset);
|
||||||
|
}
|
||||||
|
return makeVectorMatcher(matchers);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean canVectorize()
|
||||||
|
{
|
||||||
|
for (FilterBundle.MatcherBundle bundle : allMatcherBundles) {
|
||||||
|
if (!bundle.canVectorize()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public BitmapColumnIndex getBitmapColumnIndex(ColumnIndexSelector selector)
|
||||||
|
{
|
||||||
|
if (filters.size() == 1) {
|
||||||
|
return Iterables.getOnlyElement(filters).getBitmapColumnIndex(selector);
|
||||||
|
}
|
||||||
|
|
||||||
|
List<BitmapColumnIndex> bitmapColumnIndices = new ArrayList<>(filters.size());
|
||||||
|
ColumnIndexCapabilities merged = new SimpleColumnIndexCapabilities(true, true);
|
||||||
|
for (Filter filter : filters) {
|
||||||
|
BitmapColumnIndex index = filter.getBitmapColumnIndex(selector);
|
||||||
|
if (index == null) {
|
||||||
|
// all or nothing
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
merged = merged.merge(index.getIndexCapabilities());
|
||||||
|
bitmapColumnIndices.add(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
final ColumnIndexCapabilities finalMerged = merged;
|
||||||
|
return new BitmapColumnIndex()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public ColumnIndexCapabilities getIndexCapabilities()
|
||||||
|
{
|
||||||
|
return finalMerged;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int estimatedComputeCost()
|
||||||
|
{
|
||||||
|
// There's no additional cost on OR filter, cost in child filters would be summed.
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public <T> T computeBitmapResult(BitmapResultFactory<T> bitmapResultFactory, boolean includeUnknown)
|
||||||
|
{
|
||||||
|
return bitmapResultFactory.union(
|
||||||
|
() -> bitmapColumnIndices.stream()
|
||||||
|
.map(x -> x.computeBitmapResult(bitmapResultFactory, includeUnknown))
|
||||||
|
.iterator()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nullable
|
||||||
|
@Override
|
||||||
|
public <T> T computeBitmapResult(
|
||||||
|
BitmapResultFactory<T> bitmapResultFactory,
|
||||||
|
int applyRowCount,
|
||||||
|
int totalRowCount,
|
||||||
|
boolean includeUnknown
|
||||||
|
)
|
||||||
|
{
|
||||||
|
List<T> results = Lists.newArrayListWithCapacity(bitmapColumnIndices.size());
|
||||||
|
for (BitmapColumnIndex index : bitmapColumnIndices) {
|
||||||
|
final T r = index.computeBitmapResult(bitmapResultFactory, applyRowCount, totalRowCount, includeUnknown);
|
||||||
|
if (r == null) {
|
||||||
|
// all or nothing
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
results.add(r);
|
||||||
|
}
|
||||||
|
return bitmapResultFactory.union(results);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ValueMatcher makeMatcher(ColumnSelectorFactory factory)
|
||||||
|
{
|
||||||
|
final ValueMatcher[] matchers = new ValueMatcher[filters.size()];
|
||||||
|
|
||||||
|
int i = 0;
|
||||||
|
for (Filter filter : filters) {
|
||||||
|
matchers[i++] = filter.makeMatcher(factory);
|
||||||
|
}
|
||||||
|
return makeMatcher(matchers);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public VectorValueMatcher makeVectorMatcher(final VectorColumnSelectorFactory factory)
|
||||||
|
{
|
||||||
|
final VectorValueMatcher[] matchers = new VectorValueMatcher[filters.size()];
|
||||||
|
|
||||||
|
int i = 0;
|
||||||
|
for (Filter filter : filters) {
|
||||||
|
matchers[i++] = filter.makeVectorMatcher(factory);
|
||||||
|
}
|
||||||
|
return makeVectorMatcher(matchers);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean canVectorizeMatcher(ColumnInspector inspector)
|
||||||
|
{
|
||||||
|
return filters.stream().allMatch(filter -> filter.canVectorizeMatcher(inspector));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public LinkedHashSet<Filter> getFilters()
|
||||||
|
{
|
||||||
|
return filters;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean supportsRequiredColumnRewrite()
|
||||||
|
{
|
||||||
|
for (Filter filter : filters) {
|
||||||
|
if (!filter.supportsRequiredColumnRewrite()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Filter rewriteRequiredColumns(Map<String, String> columnRewrites)
|
||||||
|
{
|
||||||
|
final List<Filter> newFilters = new ArrayList<>(filters.size());
|
||||||
|
for (Filter filter : filters) {
|
||||||
|
newFilters.add(filter.rewriteRequiredColumns(columnRewrites));
|
||||||
|
}
|
||||||
|
return new OrFilter(newFilters);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString()
|
||||||
|
{
|
||||||
|
return StringUtils.format("(%s)", OR_JOINER.join(filters));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object o)
|
||||||
|
{
|
||||||
|
if (this == o) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (o == null || getClass() != o.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
OrFilter orFilter = (OrFilter) o;
|
||||||
|
return Objects.equals(getFilters(), orFilter.getFilters());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode()
|
||||||
|
{
|
||||||
|
return Objects.hash(getFilters());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,14 @@ public interface BitmapColumnIndex
|
||||||
{
|
{
|
||||||
ColumnIndexCapabilities getIndexCapabilities();
|
ColumnIndexCapabilities getIndexCapabilities();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an estimated cost for computing the bitmap result.
|
||||||
|
*/
|
||||||
|
default int estimatedComputeCost()
|
||||||
|
{
|
||||||
|
return Integer.MAX_VALUE;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compute a bitmap result wrapped with the {@link BitmapResultFactory} representing the rows matched by this index.
|
* Compute a bitmap result wrapped with the {@link BitmapResultFactory} representing the rows matched by this index.
|
||||||
* If building a cursor, use {@link #computeBitmapResult(BitmapResultFactory, int, int, boolean)} instead.
|
* If building a cursor, use {@link #computeBitmapResult(BitmapResultFactory, int, int, boolean)} instead.
|
||||||
|
@ -45,7 +53,6 @@ public interface BitmapColumnIndex
|
||||||
* to true, bitmaps returned by this method should include true bits for any rows where
|
* to true, bitmaps returned by this method should include true bits for any rows where
|
||||||
* the matching result is 'unknown', such as from the input being null valued.
|
* the matching result is 'unknown', such as from the input being null valued.
|
||||||
* See {@link NullHandling#useThreeValueLogic()}.
|
* See {@link NullHandling#useThreeValueLogic()}.
|
||||||
*
|
|
||||||
* @return bitmap result representing rows matched by this index
|
* @return bitmap result representing rows matched by this index
|
||||||
*/
|
*/
|
||||||
<T> T computeBitmapResult(
|
<T> T computeBitmapResult(
|
||||||
|
@ -69,7 +76,6 @@ public interface BitmapColumnIndex
|
||||||
* set to true, bitmaps returned by this method should include true bits for any rows where
|
* set to true, bitmaps returned by this method should include true bits for any rows where
|
||||||
* the matching result is 'unknown', such as from the input being null valued.
|
* the matching result is 'unknown', such as from the input being null valued.
|
||||||
* See {@link NullHandling#useThreeValueLogic()}.
|
* See {@link NullHandling#useThreeValueLogic()}.
|
||||||
*
|
|
||||||
* @return bitmap result representing rows matched by this index
|
* @return bitmap result representing rows matched by this index
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
|
|
|
@ -43,7 +43,12 @@ import org.junit.Before;
|
||||||
import org.junit.Rule;
|
import org.junit.Rule;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.rules.TemporaryFolder;
|
import org.junit.rules.TemporaryFolder;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.Parameterized;
|
||||||
|
import org.junit.runners.Parameterized.Parameter;
|
||||||
|
import org.junit.runners.Parameterized.Parameters;
|
||||||
|
|
||||||
|
@RunWith(Parameterized.class)
|
||||||
public class FilterBundleTest extends InitializedNullHandlingTest
|
public class FilterBundleTest extends InitializedNullHandlingTest
|
||||||
{
|
{
|
||||||
private Closer closer;
|
private Closer closer;
|
||||||
|
@ -53,6 +58,15 @@ public class FilterBundleTest extends InitializedNullHandlingTest
|
||||||
@Rule
|
@Rule
|
||||||
public TemporaryFolder tmpDir = new TemporaryFolder();
|
public TemporaryFolder tmpDir = new TemporaryFolder();
|
||||||
|
|
||||||
|
@Parameters
|
||||||
|
public static Object[] flags()
|
||||||
|
{
|
||||||
|
return new Object[]{false, true};
|
||||||
|
}
|
||||||
|
|
||||||
|
@Parameter
|
||||||
|
public boolean cursorAutoArrangeFilters;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
public void setUp()
|
public void setUp()
|
||||||
{
|
{
|
||||||
|
@ -317,8 +331,7 @@ public class FilterBundleTest extends InitializedNullHandlingTest
|
||||||
|
|
||||||
protected FilterBundle makeFilterBundle(final Filter filter)
|
protected FilterBundle makeFilterBundle(final Filter filter)
|
||||||
{
|
{
|
||||||
return filter.makeFilterBundle(
|
return new FilterBundle.Builder(filter, indexSelector, cursorAutoArrangeFilters).build(
|
||||||
indexSelector,
|
|
||||||
new DefaultBitmapResultFactory(bitmapFactory),
|
new DefaultBitmapResultFactory(bitmapFactory),
|
||||||
indexSelector.getNumRows(),
|
indexSelector.getNumRows(),
|
||||||
indexSelector.getNumRows(),
|
indexSelector.getNumRows(),
|
||||||
|
|
Loading…
Reference in New Issue