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

com.evento.application.EventoBundle Maven / Gradle / Ivy

Go to download

Evento Framework - Bundle. The library to build a RECQ System based on Evento Server

There is a newer version: ev1.10.3
Show newest version
package com.evento.application;

import com.evento.application.manager.*;
import com.evento.application.performance.TracingAgent;
import com.evento.application.performance.Track;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.module.jsonSchema.JsonSchemaGenerator;
import javassist.util.proxy.MethodHandler;
import javassist.util.proxy.ProxyFactory;
import lombok.AccessLevel;
import lombok.Getter;
import lombok.Setter;
import lombok.experimental.Accessors;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import com.evento.application.bus.EventoServerClient;
import com.evento.application.bus.EventoServerMessageBusConfiguration;
import com.evento.application.proxy.GatewayTelemetryProxy;
import com.evento.application.proxy.InvokerWrapper;
import com.evento.common.documentation.Domain;
import com.evento.common.messaging.bus.EventoServer;
import com.evento.common.messaging.consumer.ConsumerStateStore;
import com.evento.common.messaging.consumer.impl.InMemoryConsumerStateStore;
import com.evento.common.messaging.gateway.CommandGateway;
import com.evento.common.messaging.gateway.CommandGatewayImpl;
import com.evento.common.messaging.gateway.QueryGateway;
import com.evento.common.messaging.gateway.QueryGatewayImpl;
import com.evento.common.modeling.annotations.handler.InvocationHandler;
import com.evento.common.modeling.annotations.handler.SagaEventHandler;
import com.evento.common.modeling.bundle.types.ComponentType;
import com.evento.common.modeling.bundle.types.HandlerType;
import com.evento.common.modeling.bundle.types.PayloadType;
import com.evento.common.modeling.messaging.message.application.*;
import com.evento.common.modeling.messaging.message.internal.discovery.BundleRegistration;
import com.evento.common.modeling.messaging.message.internal.discovery.RegisteredHandler;
import com.evento.common.modeling.messaging.payload.DomainEvent;
import com.evento.common.modeling.messaging.query.Multiple;
import com.evento.common.performance.AutoscalingProtocol;
import com.evento.common.performance.PerformanceService;
import com.evento.common.performance.RemotePerformanceService;
import com.evento.common.serialization.ObjectMapperUtils;
import org.reflections.Reflections;
import org.reflections.util.ConfigurationBuilder;

import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.time.Instant;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Semaphore;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.BiFunction;
import java.util.function.Function;

/**
 * The EventoBundle class represents a bundle of components and services related to event handling.
 * It provides functionality for starting saga event consumers and projector event consumers.
 */
@Getter
public class EventoBundle {

    private static final Logger logger = LogManager.getLogger(EventoBundle.class);
    private final String basePackage;
    private final String bundleId;
    private final String instanceId;
    private final PerformanceService performanceService;
    private final AggregateManager aggregateManager;
    private final ServiceManager serviceManager;
    private final ProjectionManager projectionManager;
    private final ProjectorManager projectorManager;
    private final ObserverManager observerManager;
    private final SagaManager sagaManager;
    private final transient CommandGateway commandGateway;
    private final transient QueryGateway queryGateway;
    private final TracingAgent tracingAgent;

    private EventoBundle(
            String basePackage,
            String bundleId,
            String instanceId,
            AggregateManager aggregateManager,
            ProjectionManager projectionManager,
            SagaManager sagaManager,
            CommandGateway commandGateway,
            QueryGateway queryGateway,
            PerformanceService performanceService,
            ServiceManager serviceManager, ProjectorManager projectorManager,
            ObserverManager observerManager, TracingAgent tracingAgent

    ) {
        this.basePackage = basePackage;
        this.bundleId = bundleId;
        this.instanceId = instanceId;
        this.aggregateManager = aggregateManager;
        this.projectionManager = projectionManager;
        this.sagaManager = sagaManager;
        this.performanceService = performanceService;
        this.commandGateway = commandGateway;
        this.queryGateway = queryGateway;
        this.serviceManager = serviceManager;
        this.projectorManager = projectorManager;
        this.tracingAgent = tracingAgent;
        this.observerManager = observerManager;
    }

    /**
     * Creates a GatewayTelemetryProxy with the provided parameters.
     *
     * @param commandGateway     The command gateway to proxy.
     * @param queryGateway       The query gateway to proxy.
     * @param bundleId           The bundle identifier.
     * @param performanceService The performance service for tracking metrics.
     * @param tracingAgent       The tracing agent for correlating and tracking.
     * @param componentName      The name of the component associated with the proxy.
     * @param handledMessage     The message being handled by the proxy.
     * @return The created GatewayTelemetryProxy instance.
     */
    private static GatewayTelemetryProxy createGatewayTelemetryProxy(
            CommandGateway commandGateway,
            QueryGateway queryGateway,
            String bundleId,
            String instanceId,
            PerformanceService performanceService,
            TracingAgent tracingAgent,
            String componentName, Message handledMessage) {
        return new GatewayTelemetryProxy(commandGateway, queryGateway, bundleId, performanceService,
                componentName, handledMessage, tracingAgent, instanceId);
    }


    /**
     * Retrieves an instance of the specified InvokerWrapper class.
     *
     * @param invokerClass The class of the InvokerWrapper to retrieve.
     * @param           The type of the InvokerWrapper.
     * @return An instance of the specified InvokerWrapper class.
     * @throws RuntimeException if an error occurs while creating the instance.
     */
    @SuppressWarnings("unchecked")
    public  T getInvoker(Class invokerClass) {
        ProxyFactory factory = new ProxyFactory();
        factory.setSuperclass(invokerClass);
        var h = new MethodHandler() {
            @Override
            public Object invoke(Object self, Method method, Method proceed, Object[] args) throws Exception {

                if (method.getDeclaredAnnotation(InvocationHandler.class) != null) {
                    var payload = new InvocationMessage(
                            invokerClass, method, args
                    );
                    var gProxy = createGatewayTelemetryProxy(
                            commandGateway,
                            queryGateway,
                            bundleId, instanceId,
                            performanceService,
                            tracingAgent,
                            invokerClass.getSimpleName(),
                            payload
                    );
                    ProxyFactory factory = new ProxyFactory();
                    factory.setSuperclass(invokerClass);
                    var target = factory.create(new Class[0], new Object[]{},
                            (s, m, p, a) -> {
                                if (m.getName().equals("getCommandGateway")) {
                                    return gProxy;
                                }
                                if (m.getName().equals("getQueryGateway")) {
                                    return gProxy;
                                }
                                return p.invoke(s, a);
                            });
                    var start = Instant.now();
                    return tracingAgent.track(payload, invokerClass.getSimpleName(),
                            method.getDeclaredAnnotation(Track.class),
                            () -> {
                                Object result;
                                try{
                                    result = proceed.invoke(target, args);
                                }catch (Exception e){
                                    gProxy.sendServiceTimeMetric(start);
                                    throw e;
                                }
                                if(result instanceof CompletableFuture cf){
                                    result = cf.whenComplete((s,f) -> gProxy.sendServiceTimeMetric(start));
                                }else{
                                    gProxy.sendServiceTimeMetric(start);
                                }
                                return result;
                            });
                }

                return proceed.invoke(self, args);
            }
        };

        try {
            return (T) factory.create(new Class[0], new Object[]{}, h);
        } catch (NoSuchMethodException | InstantiationException | IllegalAccessException |
                 InvocationTargetException e) {
            throw new RuntimeException(e);
        }

    }

    /**
     * Retrieves information about the application.
     *
     * @return An instance of the ApplicationInfo class containing the following information:
     *         - basePackage: The base package of the application.
     *         - bundleId: The bundle identifier of the application.
     *         - aggregateMessageHandlers: The set of aggregate message handlers in the application.
     *         - serviceMessageHandlers: The set of service message handlers in the application.
     *         - projectionMessageHandlers: The set of projection message handlers in the application.
     *         - projectorMessageHandlers: The set of projector message handlers in the application.
     *         - sagaMessageHandlers: The set of saga message handlers in the application.
     */
    public ApplicationInfo getAppInfo() {
        var info = new ApplicationInfo();
        info.basePackage = basePackage;
        info.bundleId = bundleId;
        info.aggregateMessageHandlers = aggregateManager.getHandlers().keySet();
        info.serviceMessageHandlers = serviceManager.getHandlers().keySet();
        info.projectionMessageHandlers = projectionManager.getHandlers().keySet();
        info.projectorMessageHandlers = projectorManager.getHandlers().keySet();
        info.sagaMessageHandlers = sagaManager.getHandlers().keySet();
        return info;
    }
    /**
     * Represents information about an application.
     */
    public static class ApplicationInfo {
        /**
         * Represents the base package that is used for scanning components in an application.
         */
        public String basePackage;
        /**
         * Represents the bundle identifier of an application.
         */
        public String bundleId;

