io.hekate.messaging.MessagingChannel 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.messaging;
import io.hekate.cluster.ClusterFilterSupport;
import io.hekate.cluster.ClusterView;
import io.hekate.core.HekateException;
import io.hekate.messaging.loadbalance.DefaultLoadBalancer;
import io.hekate.messaging.loadbalance.LoadBalancer;
import io.hekate.messaging.operation.AckMode;
import io.hekate.messaging.operation.Aggregate;
import io.hekate.messaging.operation.AggregateFuture;
import io.hekate.messaging.operation.AggregateResult;
import io.hekate.messaging.operation.Broadcast;
import io.hekate.messaging.operation.BroadcastFuture;
import io.hekate.messaging.operation.BroadcastResult;
import io.hekate.messaging.operation.Request;
import io.hekate.messaging.operation.RequestFuture;
import io.hekate.messaging.operation.ResponsePart;
import io.hekate.messaging.operation.Send;
import io.hekate.messaging.operation.SendFuture;
import io.hekate.messaging.operation.Subscribe;
import io.hekate.messaging.operation.SubscribeCallback;
import io.hekate.messaging.operation.SubscribeFuture;
import io.hekate.partition.PartitionMapper;
import io.hekate.partition.RendezvousHashMapper;
import java.util.List;
import java.util.concurrent.Executor;
/**
* Messaging channel.
*
*
* This interface represents a channel for exchanging messages with remote nodes.
*
*
*
* Instances of this interface can be obtained via the {@link MessagingService#channel(String, Class)} method.
*
*
*
* For more details about messaging and channels please see the documentation of {@link MessagingService} interface.
*
*
* @param Base type fo messages that can be handled by channels.
*
* @see MessagingService#channel(String, Class)
*/
public interface MessagingChannel extends ClusterFilterSupport> {
/**
* Creates a new {@link Send} operation.
*
*
* Send operation doesn't assume any response to be received from the destination node. If request-response type of communication is
* required then consider using the {@link #newRequest(Object)} method.
*
*
*
* By default, this operation will not wait for the message to be processed on the receiver side. It is possible to change this behavior
* via {@link Send#withAckMode(AckMode)} method.
*
*
* @param message Message to be sent.
*
* @return New operation.
*
* @see Send#submit()
*/
Send newSend(T message);
/**
* Creates a new {@link Request} operation.
*
* @param request Request.
*
* @return New operation.
*
* @see Request#submit()
*/
Request newRequest(T request);
/**
* Creates a new {@link Subscribe} operation.
*
* @param request Subscription request.
*
* @return New operation.
*
* @see Subscribe#submit(SubscribeCallback)
*/
Subscribe newSubscribe(T request);
/**
* Creates a new {@link Broadcast} operation.
*
*
* By default, this operation will not wait for the message to be processed on the receiver side. It is possible to change this behavior
* via {@link Broadcast#withAckMode(AckMode)} method.
*
*
* @param request Message to broadcast.
*
* @return New operation.
*
* @see Broadcast#submit()
*/
Broadcast newBroadcast(T request);
/**
* Creates a new {@link Aggregate} operation.
*
* @param request Aggregation request.
*
* @return New operation.
*
* @see Aggregate#submit()
*/
Aggregate newAggregate(T request);
/**
* Returns the universally unique identifier of this channel.
*
* @return Universally unique identifier of this channel.
*/
MessagingChannelId id();
/**
* Returns the channel name.
*
* @return Channel name.
*
* @see MessagingChannelConfig#setName(String)
*/
String name();
/**
* Returns the base type of messages that can be transferred through this channel.
*
* @return Base type of messages that can be transferred through this channel.
*
* @see MessagingChannelConfig#MessagingChannelConfig(Class)
*/
Class baseType();
/**
* Returns the size of a thread pool for handling NIO-based socket connections
* (see {@link MessagingChannelConfig#setNioThreads(int)}).
*
* @return Size of a thread pool for handling NIO-based socket connections.
*/
int nioThreads();
/**
* Returns the worker thread pool size (see {@link MessagingChannelConfig#setWorkerThreads(int)}).
*
* @return Worker thread pool size.
*/
int workerThreads();
/**
* Returns the partition mapper of this channel.
*
* @return Mapper.
*
* @see MessagingChannelConfig#setPartitions(int)
* @see MessagingChannelConfig#setBackupNodes(int)
* @see #withPartitions(int, int)
*/
PartitionMapper partitions();
/**
* Returns a copy of this channel that will use a {@link RendezvousHashMapper} with the specified options.
*
* @param partitions See {@link RendezvousHashMapper.Builder#withPartitions(int)}.
* @param backupNodes See {@link RendezvousHashMapper.Builder#withBackupNodes(int)}.
*
* @return Channel wrapper.
*
* @see #partitions()
*/
MessagingChannel withPartitions(int partitions, int backupNodes);
/**
* Returns a copy of this channel that will use the specified {@link PartitionMapper}.
*
* @param mapper Partition mapper.
*
* @return Channel wrapper.
*
* @see #partitions()
*/
MessagingChannel withPartitions(PartitionMapper mapper);
/**
* Returns a copy of this channel that will use the specified load balancer and will inherit all other options from this instance.
*
*
* If not specified or set to {@code null} then {@link DefaultLoadBalancer} will be used.
*
*
* @param balancer Load balancer.
*
* @return Channel wrapper.
*
* @see MessagingChannelConfig#setLoadBalancer(LoadBalancer)
*/
MessagingChannel withLoadBalancer(LoadBalancer balancer);
/**
* Returns the cluster view of this channel.
*
*
* The returned cluster view contains only those nodes that have a {@link MessagingChannelConfig#setReceiver(MessageReceiver) receiver}
* and do match the channel's {@link ClusterFilterSupport filtering} criteria.
*
*
* @return Cluster view.
*/
ClusterView cluster();
/**
* Returns a copy of this channel that will use the specified cluster view and will inherit all other options from this instance.
*
* @param cluster Cluster view.
*
* @return Channel wrapper.
*/
MessagingChannel withCluster(ClusterView cluster);
/**
* Returns the executor of this channel.
*
* @return Asynchronous task executor of this channel.
*
* @see MessagingChannelConfig#setWorkerThreads(int)
*/
Executor executor();
/**
* Asynchronously sends the specified message with a mandatory acknowledgement.
*
* @param msg Message.
*
* @return Future object to be completed after receiving an acknowledgement.
*/
default SendFuture sendAsync(T msg) {
return newSend(msg)
.withAckMode(AckMode.REQUIRED)
.submit();
}
/**
* Asynchronously sends the specified message with the specified acknowledgement mode.
*
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @return Operation future.
*/
default SendFuture sendAsync(T msg, AckMode ackMode) {
return newSend(msg)
.withAckMode(ackMode)
.submit();
}
/**
* Asynchronously sends the specified message with a mandatory acknowledgement.
*
* @param affinityKey Affinity key (see {@link Send#withAffinity(Object)}).
* @param msg Message.
*
* @return Future object to be completed after receiving an acknowledgement.
*/
default SendFuture sendAsync(Object affinityKey, T msg) {
return newSend(msg)
.withAckMode(AckMode.REQUIRED)
.withAffinity(affinityKey)
.submit();
}
/**
* Asynchronously sends the specified message with the specified acknowledgement mode.
*
* @param affinityKey Affinity key (see {@link Send#withAffinity(Object)}).
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @return Operation future.
*/
default SendFuture sendAsync(Object affinityKey, T msg, AckMode ackMode) {
return newSend(msg)
.withAckMode(ackMode)
.withAffinity(affinityKey)
.submit();
}
/**
* Synchronously sends the specified message and awaits for an acknowledgement.
*
* @param msg Message.
*
* @throws HekateException If operation failed.
*/
default void send(T msg) throws HekateException {
newSend(msg)
.withAckMode(AckMode.REQUIRED)
.sync();
}
/**
* Synchronously sends the specified message with the specified acknowledgement mode.
*
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @throws HekateException If operation failed.
*/
default void send(T msg, AckMode ackMode) throws HekateException {
newSend(msg)
.withAckMode(ackMode)
.sync();
}
/**
* Synchronously sends the specified message and awaits for an acknowledgement.
*
* @param affinityKey Affinity key (see {@link Send#withAffinity(Object)}).
* @param msg Message.
*
* @throws HekateException If operation failed.
*/
default void send(Object affinityKey, T msg) throws HekateException {
newSend(msg)
.withAckMode(AckMode.REQUIRED)
.withAffinity(affinityKey)
.sync();
}
/**
* Synchronously sends the specified message with the specified acknowledgement mode.
*
* @param affinityKey Affinity key (see {@link Send#withAffinity(Object)}).
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @throws HekateException If operation failed.
*/
default void send(Object affinityKey, T msg, AckMode ackMode) throws HekateException {
newSend(msg)
.withAckMode(ackMode)
.withAffinity(affinityKey)
.sync();
}
/**
* Asynchronously sends the specified request.
*
* @param msg Request message.
*
* @return Future object to be completed after receiving a response.
*/
default RequestFuture requestAsync(T msg) {
return newRequest(msg).submit();
}
/**
* Asynchronously sends the specified request.
*
* @param affinityKey Affinity key (see {@link Request#withAffinity(Object)}).
* @param msg Request message.
*
* @return Future object to be completed after receiving a response.
*/
default RequestFuture requestAsync(Object affinityKey, T msg) {
return newRequest(msg)
.withAffinity(affinityKey)
.submit();
}
/**
* Synchronously sends the specified request and awaits for a response.
*
* @param msg Request message.
*
* @return Response.
*
* @throws HekateException If operation failed.
*/
default T request(T msg) throws HekateException {
return newRequest(msg).response();
}
/**
* Synchronously sends the specified request and awaits for a response.
*
* @param affinityKey Affinity key (see {@link Request#withAffinity(Object)}).
* @param msg Request message.
*
* @return Response.
*
* @throws HekateException If operation failed.
*/
default T request(Object affinityKey, T msg) throws HekateException {
return newRequest(msg)
.withAffinity(affinityKey)
.response();
}
/**
* Asynchronously sends the specified subscription request.
*
* @param msg Subscription request message.
* @param callback Subscription callback.
*
* @return Future object to be completed after receiving the final response (see {@link ResponsePart#isLastPart()}).
*/
default SubscribeFuture subscribeAsync(T msg, SubscribeCallback callback) {
return newSubscribe(msg).submit(callback);
}
/**
* Asynchronously sends the specified subscription request.
*
* @param affinityKey Affinity key (see {@link Subscribe#withAffinity(Object)}).
* @param msg Subscription request message.
* @param callback Subscription callback.
*
* @return Future object to be completed after receiving the final response (see {@link ResponsePart#isLastPart()}).
*/
default SubscribeFuture subscribeAsync(Object affinityKey, T msg, SubscribeCallback callback) {
return newSubscribe(msg)
.withAffinity(affinityKey)
.submit(callback);
}
/**
* Synchronously sends the specified subscription request and accumulates all responses.
*
*
* Notice: This method blocks until the final response is received (see {@link ResponsePart#isLastPart()}) and accumulates all
* intermediate parts in memory.
*
*
* @param msg Subscription request message.
*
* @return All response parts that were received.
*
* @throws HekateException If operation failed.
*/
default List subscribe(T msg) throws HekateException {
return newSubscribe(msg).responses();
}
/**
* Synchronously sends the specified subscription request and accumulates all responses.
*
*
* Notice: This method blocks until the final response is received (see {@link ResponsePart#isLastPart()}) and accumulates all
* intermediate parts in memory.
*
*
* @param affinityKey Affinity key (see {@link Subscribe#withAffinity(Object)}).
* @param msg Subscription request message.
*
* @return All response parts that were received.
*
* @throws HekateException If operation failed.
*/
default List subscribe(Object affinityKey, T msg) throws HekateException {
return newSubscribe(msg)
.withAffinity(affinityKey)
.responses();
}
/**
* Asynchronously broadcasts the specified message with a mandatory acknowledgement.
*
* @param msg Message.
*
* @return Future object to be completed after receiving acknowledgements from all nodes.
*/
default BroadcastFuture broadcastAsync(T msg) {
return newBroadcast(msg)
.withAckMode(AckMode.REQUIRED)
.submit();
}
/**
* Asynchronously broadcasts the specified message with the specified acknowledgement mode.
*
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @return Operation future.
*/
default BroadcastFuture broadcastAsync(T msg, AckMode ackMode) {
return newBroadcast(msg)
.withAckMode(ackMode)
.submit();
}
/**
* Asynchronously broadcasts the specified message with a mandatory acknowledgement.
*
* @param affinityKey Affinity key (see {@link Broadcast#withAffinity(Object)}).
* @param msg Message.
*
* @return Future object to be completed after receiving acknowledgements from all nodes.
*/
default BroadcastFuture broadcastAsync(Object affinityKey, T msg) {
return newBroadcast(msg)
.withAckMode(AckMode.REQUIRED)
.withAffinity(affinityKey)
.submit();
}
/**
* Asynchronously broadcasts the specified message with the specified acknowledgement mode.
*
* @param affinityKey Affinity key (see {@link Broadcast#withAffinity(Object)}).
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @return Operation future.
*/
default BroadcastFuture broadcastAsync(Object affinityKey, T msg, AckMode ackMode) {
return newBroadcast(msg)
.withAckMode(ackMode)
.withAffinity(affinityKey)
.submit();
}
/**
* Synchronously broadcasts the specified message with a mandatory acknowledgement.
*
* @param msg Message.
*
* @return Broadcast result.
*
* @throws HekateException If operation failed.
*/
default BroadcastResult broadcast(T msg) throws HekateException {
return newBroadcast(msg)
.withAckMode(AckMode.REQUIRED)
.sync();
}
/**
* Synchronously broadcasts the specified message with the specified acknowledgement mode.
*
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @return Broadcast result.
*
* @throws HekateException If operation failed.
*/
default BroadcastResult broadcast(T msg, AckMode ackMode) throws HekateException {
return newBroadcast(msg)
.withAckMode(ackMode)
.sync();
}
/**
* Synchronously broadcasts the specified message with a mandatory acknowledgement.
*
* @param affinityKey Affinity key (see {@link Broadcast#withAffinity(Object)}).
* @param msg Message.
*
* @return Broadcast result.
*
* @throws HekateException If operation failed.
*/
default BroadcastResult broadcast(Object affinityKey, T msg) throws HekateException {
return newBroadcast(msg)
.withAckMode(AckMode.REQUIRED)
.withAffinity(affinityKey).sync();
}
/**
* Synchronously broadcasts the specified message with the specified acknowledgement mode.
*
* @param affinityKey Affinity key (see {@link Broadcast#withAffinity(Object)}).
* @param msg Message.
* @param ackMode Acknowledgement mode.
*
* @return Broadcast result.
*
* @throws HekateException If operation failed.
*/
default BroadcastResult broadcast(Object affinityKey, T msg, AckMode ackMode) throws HekateException {
return newBroadcast(msg)
.withAckMode(ackMode)
.withAffinity(affinityKey)
.sync();
}
/**
* Asynchronously submits the specified aggregation request.
*
* @param msg Aggregation request message.
*
* @return Future object to be completed after receiving responses from all nodes.
*/
default AggregateFuture aggregateAsync(T msg) {
return newAggregate(msg).submit();
}
/**
* Asynchronously submits the specified aggregation request.
*
* @param affinityKey Affinity key (see {@link Aggregate#withAffinity(Object)}).
* @param msg Aggregation request message.
*
* @return Future object to be completed after receiving responses from all nodes.
*/
default AggregateFuture aggregateAsync(Object affinityKey, T msg) {
return newAggregate(msg).withAffinity(affinityKey).submit();
}
/**
* Synchronously submits the specified aggregation request.
*
* @param msg Aggregation request message.
*
* @return Aggregation result.
*
* @throws HekateException If operation failed.
*/
default AggregateResult aggregate(T msg) throws HekateException {
return newAggregate(msg).get();
}
/**
* Synchronously submits the specified aggregation request.
*
* @param affinityKey Affinity key (see {@link Aggregate#withAffinity(Object)}).
* @param msg Aggregation request message.
*
* @return Aggregation result.
*
* @throws HekateException If operation failed.
*/
default AggregateResult aggregate(Object affinityKey, T msg) throws HekateException {
return newAggregate(msg).withAffinity(affinityKey).get();
}
}