//$Id: SaveOrUpdateTest.java 10977 2006-12-12 23:28:04Z steve.ebersole@jboss.com $
package org.hibernate.test.ops;
import junit.framework.Test;
import org.hibernate.Hibernate;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.hibernate.HibernateException;
import org.hibernate.testing.junit.functional.FunctionalTestCase;
import org.hibernate.testing.junit.functional.FunctionalTestClassTestSuite;
import org.hibernate.intercept.FieldInterceptionHelper;
import org.hibernate.cfg.Configuration;
import org.hibernate.cfg.Environment;
import org.hibernate.criterion.Projections;
/**
* @author Gavin King
*/
public class SaveOrUpdateTest extends FunctionalTestCase {
public SaveOrUpdateTest(String str) {
super( str );
}
public void testSaveOrUpdateDeepTree() {
clearCounts();
Session s = openSession();
Transaction tx = s.beginTransaction();
Node root = new Node( "root" );
Node child = new Node( "child" );
Node grandchild = new Node( "grandchild" );
root.addChild( child );
child.addChild( grandchild );
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 3 );
assertUpdateCount( 0 );
clearCounts();
grandchild.setDescription( "the grand child" );
Node grandchild2 = new Node( "grandchild2" );
child.addChild( grandchild2 );
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 1 );
assertUpdateCount( 1 );
clearCounts();
Node child2 = new Node( "child2" );
Node grandchild3 = new Node( "grandchild3" );
child2.addChild( grandchild3 );
root.addChild( child2 );
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 2 );
assertUpdateCount( 0 );
clearCounts();
s = openSession();
tx = s.beginTransaction();
s.delete( grandchild );
s.delete( grandchild2 );
s.delete( grandchild3 );
s.delete( child );
s.delete( child2 );
s.delete( root );
tx.commit();
s.close();
}
public void testSaveOrUpdateDeepTreeWithGeneratedId() {
boolean instrumented = FieldInterceptionHelper.isInstrumented( new NumberedNode() );
clearCounts();
Session s = openSession();
Transaction tx = s.beginTransaction();
NumberedNode root = new NumberedNode( "root" );
NumberedNode child = new NumberedNode( "child" );
NumberedNode grandchild = new NumberedNode( "grandchild" );
root.addChild( child );
child.addChild( grandchild );
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 3 );
assertUpdateCount( 0 );
clearCounts();
child = ( NumberedNode ) root.getChildren().iterator().next();
grandchild = ( NumberedNode ) child.getChildren().iterator().next();
grandchild.setDescription( "the grand child" );
NumberedNode grandchild2 = new NumberedNode( "grandchild2" );
child.addChild( grandchild2 );
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 1 );
assertUpdateCount( instrumented ? 1 : 3 );
clearCounts();
NumberedNode child2 = new NumberedNode( "child2" );
NumberedNode grandchild3 = new NumberedNode( "grandchild3" );
child2.addChild( grandchild3 );
root.addChild( child2 );
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 2 );
assertUpdateCount( instrumented ? 0 : 4 );
clearCounts();
s = openSession();
tx = s.beginTransaction();
s.createQuery( "delete from NumberedNode where name like 'grand%'" ).executeUpdate();
s.createQuery( "delete from NumberedNode where name like 'child%'" ).executeUpdate();
s.createQuery( "delete from NumberedNode" ).executeUpdate();
tx.commit();
s.close();
}
public void testSaveOrUpdateTree() {
clearCounts();
Session s = openSession();
Transaction tx = s.beginTransaction();
Node root = new Node( "root" );
Node child = new Node( "child" );
root.addChild( child );
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 2 );
clearCounts();
root.setDescription( "The root node" );
child.setDescription( "The child node" );
Node secondChild = new Node( "second child" );
root.addChild( secondChild );
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 1 );
assertUpdateCount( 2 );
s = openSession();
tx = s.beginTransaction();
s.createQuery( "delete from Node where parent is not null" ).executeUpdate();
s.createQuery( "delete from Node" ).executeUpdate();
tx.commit();
s.close();
}
public void testSaveOrUpdateTreeWithGeneratedId() {
clearCounts();
Session s = openSession();
Transaction tx = s.beginTransaction();
NumberedNode root = new NumberedNode( "root" );
NumberedNode child = new NumberedNode( "child" );
root.addChild( child );
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 2 );
clearCounts();
root.setDescription( "The root node" );
child.setDescription( "The child node" );
NumberedNode secondChild = new NumberedNode( "second child" );
root.addChild( secondChild );
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 1 );
assertUpdateCount( 2 );
s = openSession();
tx = s.beginTransaction();
s.createQuery( "delete from NumberedNode where parent is not null" ).executeUpdate();
s.createQuery( "delete from NumberedNode" ).executeUpdate();
tx.commit();
s.close();
}
public void testSaveOrUpdateManaged() {
Session s = openSession();
Transaction tx = s.beginTransaction();
NumberedNode root = new NumberedNode( "root" );
s.saveOrUpdate( root );
tx.commit();
tx = s.beginTransaction();
NumberedNode child = new NumberedNode( "child" );
root.addChild( child );
s.saveOrUpdate( root );
assertFalse( s.contains( child ) );
s.flush();
assertTrue( s.contains( child ) );
tx.commit();
assertTrue( root.getChildren().contains( child ) );
assertEquals( root.getChildren().size(), 1 );
tx = s.beginTransaction();
assertEquals(
s.createCriteria( NumberedNode.class )
.setProjection( Projections.rowCount() )
.uniqueResult(),
new Long( 2 )
);
s.delete( root );
s.delete( child );
tx.commit();
s.close();
}
public void testSaveOrUpdateGot() {
boolean instrumented = FieldInterceptionHelper.isInstrumented( new NumberedNode() );
Session s = openSession();
Transaction tx = s.beginTransaction();
NumberedNode root = new NumberedNode( "root" );
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 1 );
assertUpdateCount( 0 );
clearCounts();
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 0 );
assertUpdateCount( instrumented ? 0 : 1 );
s = openSession();
tx = s.beginTransaction();
root = ( NumberedNode ) s.get( NumberedNode.class, new Long( root.getId() ) );
Hibernate.initialize( root.getChildren() );
tx.commit();
s.close();
clearCounts();
s = openSession();
tx = s.beginTransaction();
NumberedNode child = new NumberedNode( "child" );
root.addChild( child );
s.saveOrUpdate( root );
assertTrue( s.contains( child ) );
tx.commit();
assertInsertCount( 1 );
assertUpdateCount( instrumented ? 0 : 1 );
tx = s.beginTransaction();
assertEquals(
s.createCriteria( NumberedNode.class )
.setProjection( Projections.rowCount() )
.uniqueResult(),
new Long( 2 )
);
s.delete( root );
s.delete( child );
tx.commit();
s.close();
}
public void testSaveOrUpdateGotWithMutableProp() {
Session s = openSession();
Transaction tx = s.beginTransaction();
Node root = new Node( "root" );
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 1 );
assertUpdateCount( 0 );
clearCounts();
s = openSession();
tx = s.beginTransaction();
s.saveOrUpdate( root );
tx.commit();
s.close();
assertInsertCount( 0 );
assertUpdateCount( 0 );
s = openSession();
tx = s.beginTransaction();
root = ( Node ) s.get( Node.class, "root" );
Hibernate.initialize( root.getChildren() );
tx.commit();
s.close();
clearCounts();
s = openSession();
tx = s.beginTransaction();
Node child = new Node( "child" );
root.addChild( child );
s.saveOrUpdate( root );
assertTrue( s.contains( child ) );
tx.commit();
assertInsertCount( 1 );
assertUpdateCount( 1 ); //note: will fail here if no second-level cache
tx = s.beginTransaction();
assertEquals(
s.createCriteria( Node.class )
.setProjection( Projections.rowCount() )
.uniqueResult(),
new Long( 2 )
);
s.delete( root );
s.delete( child );
tx.commit();
s.close();
}
public void testEvictThenSaveOrUpdate() {
Session s = openSession();
s.getTransaction().begin();
Node parent = new Node( "1:parent" );
Node child = new Node( "2:child" );
Node grandchild = new Node( "3:grandchild" );
parent.addChild( child );
child.addChild( grandchild );
s.saveOrUpdate( parent );
s.getTransaction().commit();
s.close();
Session s1 = openSession();
s1.getTransaction().begin();
child = ( Node ) s1.load( Node.class, "2:child" );
assertTrue( s1.contains( child ) );
assertFalse( Hibernate.isInitialized( child ) );
assertTrue( s1.contains( child.getParent() ) );
assertTrue( Hibernate.isInitialized( child ) );
assertFalse( Hibernate.isInitialized( child.getChildren() ) );
assertFalse( Hibernate.isInitialized( child.getParent() ) );
assertTrue( s1.contains( child ) );
s1.evict( child );
assertFalse( s1.contains( child ) );
assertTrue( s1.contains( child.getParent() ) );
Session s2 = openSession();
try {
s2.getTransaction().begin();
s2.saveOrUpdate( child );
fail();
}
catch ( HibernateException ex ) {
// expected because parent is connected to s1
}
finally {
s2.getTransaction().rollback();
}
s2.close();
s1.evict( child.getParent() );
assertFalse( s1.contains( child.getParent() ) );
s2 = openSession();
s2.getTransaction().begin();
s2.saveOrUpdate( child );
assertTrue( s2.contains( child ) );
assertFalse( s1.contains( child ) );
assertTrue( s2.contains( child.getParent() ) );
assertFalse( s1.contains( child.getParent() ) );
assertFalse( Hibernate.isInitialized( child.getChildren() ) );
assertFalse( Hibernate.isInitialized( child.getParent() ) );
assertEquals( 1, child.getChildren().size() );
assertEquals( "1:parent", child.getParent().getName() );
assertTrue( Hibernate.isInitialized( child.getChildren() ) );
assertFalse( Hibernate.isInitialized( child.getParent() ) );
assertNull( child.getParent().getDescription() );
assertTrue( Hibernate.isInitialized( child.getParent() ) );
s1.getTransaction().commit();
s2.getTransaction().commit();
s1.close();
s2.close();
s = openSession();
s.beginTransaction();
s.delete( s.get( Node.class, "3:grandchild" ) );
s.delete( s.get( Node.class, "2:child" ) );
s.delete( s.get( Node.class, "1:parent" ) );
s.getTransaction().commit();
s.close();
}
private void clearCounts() {
getSessions().getStatistics().clear();
}
private void assertInsertCount(int count) {
int inserts = ( int ) getSessions().getStatistics().getEntityInsertCount();
assertEquals( count, inserts );
}
private void assertUpdateCount(int count) {
int updates = ( int ) getSessions().getStatistics().getEntityUpdateCount();
assertEquals( count, updates );
}
public void configure(Configuration cfg) {
cfg.setProperty( Environment.GENERATE_STATISTICS, "true" );
cfg.setProperty( Environment.STATEMENT_BATCH_SIZE, "0" );
}
public String[] getMappings() {
return new String[] {"ops/Node.hbm.xml"};
}
public static Test suite() {
return new FunctionalTestClassTestSuite( SaveOrUpdateTest.class );
}
}