Merge remote-tracking branch 'upstream/main' into wip/6.0
This commit is contained in:
commit
3f132da268
|
@ -260,6 +260,7 @@ public abstract class AbstractPersistentCollection<E> implements Serializable, P
|
|||
}
|
||||
finally {
|
||||
if ( tempSession != null ) {
|
||||
|
||||
// make sure the just opened temp session gets closed!
|
||||
isTempSession = false;
|
||||
session = originalSession;
|
||||
|
@ -274,6 +275,11 @@ public abstract class AbstractPersistentCollection<E> implements Serializable, P
|
|||
LOG.warn( "Unable to close temporary session used to load lazy collection associated to no session" );
|
||||
}
|
||||
}
|
||||
else {
|
||||
if ( !session.isTransactionInProgress() ) {
|
||||
session.getJdbcCoordinator().afterTransaction();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -232,10 +232,12 @@ public final class Cascade {
|
|||
}
|
||||
}
|
||||
else if ( type.isComponentType() ) {
|
||||
if ( componentPath == null ) {
|
||||
if ( componentPath == null && propertyName != null ) {
|
||||
componentPath = new ArrayList<>();
|
||||
}
|
||||
componentPath.add( propertyName );
|
||||
if ( componentPath != null ) {
|
||||
componentPath.add( propertyName );
|
||||
}
|
||||
cascadeComponent(
|
||||
action,
|
||||
cascadePoint,
|
||||
|
@ -246,7 +248,9 @@ public final class Cascade {
|
|||
(CompositeType) type,
|
||||
anything
|
||||
);
|
||||
componentPath.remove( componentPath.size() - 1 );
|
||||
if ( componentPath != null ) {
|
||||
componentPath.remove( componentPath.size() - 1 );
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -293,17 +297,24 @@ public final class Cascade {
|
|||
// Since the loadedState in the EntityEntry is a flat domain type array
|
||||
// We first have to extract the component object and then ask the component type
|
||||
// recursively to give us the value of the sub-property of that object
|
||||
loadedValue = entry.getLoadedValue( componentPath.get( 0 ) );
|
||||
ComponentType componentType = (ComponentType) entry.getPersister().getPropertyType( componentPath.get( 0 ) );
|
||||
if ( componentPath.size() != 1 ) {
|
||||
for ( int i = 1; i < componentPath.size(); i++ ) {
|
||||
final int subPropertyIndex = componentType.getPropertyIndex( componentPath.get( i ) );
|
||||
loadedValue = componentType.getPropertyValue( loadedValue, subPropertyIndex );
|
||||
componentType = (ComponentType) componentType.getSubtypes()[subPropertyIndex];
|
||||
final Type propertyType = entry.getPersister().getPropertyType( componentPath.get(0) );
|
||||
if ( propertyType instanceof ComponentType ) {
|
||||
loadedValue = entry.getLoadedValue( componentPath.get( 0 ) );
|
||||
ComponentType componentType = (ComponentType) propertyType;
|
||||
if ( componentPath.size() != 1 ) {
|
||||
for ( int i = 1; i < componentPath.size(); i++ ) {
|
||||
final int subPropertyIndex = componentType.getPropertyIndex( componentPath.get( i ) );
|
||||
loadedValue = componentType.getPropertyValue( loadedValue, subPropertyIndex );
|
||||
componentType = (ComponentType) componentType.getSubtypes()[subPropertyIndex];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
loadedValue = componentType.getPropertyValue( loadedValue, componentType.getPropertyIndex( propertyName ) );
|
||||
loadedValue = componentType.getPropertyValue( loadedValue, componentType.getPropertyIndex( propertyName ) );
|
||||
}
|
||||
else {
|
||||
// Association is probably defined in an element collection, so we can't do orphan removals
|
||||
loadedValue = null;
|
||||
}
|
||||
}
|
||||
|
||||
// orphaned if the association was nulled (child == null) or receives a new value while the
|
||||
|
@ -538,7 +549,7 @@ public final class Cascade {
|
|||
itr.next(),
|
||||
elemType,
|
||||
style,
|
||||
null,
|
||||
collectionType.getRole().substring( collectionType.getRole().lastIndexOf('.') + 1 ),
|
||||
anything,
|
||||
isCascadeDeleteEnabled
|
||||
);
|
||||
|
|
|
@ -160,22 +160,29 @@ public abstract class AbstractLazyInitializer implements LazyInitializer {
|
|||
@Override
|
||||
public final void initialize() throws HibernateException {
|
||||
if ( !initialized ) {
|
||||
if ( allowLoadOutsideTransaction ) {
|
||||
permissiveInitialization();
|
||||
try {
|
||||
if ( allowLoadOutsideTransaction ) {
|
||||
permissiveInitialization();
|
||||
}
|
||||
else if ( session == null ) {
|
||||
throw new LazyInitializationException( "could not initialize proxy [" + entityName + "#" + id + "] - no Session" );
|
||||
}
|
||||
else if ( !session.isOpenOrWaitingForAutoClose() ) {
|
||||
throw new LazyInitializationException( "could not initialize proxy [" + entityName + "#" + id + "] - the owning Session was closed" );
|
||||
}
|
||||
else if ( !session.isConnected() ) {
|
||||
throw new LazyInitializationException( "could not initialize proxy [" + entityName + "#" + id + "] - the owning Session is disconnected" );
|
||||
}
|
||||
else {
|
||||
target = session.immediateLoad( entityName, id );
|
||||
initialized = true;
|
||||
checkTargetState( session );
|
||||
}
|
||||
}
|
||||
else if ( session == null ) {
|
||||
throw new LazyInitializationException( "could not initialize proxy [" + entityName + "#" + id + "] - no Session" );
|
||||
}
|
||||
else if ( !session.isOpenOrWaitingForAutoClose() ) {
|
||||
throw new LazyInitializationException( "could not initialize proxy [" + entityName + "#" + id + "] - the owning Session was closed" );
|
||||
}
|
||||
else if ( !session.isConnected() ) {
|
||||
throw new LazyInitializationException( "could not initialize proxy [" + entityName + "#" + id + "] - the owning Session is disconnected" );
|
||||
}
|
||||
else {
|
||||
target = session.immediateLoad( entityName, id );
|
||||
initialized = true;
|
||||
checkTargetState(session);
|
||||
finally {
|
||||
if ( session != null && !session.isTransactionInProgress() ) {
|
||||
session.getJdbcCoordinator().afterTransaction();
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
|
|
|
@ -0,0 +1,365 @@
|
|||
/*
|
||||
* Hibernate, Relational Persistence for Idiomatic Java
|
||||
*
|
||||
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
|
||||
* See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
|
||||
*/
|
||||
package org.hibernate.test.connections;
|
||||
|
||||
import java.sql.Connection;
|
||||
import java.sql.DriverManager;
|
||||
import java.sql.SQLException;
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Properties;
|
||||
import java.util.Set;
|
||||
import javax.persistence.Entity;
|
||||
import javax.persistence.EntityManager;
|
||||
import javax.persistence.FetchType;
|
||||
import javax.persistence.Id;
|
||||
import javax.persistence.JoinColumn;
|
||||
import javax.persistence.ManyToOne;
|
||||
import javax.persistence.OneToMany;
|
||||
import javax.persistence.Query;
|
||||
import javax.sql.DataSource;
|
||||
|
||||
import org.hibernate.annotations.Fetch;
|
||||
import org.hibernate.annotations.FetchMode;
|
||||
import org.hibernate.annotations.LazyCollection;
|
||||
import org.hibernate.annotations.LazyCollectionOption;
|
||||
import org.hibernate.annotations.LazyToOne;
|
||||
import org.hibernate.annotations.LazyToOneOption;
|
||||
import org.hibernate.cfg.AvailableSettings;
|
||||
import org.hibernate.cfg.Environment;
|
||||
import org.hibernate.engine.jdbc.connections.internal.UserSuppliedConnectionProviderImpl;
|
||||
import org.hibernate.jpa.test.BaseEntityManagerFunctionalTestCase;
|
||||
import org.hibernate.jpa.test.connection.BaseDataSource;
|
||||
import org.hibernate.resource.jdbc.spi.PhysicalConnectionHandlingMode;
|
||||
|
||||
import org.hibernate.testing.TestForIssue;
|
||||
import org.junit.Assert;
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
|
||||
import static junit.framework.TestCase.assertTrue;
|
||||
import static org.hibernate.testing.transaction.TransactionUtil.doInJPA;
|
||||
import static org.junit.Assert.assertEquals;
|
||||
import static org.mockito.Mockito.spy;
|
||||
|
||||
/**
|
||||
* @author Selaron
|
||||
*/
|
||||
@TestForIssue(jiraKey = "HHH-4808")
|
||||
public class LazyLoadingConnectionCloseTest extends BaseEntityManagerFunctionalTestCase {
|
||||
|
||||
private ConnectionProviderDecorator connectionProvider;
|
||||
|
||||
@Override
|
||||
protected Class<?>[] getAnnotatedClasses() {
|
||||
return new Class[] { SimpleEntity.class, ChildEntity.class };
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Map getConfig() {
|
||||
Map config = super.getConfig();
|
||||
config.put( AvailableSettings.ENABLE_LAZY_LOAD_NO_TRANS, "true" );
|
||||
|
||||
config.put( AvailableSettings.CONNECTION_HANDLING, PhysicalConnectionHandlingMode.DELAYED_ACQUISITION_AND_RELEASE_AFTER_STATEMENT );
|
||||
|
||||
config.put( AvailableSettings.AUTOCOMMIT, "false" );
|
||||
|
||||
connectionProvider = new ConnectionProviderDecorator( getDataSource() );
|
||||
config.put( AvailableSettings.CONNECTION_PROVIDER, connectionProvider );
|
||||
return config;
|
||||
}
|
||||
|
||||
@Before
|
||||
public void setUp() {
|
||||
doInJPA( this::entityManagerFactory, entityManager -> {
|
||||
final SimpleEntity entity = new SimpleEntity();
|
||||
entity.setId( 1L );
|
||||
entity.setName( "TheParent" );
|
||||
|
||||
final ChildEntity c1 = new ChildEntity();
|
||||
c1.setId( 1L );
|
||||
c1.setParent( entity );
|
||||
c1.setName( "child1" );
|
||||
|
||||
final ChildEntity c2 = new ChildEntity();
|
||||
c2.setId( 2L );
|
||||
c2.setParent( entity );
|
||||
c2.setName( "child2" );
|
||||
|
||||
entityManager.persist( entity );
|
||||
entityManager.persist( c1 );
|
||||
entityManager.persist( c2 );
|
||||
} );
|
||||
}
|
||||
|
||||
/**
|
||||
* Tests connections get closed after transaction commit.
|
||||
*/
|
||||
@Test
|
||||
public void testConnectionCloseAfterTx() {
|
||||
connectionProvider.clear();
|
||||
EntityManager entityManager = getOrCreateEntityManager();
|
||||
|
||||
try {
|
||||
entityManager.getTransaction().begin();
|
||||
try {
|
||||
|
||||
final Query qry = entityManager.createQuery( "FROM SimpleEntity" );
|
||||
final List<SimpleEntity> entities = qry.getResultList();
|
||||
final SimpleEntity entity = entities.get( 0 );
|
||||
assertEquals( 1, connectionProvider.getCurrentOpenConnections() );
|
||||
}
|
||||
catch (Exception e) {
|
||||
if ( entityManager.getTransaction().isActive() ) {
|
||||
entityManager.getTransaction().rollback();
|
||||
}
|
||||
throw e;
|
||||
}
|
||||
finally {
|
||||
if ( entityManager.getTransaction().isActive() ) {
|
||||
entityManager.getTransaction().commit();
|
||||
}
|
||||
}
|
||||
assertTrue( connectionProvider.areAllConnectionClosed() );
|
||||
}
|
||||
finally {
|
||||
entityManager.close();
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Tests connections get closed after lazy collection initialization.
|
||||
*/
|
||||
@Test
|
||||
public void testConnectionCloseAfterLazyCollectionInit() {
|
||||
connectionProvider.clear();
|
||||
EntityManager entityManager = getOrCreateEntityManager();
|
||||
|
||||
try {
|
||||
final Query qry = entityManager.createQuery( "FROM SimpleEntity" );
|
||||
final List<SimpleEntity> entities = qry.getResultList();
|
||||
final SimpleEntity entity = entities.get( 0 );
|
||||
|
||||
// assert no connection is open
|
||||
assertTrue( connectionProvider.areAllConnectionClosed() );
|
||||
|
||||
final int oldOpenedConnections = connectionProvider.getTotalOpenedConnectionCount();
|
||||
final Set<ChildEntity> lazyChildren = entity.getChildren();
|
||||
|
||||
// this will initialize the collection and such trigger a query
|
||||
lazyChildren.stream().findAny();
|
||||
|
||||
// assert a connection had been opened
|
||||
Assert.assertTrue( oldOpenedConnections < connectionProvider.getTotalOpenedConnectionCount() );
|
||||
|
||||
// assert there's no remaining connection left.
|
||||
assertTrue( connectionProvider.areAllConnectionClosed() );
|
||||
|
||||
}
|
||||
finally {
|
||||
entityManager.close();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Tests connections get closed after transaction commit.
|
||||
*/
|
||||
@Test
|
||||
public void testConnectionCloseAfterLazyPojoPropertyInit() {
|
||||
connectionProvider.clear();
|
||||
EntityManager entityManager = getOrCreateEntityManager();
|
||||
|
||||
try {
|
||||
final Query qry = entityManager.createQuery( "FROM ChildEntity" );
|
||||
final List<ChildEntity> entities = qry.getResultList();
|
||||
final ChildEntity entity = entities.get( 0 );
|
||||
|
||||
// assert no connection is open
|
||||
assertTrue( connectionProvider.areAllConnectionClosed() );
|
||||
|
||||
final int oldOpenedConnections = connectionProvider.getTotalOpenedConnectionCount();
|
||||
|
||||
final SimpleEntity parent = entity.getParent();
|
||||
// this will initialize the collection and such trigger a query
|
||||
parent.getName();
|
||||
// assert a connection had been opened
|
||||
Assert.assertTrue( oldOpenedConnections < connectionProvider.getTotalOpenedConnectionCount() );
|
||||
|
||||
|
||||
// assert there's no remaining connection left.
|
||||
assertTrue( connectionProvider.areAllConnectionClosed() );
|
||||
}
|
||||
finally {
|
||||
entityManager.close();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Tests connections get closed after transaction commit.
|
||||
*/
|
||||
@Test
|
||||
public void testConnectionCloseAfterQueryWithoutTx() {
|
||||
connectionProvider.clear();
|
||||
EntityManager entityManager = getOrCreateEntityManager();
|
||||
|
||||
try {
|
||||
|
||||
final int oldOpenedConnections = connectionProvider.getTotalOpenedConnectionCount();
|
||||
final List<ChildEntity> childrenByQuery = entityManager.createQuery( "FROM ChildEntity" ).getResultList();
|
||||
assertTrue( childrenByQuery.size() > 0 );
|
||||
|
||||
// assert a connection had been opened
|
||||
assertTrue( oldOpenedConnections < connectionProvider.getTotalOpenedConnectionCount() );
|
||||
// assert there's no remaining connection left.
|
||||
assertTrue( connectionProvider.areAllConnectionClosed() );
|
||||
}
|
||||
finally {
|
||||
entityManager.close();
|
||||
}
|
||||
}
|
||||
|
||||
@Entity(name = "SimpleEntity")
|
||||
public static class SimpleEntity {
|
||||
private Long id;
|
||||
|
||||
private String name;
|
||||
|
||||
Set<ChildEntity> children = new HashSet<>();
|
||||
|
||||
@Id
|
||||
public Long getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(final Long id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public void setName(final String name) {
|
||||
this.name = name;
|
||||
}
|
||||
|
||||
@OneToMany(targetEntity = ChildEntity.class, mappedBy = "parent")
|
||||
@LazyCollection(LazyCollectionOption.EXTRA)
|
||||
@Fetch(FetchMode.SELECT)
|
||||
public Set<ChildEntity> getChildren() {
|
||||
return children;
|
||||
}
|
||||
|
||||
public void setChildren(final Set<ChildEntity> children) {
|
||||
this.children = children;
|
||||
}
|
||||
}
|
||||
|
||||
@Entity(name = "ChildEntity")
|
||||
public static class ChildEntity {
|
||||
private Long id;
|
||||
|
||||
private String name;
|
||||
|
||||
private SimpleEntity parent;
|
||||
|
||||
@Id
|
||||
public Long getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(final Long id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public void setName(final String name) {
|
||||
this.name = name;
|
||||
}
|
||||
|
||||
@ManyToOne(fetch = FetchType.LAZY)
|
||||
@JoinColumn
|
||||
@LazyToOne(LazyToOneOption.PROXY)
|
||||
public SimpleEntity getParent() {
|
||||
return parent;
|
||||
}
|
||||
|
||||
public void setParent(final SimpleEntity parent) {
|
||||
this.parent = parent;
|
||||
}
|
||||
}
|
||||
|
||||
private BaseDataSource getDataSource() {
|
||||
final Properties connectionProps = new Properties();
|
||||
connectionProps.put( "user", Environment.getProperties().getProperty( Environment.USER ) );
|
||||
connectionProps.put( "password", Environment.getProperties().getProperty( Environment.PASS ) );
|
||||
|
||||
final String url = Environment.getProperties().getProperty( Environment.URL );
|
||||
return new BaseDataSource() {
|
||||
@Override
|
||||
public Connection getConnection() throws SQLException {
|
||||
return DriverManager.getConnection( url, connectionProps );
|
||||
}
|
||||
|
||||
@Override
|
||||
public Connection getConnection(String username, String password) throws SQLException {
|
||||
return DriverManager.getConnection( url, connectionProps );
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
public static class ConnectionProviderDecorator extends UserSuppliedConnectionProviderImpl {
|
||||
|
||||
private final DataSource dataSource;
|
||||
|
||||
private int connectionCount;
|
||||
private int openConnections;
|
||||
|
||||
private Connection connection;
|
||||
|
||||
public ConnectionProviderDecorator(DataSource dataSource) {
|
||||
this.dataSource = dataSource;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Connection getConnection() throws SQLException {
|
||||
connectionCount++;
|
||||
openConnections++;
|
||||
connection = spy( dataSource.getConnection() );
|
||||
return connection;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void closeConnection(Connection connection) throws SQLException {
|
||||
connection.close();
|
||||
openConnections--;
|
||||
}
|
||||
|
||||
public int getTotalOpenedConnectionCount() {
|
||||
return this.connectionCount;
|
||||
}
|
||||
|
||||
public int getCurrentOpenConnections() {
|
||||
return openConnections;
|
||||
}
|
||||
|
||||
public boolean areAllConnectionClosed() {
|
||||
return openConnections == 0;
|
||||
}
|
||||
|
||||
public void clear() {
|
||||
connectionCount = 0;
|
||||
openConnections = 0;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,53 @@
|
|||
package org.hibernate.test.orphan.elementcollection;
|
||||
|
||||
import java.util.Collections;
|
||||
|
||||
import org.hibernate.Session;
|
||||
import org.hibernate.SessionFactory;
|
||||
import org.hibernate.boot.Metadata;
|
||||
import org.hibernate.boot.MetadataSources;
|
||||
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
|
||||
import org.hibernate.testing.TestForIssue;
|
||||
import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
|
||||
import org.hibernate.testing.transaction.TransactionUtil;
|
||||
import org.junit.Before;
|
||||
import org.junit.Test;
|
||||
|
||||
@TestForIssue(jiraKey = "HHH-14597")
|
||||
public class ElementCollectionOrphanTest extends BaseCoreFunctionalTestCase {
|
||||
|
||||
@Override
|
||||
protected String[] getMappings() {
|
||||
return new String[] { "orphan/elementcollection/student.hbm.xml" };
|
||||
}
|
||||
|
||||
@Test
|
||||
public void setCompositeElementTest() {
|
||||
TransactionUtil.doInHibernate(
|
||||
this::sessionFactory,
|
||||
session -> {
|
||||
EnrollableClass aClass = new EnrollableClass();
|
||||
aClass.setId("123");
|
||||
aClass.setName("Math");
|
||||
session.save(aClass);
|
||||
|
||||
Student aStudent = new Student();
|
||||
aStudent.setId("s1");
|
||||
aStudent.setFirstName("John");
|
||||
aStudent.setLastName("Smith");
|
||||
|
||||
EnrolledClassSeat seat = new EnrolledClassSeat();
|
||||
seat.setId("seat1");
|
||||
seat.setRow(10);
|
||||
seat.setColumn(5);
|
||||
|
||||
StudentEnrolledClass enrClass = new StudentEnrolledClass();
|
||||
enrClass.setEnrolledClass(aClass);
|
||||
enrClass.setClassStartTime(130);
|
||||
enrClass.setSeat(seat);
|
||||
aStudent.setEnrolledClasses(Collections.singleton(enrClass));
|
||||
session.save(aStudent);
|
||||
}
|
||||
);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,34 @@
|
|||
package org.hibernate.test.orphan.elementcollection;
|
||||
|
||||
import javax.persistence.Column;
|
||||
import javax.persistence.Entity;
|
||||
import javax.persistence.Id;
|
||||
import javax.persistence.Table;
|
||||
|
||||
@Entity
|
||||
@Table(name = "ENROLLABLECLASS")
|
||||
public class EnrollableClass {
|
||||
|
||||
@Id
|
||||
@Column(name = "id")
|
||||
private String id;
|
||||
|
||||
@Column(name = "name")
|
||||
private String name;
|
||||
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(String id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public String getName() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public void setName(String name) {
|
||||
this.name = name;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,31 @@
|
|||
package org.hibernate.test.orphan.elementcollection;
|
||||
|
||||
public class EnrolledClassSeat {
|
||||
private String id;
|
||||
private int row;
|
||||
private int column;
|
||||
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(String id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public int getRow() {
|
||||
return row;
|
||||
}
|
||||
|
||||
public void setRow(int row) {
|
||||
this.row = row;
|
||||
}
|
||||
|
||||
public int getColumn() {
|
||||
return column;
|
||||
}
|
||||
|
||||
public void setColumn(int column) {
|
||||
this.column = column;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,43 @@
|
|||
package org.hibernate.test.orphan.elementcollection;
|
||||
|
||||
import java.util.Set;
|
||||
|
||||
public class Student {
|
||||
|
||||
private String id;
|
||||
private String firstName;
|
||||
private String lastName;
|
||||
private Set< StudentEnrolledClass > enrolledClasses;
|
||||
|
||||
public String getId() {
|
||||
return id;
|
||||
}
|
||||
|
||||
public void setId(String id) {
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public String getFirstName() {
|
||||
return firstName;
|
||||
}
|
||||
|
||||
public void setFirstName(String firstName) {
|
||||
this.firstName = firstName;
|
||||
}
|
||||
|
||||
public String getLastName() {
|
||||
return lastName;
|
||||
}
|
||||
|
||||
public void setLastName(String lastName) {
|
||||
this.lastName = lastName;
|
||||
}
|
||||
|
||||
public Set<StudentEnrolledClass> getEnrolledClasses() {
|
||||
return enrolledClasses;
|
||||
}
|
||||
|
||||
public void setEnrolledClasses(Set<StudentEnrolledClass> enrolledClasses) {
|
||||
this.enrolledClasses = enrolledClasses;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,31 @@
|
|||
package org.hibernate.test.orphan.elementcollection;
|
||||
|
||||
public class StudentEnrolledClass {
|
||||
private EnrollableClass enrolledClass;
|
||||
private int classStartTime;
|
||||
private EnrolledClassSeat seat;
|
||||
|
||||
public EnrollableClass getEnrolledClass() {
|
||||
return enrolledClass;
|
||||
}
|
||||
|
||||
public void setEnrolledClass(EnrollableClass enrolledClass) {
|
||||
this.enrolledClass = enrolledClass;
|
||||
}
|
||||
|
||||
public int getClassStartTime() {
|
||||
return classStartTime;
|
||||
}
|
||||
|
||||
public void setClassStartTime(int classStartTime) {
|
||||
this.classStartTime = classStartTime;
|
||||
}
|
||||
|
||||
public EnrolledClassSeat getSeat() {
|
||||
return seat;
|
||||
}
|
||||
|
||||
public void setSeat(EnrolledClassSeat seat) {
|
||||
this.seat = seat;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,45 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
|
||||
<!DOCTYPE hibernate-mapping PUBLIC
|
||||
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
|
||||
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
|
||||
|
||||
<hibernate-mapping>
|
||||
<class name="org.hibernate.test.orphan.elementcollection.Student" table="STUDENT" lazy="true" >
|
||||
|
||||
<id name="id" column="id">
|
||||
<generator class="assigned"/>
|
||||
</id>
|
||||
|
||||
<property name="firstName" column="FIRSTNAME" />
|
||||
<property name="lastName" column="LASTNAME" />
|
||||
|
||||
<set name="enrolledClasses" table="STUDENT_CLASS" lazy="true" fetch="subselect" cascade="all">
|
||||
<key column="STUDENT_ID" />
|
||||
|
||||
<composite-element class="org.hibernate.test.orphan.elementcollection.StudentEnrolledClass" >
|
||||
<many-to-one class="org.hibernate.test.orphan.elementcollection.EnrollableClass" name="enrolledClass"
|
||||
column="ENROLLED_CLASS_ID" not-null="true" lazy="false" unique="false" cascade="none" />
|
||||
<property name="classStartTime" column="STARTTIME" />
|
||||
<many-to-one class="org.hibernate.test.orphan.elementcollection.EnrolledClassSeat" name="seat"
|
||||
column="ENROLLED_CLASS_SEAT_ID" not-null="true" lazy="false" unique="true" cascade="all-delete-orphan" />
|
||||
</composite-element>
|
||||
</set>
|
||||
</class>
|
||||
|
||||
<class name="org.hibernate.test.orphan.elementcollection.EnrollableClass" lazy="true" >
|
||||
|
||||
<id name="id" column="id">
|
||||
<generator class="assigned"/>
|
||||
</id>
|
||||
<property name="name" column="NAME" />
|
||||
</class>
|
||||
|
||||
<class name="org.hibernate.test.orphan.elementcollection.EnrolledClassSeat" lazy="true" >
|
||||
<id name="id" column="id">
|
||||
<generator class="assigned"/>
|
||||
</id>
|
||||
<property name="row" column="SEAT_ROW" />
|
||||
<property name="column" column="SEAT_COLUMN" />
|
||||
</class>
|
||||
</hibernate-mapping>
|
Loading…
Reference in New Issue