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

org.elasticsearch.index.shard.IndexLongFieldRange Maven / Gradle / Ivy

/*
 * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
 * or more contributor license agreements. Licensed under the "Elastic License
 * 2.0", the "GNU Affero General Public License v3.0 only", and the "Server Side
 * Public License v 1"; you may not use this file except in compliance with, at
 * your election, the "Elastic License 2.0", the "GNU Affero General Public
 * License v3.0 only", or the "Server Side Public License, v 1".
 */

package org.elasticsearch.index.shard;

import org.elasticsearch.common.io.stream.StreamInput;
import org.elasticsearch.common.io.stream.StreamOutput;
import org.elasticsearch.common.io.stream.Writeable;
import org.elasticsearch.core.Nullable;
import org.elasticsearch.xcontent.ToXContentFragment;
import org.elasticsearch.xcontent.XContentBuilder;
import org.elasticsearch.xcontent.XContentParser;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.stream.IntStream;

/**
 * Class representing an (inclusive) range of {@code long} values in a field in an index which may comprise multiple shards. This
 * information is accumulated shard-by-shard, and we keep track of which shards are represented in this value. Only once all shards are
 * represented should this information be considered accurate for the index.
 */
public class IndexLongFieldRange implements Writeable, ToXContentFragment {

    /**
     * Sentinel value indicating that no information is currently available, for instance because the index has just been created.
     */
    public static final IndexLongFieldRange NO_SHARDS = new IndexLongFieldRange(new int[0], Long.MAX_VALUE, Long.MIN_VALUE);

    /**
     * Sentinel value indicating an empty range, for instance because the field is missing or has no values in any shard.
     */
    public static final IndexLongFieldRange EMPTY = new IndexLongFieldRange(null, Long.MAX_VALUE, Long.MIN_VALUE);

    /**
     * Sentinel value indicating the actual range is unknown, for instance because more docs may be added in future.
     */
    public static final IndexLongFieldRange UNKNOWN = new IndexLongFieldRange(null, Long.MIN_VALUE, Long.MAX_VALUE);

    @Nullable // if this range includes all shards
    private final int[] shards;
    private final long min, max;

    private IndexLongFieldRange(int[] shards, long min, long max) {
        assert (min == Long.MAX_VALUE && max == Long.MIN_VALUE) || min <= max : min + " vs " + max;
        assert shards == null || shards.length > 0 || (min == Long.MAX_VALUE && max == Long.MIN_VALUE) : Arrays.toString(shards);
        assert shards == null || Arrays.equals(shards, Arrays.stream(shards).sorted().distinct().toArray()) : Arrays.toString(shards);
        this.shards = shards;
        this.min = min;
        this.max = max;
    }

    /**
     * @return whether this range includes information from all shards yet.
     */
    public boolean isComplete() {
        return shards == null;
    }

    /**
     * @return whether this range includes information from all shards and can be used meaningfully.
     */
    public boolean containsAllShardRanges() {
        return isComplete() && this != IndexLongFieldRange.UNKNOWN;
    }

    // exposed for testing
    int[] getShards() {
        return shards;
    }

    // exposed for testing
    long getMinUnsafe() {
        return min;
    }

    // exposed for testing
    long getMaxUnsafe() {
        return max;
    }

    /**
     * @return the (inclusive) minimum of this range.
     */
    public long getMin() {
        assert shards == null : "min is meaningless if we don't have data from all shards yet";
        assert this != EMPTY : "min is meaningless if range is empty";
        assert this != UNKNOWN : "min is meaningless if range is unknown";
        return min;
    }

    /**
     * @return the (inclusive) maximum of this range.
     */
    public long getMax() {
        assert shards == null : "max is meaningless if we don't have data from all shards yet";
        assert this != EMPTY : "max is meaningless if range is empty";
        assert this != UNKNOWN : "max is meaningless if range is unknown";
        return max;
    }

    private static final byte WIRE_TYPE_OTHER = (byte) 0;
    private static final byte WIRE_TYPE_NO_SHARDS = (byte) 1;
    private static final byte WIRE_TYPE_UNKNOWN = (byte) 2;
    private static final byte WIRE_TYPE_EMPTY = (byte) 3;

    @Override
    public void writeTo(StreamOutput out) throws IOException {
        if (this == NO_SHARDS) {
            out.writeByte(WIRE_TYPE_NO_SHARDS);
        } else if (this == UNKNOWN) {
            out.writeByte(WIRE_TYPE_UNKNOWN);
        } else if (this == EMPTY) {
            out.writeByte(WIRE_TYPE_EMPTY);
        } else {
            out.writeByte(WIRE_TYPE_OTHER);
            if (shards == null) {
                out.writeBoolean(false);
            } else {
                out.writeBoolean(true);
                out.writeVIntArray(shards);
            }
            out.writeZLong(min);
            out.writeZLong(max);
        }
    }

    public static IndexLongFieldRange readFrom(StreamInput in) throws IOException {
        final byte type = in.readByte();
        return switch (type) {
            case WIRE_TYPE_NO_SHARDS -> NO_SHARDS;
            case WIRE_TYPE_UNKNOWN -> UNKNOWN;
            case WIRE_TYPE_EMPTY -> EMPTY;
            case WIRE_TYPE_OTHER -> new IndexLongFieldRange(in.readBoolean() ? in.readVIntArray() : null, in.readZLong(), in.readZLong());
            default -> throw new IllegalStateException("type [" + type + "] not known");
        };
    }

    @Override
    public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
        if (this == UNKNOWN) {
            builder.field("unknown", true);
        } else if (this == EMPTY) {
            builder.field("empty", true);
        } else if (this == NO_SHARDS) {
            builder.startArray("shards");
            builder.endArray();
        } else {
            builder.field("min", min);
            builder.field("max", max);
            if (shards != null) {
                builder.startArray("shards");
                for (int shard : shards) {
                    builder.value(shard);
                }
                builder.endArray();
            }
        }
        return builder;
    }

    public static IndexLongFieldRange fromXContent(XContentParser parser) throws IOException {
        XContentParser.Token token;
        String currentFieldName = null;
        Boolean isUnknown = null;
        Boolean isEmpty = null;
        Long min = null;
        Long max = null;
        List shardsList = null;
        while ((token = parser.nextToken()) != XContentParser.Token.END_OBJECT) {
            if (token == XContentParser.Token.FIELD_NAME) {
                currentFieldName = parser.currentName();
            } else if (token.isValue()) {
                if ("unknown".equals(currentFieldName)) {
                    if (Boolean.FALSE.equals(isUnknown)) {
                        throw new IllegalArgumentException("unexpected field 'unknown'");
                    } else {
                        isUnknown = Boolean.TRUE;
                        isEmpty = Boolean.FALSE;
                    }
                } else if ("empty".equals(currentFieldName)) {
                    if (Boolean.FALSE.equals(isEmpty)) {
                        throw new IllegalArgumentException("unexpected field 'empty'");
                    } else {
                        isUnknown = Boolean.FALSE;
                        isEmpty = Boolean.TRUE;
                    }
                } else if ("min".equals(currentFieldName)) {
                    if (Boolean.TRUE.equals(isUnknown) || Boolean.TRUE.equals(isEmpty)) {
                        throw new IllegalArgumentException("unexpected field 'min'");
                    } else {
                        isUnknown = Boolean.FALSE;
                        isEmpty = Boolean.FALSE;
                        min = parser.longValue();
                    }
                } else if ("max".equals(currentFieldName)) {
                    if (Boolean.TRUE.equals(isUnknown) || Boolean.TRUE.equals(isEmpty)) {
                        throw new IllegalArgumentException("unexpected field 'max'");
                    } else {
                        isUnknown = Boolean.FALSE;
                        isEmpty = Boolean.FALSE;
                        max = parser.longValue();
                    }
                }
            } else if (token == XContentParser.Token.START_ARRAY) {
                if ("shards".equals(currentFieldName)) {
                    if (Boolean.TRUE.equals(isUnknown) || Boolean.TRUE.equals(isEmpty) || shardsList != null) {
                        throw new IllegalArgumentException("unexpected array 'shards'");
                    } else {
                        isUnknown = Boolean.FALSE;
                        isEmpty = Boolean.FALSE;
                        shardsList = new ArrayList<>();
                        while ((token = parser.nextToken()) != XContentParser.Token.END_ARRAY) {
                            if (token.isValue()) {
                                shardsList.add(parser.intValue());
                            }
                        }
                    }
                } else {
                    throw new IllegalArgumentException("Unexpected array: " + currentFieldName);
                }
            } else {
                throw new IllegalArgumentException("Unexpected token: " + token);
            }
        }

        if (Boolean.TRUE.equals(isUnknown)) {
            // noinspection ConstantConditions this assertion is always true but left here for the benefit of readers
            assert min == null && max == null && shardsList == null && Boolean.FALSE.equals(isEmpty);
            return UNKNOWN;
        } else if (Boolean.TRUE.equals(isEmpty)) {
            // noinspection ConstantConditions this assertion is always true but left here for the benefit of readers
            assert min == null && max == null && shardsList == null && Boolean.FALSE.equals(isUnknown);
            return EMPTY;
        } else if (shardsList != null && shardsList.isEmpty()) {
            // noinspection ConstantConditions this assertion is always true but left here for the benefit of readers
            assert min == null && max == null && Boolean.FALSE.equals(isEmpty) && Boolean.FALSE.equals(isUnknown);
            return NO_SHARDS;
        } else if (min != null) {
            // noinspection ConstantConditions this assertion is always true but left here for the benefit of readers
            assert Boolean.FALSE.equals(isUnknown) && Boolean.FALSE.equals(isEmpty);
            if (max == null) {
                throw new IllegalArgumentException("field 'max' unexpectedly missing");
            }
            final int[] shards;
            if (shardsList != null) {
                shards = shardsList.stream().mapToInt(i -> i).toArray();
                assert shards.length > 0;
            } else {
                shards = null;
            }
            return new IndexLongFieldRange(shards, min, max);
        } else {
            throw new IllegalArgumentException("field range contents unexpectedly missing");
        }
    }

    public IndexLongFieldRange extendWithShardRange(int shardId, int shardCount, ShardLongFieldRange shardFieldRange) {
        if (shardFieldRange == ShardLongFieldRange.UNKNOWN) {
            assert shards == null ? this == UNKNOWN : Arrays.stream(shards).noneMatch(i -> i == shardId);
            return UNKNOWN;
        }
        if (shards == null || Arrays.stream(shards).anyMatch(i -> i == shardId)) {
            assert shardFieldRange == ShardLongFieldRange.EMPTY || min <= shardFieldRange.getMin() && shardFieldRange.getMax() <= max;
            return this;
        }
        final int[] newShards;
        if (shards.length == shardCount - 1) {
            assert Arrays.equals(shards, IntStream.range(0, shardCount).filter(i -> i != shardId).toArray())
                : Arrays.toString(shards) + " + " + shardId;
            if (shardFieldRange == ShardLongFieldRange.EMPTY && min == EMPTY.min && max == EMPTY.max) {
                return EMPTY;
            }
            newShards = null;
        } else {
            newShards = IntStream.concat(Arrays.stream(this.shards), IntStream.of(shardId)).sorted().toArray();
        }
        if (shardFieldRange == ShardLongFieldRange.EMPTY) {
            return new IndexLongFieldRange(newShards, min, max);
        } else {
            return new IndexLongFieldRange(newShards, Math.min(shardFieldRange.getMin(), min), Math.max(shardFieldRange.getMax(), max));
        }
    }

    @Override
    public String toString() {
        if (this == NO_SHARDS) {
            return "NO_SHARDS";
        } else if (this == UNKNOWN) {
            return "UNKNOWN";
        } else if (this == EMPTY) {
            return "EMPTY";
        } else if (shards == null) {
            return "[" + min + "-" + max + "]";
        } else {
            return "[" + min + "-" + max + ", shards=" + Arrays.toString(shards) + "]";
        }
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        if (this == EMPTY || this == UNKNOWN || this == NO_SHARDS || o == EMPTY || o == UNKNOWN || o == NO_SHARDS) return false;
        IndexLongFieldRange that = (IndexLongFieldRange) o;
        return min == that.min && max == that.max && Arrays.equals(shards, that.shards);
    }

    @Override
    public int hashCode() {
        int result = Objects.hash(min, max);
        result = 31 * result + Arrays.hashCode(shards);
        return result;
    }

    /**
     * Remove the given shard from the set of known shards, possibly without adjusting the min and max. Used when allocating a stale primary
     * which may have a different range from the original, so we must allow the range to grow. Note that this doesn't usually allow the
     * range to shrink, so we may in theory hit this shard more than needed after allocating a stale primary.
     */
    public IndexLongFieldRange removeShard(int shardId, int numberOfShards) {
        assert 0 <= shardId && shardId < numberOfShards : shardId + " vs " + numberOfShards;

        if (shards != null && Arrays.stream(shards).noneMatch(i -> i == shardId)) {
            return this;
        }
        if (shards == null && numberOfShards == 1) {
            return NO_SHARDS;
        }
        if (this == UNKNOWN) {
            return this;
        }
        if (shards != null && shards.length == 1 && shards[0] == shardId) {
            return NO_SHARDS;
        }

        final IntStream currentShards = shards == null ? IntStream.range(0, numberOfShards) : Arrays.stream(shards);
        return new IndexLongFieldRange(currentShards.filter(i -> i != shardId).toArray(), min, max);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy