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

io.hekate.partition.UpdatablePartitionMapper Maven / Gradle / Ivy

/*
 * Copyright 2022 The Hekate Project
 *
 * The Hekate Project licenses this file to you 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.hekate.partition;

import io.hekate.cluster.ClusterTopology;
import io.hekate.cluster.ClusterTopologySupport;
import io.hekate.core.internal.util.ArgAssert;
import io.hekate.util.format.ToString;
import io.hekate.util.format.ToStringIgnore;
import java.util.Objects;
import java.util.function.Function;
import java.util.stream.IntStream;

/**
 * Updatable implementation of {@link PartitionMapper} interface.
 *
 * 

* This class is intended for use cases when it is required to manually control partitions mapping (as opposed to automatic mapping that is * provided by the {@link RendezvousHashMapper}). *

* *

* Partitions mapping can be updated via the {@link #update(ClusterTopology, Function)} method. *

*/ public class UpdatablePartitionMapper extends PartitionMapperBase { private static final class Snapshot extends PartitionMapperBase { private final ClusterTopology topology; private final Partition[] partitions; public Snapshot(ClusterTopology topology, Partition[] partitions, int backupSize) { super(partitions.length, backupSize); this.topology = topology; this.partitions = partitions; } public static Snapshot empty(int partitions, int backupSize) { return new Snapshot( ClusterTopology.empty(), IntStream.range(0, partitions) .mapToObj(DefaultPartition::empty) .toArray(Partition[]::new), backupSize ); } @Override public PartitionMapper snapshot() { return this; } @Override public boolean isSnapshot() { return true; } @Override public Partition partition(int id) { return partitions[id]; } @Override public ClusterTopology topology() { return topology; } @Override public PartitionMapper copy(ClusterTopologySupport cluster) throws UnsupportedOperationException { throw new UnsupportedOperationException("Snapshot doesn't support copying."); } @Override public String toString() { return ToString.format(UpdatablePartitionMapper.class, this); } } @ToStringIgnore private volatile Snapshot snapshot; /** * Constructs a new instance. * * @param partitions See {@link #partitions()}. * @param backupSize See {@link #backupNodes()}. */ public UpdatablePartitionMapper(int partitions, int backupSize) { super(partitions, backupSize); snapshot = Snapshot.empty(partitions, backupSize); } /** * Updates partitions mapping by applying the specified {@code update} function to each partition and using its result as a new mapping. * * @param topology New cluster topology (see {@link #topology()}). * @param update Partitions update function. */ public void update(ClusterTopology topology, Function update) { ArgAssert.notNull(topology, "Topology"); ArgAssert.notNull(update, "Update function"); Snapshot oldSnapshot = this.snapshot; Partition[] newPartitions = new Partition[partitions()]; for (int i = 0; i < newPartitions.length; i++) { Partition newPartition = update.apply(oldSnapshot.partition(i)); Objects.requireNonNull(newPartition, "Partition mapper function returned null."); newPartitions[i] = newPartition; } this.snapshot = new Snapshot(topology, newPartitions, backupNodes()); } @Override public Partition partition(int id) { return snapshot.partition(id); } @Override public PartitionMapper snapshot() { return snapshot; } @Override public boolean isSnapshot() { return false; } @Override public ClusterTopology topology() { return snapshot().topology(); } @Override public PartitionMapper copy(ClusterTopologySupport cluster) throws UnsupportedOperationException { throw new UnsupportedOperationException(getClass().getSimpleName() + " doesn't support copying."); } @Override public String toString() { return ToString.format(this); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy