HHH-16573 Reproduce NPE with embeddable element collection with updateable = false

This commit is contained in:
Yoann Rodière 2023-05-09 11:37:51 +02:00 committed by Andrea Boriero
parent edd825d133
commit b6721961dd
2 changed files with 268 additions and 0 deletions

View File

@ -0,0 +1,133 @@
package org.hibernate.orm.test.embeddable;
import java.util.ArrayList;
import java.util.List;
import org.hibernate.testing.orm.junit.DomainModel;
import org.hibernate.testing.orm.junit.JiraKey;
import org.hibernate.testing.orm.junit.SessionFactory;
import org.hibernate.testing.orm.junit.SessionFactoryScope;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Test;
import jakarta.persistence.Column;
import jakarta.persistence.ElementCollection;
import jakarta.persistence.Embeddable;
import jakarta.persistence.Entity;
import jakarta.persistence.FetchType;
import jakarta.persistence.GeneratedValue;
import jakarta.persistence.Id;
import static org.assertj.core.api.Assertions.assertThat;
@DomainModel(annotatedClasses = {
EmbeddableAsElementCollectionWithUpdatableFalseTest.MyEntity.class,
EmbeddableAsElementCollectionWithUpdatableFalseTest.MyEmbeddable.class
})
@SessionFactory
@JiraKey("HHH-16573")
public class EmbeddableAsElementCollectionWithUpdatableFalseTest {
@AfterEach
public void tearDown(SessionFactoryScope scope) {
scope.inTransaction(
session -> {
session.createMutationQuery( "delete from MyEntity" ).executeUpdate();
}
);
}
@Test
public void test(SessionFactoryScope scope) {
var id = scope.fromTransaction( session -> {
var entity = new MyEntity();
session.persist( entity );
return entity.getId();
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
assertThat( entity.getMyEmbeddables() ).isEmpty();
entity.getMyEmbeddables().add( new MyEmbeddable( "first" ) );
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
List<MyEmbeddable> myEmbeddables = entity.getMyEmbeddables();
assertThat( myEmbeddables ).hasSize( 1 );
} );
}
@Test
public void insertTest(SessionFactoryScope scope) {
var id = scope.fromTransaction( session -> {
var entity = new MyEntity();
entity.getMyEmbeddables().add( new MyEmbeddable( "first" ) );
entity.getMyEmbeddables().add( new MyEmbeddable( "third" ) );
session.persist( entity );
return entity.getId();
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
assertThat( entity.getMyEmbeddables() ).hasSize( 2 );
entity.getMyEmbeddables().add( 1, new MyEmbeddable( "second" ) );
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
List<MyEmbeddable> myEmbeddables = entity.getMyEmbeddables();
assertThat( myEmbeddables ).hasSize( 3 );
assertThat( myEmbeddables.get( 0 ).getEmbeddedProperty() ).isEqualTo( "first" );
assertThat( myEmbeddables.get( 1 ).getEmbeddedProperty() ).isEqualTo( "second" );
assertThat( myEmbeddables.get( 2 ).getEmbeddedProperty() ).isEqualTo( "third" );
} );
}
@Entity(name = "MyEntity")
static class MyEntity {
@Id
@GeneratedValue
private Integer id;
@ElementCollection(fetch = FetchType.LAZY)
private List<MyEmbeddable> myEmbeddables = new ArrayList<>();
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public List<MyEmbeddable> getMyEmbeddables() {
return myEmbeddables;
}
}
@Embeddable
public static class MyEmbeddable {
@Column(updatable = false)
private String embeddedProperty;
public MyEmbeddable() {
}
public MyEmbeddable(String embeddedProperty) {
this.embeddedProperty = embeddedProperty;
}
public String getEmbeddedProperty() {
return embeddedProperty;
}
public void setEmbeddedProperty(String embeddedProperty) {
this.embeddedProperty = embeddedProperty;
}
}
}

View File

@ -0,0 +1,135 @@
package org.hibernate.orm.test.embeddable;
import java.util.ArrayList;
import java.util.List;
import org.hibernate.testing.orm.junit.DomainModel;
import org.hibernate.testing.orm.junit.JiraKey;
import org.hibernate.testing.orm.junit.SessionFactory;
import org.hibernate.testing.orm.junit.SessionFactoryScope;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Test;
import jakarta.persistence.Column;
import jakarta.persistence.ElementCollection;
import jakarta.persistence.Embeddable;
import jakarta.persistence.Entity;
import jakarta.persistence.FetchType;
import jakarta.persistence.GeneratedValue;
import jakarta.persistence.Id;
import jakarta.persistence.OrderColumn;
import static org.assertj.core.api.Assertions.assertThat;
@DomainModel(annotatedClasses = {
EmbeddableAsOrderedElementCollectionWithUpdatableFalseTest.MyEntity.class,
EmbeddableAsOrderedElementCollectionWithUpdatableFalseTest.MyEmbeddable.class
})
@SessionFactory
@JiraKey("HHH-16573")
public class EmbeddableAsOrderedElementCollectionWithUpdatableFalseTest {
@AfterEach
public void tearDown(SessionFactoryScope scope) {
scope.inTransaction(
session -> {
session.createMutationQuery( "delete from MyEntity" ).executeUpdate();
}
);
}
@Test
public void test(SessionFactoryScope scope) {
var id = scope.fromTransaction( session -> {
var entity = new MyEntity();
session.persist( entity );
return entity.getId();
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
assertThat( entity.getMyEmbeddables() ).isEmpty();
entity.getMyEmbeddables().add( new MyEmbeddable( "first" ) );
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
List<MyEmbeddable> myEmbeddables = entity.getMyEmbeddables();
assertThat( myEmbeddables ).hasSize( 1 );
} );
}
@Test
public void insertTest(SessionFactoryScope scope) {
var id = scope.fromTransaction( session -> {
var entity = new MyEntity();
entity.getMyEmbeddables().add( new MyEmbeddable( "first" ) );
entity.getMyEmbeddables().add( new MyEmbeddable( "third" ) );
session.persist( entity );
return entity.getId();
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
assertThat( entity.getMyEmbeddables() ).hasSize( 2 );
entity.getMyEmbeddables().add( 1, new MyEmbeddable( "second" ) );
} );
scope.inTransaction( session -> {
MyEntity entity = session.find( MyEntity.class, id );
assertThat( entity ).isNotNull();
List<MyEmbeddable> myEmbeddables = entity.getMyEmbeddables();
assertThat( myEmbeddables ).hasSize( 3 );
assertThat( myEmbeddables.get( 0 ).getEmbeddedProperty() ).isEqualTo( "first" );
assertThat( myEmbeddables.get( 1 ).getEmbeddedProperty() ).isEqualTo( "second" );
assertThat( myEmbeddables.get( 2 ).getEmbeddedProperty() ).isEqualTo( "third" );
} );
}
@Entity(name = "MyEntity")
static class MyEntity {
@Id
@GeneratedValue
private Integer id;
@ElementCollection(fetch = FetchType.LAZY)
@OrderColumn
private List<MyEmbeddable> myEmbeddables = new ArrayList<>();
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public List<MyEmbeddable> getMyEmbeddables() {
return myEmbeddables;
}
}
@Embeddable
public static class MyEmbeddable {
@Column(updatable = false)
private String embeddedProperty;
public MyEmbeddable() {
}
public MyEmbeddable(String embeddedProperty) {
this.embeddedProperty = embeddedProperty;
}
public String getEmbeddedProperty() {
return embeddedProperty;
}
public void setEmbeddedProperty(String embeddedProperty) {
this.embeddedProperty = embeddedProperty;
}
}
}