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

org.axonframework.spring.eventsourcing.SpringAggregateSnapshotter Maven / Gradle / Ivy

/*
 * Copyright (c) 2010-2023. Axon Framework
 *
 * 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 org.axonframework.spring.eventsourcing;

import org.axonframework.common.AxonConfigurationException;
import org.axonframework.common.transaction.TransactionManager;
import org.axonframework.eventsourcing.*;
import org.axonframework.eventsourcing.eventstore.EventStore;
import org.axonframework.messaging.annotation.HandlerDefinition;
import org.axonframework.messaging.annotation.ParameterResolverFactory;
import org.axonframework.modelling.command.RepositoryProvider;
import org.axonframework.tracing.SpanFactory;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;

import static org.springframework.beans.factory.BeanFactoryUtils.beansOfTypeIncludingAncestors;

import java.util.Collections;
import java.util.Objects;
import java.util.Optional;
import java.util.concurrent.Executor;
import java.util.function.Function;
import javax.annotation.Nonnull;

/**
 * Snapshotter implementation that uses the AggregateRoot as state for the snapshot. Unlike the
 * {@link AggregateSnapshotter}, this implementation lazily retrieves AggregateFactories from the Application
 * Context when snapshot requests are made.
 * 

* Instead of configuring directly, consider using the {@link SpringAggregateSnapshotterFactoryBean}, which * configures this class using values available in the application context. * * @author Allard Buijze * @see SpringAggregateSnapshotterFactoryBean * @since 3.0 */ public class SpringAggregateSnapshotter extends AggregateSnapshotter implements ApplicationContextAware { private ApplicationContext applicationContext; /** * Instantiate a {@link SpringAggregateSnapshotter} based on the fields contained in the {@link Builder}. The * {@link AggregateFactory} instances are lazily retrieved by the {@link ApplicationContext}. *

* Will assert that the {@link EventStore}, {@link ParameterResolverFactory} and {@link HandlerDefinition} are not * {@code null}, and will throw an {@link AxonConfigurationException} if any of them is {@code null}. * * @param builder the {@link Builder} used to instantiate a {@link SpringAggregateSnapshotter} instance */ protected SpringAggregateSnapshotter(Builder builder) { super(builder); } /** * Instantiate a Builder to be able to create a {@link SpringAggregateSnapshotter}. The {@link AggregateFactory} * instances are lazily retrieved by the {@link ApplicationContext}. *

* The {@link Executor} is defaulted to an {@link org.axonframework.common.DirectExecutor#INSTANCE}, the * {@link TransactionManager} defaults to a {@link org.axonframework.common.transaction.NoTransactionManager} and * the {@link SpanFactory} defaults to a {@link org.axonframework.tracing.NoOpSpanFactory}. Additionally, this * Builder has convenience functions to default the {@link ParameterResolverFactory} and {@link HandlerDefinition} * based on instances of these available on the classpath in case these are not provided (respectively * {@link Builder#buildParameterResolverFactory()} and {@link Builder#buildHandlerDefinition()}). Upon instantiation * of a {@link AggregateSnapshotter}, it is recommended to use these function to set those fields. *

* The {@link EventStore} is a hard requirement and as such should be provided. * * @return a Builder to be able to create a {@link SpringAggregateSnapshotter} * @see org.axonframework.messaging.annotation.ClasspathParameterResolverFactory * @see org.axonframework.messaging.annotation.ClasspathHandlerDefinition */ public static Builder builder() { return new Builder(); } @Override protected AggregateFactory getAggregateFactory(Class aggregateType) { AggregateFactory aggregateFactory = super.getAggregateFactory(aggregateType); if (aggregateFactory == null) { Optional factory = beansOfTypeIncludingAncestors(applicationContext, AggregateFactory.class).values().stream() .filter(af -> Objects.equals(af.getAggregateType(), aggregateType)) .findFirst(); if (!factory.isPresent()) { factory = beansOfTypeIncludingAncestors(applicationContext, EventSourcingRepository.class).values().stream() .map((Function) EventSourcingRepository::getAggregateFactory) .filter(af -> Objects.equals(af.getAggregateType(), aggregateType)) .findFirst(); if (factory.isPresent()) { aggregateFactory = factory.get(); registerAggregateFactory(aggregateFactory); } } if (factory.isPresent()) { aggregateFactory = factory.get(); registerAggregateFactory(aggregateFactory); } } return aggregateFactory; } @Override public void setApplicationContext(@Nonnull ApplicationContext applicationContext) throws BeansException { this.applicationContext = applicationContext; } /** * Builder class to instantiate a {@link SpringAggregateSnapshotter}. The {@link AggregateFactory} instances are * lazily retrieved by the {@link ApplicationContext}. *

* The {@link Executor} is defaulted to an {@link org.axonframework.common.DirectExecutor#INSTANCE}, the * {@link TransactionManager} defaults to a {@link org.axonframework.common.transaction.NoTransactionManager} and * the {@link SpanFactory} defaults to a {@link org.axonframework.tracing.NoOpSpanFactory}. Additionally, this * Builder has convenience functions to default the {@link ParameterResolverFactory} and {@link HandlerDefinition} * based on instances of these available on the classpath in case these are not provided (respectively * {@link Builder#buildParameterResolverFactory()} and {@link Builder#buildHandlerDefinition()}). Upon instantiation * of a {@link AggregateSnapshotter}, it is recommended to use these function to set those fields. *

* The {@link EventStore} is a hard requirement and as such should be provided. * * @see org.axonframework.messaging.annotation.ClasspathParameterResolverFactory * @see org.axonframework.messaging.annotation.ClasspathHandlerDefinition */ public static class Builder extends AggregateSnapshotter.Builder { public Builder() { aggregateFactories(Collections.emptyList()); } @Override public Builder eventStore(EventStore eventStore) { super.eventStore(eventStore); return this; } @Override public Builder executor(Executor executor) { super.executor(executor); return this; } @Override public Builder transactionManager(TransactionManager transactionManager) { super.transactionManager(transactionManager); return this; } @Override public Builder repositoryProvider(RepositoryProvider repositoryProvider) { super.repositoryProvider(repositoryProvider); return this; } @Override public Builder parameterResolverFactory(ParameterResolverFactory parameterResolverFactory) { super.parameterResolverFactory(parameterResolverFactory); return this; } @Override public Builder handlerDefinition(HandlerDefinition handlerDefinition) { super.handlerDefinition(handlerDefinition); return this; } @Override public Builder spanFactory(@Nonnull SpanFactory spanFactory) { super.spanFactory(spanFactory); return this; } @Override public Builder spanFactory(@Nonnull SnapshotterSpanFactory spanFactory) { super.spanFactory(spanFactory); return this; } /** * Initializes a {@link SpringAggregateSnapshotter} as specified through this Builder. * * @return a {@link SpringAggregateSnapshotter} as specified through this Builder */ public SpringAggregateSnapshotter build() { return new SpringAggregateSnapshotter(this); } /** * Validates whether the fields contained in this Builder are set accordingly. * * @throws AxonConfigurationException if one field is asserted to be incorrect according to the Builder's * specifications */ @Override protected void validate() throws AxonConfigurationException { super.validate(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy