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

io.fluxcapacitor.axonclient.common.configuration.AbstractFluxCapacitorConfiguration Maven / Gradle / Ivy

Go to download

A Java client library that enables Axonframework components to interact via Flux Capacitor. Axonframework is a CQRS framework for Java. For more information about Axonframework see http://axonframework.org

There is a newer version: 0.118
Show newest version
/*
 * Copyright (c) 2016-2017 Flux Capacitor.
 *
 * 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.fluxcapacitor.axonclient.common.configuration;

import io.fluxcapacitor.axonclient.commandhandling.CommandProcessor;
import io.fluxcapacitor.axonclient.commandhandling.FluxCapacitorCommandBus;
import io.fluxcapacitor.axonclient.commandhandling.result.ResultProcessor;
import io.fluxcapacitor.axonclient.common.serialization.AxonMessageSerializer;
import io.fluxcapacitor.axonclient.common.serialization.DefaultAxonMessageSerializer;
import io.fluxcapacitor.axonclient.eventhandling.FluxCapacitorEventProcessor;
import io.fluxcapacitor.axonclient.eventhandling.FluxCapacitorEventStore;
import io.fluxcapacitor.common.MessageType;
import io.fluxcapacitor.javaclient.configuration.client.Client;
import io.fluxcapacitor.javaclient.eventsourcing.client.EventStoreClient;
import io.fluxcapacitor.javaclient.keyvalue.client.KeyValueClient;
import io.fluxcapacitor.javaclient.publishing.client.GatewayClient;
import io.fluxcapacitor.javaclient.tracking.client.TrackingClient;
import org.axonframework.commandhandling.SimpleCommandBus;
import org.axonframework.commandhandling.distributed.AnnotationRoutingStrategy;
import org.axonframework.commandhandling.distributed.RoutingStrategy;
import org.axonframework.commandhandling.distributed.UnresolvedRoutingKeyPolicy;
import org.axonframework.config.Component;
import org.axonframework.config.Configuration;
import org.axonframework.config.Configurer;
import org.axonframework.config.EventHandlingConfiguration;
import org.axonframework.config.ModuleConfiguration;
import org.axonframework.config.SagaConfiguration;
import org.axonframework.eventhandling.EventProcessor;
import org.axonframework.eventhandling.ListenerInvocationErrorHandler;
import org.axonframework.eventhandling.LoggingErrorHandler;
import org.axonframework.eventhandling.PropagatingErrorHandler;
import org.axonframework.eventhandling.SimpleEventHandlerInvoker;
import org.axonframework.messaging.unitofwork.RollbackConfigurationType;
import org.axonframework.monitoring.MessageMonitor;
import org.axonframework.serialization.upcasting.event.EventUpcasterChain;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.Field;
import java.util.concurrent.atomic.AtomicReference;

import static io.fluxcapacitor.common.reflection.ReflectionUtils.ensureAccessible;
import static java.lang.String.format;

public abstract class AbstractFluxCapacitorConfiguration implements FluxCapacitorConfiguration {

    private final Client client;
    private final AtomicReference eventConsumerService = new AtomicReference<>();

    protected AbstractFluxCapacitorConfiguration(Client client) {
        this.client = client;
    }

    @Override
    public Configurer configure(Configurer configurer) {
        SimpleCommandBus localCommandBus = new SimpleCommandBus();
        configurer = configurer
                .registerComponent(AxonMessageSerializer.class, c -> new DefaultAxonMessageSerializer(
                        c.serializer(), c.getComponent(EventUpcasterChain.class, EventUpcasterChain::new)))
                .registerComponent(ResultProcessor.class,
                                   c -> new ResultProcessor(c.getComponent(AxonMessageSerializer.class),
                                                            createConsumerService(MessageType.RESULT),
                                                            format("%s/result", client.name())))
                .registerComponent(CommandProcessor.class,
                                   c -> new CommandProcessor(
                                           c.getComponent(AxonMessageSerializer.class),
                                           localCommandBus, createProducerService(MessageType.RESULT),
                                           format("%s/command", client.name()),
                                           createConsumerService(MessageType.COMMAND)))
                .registerComponent(RoutingStrategy.class,
                                   c -> new AnnotationRoutingStrategy(UnresolvedRoutingKeyPolicy.RANDOM_KEY))
                .configureCommandBus(c -> new FluxCapacitorCommandBus(
                        createProducerService(MessageType.COMMAND), c.getComponent(ResultProcessor.class),
                        c.getComponent(AxonMessageSerializer.class),
                        c.getComponent(RoutingStrategy.class), client.id(), localCommandBus))
                .registerComponent(FluxCapacitorEventStore.class, this::createEventStore)
                .configureEventStore(c -> c.getComponent(FluxCapacitorEventStore.class))
                .registerModule(new FluxCapacitorModuleConfiguration());
        configurer = configureEventHandling(configurer);
        configurer = configureSagaManagers(configurer);
        return configurer;
    }

    @Override
    public EventHandlingConfiguration configure(EventHandlingConfiguration eventHandlingConfiguration) {
        return eventHandlingConfiguration.registerEventProcessorFactory(
                (c, name, eventListeners) -> {
                    String processorName = format("%s/%s", client.name(), name);
                    return new FluxCapacitorEventProcessor(
                            processorName,
                            new SimpleEventHandlerInvoker(eventListeners, c.parameterResolverFactory(),
                                                          c.getComponent(ListenerInvocationErrorHandler.class,
                                                                         LoggingErrorHandler::new)),
                            RollbackConfigurationType.ANY_THROWABLE, PropagatingErrorHandler.INSTANCE,
                            c.messageMonitor(FluxCapacitorEventProcessor.class, processorName),
                            getEventConsumerService(), c.getComponent(AxonMessageSerializer.class), 1);
                });
    }

    protected Configurer configureEventHandling(Configurer configurer) {
        Configuration configuration = getField("config", configurer);
        configuration.getModules().forEach(m -> {
            if (m instanceof EventHandlingConfiguration) {
                configure((EventHandlingConfiguration) m);
            }
        });
        return configurer;
    }

    protected Configurer configureSagaManagers(Configurer configurer) {
        Configuration configuration = getField("config", configurer);
        configuration.getModules().forEach(m -> {
            if (m instanceof SagaConfiguration) {
                SagaConfiguration sagaConfig = (SagaConfiguration) m;
                Component processorComponent = getField("processor", sagaConfig);
                String name = getField("name", processorComponent);
                processorComponent.update(c -> {
                    String processorName = format("%s/%s", client.name(), name);
                    Logger logger = LoggerFactory.getLogger(processorName);
                    return new FluxCapacitorEventProcessor(
                            processorName, sagaConfig.getSagaManager(),
                            RollbackConfigurationType.ANY_THROWABLE,
                            errorContext -> logger.error("Failed to handle events on saga", errorContext.error()),
                            c.messageMonitor(FluxCapacitorEventProcessor.class, processorName), getEventConsumerService(),
                            c.getComponent(AxonMessageSerializer.class), 1);
                });
            }
        });
        return configurer;
    }

    protected abstract TrackingClient createConsumerService(MessageType type);

    protected abstract GatewayClient createProducerService(MessageType type);

    protected abstract EventStoreClient createEventStore();

    protected abstract KeyValueClient createKeyValueClient();

    private TrackingClient getEventConsumerService() {
        return eventConsumerService
                .updateAndGet(service -> service == null ? createConsumerService(MessageType.EVENT) : service);
    }

    @SuppressWarnings("unchecked")
    protected FluxCapacitorEventStore createEventStore(Configuration configuration) {
        MessageMonitor monitor = configuration.messageMonitor(FluxCapacitorEventStore.class, "eventStore");
        return new FluxCapacitorEventStore(monitor, createEventStore(), createKeyValueClient(),
                                           configuration.getComponent(AxonMessageSerializer.class));
    }

    @SuppressWarnings("unchecked")
    protected static  T getField(String location, Object instance) {
        try {
            String[] paths = location.split("/");
            Object object = instance;
            for (String path : paths) {
                Field field = ensureAccessible(object.getClass().getDeclaredField(path));
                object = field.get(object);
            }
            return (T) object;
        } catch (Exception e) {
            throw new IllegalArgumentException(String.format("Could not find %s on instance %s", location, instance));
        }
    }

    protected static class FluxCapacitorModuleConfiguration implements ModuleConfiguration {

        private Configuration config;

        @Override
        public void initialize(Configuration config) {
            this.config = config;
        }

        @Override
        public void start() {
            config.getComponent(ResultProcessor.class).start();
            config.getComponent(CommandProcessor.class).start();
        }

        @Override
        public void shutdown() {
            config.getComponent(CommandProcessor.class).shutDown();
            config.getComponent(ResultProcessor.class).shutDown();
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy