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

io.fluxcapacitor.javaclient.modeling.LazyAggregateRoot Maven / Gradle / Ivy

There is a newer version: 0.1072.0
Show newest version
/*
 * Copyright (c) Flux Capacitor IP B.V. or its affiliates. All Rights Reserved.
 *
 * 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.javaclient.modeling;

import io.fluxcapacitor.javaclient.common.Message;
import io.fluxcapacitor.javaclient.common.serialization.DeserializingMessage;
import io.fluxcapacitor.javaclient.persisting.eventsourcing.AggregateEventStream;
import lombok.With;

import java.time.Instant;
import java.util.Collection;
import java.util.Iterator;
import java.util.Objects;
import java.util.function.UnaryOperator;

public class LazyAggregateRoot implements AggregateRoot {
    @With
    private final ImmutableAggregateRoot delegate;

    public static  LazyAggregateRoot from(ImmutableAggregateRoot delegate) {
        return new LazyAggregateRoot<>(delegate.toBuilder().value(null).build());
    }

    protected LazyAggregateRoot(ImmutableAggregateRoot delegate) {
        this.delegate = delegate;
        if (!rootAnnotation().eventSourced()) {
            throw new IllegalStateException("Cannot create lazy aggregate: event sourcing is disabled.");
        }
    }

    @Override
    public Entity withEventIndex(Long index, String messageId) {
        return withDelegate((ImmutableAggregateRoot) delegate.withEventIndex(index, messageId));
    }

    @Override
    public long sequenceNumber() {
        return delegate.sequenceNumber();
    }

    @Override
    public Instant timestamp() {
        return delegate.timestamp();
    }

    @Override
    public Entity previous() {
        return delegate.previous();
    }

    /*
        Requires loading events
     */

    @Override
    public T get() {
        var start = getLastCheckpoint();
        String targetEventId = lastEventId();
        AggregateEventStream events = delegate.eventStore().getEvents(
                id(), start.sequenceNumber(), (int) (sequenceNumber() - start.sequenceNumber()),
                rootAnnotation().ignoreUnknownEvents());
        Iterator iterator = events.iterator();
        Entity result = start;
        boolean eventReached = false;
        while (iterator.hasNext()) {
            DeserializingMessage nextEvent = iterator.next();
            boolean lastEventId = Objects.equals(targetEventId, nextEvent.getMessageId());
            if (eventReached && !lastEventId) {
                break;
            }
            if (lastEventId) {
                eventReached = true;
            }
            result = result.apply(nextEvent);
        }
        return result.get();
    }

    protected Entity getLastCheckpoint() {
        Entity result = previous();
        while (result instanceof LazyAggregateRoot) {
            result = result.previous();
        }
        if (result == null) {
            throw new IllegalStateException("Failed to get last checkpoint for aggregate: " + id());
        }
        return result;
    }

    @Override
    public Collection aliases() {
        return delegate.toBuilder().value(get()).build().aliases();
    }

    @Override
    public Collection> entities() {
        return delegate.toBuilder().value(get()).build().entities();
    }

    /*
        Simply delegation
     */

    @Override
    public Object id() {
        return delegate.id();
    }

    @Override
    public Class type() {
        return delegate.type();
    }

    @Override
    public String idProperty() {
        return delegate.idProperty();
    }

    @Override
    public String lastEventId() {
        return delegate.lastEventId();
    }

    @Override
    public Long lastEventIndex() {
        return delegate.lastEventIndex();
    }

    /*
        Unsupported operations
     */

    @Override
    public Entity update(UnaryOperator function) {
        throw new UnsupportedOperationException("This aggregate is read-only.");
    }

    @Override
    public Entity apply(Message eventMessage) {
        throw new UnsupportedOperationException("This aggregate is read-only.");
    }

    @Override
    public Entity commit() {
        throw new UnsupportedOperationException("This aggregate is read-only.");
    }

    @Override
    public Entity withType(Class type) {
        throw new UnsupportedOperationException("This aggregate is read-only.");
    }

    @Override
    public Entity withSequenceNumber(long sequenceNumber) {
        throw new UnsupportedOperationException("This aggregate is read-only.");
    }

    @Override
    public  Entity assertLegal(Object command) throws E {
        throw new UnsupportedOperationException("This aggregate is read-only.");
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy