
net.openhft.chronicle.hash.replication.ReplicationHub Maven / Gradle / Ivy
/*
* Copyright (C) 2015 higherfrequencytrading.com
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see .
*/
package net.openhft.chronicle.hash.replication;
import net.openhft.chronicle.hash.ChronicleHash;
import net.openhft.chronicle.hash.ChronicleHashBuilder;
import net.openhft.chronicle.hash.ChronicleHashInstanceBuilder;
import net.openhft.chronicle.map.ChronicleMap;
import net.openhft.chronicle.set.ChronicleSet;
import org.jetbrains.annotations.NotNull;
import java.util.Arrays;
/**
* This class holds all configurations of
* multicontainer replication, which is usable, when you want to replicate several {@linkplain
* ChronicleHash ChronicleHashes} ({@linkplain ChronicleMap maps}, {@linkplain ChronicleSet sets},
* etc.) between same servers. {@code ReplicationHub} allows to share TCP/UDP connection, sockets,
* buffers, worker threads for that, considerably reducing resources usage and increasing cumulative
* (for all replicated containers) replication throughput. See
* the corresponding section in ChronicleMap manual for more information.
*
*
Create instances of this class using this pattern:
{@code
* ReplicationHub hub = ReplicationHub.builder()
* .tcpTransportAndNetwork(tcpConfig)
* // more configurations...
* .createWithId(identifierOfThisServerWithinTheChronicleReplicationNetwork);}
* Then, given you prepared a {@link ChronicleHashBuilder builder} to create a {@link ChronicleHash
* }, configure it's replication via channel of this hub like this: {@code
* ChronicleMap myMap = builder.instance()
* .replicatedViaChannel(hub.createChannel(myMapChannelId))
* .persistedTo(myMapFile) // optional, for this example
* .create();}
*
* @see ChronicleHashInstanceBuilder#replicatedViaChannel(ReplicationChannel)
* @see ReplicationChannel
*/
public final class ReplicationHub extends AbstractReplication {
private final ReplicationChannel[] channels;
private ReplicationHub(byte localIdentifier, Builder builder) {
super(localIdentifier, builder);
channels = new ReplicationChannel[builder.maxNumberOfChannels];
}
/**
* Creates and returns a new {@link ReplicationHub.Builder}.
*
* @return a new {@link ReplicationHub.Builder}
*/
@NotNull
public static Builder builder() {
return new Builder();
}
@Override
public String toString() {
return "ReplicationHub{" + super.toString() +
", channels=" + Arrays.toString(channels) + '}';
}
/**
* Returns the maximum number of channels could be {@linkplain #createChannel(int) created} for
* this {@code ReplicationHub}.
*
* @return the maximum number of channels
* @see Builder#maxNumberOfChannels()
* @see #createChannel(int)
*/
public int maxNumberOfChannels() {
return channels.length;
}
/**
* Creates a new {@link ReplicationChannel} in this {@code ReplicationHub} with the given
* identifier. Identifier shouldn't be lesser than zero and greater or equal to {@code
* maxNumberOfChannels() - 1}. On a {@code ReplicationHub} instance, {@code ReplicationChannel}
* could be created only once for each possible {@code channelId} value.
*
* @param channelId the identifier of the channel. Should be equal for replicated containers on
* different nodes (servers)
* @return a new {@code ReplicationChannel} instance, in fact just incapsulating the given
* {@code channelId} for this {@code ReplicationHub}
* @throws IllegalArgumentException if the specified {@code channelId} is out of
[0,
* {@link #maxNumberOfChannels()})
range
* @throws IllegalStateException if {@code ReplicationChannel} with the specified {@code
* channelId} has already been acquired on this {@code
* ReplicationHub}
* @see ReplicationChannel
*/
public synchronized ReplicationChannel createChannel(int channelId) {
if (channelId < 0)
throw new IllegalArgumentException("channelId should be positive");
if (channelId >= maxNumberOfChannels())
throw new IllegalArgumentException("maxNumberOfChannels is configured (or defaulted) " +
"to " + maxNumberOfChannels() + ", channelId=" + channelId + " is requested");
if (channels[channelId] != null)
throw new IllegalStateException("The requested channelId=" + channelId +
" is already in use");
ReplicationChannel channel = new ReplicationChannel(this, channelId);
channels[channelId] = channel;
return channel;
}
/**
* Builder of {@link ReplicationHub}s.
*/
public static final class Builder extends AbstractReplication.Builder {
private int maxNumberOfChannels = 128;
private Builder() {
}
/**
* Configures the maximum number of channels could be {@linkplain #createChannel(int)}
* created for {@code ReplicationHub}s, created by this builder.
*
*
Default value is {@code 128}.
*
* @param maxNumberOfChannels {@link ReplicationHub#maxNumberOfChannels()} of {@link
* ReplicationHub}s, created by this builder
* @return this builder object back, for chaining
* @see ReplicationHub#maxNumberOfChannels()
*/
@NotNull
public Builder maxNumberOfChannels(int maxNumberOfChannels) {
this.maxNumberOfChannels = maxNumberOfChannels;
return this;
}
@NotNull
@Override
public ReplicationHub createWithId(byte identifier) {
check(identifier);
return new ReplicationHub(identifier, this);
}
}
}