/* * 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. */ package com.facebook.presto.operator.index; import com.facebook.presto.operator.project.InputChannels; import com.facebook.presto.operator.project.PageFilter; import com.facebook.presto.operator.project.SelectedPositions; import com.facebook.presto.spi.ConnectorSession; import com.facebook.presto.spi.Page; import com.facebook.presto.spi.block.Block; import com.facebook.presto.spi.type.Type; import com.google.common.collect.ImmutableList; import java.util.List; import static com.google.common.base.Preconditions.checkArgument; import static java.util.Objects.requireNonNull; /** * Filters out rows that do not match the values from the specified tuple */ public class TuplePageFilter implements PageFilter { private final Page tuplePage; private final InputChannels inputChannels; private final List<Type> types; private boolean[] selectedPositions = new boolean[0]; public TuplePageFilter(Page tuplePage, List<Type> types, List<Integer> inputChannels) { requireNonNull(tuplePage, "tuplePage is null"); requireNonNull(types, "types is null"); requireNonNull(inputChannels, "inputChannels is null"); checkArgument(tuplePage.getPositionCount() == 1, "tuplePage should only have one position"); checkArgument(tuplePage.getChannelCount() == inputChannels.size(), "tuplePage and inputChannels have different number of channels"); checkArgument(types.size() == inputChannels.size(), "types and inputChannels have different number of channels"); this.tuplePage = tuplePage; this.types = ImmutableList.copyOf(types); this.inputChannels = new InputChannels(inputChannels); } @Override public boolean isDeterministic() { return true; } @Override public InputChannels getInputChannels() { return inputChannels; } @Override public SelectedPositions filter(ConnectorSession session, Page page) { if (selectedPositions.length < page.getPositionCount()) { selectedPositions = new boolean[page.getPositionCount()]; } for (int position = 0; position < page.getPositionCount(); position++) { selectedPositions[position] = matches(page, position); } return PageFilter.positionsArrayToSelectedPositions(selectedPositions, page.getPositionCount()); } private boolean matches(Page page, int position) { for (int channel = 0; channel < inputChannels.size(); channel++) { Type type = types.get(channel); Block outputBlock = page.getBlock(channel); Block singleTupleBlock = tuplePage.getBlock(channel); if (!type.equalTo(singleTupleBlock, 0, outputBlock, position)) { return false; } } return true; } }