        /**
         * Represents a set of aggregate message handlers.
         */
        public Set aggregateMessageHandlers;
        /**
         * The set of service message handlers.
         */
        public Set serviceMessageHandlers;
        /**
         * Represents a set of projection message handlers.
         */
        public Set projectionMessageHandlers;
        /**
         * A set of available message handlers for the projector.
         */
        public Set projectorMessageHandlers;
        /**
         * A set of string values representing the saga message handlers.
         *
         */
        public Set sagaMessageHandlers;
    }

    /**
     * The Builder class is responsible for constructing an EventoBundle and starting the Evento application.
     */
    @Getter
    @Setter
    @Accessors(chain = true)
    public static class Builder {
        private Package basePackage;
        private String bundleId;
        private String instanceId;
        private long bundleVersion = 1;
        private Function, Object> injector;

        private Function autoscalingProtocolBuilder;
        private BiFunction consumerStateStoreBuilder;
        private Function commandGatewayBuilder  = CommandGatewayImpl::new;
        @Setter(AccessLevel.NONE)
        private CommandGateway commandGateway;

        private Function queryGatewayBuilder  = QueryGatewayImpl::new;
        @Setter(AccessLevel.NONE)
        private QueryGateway queryGateway;

        private Function performanceServiceBuilder = eventoServer -> new RemotePerformanceService(eventoServer, 1);
        @Setter(AccessLevel.NONE)
        private PerformanceService performanceService;

        private int sssFetchSize = 1000;
        private int sssFetchDelay = 1000;

        private TracingAgent tracingAgent;

        private EventoServerMessageBusConfiguration eventoServerMessageBusConfiguration;

        private ObjectMapper objectMapper = ObjectMapperUtils.getPayloadObjectMapper();
        private Map> contexts = new HashMap<>();

        /**
         * The Builder class represents a builder for constructing objects.
         */
        private Builder() {
        }

        /**
         * Returns a new instance of the Builder class.
         *
         * @return a new instance of the Builder class
         */
        public static Builder builder() {
            return new Builder();
        }

        /**
         * Starts the Evento Application.
         *
         * @return the EventoBundle representing the started application
         * @throws Exception if there is an error during initialization
         */
        public EventoBundle start() throws Exception {
            if (basePackage == null) {
                throw new IllegalArgumentException("Invalid basePackage");
            }
            if (bundleId == null || bundleId.isBlank() || bundleId.isEmpty()) {
                throw new IllegalArgumentException("Invalid bundleId");
            }
            if (eventoServerMessageBusConfiguration == null) {
                throw new IllegalArgumentException("Invalid messageBusConfiguration");
            }

            if (injector == null) {
                injector = clz -> null;
            }


            if (instanceId == null || instanceId.isBlank() || instanceId.isEmpty()) {
                instanceId = UUID.randomUUID().toString();
            }


            if (sssFetchSize < 1) {
                sssFetchSize = 1;
            }
            if (sssFetchDelay < 100) {
                sssFetchDelay = 100;
            }
            if (tracingAgent == null) {
                tracingAgent = new TracingAgent(bundleId, bundleVersion);
            }


            var isShuttingDown = new AtomicBoolean();

            var aggregateManager = new AggregateManager(
                    bundleId,
                    (c, p) -> createGatewayTelemetryProxy(commandGateway, queryGateway, bundleId, instanceId, performanceService,
                            tracingAgent, c, p),
                    tracingAgent
            );
            var serviceManager = new ServiceManager(
                    bundleId,
                    (c, p) -> createGatewayTelemetryProxy(commandGateway, queryGateway, bundleId, instanceId, performanceService,
                            tracingAgent, c, p),
                    tracingAgent
            );
            var projectionManager = new ProjectionManager(
                    bundleId,
                    (c, p) -> createGatewayTelemetryProxy(commandGateway, queryGateway, bundleId, instanceId, performanceService,
                            tracingAgent, c, p),
                    tracingAgent
            );
            var projectorManager = new ProjectorManager(
                    bundleId,
                    (c, p) -> createGatewayTelemetryProxy(commandGateway, queryGateway, bundleId, instanceId, performanceService,
                            tracingAgent, c, p),
                    tracingAgent,
                    isShuttingDown::get,
                    sssFetchSize,
                    sssFetchDelay
            );
            var sagaManager = new SagaManager(
                    bundleId,
                    (c, p) -> createGatewayTelemetryProxy(commandGateway, queryGateway, bundleId, instanceId, performanceService,
                            tracingAgent, c, p),
                    tracingAgent,
                    isShuttingDown::get,
                    sssFetchSize,
                    sssFetchDelay
            );
            var observerManager = new ObserverManager(
                    bundleId,
                    (c, p) -> createGatewayTelemetryProxy(commandGateway, queryGateway, bundleId, instanceId, performanceService,
                            tracingAgent, c, p),
                    tracingAgent,
                    isShuttingDown::get,
                    sssFetchSize,
                    sssFetchDelay
            );
            var invokerManager = new InvokerManager();

            logger.info("Discovery handlers in %s".formatted(basePackage));
            Reflections reflections = new Reflections((new ConfigurationBuilder().forPackages(basePackage.getName())));

            aggregateManager.parse(reflections, injector);
            serviceManager.parse(reflections, injector);
            projectionManager.parse(reflections, injector);
            projectorManager.parse(reflections, injector);
            sagaManager.parse(reflections, injector);
            observerManager.parse(reflections, injector);
            invokerManager.parse(reflections);

            logger.info("Discovery Complete");

            var handlers = new ArrayList();
            var payloads = new HashSet>();
            aggregateManager.getHandlers().forEach((k, v) -> {
                var r = v.getAggregateCommandHandler(k).getReturnType().getSimpleName();
                handlers.add(new RegisteredHandler(
                        ComponentType.Aggregate,
                        v.getRef().getClass().getSimpleName(),
                        HandlerType.AggregateCommandHandler,
                        PayloadType.DomainCommand,
                        k,
                        r,
                        false,
                        null
                ));
                var esh = v.getEventSourcingHandler(r);
                if (esh != null) {
                    handlers.add(new RegisteredHandler(
                            ComponentType.Aggregate,
                            v.getRef().getClass().getSimpleName(),
                            HandlerType.EventSourcingHandler,
                            PayloadType.DomainEvent,
                            r,
                            null,
                            false,
                            null
                    ));

                }
                payloads.add(v.getAggregateCommandHandler(k).getParameterTypes()[0]);
                payloads.add(v.getAggregateCommandHandler(k).getReturnType());
            });
            serviceManager.getHandlers().forEach((k, v) -> {
                var r = v.getAggregateCommandHandler(k).getReturnType().getSimpleName();
                handlers.add(new RegisteredHandler(
                        ComponentType.Service,
                        v.getRef().getClass().getSimpleName(),
                        HandlerType.CommandHandler,
                        PayloadType.ServiceCommand,
                        k,
                        r.equals("void") ? null : r,
                        false,
                        null
                ));
                payloads.add(v.getAggregateCommandHandler(k).getParameterTypes()[0]);
                payloads.add(v.getAggregateCommandHandler(k).getReturnType());
            });
            projectorManager.getHandlers().forEach((k, v) -> v.forEach((k1, v1) -> {
                handlers.add(new RegisteredHandler(
                        ComponentType.Projector,
                        v1.getRef().getClass().getSimpleName(),
                        HandlerType.EventHandler,
                        v1.getEventHandler(k).getParameterTypes()[0].getSuperclass().isAssignableFrom(DomainEvent.class) ? PayloadType.DomainEvent : PayloadType.ServiceEvent,
                        k,
                        null,
                        false,
                        null
                ));
                payloads.add(v1.getEventHandler(k).getParameterTypes()[0]);
            }));
            observerManager.getHandlers().forEach((k, v) -> v.forEach((k1, v1) -> {
                handlers.add(new RegisteredHandler(
                        ComponentType.Observer,
                        v1.getRef().getClass().getSimpleName(),
                        HandlerType.EventHandler,
                        v1.getEventHandler(k).getParameterTypes()[0].getSuperclass().isAssignableFrom(DomainEvent.class) ? PayloadType.DomainEvent : PayloadType.ServiceEvent,
                        k,
                        null,
                        false,
                        null
                ));
                payloads.add(v1.getEventHandler(k).getParameterTypes()[0]);
            }));
            sagaManager.getHandlers().forEach((k, v) -> v.forEach((k1, v1) -> {
                handlers.add(new RegisteredHandler(
                        ComponentType.Saga,
                        v1.getRef().getClass().getSimpleName(),
                        HandlerType.SagaEventHandler,
                        v1.getSagaEventHandler(k).getParameterTypes()[0].getSuperclass().isAssignableFrom(DomainEvent.class) ? PayloadType.DomainEvent : PayloadType.ServiceEvent,
                        k,
                        null,
                        false,
                        v1.getSagaEventHandler(k).getAnnotation(SagaEventHandler.class).associationProperty()
                ));
                payloads.add(v1.getSagaEventHandler(k).getParameterTypes()[0]);
            }));
            projectionManager.getHandlers().forEach((k, v) -> {
                var r = v.getQueryHandler(k).getReturnType();
                handlers.add(new RegisteredHandler(
                        ComponentType.Projection,
                        v.getRef().getClass().getSimpleName(),
                        HandlerType.QueryHandler,
                        PayloadType.Query,
                        k,
                        ((Class) (((ParameterizedType) v.getQueryHandler(k).getGenericReturnType()).getActualTypeArguments()[0])).getSimpleName(),
                        r.isAssignableFrom(Multiple.class),
                        null
                ));
                payloads.add(v.getQueryHandler(k).getParameterTypes()[0]);
                payloads.add(((Class) ((ParameterizedType) v.getQueryHandler(k).getGenericReturnType()).getActualTypeArguments()[0]));
            });
            handlers.addAll(invokerManager.getHandlers());
            ObjectMapper mapper = new ObjectMapper();
            JsonSchemaGenerator schemaGen = new JsonSchemaGenerator(mapper);
            var payloadInfo = new HashMap();
            for (Class p : payloads) {
                if (p == null) continue;
                var info = new String[2];
                payloadInfo.put(p.getSimpleName(), info);
                try {
                    info[0] = mapper.writeValueAsString(schemaGen.generateSchema(p));
                } catch (Exception ignored) {
                }
                if (p.getAnnotation(Domain.class) != null) {
                    info[1] = p.getAnnotation(Domain.class).name();
                }
            }
            var registration = new BundleRegistration(
                    bundleId,
                    bundleVersion,
                    instanceId,
                    handlers,
                    payloadInfo
            );

            logger.info("Starting EventoApplication %s".formatted(bundleId));
            logger.info("Connecting to Evento Server...");
            var eventoServer =
                    new EventoServerClient.Builder(
                            registration,
                            objectMapper,
                            eventoServerMessageBusConfiguration.getAddresses(),
                            (body) -> switch (body) {
                                case DecoratedDomainCommandMessage cm -> aggregateManager.handle(cm);
                                case ServiceCommandMessage sm -> serviceManager.handle(sm);
                                case QueryMessage qm -> projectionManager.handle(qm);
                                case null, default -> throw new RuntimeException("Invalid request body: " + body);
                            }
                    )
                            .setMaxReconnectAttempts(eventoServerMessageBusConfiguration.getMaxReconnectAttempts())
                            .setReconnectDelayMillis(eventoServerMessageBusConfiguration.getReconnectDelayMillis())
                            .setMaxDisableAttempts(eventoServerMessageBusConfiguration.getMaxDisableAttempts())
                            .setDisableDelayMillis(eventoServerMessageBusConfiguration.getDisableDelayMillis())
                            .setMaxRetryAttempts(eventoServerMessageBusConfiguration.getMaxRetryAttempts())
                            .setRetryDelayMillis(eventoServerMessageBusConfiguration.getRetryDelayMillis())
                            .connect();

            if(autoscalingProtocolBuilder == null){
                autoscalingProtocolBuilder = (e) -> new AutoscalingProtocol(e) {

                    @Override
                    public void arrival() {

                    }

                    @Override
                    public void departure() {

                    }
                };
            }
            var autoscalingProtocol = autoscalingProtocolBuilder.apply(eventoServer);
            logger.info("Autoscaling protocol: %s".formatted(autoscalingProtocol.getClass().getName()));
            tracingAgent.setAutoscalingProtocol(autoscalingProtocol);

            if(performanceService == null) {
                performanceService = performanceServiceBuilder.apply(eventoServer);
            }
            if (consumerStateStoreBuilder == null) {
                consumerStateStoreBuilder = InMemoryConsumerStateStore::new;
            }
            if (commandGateway == null) {
                commandGateway = commandGatewayBuilder.apply(eventoServer);
            }
            if (queryGateway == null) {
                queryGateway = queryGatewayBuilder.apply(eventoServer);
            }


            var css = consumerStateStoreBuilder.apply(eventoServer, performanceService);
            EventoBundle eventoBundle = new EventoBundle(
                    basePackage.getName(),
                    bundleId, instanceId,
                    aggregateManager, projectionManager, sagaManager, commandGateway,
                    queryGateway,
                    performanceService,
                    serviceManager, projectorManager, observerManager, tracingAgent);
            logger.info("Starting projector consumers...");
            var start = Instant.now();
            var wait = new Semaphore(0);
            eventoBundle.startProjectorEventConsumers(wait::release, css, contexts);
            var startThread = new Thread(() -> {
                try {
                    wait.acquire();
                    logger.info("All Projector Consumers head Reached! (in " + (Instant.now().toEpochMilli() - start.toEpochMilli()) + " millis)");
                    logger.info("Sending registration to enable the Bundle");
                    eventoServer.enable();
                    eventoBundle.startSagaEventConsumers(css, contexts);
                    eventoBundle.startObserverEventConsumers(css, contexts);
                    logger.info("Application Started!");
                }catch (Exception e){
                    logger.error("Error during startup", e);
                    System.exit(1);
                }
            });
            startThread.setName("Start Bundle Thread");
            startThread.start();
            return eventoBundle;

        }
    }

