Skip to content

Commit

Permalink
Fix mapping with embedded field with generic type (#3223)
Browse files Browse the repository at this point in the history
  • Loading branch information
radovanradic authored Nov 13, 2024
1 parent 4179f17 commit 3341071
Show file tree
Hide file tree
Showing 14 changed files with 209 additions and 1 deletion.
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,9 @@
package io.micronaut.data.jdbc.h2

import groovy.transform.Memoized
import io.micronaut.data.tck.entities.embedded.BookEntity
import io.micronaut.data.tck.entities.embedded.BookState
import io.micronaut.data.tck.entities.embedded.ResourceEntity
import io.micronaut.data.tck.repositories.AuthorRepository
import io.micronaut.data.tck.repositories.BasicTypesRepository
import io.micronaut.data.tck.repositories.BookDtoRepository
Expand Down Expand Up @@ -113,6 +116,9 @@ class H2RepositorySpec extends AbstractRepositorySpec implements H2TestPropertyP
@Shared
H2EntityWithIdClass2Repository entityWithIdClass2Repo = context.getBean(H2EntityWithIdClass2Repository)

@Shared
H2BookEntityRepository bookEntityRepository = context.getBean(H2BookEntityRepository)

@Override
EntityWithIdClassRepository getEntityWithIdClassRepository() {
return entityWithIdClassRepo
Expand Down Expand Up @@ -293,4 +299,12 @@ class H2RepositorySpec extends AbstractRepositorySpec implements H2TestPropertyP
cleanupData()
}

void "find by embedded entity field"() {
when:
def bookEntity = new BookEntity(1L, new ResourceEntity<BookState>("1984", BookState.BORROWED))
bookEntityRepository.save(bookEntity)
def result = bookEntityRepository.findAllByResourceState(BookState.BORROWED)
then:
result
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package io.micronaut.data.jdbc.h2;

import io.micronaut.data.jdbc.annotation.JdbcRepository;
import io.micronaut.data.model.query.builder.sql.Dialect;
import io.micronaut.data.tck.repositories.embedded.BookEntityRepository;

@JdbcRepository(dialect = Dialect.H2)
public interface H2BookEntityRepository extends BookEntityRepository {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
package io.micronaut.data.jdbc.h2;

import io.micronaut.data.jdbc.annotation.JdbcRepository;
import io.micronaut.data.model.query.builder.sql.Dialect;
import io.micronaut.data.tck.repositories.embedded.HouseEntityRepository;

@JdbcRepository(dialect = Dialect.H2)
public interface H2HouseEntityRepository extends HouseEntityRepository {
}
Original file line number Diff line number Diff line change
Expand Up @@ -184,7 +184,8 @@ public void visitClass(ClassElement element, VisitorContext context) {

@Override
public SourcePersistentEntity apply(ClassElement classElement) {
return entityMap.computeIfAbsent(classElement.getName(), s -> {
String classNameKey = getClassNameKey(classElement);
return entityMap.computeIfAbsent(classNameKey, s -> {
if (classElement.hasAnnotation("io.micronaut.data.annotation.Embeddable")) {
embeddedMappedEntityVisitor.visitClass(classElement, context);
} else {
Expand Down Expand Up @@ -755,4 +756,32 @@ private void annotateQueryResultIfApplicable(MethodElement element, MethodMatchI
}
}
}

/**
* Generates key for the entityMap using {@link ClassElement}.
* If class element has generic types then will use all bound generic types in the key like
* for example {@code Entity<CustomKeyType, CustomValueType>} and for non-generic class element
* will just return class name.
* This is needed when there are for example multiple embedded fields with the same type
* but different generic type argument.
*
* @param classElement The class element
* @return The key for entityMap created from the class element
*/
private String getClassNameKey(ClassElement classElement) {
List<? extends ClassElement> boundGenericTypes = classElement.getBoundGenericTypes();
if (CollectionUtils.isNotEmpty(boundGenericTypes)) {
StringBuilder keyBuff = new StringBuilder(classElement.getName());
keyBuff.append("<");
for (ClassElement boundGenericType : boundGenericTypes) {
keyBuff.append(boundGenericType.getName());
keyBuff.append(",");
}
keyBuff.deleteCharAt(keyBuff.length() - 1);
keyBuff.append(">");
return keyBuff.toString();
} else {
return classElement.getName();
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -1992,4 +1992,28 @@ interface TestRepository extends GenericRepository<Book, Long> {
expect:
getQuery(findByAuthorInListMethod) == "SELECT book_.`id`,book_.`author_id`,book_.`genre_id`,book_.`title`,book_.`total_pages`,book_.`publisher_id`,book_.`last_updated` FROM `book` book_ WHERE (book_.`author_id` IN (?))"
}

void "test repository with reused embedded entity"() {
when:
buildRepository('test.TestRepository', """
import io.micronaut.data.jdbc.annotation.JdbcRepository;
import io.micronaut.data.repository.GenericRepository;
import io.micronaut.data.model.query.builder.sql.Dialect;
import io.micronaut.data.tck.entities.embedded.BookEntity;
import io.micronaut.data.tck.entities.embedded.BookState;
import io.micronaut.data.tck.entities.embedded.HouseEntity;
import io.micronaut.data.tck.entities.embedded.HouseState;
import java.util.List;
@JdbcRepository(dialect = Dialect.POSTGRES)
interface TestRepository extends GenericRepository<HouseEntity, Long> {
List<HouseEntity> findAllByResourceState(HouseState state);
}
@JdbcRepository(dialect = Dialect.POSTGRES)
interface OtherRepository extends GenericRepository<BookEntity, Long> {
List<BookEntity> findAllByResourceState(BookState state);
}
""")
then:
noExceptionThrown()
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
package io.micronaut.data.tck.entities.embedded;

public interface BaseEntity<I, S extends Enum<S>> {

I id();

ResourceEntity<S> resource();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
package io.micronaut.data.tck.entities.embedded;

import io.micronaut.data.annotation.Id;
import io.micronaut.data.annotation.MappedEntity;
import jakarta.persistence.Embedded;

@MappedEntity
public record BookEntity(
@Id Long id,
@Embedded ResourceEntity<BookState> resource
) implements BaseEntity<Long, BookState> {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
package io.micronaut.data.tck.entities.embedded;

public enum BookState {
BORROWED,
READ,
RETURNED
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
package io.micronaut.data.tck.entities.embedded;

import io.micronaut.data.annotation.Id;
import io.micronaut.data.annotation.MappedEntity;
import jakarta.persistence.Embedded;

@MappedEntity
public record HouseEntity(
@Id Long id,
@Embedded ResourceEntity<HouseState> resource
) implements BaseEntity<Long, HouseState> {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
package io.micronaut.data.tck.entities.embedded;

public enum HouseState {
BUILDING,
FINISHED
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
package io.micronaut.data.tck.entities.embedded;

import io.micronaut.data.annotation.Embeddable;
import io.micronaut.data.annotation.TypeDef;
import io.micronaut.data.model.DataType;
import jakarta.persistence.Convert;

@Embeddable
public record ResourceEntity<S extends Enum<S>>(

String displayName,

@TypeDef(type = DataType.STRING)
@Convert(converter = StateConverter.class)
S state
) {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
package io.micronaut.data.tck.entities.embedded;

import jakarta.persistence.AttributeConverter;
import jakarta.persistence.Converter;

@Converter
public class StateConverter<Enum> implements AttributeConverter<java.lang.Enum, String> {

@Override
public String convertToDatabaseColumn(java.lang.Enum anEnum) {
if (anEnum == null) {
return null;
}
return anEnum.name();
}

@Override
public java.lang.Enum convertToEntityAttribute(String string) {
if (string == null) {
return null;
}
// Because enum generics in ResourceEntity then implement this
// simple converter just to be able to run tests
if (string.equals("BORROWED")) {
return BookState.BORROWED;
} else if (string.equals("READ")) {
return BookState.READ;
} else if (string.equals("RETURNED")) {
return BookState.RETURNED;
} else if (string.equals("BUILDING")) {
return HouseState.BUILDING;
} else if (string.equals("FINISHED")) {
return HouseState.FINISHED;
}
throw new IllegalStateException("Unexpected enum value: " + string);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
package io.micronaut.data.tck.repositories.embedded;

import io.micronaut.data.repository.CrudRepository;
import io.micronaut.data.tck.entities.embedded.BookEntity;
import io.micronaut.data.tck.entities.embedded.BookState;

import java.util.List;

public interface BookEntityRepository extends CrudRepository<BookEntity, Long> {

List<BookEntity> findAllByResourceState(BookState state);
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
package io.micronaut.data.tck.repositories.embedded;

import io.micronaut.data.repository.CrudRepository;
import io.micronaut.data.tck.entities.embedded.HouseEntity;
import io.micronaut.data.tck.entities.embedded.HouseState;

import java.util.List;

public interface HouseEntityRepository extends CrudRepository<HouseEntity, Long> {

List<HouseEntity> findAllByResourceState(HouseState state);
}

0 comments on commit 3341071

Please sign in to comment.