/*
* 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;
}
}