    /**
     * Starts the saga event consumers for the registered SagaReferences. Each SagaReference is checked for
     * the Saga annotation, and for each context specified in the annotation, a new SagaEventConsumer is created
     * and started in a new thread.
     *
     * @param consumerStateStore the consumer state store to track the state of event consumers
     * @param contexts the component contexts associations
     */
    private void startSagaEventConsumers(ConsumerStateStore consumerStateStore, Map> contexts) {
        sagaManager.startSagaEventConsumers(consumerStateStore, contexts);
    }


    /**
     * Starts the projector event consumers for the specified contexts.
     *
     * @param onAllHeadReached   The callback to be executed when all heads are reached.
     * @param consumerStateStore The consumer state store to track the state of event consumers.
     * @param contexts           The contexts for which the projector event consumers should be started.
     */
    private void startProjectorEventConsumers(Runnable onAllHeadReached, ConsumerStateStore consumerStateStore, Map> contexts) {
        projectorManager.startEventConsumers(onAllHeadReached, consumerStateStore, contexts);
    }


    /**
     * Starts the observer event consumers for the registered ObserverReferences. Each ObserverReference is checked for
     * the Observer annotation, and for each context specified in the annotation, a new ObserverEventConsumer is created
     * and started in a new thread.
     *
     * @param consumerStateStore The consumer state store to track the state of event consumers.
     * @param contexts the component contexts associations
     */
    private void startObserverEventConsumers(ConsumerStateStore consumerStateStore, Map> contexts) {
        observerManager.startEventConsumers(consumerStateStore, contexts);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy