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

io.gravitee.plugin.endpoint.mock.MockEndpointConnector Maven / Gradle / Ivy

/*
 * Copyright © 2015 The Gravitee team (http://gravitee.io)
 *
 * 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 io.gravitee.plugin.endpoint.mock;

import io.gravitee.common.http.HttpHeader;
import io.gravitee.gateway.api.http.HttpHeaders;
import io.gravitee.gateway.reactive.api.ConnectorMode;
import io.gravitee.gateway.reactive.api.connector.endpoint.async.HttpEndpointAsyncConnector;
import io.gravitee.gateway.reactive.api.context.InternalContextAttributes;
import io.gravitee.gateway.reactive.api.context.http.HttpExecutionContext;
import io.gravitee.gateway.reactive.api.message.DefaultMessage;
import io.gravitee.gateway.reactive.api.message.Message;
import io.gravitee.gateway.reactive.api.qos.Qos;
import io.gravitee.gateway.reactive.api.qos.QosCapability;
import io.gravitee.gateway.reactive.api.tracing.message.TracingMessageAttribute;
import io.gravitee.gateway.reactive.api.tracing.message.TracingMessageOperationType;
import io.gravitee.plugin.endpoint.mock.configuration.MockEndpointConnectorConfiguration;
import io.reactivex.rxjava3.core.Completable;
import io.reactivex.rxjava3.core.Flowable;
import io.reactivex.rxjava3.core.Maybe;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;

/**
 * @author GraviteeSource Team
 */
@AllArgsConstructor
@Slf4j
public class MockEndpointConnector extends HttpEndpointAsyncConnector {

    static final Set SUPPORTED_MODES = Set.of(ConnectorMode.PUBLISH, ConnectorMode.SUBSCRIBE);
    static final Set SUPPORTED_QOS = Set.of(Qos.NONE, Qos.AUTO, Qos.AT_LEAST_ONCE, Qos.AT_MOST_ONCE);
    private static final Set SUPPORTED_QOS_CAPABILITIES = Set.of(
        QosCapability.AUTO_ACK,
        QosCapability.MANUAL_ACK,
        QosCapability.RECOVER
    );
    private static final String ENDPOINT_ID = "mock";
    protected final MockEndpointConnectorConfiguration configuration;

    @Override
    public String id() {
        return ENDPOINT_ID;
    }

    @Override
    public Set supportedModes() {
        return SUPPORTED_MODES;
    }

    @Override
    public Set supportedQos() {
        return SUPPORTED_QOS;
    }

    @Override
    public Set supportedQosCapabilities() {
        return SUPPORTED_QOS_CAPABILITIES;
    }

    @Override
    public Completable subscribe(final HttpExecutionContext ctx) {
        return Completable.fromRunnable(() -> {
            final Integer messagesLimitCount = ctx.getInternalAttribute(InternalContextAttributes.ATTR_INTERNAL_MESSAGES_LIMIT_COUNT);
            final Long messagesLimitDurationMs = ctx.getInternalAttribute(
                InternalContextAttributes.ATTR_INTERNAL_MESSAGES_LIMIT_DURATION_MS
            );

            final String messagesResumeLastId = ctx.getInternalAttribute(InternalContextAttributes.ATTR_INTERNAL_MESSAGES_RECOVERY_LAST_ID);

            final Integer maximumPublishedMessages = configuration.getMessageCount();

            ctx
                .response()
                .messages(generateMessageFlow(messagesLimitCount, maximumPublishedMessages, messagesLimitDurationMs, messagesResumeLastId));
        });
    }

    @Override
    public Completable publish(final HttpExecutionContext ctx) {
        return Completable.defer(() ->
            ctx
                .request()
                .onMessage(message -> {
                    log.info("Received message: {}", message.content() != null ? message.content().toString() : null);
                    return Maybe.just(message);
                })
        );
    }

    private Flowable generateMessageFlow(
        final Integer messagesLimitCount,
        final Integer maximumPublishedMessages,
        final Long messagesLimitDurationMs,
        final String lastId
    ) {
        final long stateInitValue = getStateInitValue(lastId);

        Flowable messageFlow = Flowable
            .generate(
                () -> stateInitValue,
                (state, emitter) -> {
                    if (
                        // If we have no published message limit or state is before the limit
                        (maximumPublishedMessages == null || state < maximumPublishedMessages) &&
                        // And the entrypoint has no limit or state minus lastId is less than limit, then emit a message
                        (messagesLimitCount == null || (state - stateInitValue) < messagesLimitCount)
                    ) {
                        DefaultMessage.DefaultMessageBuilder messageBuilder = DefaultMessage
                            .builder()
                            .content(configuration.getMessageContent())
                            .id(Long.toString(state));
                        // handle optional params
                        if (configuration.getHeaders() != null) {
                            HttpHeaders headers = HttpHeaders.create();
                            configuration.getHeaders().forEach(h -> headers.add(h.getName(), h.getValue()));
                            messageBuilder.headers(headers);
                        }
                        if (configuration.getMetadata() != null) {
                            messageBuilder.metadata(
                                configuration.getMetadata().stream().collect(Collectors.toMap(HttpHeader::getName, HttpHeader::getValue))
                            );
                        }
                        messageBuilder.tracingAttributes(
                            Map.of(
                                TracingMessageAttribute.MESSAGING_OPERATION_NAME.key(),
                                "receive",
                                TracingMessageAttribute.MESSAGING_OPERATION_TYPE.key(),
                                TracingMessageOperationType.RECEIVE.value(),
                                TracingMessageAttribute.MESSAGING_SYSTEM.key(),
                                "mock",
                                TracingMessageAttribute.MESSAGING_MESSAGE_BODY_SIZE.key(),
                                messageBuilder.content() != null ? String.valueOf(messageBuilder.content().length()) : "0"
                            )
                        );
                        emitter.onNext(messageBuilder.build());
                    } else {
                        emitter.onComplete();
                    }
                    return state + 1;
                }
            )
            .delay(configuration.getMessageInterval(), TimeUnit.MILLISECONDS)
            .rebatchRequests(1);

        if (messagesLimitDurationMs != null) {
            messageFlow = messageFlow.take(messagesLimitDurationMs, TimeUnit.MILLISECONDS);
        }

        return messageFlow;
    }

    private long getStateInitValue(final String lastId) {
        long stateInitValue = 0L;
        if (lastId != null) {
            try {
                stateInitValue = Long.parseLong(lastId) + 1;
            } catch (NumberFormatException nfe) {
                log.warn("Unable to parse lastId: {}. Setting to 0", lastId);
            }
        }

        return stateInitValue;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy