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

com.hivemq.extensions.handler.tasks.AllTopicsProcessedTask Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2019-present HiveMQ GmbH
 *
 * 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.extensions.handler.tasks;

import com.google.common.base.Preconditions;
import com.google.common.util.concurrent.ListenableFuture;
import com.hivemq.extension.sdk.api.annotations.NotNull;
import com.hivemq.extensions.auth.parameter.SubscriptionAuthorizerOutputImpl;
import com.hivemq.mqtt.handler.disconnect.MqttServerDisconnector;
import com.hivemq.mqtt.handler.subscribe.IncomingSubscribeService;
import com.hivemq.mqtt.message.reason.Mqtt5DisconnectReasonCode;
import com.hivemq.mqtt.message.reason.Mqtt5SubAckReasonCode;
import com.hivemq.mqtt.message.subscribe.SUBSCRIBE;
import io.netty.channel.ChannelHandlerContext;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.List;

/**
 * @author Christoph Schäbel
 */
public class AllTopicsProcessedTask implements Runnable {

    private static final Logger log = LoggerFactory.getLogger(AllTopicsProcessedTask.class);

    private final @NotNull SUBSCRIBE msg;
    private final @NotNull List> listenableFutures;
    private final @NotNull ChannelHandlerContext ctx;
    private final @NotNull MqttServerDisconnector mqttServerDisconnector;
    private final @NotNull IncomingSubscribeService incomingSubscribeService;

    public AllTopicsProcessedTask(
            final @NotNull SUBSCRIBE msg,
            final @NotNull List> listenableFutures,
            final @NotNull ChannelHandlerContext ctx,
            final @NotNull MqttServerDisconnector mqttServerDisconnector,
            final @NotNull IncomingSubscribeService incomingSubscribeService) {

        this.msg = msg;
        this.listenableFutures = listenableFutures;
        this.ctx = ctx;
        this.mqttServerDisconnector = mqttServerDisconnector;
        this.incomingSubscribeService = incomingSubscribeService;
    }

    @Override
    public void run() {

        try {
            Preconditions.checkArgument(listenableFutures.size() == msg.getTopics().size(),
                    "The amount of futures must be equal to the amount of topics");

            final Mqtt5SubAckReasonCode[] answerCodes = new Mqtt5SubAckReasonCode[msg.getTopics().size()];
            final String[] reasonStrings = new String[msg.getTopics().size()];

            boolean authorizersPresent = false;
            for (int i = 0; i < listenableFutures.size(); i++) {
                final SubscriptionAuthorizerOutputImpl output = listenableFutures.get(i).get();

                if (output.isAuthorizerPresent()) {
                    authorizersPresent = true;
                }

                switch (output.getAuthorizationState()) {
                    case CONTINUE:
                        break;
                    case DISCONNECT:
                        disconnectClient(i, output);
                        return;
                    case FAIL:
                        if (output.getSubackReasonCode() != null) {
                            answerCodes[i] = Mqtt5SubAckReasonCode.from(output.getSubackReasonCode());
                            reasonStrings[i] = output.getReasonString();
                        } else {
                            answerCodes[i] = Mqtt5SubAckReasonCode.NOT_AUTHORIZED;
                        }
                        break;
                    case UNDECIDED:
                        if (!output.isAuthorizerPresent()) {
                            //providers never returned an authorizer, same as continue
                            break;
                        }
                        answerCodes[i] = Mqtt5SubAckReasonCode.NOT_AUTHORIZED;
                        reasonStrings[i] = "Sent a SUBSCRIBE with an unauthorized subscription";
                        break;
                    case SUCCESS:
                        answerCodes[i] = Mqtt5SubAckReasonCode.fromCode(msg.getTopics().get(i).getQoS().getQosNumber());
                        break;
                    default:
                        break;
                }
            }

            final boolean finalAuthorizersPresent = authorizersPresent;
            if (ctx.channel().isActive()) {
                ctx.executor()
                        .execute(() -> incomingSubscribeService.processSubscribe(ctx,
                                msg,
                                answerCodes,
                                reasonStrings,
                                finalAuthorizersPresent));
            }

        } catch (final Exception e) {
            log.error("Subscription authorization failed: ", e);
        }

    }

    private void disconnectClient(final int topicIndex, final @NotNull SubscriptionAuthorizerOutputImpl output) {
        final String logMessage = "A client (IP: {}) sent a SUBSCRIBE with an unauthorized subscription for topic '" +
                msg.getTopics().get(topicIndex).getTopic() +
                "'. This is not allowed. Disconnecting client.";
        final String eventLogMessage = "Sent a SUBSCRIBE with an unauthorized subscription for topic '" +
                msg.getTopics().get(topicIndex).getTopic() +
                "'";

        ctx.channel().eventLoop().execute(() -> {
            mqttServerDisconnector.disconnect(ctx.channel(),
                    logMessage,
                    eventLogMessage,
                    Mqtt5DisconnectReasonCode.from(output.getDisconnectReasonCode()),
                    output.getReasonString());
        });
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy