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

com.hazelcast.client.impl.proxy.ClientReliableTopicProxy Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2008-2024, Hazelcast, Inc. All Rights Reserved.
 *
 * 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 com.hazelcast.client.impl.proxy;

import com.hazelcast.client.config.ClientReliableTopicConfig;
import com.hazelcast.client.impl.clientside.HazelcastClientInstanceImpl;
import com.hazelcast.client.impl.spi.ClientContext;
import com.hazelcast.client.impl.spi.ClientProxy;
import com.hazelcast.internal.serialization.Data;
import com.hazelcast.internal.serialization.SerializationService;
import com.hazelcast.internal.util.ConcurrencyUtil;
import com.hazelcast.internal.util.UuidUtil;
import com.hazelcast.logging.ILogger;
import com.hazelcast.ringbuffer.OverflowPolicy;
import com.hazelcast.ringbuffer.Ringbuffer;
import com.hazelcast.spi.impl.InternalCompletableFuture;
import com.hazelcast.topic.ITopic;
import com.hazelcast.topic.LocalTopicStats;
import com.hazelcast.topic.MessageListener;
import com.hazelcast.topic.ReliableMessageListener;
import com.hazelcast.topic.TopicOverloadException;
import com.hazelcast.topic.TopicOverloadPolicy;
import com.hazelcast.topic.impl.reliable.MessageRunner;
import com.hazelcast.topic.impl.reliable.ReliableMessageListenerAdapter;
import com.hazelcast.topic.impl.reliable.ReliableTopicMessage;

import javax.annotation.Nonnull;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.UUID;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.Executor;
import java.util.stream.Collectors;

import static com.hazelcast.internal.util.ConcurrencyUtil.CALLER_RUNS;
import static com.hazelcast.internal.util.ExceptionUtil.peel;
import static com.hazelcast.internal.util.Preconditions.checkNoNullInside;
import static com.hazelcast.internal.util.Preconditions.checkNotNull;
import static com.hazelcast.ringbuffer.impl.RingbufferService.TOPIC_RB_PREFIX;
import static com.hazelcast.topic.impl.reliable.ReliableTopicService.SERVICE_NAME;
import static java.util.concurrent.TimeUnit.MILLISECONDS;

/**
 * Reliable proxy implementation of {@link ITopic}.
 * 

* Unlike other topics, a reliable topic has its own {@link com.hazelcast.ringbuffer.Ringbuffer} to store events and * has its own executor to process events. * * @param message type */ public class ClientReliableTopicProxy extends ClientProxy implements ITopic { private static final String NULL_MESSAGE_IS_NOT_ALLOWED = "Null message is not allowed!"; private static final String NULL_LISTENER_IS_NOT_ALLOWED = "Null listener is not allowed!"; private static final int MAX_BACKOFF = 2000; private static final int INITIAL_BACKOFF_MS = 100; private final ILogger logger; private final ConcurrentMap> runnersMap = new ConcurrentHashMap<>(); private final Ringbuffer ringbuffer; private final SerializationService serializationService; private final ClientReliableTopicConfig config; private final Executor executor; private final TopicOverloadPolicy overloadPolicy; public ClientReliableTopicProxy(String objectId, ClientContext context, HazelcastClientInstanceImpl client) { super(SERVICE_NAME, objectId, context); this.ringbuffer = client.getRingbuffer(TOPIC_RB_PREFIX + objectId); this.serializationService = client.getSerializationService(); this.config = client.getClientConfig().getReliableTopicConfig(objectId); this.executor = getExecutor(config); this.overloadPolicy = config.getTopicOverloadPolicy(); logger = client.getLoggingService().getLogger(getClass()); } private Executor getExecutor(ClientReliableTopicConfig config) { Executor executor = config.getExecutor(); if (executor == null) { executor = ConcurrencyUtil.getDefaultAsyncExecutor(); } return executor; } @Override public void publish(@Nonnull E payload) { checkNotNull(payload, NULL_MESSAGE_IS_NOT_ALLOWED); try { Data data = serializationService.toData(payload); ReliableTopicMessage message = new ReliableTopicMessage(data, null); switch (overloadPolicy) { case ERROR: addOrFail(message); break; case DISCARD_OLDEST: addOrOverwrite(message); break; case DISCARD_NEWEST: ringbuffer.addAsync(message, OverflowPolicy.FAIL).toCompletableFuture().get(); break; case BLOCK: addWithBackoff(Collections.singleton(message)); break; default: throw new IllegalArgumentException("Unknown overloadPolicy:" + overloadPolicy); } } catch (Exception e) { throw (RuntimeException) peel(e, null, "Failed to publish message: " + payload + " to topic:" + getName()); } } @Override public CompletionStage publishAsync(@Nonnull E payload) { checkNotNull(payload, NULL_MESSAGE_IS_NOT_ALLOWED); Collection messages = Collections.singleton(payload); return publishAllAsync(messages); } private void addOrOverwrite(ReliableTopicMessage message) throws Exception { ringbuffer.addAsync(message, OverflowPolicy.OVERWRITE).toCompletableFuture().get(); } private void addOrFail(ReliableTopicMessage message) throws Exception { long sequenceId = ringbuffer.addAsync(message, OverflowPolicy.FAIL).toCompletableFuture().get(); if (sequenceId == -1) { throw new TopicOverloadException("Failed to publish message: " + message + " on topic:" + name); } } private void addWithBackoff(Collection messages) throws Exception { long timeoutMs = INITIAL_BACKOFF_MS; for (; ; ) { long result = ringbuffer.addAllAsync(messages, OverflowPolicy.FAIL).toCompletableFuture().get(); if (result != -1) { break; } MILLISECONDS.sleep(timeoutMs); timeoutMs *= 2; if (timeoutMs > MAX_BACKOFF) { timeoutMs = MAX_BACKOFF; } } } @Nonnull @Override public UUID addMessageListener(@Nonnull MessageListener listener) { checkNotNull(listener, NULL_LISTENER_IS_NOT_ALLOWED); UUID id = UuidUtil.newUnsecureUUID(); ReliableMessageListener reliableMessageListener = toReliableMessageListener(listener); MessageRunner runner = new ClientReliableMessageRunner<>(id, reliableMessageListener, ringbuffer, name, config.getReadBatchSize(), serializationService, executor, runnersMap, logger); runnersMap.put(id, runner); runner.next(); return id; } //for testing public boolean isListenerCancelled(@Nonnull UUID registrationID) { checkNotNull(registrationID, "registrationId can't be null"); MessageRunner runner = runnersMap.get(registrationID); if (runner == null) { return true; } return runner.isCancelled(); } private ReliableMessageListener toReliableMessageListener(MessageListener listener) { if (listener instanceof ReliableMessageListener) { return (ReliableMessageListener) listener; } else { return new ReliableMessageListenerAdapter<>(listener); } } @Override public boolean removeMessageListener(@Nonnull UUID registrationId) { checkNotNull(registrationId, "registrationId can't be null"); MessageRunner runner = runnersMap.get(registrationId); if (runner == null) { return false; } runner.cancel(); return true; } @Nonnull @Override public LocalTopicStats getLocalTopicStats() { throw new UnsupportedOperationException("Locality is ambiguous for client!"); } @Override public void publishAll(@Nonnull Collection payload) { checkNotNull(payload, NULL_MESSAGE_IS_NOT_ALLOWED); checkNoNullInside(payload, NULL_MESSAGE_IS_NOT_ALLOWED); try { List messages = payload.stream() .map(m -> new ReliableTopicMessage(toData(m), null)) .collect(Collectors.toList()); switch (overloadPolicy) { case ERROR: long sequenceId = ringbuffer.addAllAsync(messages, OverflowPolicy.FAIL).toCompletableFuture().get(); if (sequenceId == -1) { throw new TopicOverloadException("Failed to publish messages: " + payload + " on topic:" + getName()); } break; case DISCARD_OLDEST: ringbuffer.addAllAsync(messages, OverflowPolicy.OVERWRITE).toCompletableFuture().get(); break; case DISCARD_NEWEST: ringbuffer.addAllAsync(messages, OverflowPolicy.FAIL).toCompletableFuture().get(); break; case BLOCK: addWithBackoff(messages); break; default: throw new IllegalArgumentException("Unknown overloadPolicy:" + overloadPolicy); } } catch (Exception e) { throw (RuntimeException) peel(e, null, "Failed to publish messages: " + payload + " to topic:" + getName()); } } @Override public CompletionStage publishAllAsync(@Nonnull Collection payload) { checkNotNull(payload, NULL_MESSAGE_IS_NOT_ALLOWED); checkNoNullInside(payload, NULL_MESSAGE_IS_NOT_ALLOWED); InternalCompletableFuture returnFuture = new InternalCompletableFuture<>(); try { List messages = payload.stream() .map(m -> new ReliableTopicMessage(toData(m), null)) .collect(Collectors.toList()); switch (overloadPolicy) { case ERROR: addAsyncOrFail(payload, returnFuture, messages); break; case DISCARD_OLDEST: addAsync(messages, OverflowPolicy.OVERWRITE); break; case DISCARD_NEWEST: addAsync(messages, OverflowPolicy.FAIL); break; case BLOCK: addAsyncAndBlock(returnFuture, messages, INITIAL_BACKOFF_MS); break; default: throw new IllegalArgumentException("Unknown overloadPolicy:" + overloadPolicy); } } catch (Exception e) { throw (RuntimeException) peel(e, null, String.format("Failed to publish messages: %s on topic: %s", payload, getName())); } return returnFuture; } private void addAsyncAndBlock(InternalCompletableFuture returnFuture, List messages, long pauseMillis) { ringbuffer.addAllAsync(messages, OverflowPolicy.FAIL).whenCompleteAsync((id, t) -> { if (t != null) { returnFuture.completeExceptionally(t); } else if (id == -1) { getContext().getTaskScheduler().schedule( () -> addAsyncAndBlock(returnFuture, messages, Math.min(pauseMillis * 2, MAX_BACKOFF)), pauseMillis, MILLISECONDS); } else { returnFuture.complete(null); } }, CALLER_RUNS); } private void addAsyncOrFail(@Nonnull Collection payload, InternalCompletableFuture returnFuture, List messages) { ringbuffer.addAllAsync(messages, OverflowPolicy.FAIL).whenCompleteAsync((id, t) -> { if (t != null) { returnFuture.completeExceptionally(t); } else if (id == -1) { returnFuture.completeExceptionally(new TopicOverloadException( "Failed to publish messages: " + payload + " on topic:" + getName())); } else { returnFuture.complete(null); } }, CALLER_RUNS); } private InternalCompletableFuture addAsync(List messages, OverflowPolicy overflowPolicy) { InternalCompletableFuture returnFuture = new InternalCompletableFuture<>(); ringbuffer.addAllAsync(messages, overflowPolicy).whenCompleteAsync((id, t) -> { if (t != null) { returnFuture.completeExceptionally(t); } else { returnFuture.complete(null); } }, CALLER_RUNS); return returnFuture; } public Ringbuffer getRingbuffer() { return ringbuffer; } @Override public String toString() { return "ITopic{" + "name='" + name + '\'' + '}'; } @Override protected void postDestroy() { // this will trigger all listeners to destroy themselves. ringbuffer.destroy(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy