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

io.evitadb.api.proxy.impl.SealedEntityProxyState Maven / Gradle / Ivy

There is a newer version: 2024.10.0
Show newest version
/*
 *
 *                         _ _        ____  ____
 *               _____   _(_) |_ __ _|  _ \| __ )
 *              / _ \ \ / / | __/ _` | | | |  _ \
 *             |  __/\ V /| | || (_| | |_| | |_) |
 *              \___| \_/ |_|\__\__,_|____/|____/
 *
 *   Copyright (c) 2023-2024
 *
 *   Licensed under the Business Source License, Version 1.1 (the "License");
 *   you may not use this file except in compliance with the License.
 *   You may obtain a copy of the License at
 *
 *   https://github.com/FgForrest/evitaDB/blob/master/LICENSE
 *
 *   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.evitadb.api.proxy.impl;

import io.evitadb.api.EvitaSessionContract;
import io.evitadb.api.exception.EntityClassInvalidException;
import io.evitadb.api.proxy.SealedEntityProxy;
import io.evitadb.api.proxy.SealedEntityReferenceProxy;
import io.evitadb.api.proxy.impl.ProxycianFactory.ProxyEntityCacheKey;
import io.evitadb.api.requestResponse.data.EntityContract;
import io.evitadb.api.requestResponse.data.EntityEditor.EntityBuilder;
import io.evitadb.api.requestResponse.data.SealedEntity;
import io.evitadb.api.requestResponse.data.mutation.EntityMutation;
import io.evitadb.api.requestResponse.data.mutation.LocalMutation;
import io.evitadb.api.requestResponse.data.structure.Entity;
import io.evitadb.api.requestResponse.data.structure.EntityDecorator;
import io.evitadb.api.requestResponse.data.structure.EntityReference;
import io.evitadb.api.requestResponse.data.structure.ExistingEntityBuilder;
import io.evitadb.api.requestResponse.data.structure.ExistingReferenceBuilder;
import io.evitadb.api.requestResponse.data.structure.InitialReferenceBuilder;
import io.evitadb.api.requestResponse.schema.EntitySchemaContract;
import io.evitadb.api.requestResponse.schema.ReferenceSchemaContract;
import io.evitadb.exception.EvitaInvalidUsageException;
import io.evitadb.exception.GenericEvitaInternalError;
import io.evitadb.utils.Assert;
import io.evitadb.utils.ReflectionLookup;
import lombok.Setter;
import one.edee.oss.proxycian.recipe.ProxyRecipe;
import one.edee.oss.proxycian.trait.ProxyStateAccessor;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.io.Serial;
import java.util.Collection;
import java.util.Map;
import java.util.Optional;
import java.util.function.Supplier;

import static java.util.Optional.ofNullable;

/**
 * Proxy state for proxies that wrap sealed entity.
 *
 * @author Jan Novotný ([email protected]), FG Forrest a.s. (c) 2023
 */
public class SealedEntityProxyState
	extends AbstractEntityProxyState
	implements SealedEntityProxy {
	@Serial private static final long serialVersionUID = 586508293856395550L;
	/**
	 * Optional reference to the {@link EntityBuilder} that is created on demand by calling {@link SealedEntity#openForWrite()}
	 * from internally wrapped entity {@link #entity()}.
	 */
	@Nullable protected EntityBuilder entityBuilder;
	/**
	 * Optional information about the last {@link EntityReference} returned from {@link EvitaSessionContract#upsertEntity(EntityMutation)},
	 * it may contain newly assigned {@link EntityContract#getPrimaryKey()} that is not available in the wrapped entity.
	 */
	@Setter private EntityReference entityReference;

	public SealedEntityProxyState(
		@Nonnull EntityContract entity,
		@Nonnull Map referencedEntitySchemas,
		@Nonnull Class proxyClass,
		@Nonnull Map recipes,
		@Nonnull Map collectedRecipes,
		@Nonnull ReflectionLookup reflectionLookup
	) {
		super(entity, referencedEntitySchemas, proxyClass, recipes, collectedRecipes, reflectionLookup);
	}

	@Nonnull
	@Override
	public String getType() {
		return entity.getType();
	}

	@Nullable
	@Override
	public Integer getPrimaryKey() {
		return ofNullable(entity.getPrimaryKey())
			.orElseGet(() -> ofNullable(entityReference)
				.map(EntityReference::getPrimaryKey)
				.orElse(null)
			);
	}

	@Nonnull
	@Override
	public EntityContract entity() {
		return entityBuilderIfPresent()
			.map(EntityContract.class::cast)
			.orElse(this.entity);
	}

	@Nonnull
	@Override
	public Optional getEntityBuilderWithCallback() {
		propagateReferenceMutations();
		return Optional.ofNullable(this.entityBuilder)
			.map(
				it -> new EntityBuilderWithCallback(
					it,
					entityReference -> {
						this.entityReference = entityReference;
						this.entityBuilder = new ExistingEntityBuilder(
							// we can cast it here, since we know that both InitialEntityBuilder and ExistingEntityBuilder
							// fabricate Entity instances
							(Entity) this.entityBuilder.toInstance()
						);
						// we need to mark all references as upserted, since they were persisted along with the entity
						this.generatedProxyObjects
							.entrySet()
							.stream()
							.filter(goEntry -> goEntry.getKey().proxyType() == ProxyType.REFERENCE)
							.flatMap(goEntry -> goEntry.getValue().proxies().stream())
							.forEach(refProxy -> ((SealedEntityReferenceProxyState)((ProxyStateAccessor)refProxy).getProxyState())
								.notifyBuilderUpserted()
							);
					}
				)
			);
	}

	@Nonnull
	public Optional entityBuilderIfPresent() {
		return ofNullable(this.entityBuilder);
	}

	@Nonnull
	public EntityBuilder getEntityBuilderWithMutations(@Nonnull Collection> mutations) {
		Assert.isPremiseValid(entityBuilder == null, "Entity builder already created!");
		if (entity instanceof EntityDecorator entityDecorator) {
			entityBuilder = new ExistingEntityBuilder(entityDecorator, mutations);
		} else if (entity instanceof Entity theEntity) {
			entityBuilder = new ExistingEntityBuilder(theEntity, mutations);
		} else if (entity instanceof EntityBuilder) {
			throw new GenericEvitaInternalError("Entity builder already created!");
		} else {
			throw new GenericEvitaInternalError("Unexpected entity type: " + entity.getClass().getName());
		}
		return entityBuilder;
	}

	@Nonnull
	public EntityBuilder entityBuilder() {
		if (entityBuilder == null) {
			if (entity instanceof EntityDecorator entityDecorator) {
				entityBuilder = new ExistingEntityBuilder(entityDecorator);
			} else if (entity instanceof Entity theEntity) {
				entityBuilder = new ExistingEntityBuilder(theEntity);
			} else if (entity instanceof EntityBuilder theBuilder) {
				entityBuilder = theBuilder;
			} else {
				throw new GenericEvitaInternalError("Unexpected entity type: " + entity.getClass().getName());
			}
		}
		return entityBuilder;
	}

	@Nonnull
	public  T createEntityReferenceProxy(
		@Nonnull EntitySchemaContract entitySchema,
		@Nonnull ReferenceSchemaContract referenceSchema,
		@Nonnull Class expectedType,
		@Nonnull ProxyType proxyType,
		int primaryKey
	) throws EntityClassInvalidException {
		final Supplier instanceSupplier = () -> {
			final Optional entityBuilderRef = entityBuilderIfPresent();
			final EntityContract entity = entityBuilderRef
				.map(EntityContract.class::cast)
				.orElseGet(this::entity);
			return entity.getReference(referenceSchema.getName(), primaryKey)
				.filter(
					ref -> entityBuilderRef
						.filter(ExistingEntityBuilder.class::isInstance)
						.map(ExistingEntityBuilder.class::cast)
						.map(eb -> eb.isPresentInBaseEntity(ref))
						.orElse(true)
				)
				.map(
					existingReference -> new ProxyWithUpsertCallback(
						ProxycianFactory.createEntityReferenceProxy(
							this.getProxyClass(), expectedType, recipes, collectedRecipes,
							this.entity, this::getPrimaryKey,
							referencedEntitySchemas,
							new ExistingReferenceBuilder(existingReference, getEntitySchema()),
							getReflectionLookup(),
							this.generatedProxyObjects
						)
					)
				)
				.orElseGet(() -> new ProxyWithUpsertCallback(
						ProxycianFactory.createEntityReferenceProxy(
							this.getProxyClass(), expectedType, recipes, collectedRecipes,
							this.entity, this::getPrimaryKey,
							getReferencedEntitySchemas(),
							new InitialReferenceBuilder(
								entitySchema,
								referenceSchema.getName(),
								primaryKey,
								referenceSchema.getCardinality(),
								referenceSchema.getReferencedEntityType()
							),
							getReflectionLookup(),
							this.generatedProxyObjects
						)
					)
				);
		};
		return generatedProxyObjects.computeIfAbsent(
				new ProxyInstanceCacheKey(referenceSchema.getName(), primaryKey, proxyType),
				key -> instanceSupplier.get()
			)
			.proxy(expectedType, instanceSupplier);
	}

	/**
	 * Method propagates all mutations in reference proxies to the {@link #entityBuilder()}.
	 */
	public void propagateReferenceMutations() {
		this.generatedProxyObjects.entrySet().stream()
			.filter(it -> it.getKey().proxyType() == ProxyType.REFERENCE)
			.flatMap(
				it -> it.getValue()
					.proxy(
						SealedEntityReferenceProxy.class,
						() -> {
							throw new EvitaInvalidUsageException("Unexpected proxy type!");
						})
					.getReferenceBuilderIfPresent()
					.stream()
			)
			.forEach(referenceBuilder -> entityBuilder().addOrReplaceReferenceMutations(referenceBuilder));
	}

	@Override
	public String toString() {
		return entity instanceof EntityBuilder eb ?
			eb.toInstance().toString() : entity.toString();
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy