com.hivemq.client.internal.mqtt.handler.publish.incoming.MqttIncomingPublishFlows Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of hivemq-mqtt-client Show documentation
Show all versions of hivemq-mqtt-client Show documentation
HiveMQ MQTT Client is an MQTT 5.0 and MQTT 3.1.1 compatible and feature-rich high-performance Java client library with different API flavours and backpressure support
/*
* Copyright 2018 dc-square and the HiveMQ MQTT Client Project
*
* 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.hivemq.client.internal.mqtt.handler.publish.incoming;
import com.hivemq.client.internal.annotations.NotThreadSafe;
import com.hivemq.client.internal.mqtt.datatypes.MqttTopicFilterImpl;
import com.hivemq.client.internal.mqtt.datatypes.MqttTopicImpl;
import com.hivemq.client.internal.mqtt.ioc.ClientScope;
import com.hivemq.client.internal.mqtt.message.publish.MqttStatefulPublish;
import com.hivemq.client.internal.mqtt.message.subscribe.MqttStatefulSubscribe;
import com.hivemq.client.internal.mqtt.message.subscribe.MqttSubscription;
import com.hivemq.client.internal.mqtt.message.subscribe.suback.MqttSubAck;
import com.hivemq.client.internal.mqtt.message.unsubscribe.MqttStatefulUnsubscribe;
import com.hivemq.client.internal.mqtt.message.unsubscribe.unsuback.MqttUnsubAck;
import com.hivemq.client.internal.mqtt.message.unsubscribe.unsuback.mqtt3.Mqtt3UnsubAckView;
import com.hivemq.client.internal.util.collections.HandleList;
import com.hivemq.client.internal.util.collections.ImmutableList;
import com.hivemq.client.mqtt.MqttGlobalPublishFilter;
import com.hivemq.client.mqtt.mqtt5.message.subscribe.suback.Mqtt5SubAckReasonCode;
import com.hivemq.client.mqtt.mqtt5.message.unsubscribe.unsuback.Mqtt5UnsubAckReasonCode;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import javax.inject.Inject;
/**
* @author Silvio Giebl
*/
@ClientScope
@NotThreadSafe
public class MqttIncomingPublishFlows {
private final @NotNull MqttSubscriptionFlows subscriptionFlows;
private final @Nullable HandleList @NotNull [] globalFlows;
@Inject
MqttIncomingPublishFlows(final @NotNull MqttSubscriptionFlows subscriptionFlows) {
this.subscriptionFlows = subscriptionFlows;
//noinspection unchecked
globalFlows = new HandleList[MqttGlobalPublishFilter.values().length];
}
public void subscribe(
final @NotNull MqttStatefulSubscribe subscribe, final @Nullable MqttSubscribedPublishFlow flow) {
final ImmutableList subscriptions = subscribe.stateless().getSubscriptions();
//noinspection ForLoopReplaceableByForEach
for (int i = 0; i < subscriptions.size(); i++) {
subscribe(subscriptions.get(i).getTopicFilter(), flow);
}
}
void subscribe(final @NotNull MqttTopicFilterImpl topicFilter, final @Nullable MqttSubscribedPublishFlow flow) {
subscriptionFlows.subscribe(topicFilter, flow);
}
public void subAck(
final @NotNull MqttStatefulSubscribe subscribe, final @NotNull MqttSubAck subAck,
final @Nullable MqttSubscribedPublishFlow flow) {
final ImmutableList subscriptions = subscribe.stateless().getSubscriptions();
final ImmutableList reasonCodes = subAck.getReasonCodes();
for (int i = 0; i < subscriptions.size(); i++) {
if (reasonCodes.get(i).isError()) {
remove(subscriptions.get(i).getTopicFilter(), flow);
}
}
}
void remove(final @NotNull MqttTopicFilterImpl topicFilter, final @Nullable MqttSubscribedPublishFlow flow) {
subscriptionFlows.remove(topicFilter, flow);
}
public void unsubscribe(final @NotNull MqttStatefulUnsubscribe unsubscribe, final @NotNull MqttUnsubAck unsubAck) {
final ImmutableList topicFilters = unsubscribe.stateless().getTopicFilters();
final ImmutableList reasonCodes = unsubAck.getReasonCodes();
final boolean areAllSuccess = reasonCodes == Mqtt3UnsubAckView.REASON_CODES_ALL_SUCCESS;
for (int i = 0; i < topicFilters.size(); i++) {
if (areAllSuccess || !reasonCodes.get(i).isError()) {
unsubscribe(topicFilters.get(i));
}
}
}
void unsubscribe(final @NotNull MqttTopicFilterImpl topicFilter) {
subscriptionFlows.unsubscribe(topicFilter, null);
}
void cancel(final @NotNull MqttSubscribedPublishFlow flow) {
subscriptionFlows.cancel(flow);
}
@NotNull HandleList findMatching(final @NotNull MqttStatefulPublish publish) {
final HandleList matchingFlows = new HandleList<>();
findMatching(publish, matchingFlows);
return matchingFlows;
}
void findMatching(
final @NotNull MqttStatefulPublish publish,
final @NotNull HandleList matchingFlows) {
final MqttTopicImpl topic = publish.stateless().getTopic();
if (subscriptionFlows.findMatching(topic, matchingFlows) || !matchingFlows.isEmpty()) {
add(matchingFlows, globalFlows[MqttGlobalPublishFilter.SUBSCRIBED.ordinal()]);
} else {
add(matchingFlows, globalFlows[MqttGlobalPublishFilter.UNSOLICITED.ordinal()]);
}
add(matchingFlows, globalFlows[MqttGlobalPublishFilter.ALL.ordinal()]);
if (matchingFlows.isEmpty()) {
add(matchingFlows, globalFlows[MqttGlobalPublishFilter.REMAINING.ordinal()]);
}
}
void subscribeGlobal(final @NotNull MqttGlobalIncomingPublishFlow flow) {
final int filter = flow.getFilter().ordinal();
HandleList globalFlow = globalFlows[filter];
if (globalFlow == null) {
globalFlow = new HandleList<>();
globalFlows[filter] = globalFlow;
}
flow.setHandle(globalFlow.add(flow));
}
void cancelGlobal(final @NotNull MqttGlobalIncomingPublishFlow flow) {
final HandleList.Handle handle = flow.getHandle();
assert handle != null;
handle.remove();
final int filter = flow.getFilter().ordinal();
final HandleList globalFlow = globalFlows[filter];
assert globalFlow != null;
if (globalFlow.isEmpty()) {
globalFlows[filter] = null;
}
}
void clear(final @NotNull Throwable cause) {
subscriptionFlows.clear(cause);
for (int i = 0; i < globalFlows.length; i++) {
final HandleList globalFlow = globalFlows[i];
if (globalFlow != null) {
for (final MqttGlobalIncomingPublishFlow flow : globalFlow) {
flow.onError(cause);
}
}
globalFlows[i] = null;
}
}
private static void add(
final @NotNull HandleList target,
final @Nullable HandleList extends MqttIncomingPublishFlow> source) {
if (source != null) {
for (final MqttIncomingPublishFlow flow : source) {
target.add(flow);
}
}
}
}