All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.trino.spi.predicate.Domain Maven / Gradle / Ivy

There is a newer version: 468
Show newest version
/*
 * 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 io.trino.spi.predicate;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import io.trino.spi.connector.ConnectorSession;
import io.trino.spi.type.Type;

import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.Optional;

import static io.airlift.slice.SizeOf.instanceSize;
import static java.lang.String.format;
import static java.util.Objects.requireNonNull;

/**
 * Defines the possible values of a single variable in terms of its valid scalar values and nullability.
 * 

* For example: *

    *
  • Domain.none() => no scalar values allowed, NULL not allowed *
  • Domain.all() => all scalar values allowed, NULL allowed *
  • Domain.onlyNull() => no scalar values allowed, NULL allowed *
  • Domain.notNull() => all scalar values allowed, NULL not allowed *
*/ public final class Domain { private static final int INSTANCE_SIZE = instanceSize(Domain.class); public static final int DEFAULT_COMPACTION_THRESHOLD = 32; private final ValueSet values; private final boolean nullAllowed; private Domain(ValueSet values, boolean nullAllowed) { this.values = requireNonNull(values, "values is null"); this.nullAllowed = nullAllowed; } @JsonCreator public static Domain create( @JsonProperty("values") ValueSet values, @JsonProperty("nullAllowed") boolean nullAllowed) { return new Domain(values, nullAllowed); } public static Domain none(Type type) { return new Domain(ValueSet.none(type), false); } public static Domain all(Type type) { return new Domain(ValueSet.all(type), true); } public static Domain onlyNull(Type type) { return new Domain(ValueSet.none(type), true); } public static Domain notNull(Type type) { return new Domain(ValueSet.all(type), false); } public static Domain singleValue(Type type, Object value) { return singleValue(type, value, false); } public static Domain singleValue(Type type, Object value, boolean nullAllowed) { return new Domain(ValueSet.of(type, value), nullAllowed); } public static Domain multipleValues(Type type, List values) { return multipleValues(type, values, false); } public static Domain multipleValues(Type type, List values, boolean nullAllowed) { if (values.isEmpty()) { throw new IllegalArgumentException("values cannot be empty"); } if (values.size() == 1) { return singleValue(type, values.get(0), nullAllowed); } return new Domain(ValueSet.of(type, values.get(0), values.subList(1, values.size()).toArray()), nullAllowed); } public Type getType() { return values.getType(); } @JsonProperty public ValueSet getValues() { return values; } @JsonProperty public boolean isNullAllowed() { return nullAllowed; } public boolean isNone() { return values.isNone() && !nullAllowed; } public boolean isAll() { return values.isAll() && nullAllowed; } public boolean isSingleValue() { return !nullAllowed && values.isSingleValue(); } public boolean isNullableSingleValue() { if (nullAllowed) { return values.isNone(); } return values.isSingleValue(); } public boolean isOnlyNull() { return values.isNone() && nullAllowed; } public Object getSingleValue() { if (!isSingleValue()) { throw new IllegalStateException("Domain is not a single value"); } return values.getSingleValue(); } public Object getNullableSingleValue() { if (!isNullableSingleValue()) { throw new IllegalStateException("Domain is not a nullable single value"); } if (nullAllowed) { return null; } return values.getSingleValue(); } public boolean includesNullableValue(Object value) { return value == null ? nullAllowed : values.containsValue(value); } public boolean isNullableDiscreteSet() { return values.isNone() ? nullAllowed : values.isDiscreteSet(); } public DiscreteSet getNullableDiscreteSet() { if (!isNullableDiscreteSet()) { throw new IllegalStateException("Domain is not a nullable discrete set"); } return new DiscreteSet( values.isNone() ? List.of() : values.getDiscreteSet(), nullAllowed); } public boolean overlaps(Domain other) { checkCompatibility(other); if (this.isNullAllowed() && other.isNullAllowed()) { return true; } return values.overlaps(other.getValues()); } public boolean contains(Domain other) { checkCompatibility(other); if (!this.isNullAllowed() && other.isNullAllowed()) { return false; } return values.contains(other.getValues()); } public Domain intersect(Domain other) { checkCompatibility(other); return new Domain(values.intersect(other.getValues()), this.isNullAllowed() && other.isNullAllowed()); } public Domain union(Domain other) { checkCompatibility(other); return new Domain(values.union(other.getValues()), this.isNullAllowed() || other.isNullAllowed()); } public static Domain union(List domains) { if (domains.isEmpty()) { throw new IllegalArgumentException("domains cannot be empty for union"); } if (domains.size() == 1) { return domains.get(0); } boolean nullAllowed = false; List valueSets = new ArrayList<>(domains.size()); for (Domain domain : domains) { valueSets.add(domain.getValues()); nullAllowed = nullAllowed || domain.nullAllowed; } ValueSet unionedValues = valueSets.get(0).union(valueSets.subList(1, valueSets.size())); return new Domain(unionedValues, nullAllowed); } public Domain complement() { return new Domain(values.complement(), !nullAllowed); } public Domain subtract(Domain other) { checkCompatibility(other); return new Domain(values.subtract(other.getValues()), this.isNullAllowed() && !other.isNullAllowed()); } private void checkCompatibility(Domain domain) { if (!getType().equals(domain.getType())) { throw new IllegalArgumentException(format("Mismatched Domain types: %s vs %s", getType(), domain.getType())); } if (values.getClass() != domain.values.getClass()) { throw new IllegalArgumentException(format("Mismatched Domain value set classes: %s vs %s", values.getClass(), domain.values.getClass())); } } @Override public int hashCode() { return Objects.hash(values, nullAllowed); } @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj == null || getClass() != obj.getClass()) { return false; } Domain other = (Domain) obj; return Objects.equals(this.values, other.values) && this.nullAllowed == other.nullAllowed; } /** * Reduces the number of discrete components in the Domain if there are too many. */ public Domain simplify() { return simplify(DEFAULT_COMPACTION_THRESHOLD); } public Domain simplify(int threshold) { Optional simplifiedValueSet = values.getValuesProcessor().transform( ranges -> { if (ranges.getRangeCount() <= threshold) { return Optional.empty(); } return Optional.of(ValueSet.ofRanges(ranges.getSpan())); }, discreteValues -> { if (discreteValues.getValuesCount() <= threshold) { return Optional.empty(); } return Optional.of(ValueSet.all(values.getType())); }, allOrNone -> Optional.empty()); if (simplifiedValueSet.isEmpty()) { return this; } return Domain.create(simplifiedValueSet.get(), nullAllowed); } @Override public String toString() { return toString(ToStringSession.INSTANCE); } public String toString(ConnectorSession session) { return toString(session, 10); } public String toString(ConnectorSession session, int limit) { if (isAll()) { return "ALL"; } if (isNone()) { return "NONE"; } if (isOnlyNull()) { return "[NULL]"; } return "[ " + (nullAllowed ? "NULL, " : "") + values.toString(session, limit) + " ]"; } public long getRetainedSizeInBytes() { return INSTANCE_SIZE + values.getRetainedSizeInBytes(); } public static class DiscreteSet { private final List nonNullValues; private final boolean containsNull; DiscreteSet(List values, boolean containsNull) { this.nonNullValues = requireNonNull(values, "values is null"); this.containsNull = containsNull; if (!containsNull && values.isEmpty()) { throw new IllegalArgumentException("Discrete set cannot be empty"); } } public List getNonNullValues() { return nonNullValues; } public boolean containsNull() { return containsNull; } } }