diff --git a/spring-orm/src/main/java/org/springframework/orm/jpa/persistenceunit/PersistenceManagedTypesBeanRegistrationAotProcessor.java b/spring-orm/src/main/java/org/springframework/orm/jpa/persistenceunit/PersistenceManagedTypesBeanRegistrationAotProcessor.java index 2a6d68a2b615..b334d327bac9 100644 --- a/spring-orm/src/main/java/org/springframework/orm/jpa/persistenceunit/PersistenceManagedTypesBeanRegistrationAotProcessor.java +++ b/spring-orm/src/main/java/org/springframework/orm/jpa/persistenceunit/PersistenceManagedTypesBeanRegistrationAotProcessor.java @@ -23,6 +23,7 @@ import jakarta.persistence.Convert; import jakarta.persistence.Converter; +import jakarta.persistence.Embedded; import jakarta.persistence.EntityListeners; import jakarta.persistence.IdClass; import jakarta.persistence.PostLoad; @@ -64,11 +65,26 @@ * @author Sebastien Deleuze * @since 6.0 */ +@SuppressWarnings("unchecked") class PersistenceManagedTypesBeanRegistrationAotProcessor implements BeanRegistrationAotProcessor { private static final List> CALLBACK_TYPES = List.of(PreUpdate.class, PostUpdate.class, PrePersist.class, PostPersist.class, PreRemove.class, PostRemove.class, PostLoad.class); + @Nullable + private static Class embeddableInstantiatorClass; + + static { + try { + embeddableInstantiatorClass = (Class) ClassUtils.forName("org.hibernate.annotations.EmbeddableInstantiator", + PersistenceManagedTypesBeanRegistrationAotProcessor.class.getClassLoader()); + } + catch (ClassNotFoundException ex) { + embeddableInstantiatorClass = null; + } + } + + @Nullable @Override public BeanRegistrationAotContribution processAheadOfTime(RegisteredBean registeredBean) { @@ -129,6 +145,7 @@ private void contributeHints(RuntimeHints hints, List managedClassNames) contributeIdClassHints(hints, managedClass); contributeConverterHints(hints, managedClass); contributeCallbackHints(hints, managedClass); + contributeHibernateHints(hints, managedClass); } catch (ClassNotFoundException ex) { throw new IllegalArgumentException("Failed to instantiate the managed class: " + managedClassName, ex); @@ -176,5 +193,23 @@ private void contributeCallbackHints(RuntimeHints hints, Class managedClass) reflection.registerMethod(method, ExecutableMode.INVOKE), method -> CALLBACK_TYPES.stream().anyMatch(method::isAnnotationPresent)); } + + @SuppressWarnings("unchecked") + private void contributeHibernateHints(RuntimeHints hints, Class managedClass) { + if (embeddableInstantiatorClass == null) { + return; + } + ReflectionHints reflection = hints.reflection(); + ReflectionUtils.doWithFields(managedClass, field -> { + Embedded embeddedAnnotation = AnnotationUtils.findAnnotation(field, Embedded.class); + if (embeddedAnnotation != null && field.getAnnotatedType().getType() instanceof Class embeddedClass) { + Annotation embeddableInstantiatorAnnotation = AnnotationUtils.findAnnotation(embeddedClass, embeddableInstantiatorClass); + if (embeddableInstantiatorAnnotation != null) { + Class embeddableInstantiatorClass = (Class) AnnotationUtils.getAnnotationAttributes(embeddableInstantiatorAnnotation).get("value"); + reflection.registerType(embeddableInstantiatorClass, MemberCategory.INVOKE_DECLARED_CONSTRUCTORS); + } + } + }); + } } }