Updated test to JUnit5 to remove reference to Dialect subclass

Moved it back for now to the old location, since it fails with a
non-related error that will have to be looked at as soon as there is an
implementation for composite sub-types

Signed-off-by: Jan Schatteman <jschatte@redhat.com>
This commit is contained in:
Jan Schatteman 2021-10-29 17:28:25 +02:00 committed by Christian Beikov
parent 599b0ba39f
commit 2d38df66fd
2 changed files with 883 additions and 899 deletions

View File

@ -1,899 +0,0 @@
/*
* 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.orm.test.sql.hand.query;
import java.io.Serializable;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import jakarta.persistence.PersistenceException;
import org.hibernate.Hibernate;
import org.hibernate.QueryException;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.hibernate.cfg.Configuration;
import org.hibernate.cfg.Environment;
import org.hibernate.dialect.AbstractHANADialect;
import org.hibernate.dialect.H2Dialect;
import org.hibernate.dialect.MySQL5Dialect;
import org.hibernate.query.NativeQuery;
import org.hibernate.query.Query;
import org.hibernate.transform.BasicTransformerAdapter;
import org.hibernate.transform.DistinctRootEntityResultTransformer;
import org.hibernate.transform.Transformers;
import org.hibernate.type.StandardBasicTypes;
import org.hibernate.testing.FailureExpected;
import org.hibernate.testing.RequiresDialect;
import org.hibernate.testing.SkipForDialect;
import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
import org.hibernate.orm.test.sql.hand.Dimension;
import org.hibernate.orm.test.sql.hand.Employment;
import org.hibernate.orm.test.sql.hand.Group;
import org.hibernate.orm.test.sql.hand.ImageHolder;
import org.hibernate.orm.test.sql.hand.Order;
import org.hibernate.orm.test.sql.hand.Organization;
import org.hibernate.orm.test.sql.hand.Person;
import org.hibernate.orm.test.sql.hand.Product;
import org.hibernate.orm.test.sql.hand.SpaceShip;
import org.hibernate.orm.test.sql.hand.Speech;
import org.hibernate.orm.test.sql.hand.TextHolder;
import org.junit.Ignore;
import org.junit.Test;
import static org.hibernate.testing.junit4.ExtraAssertions.assertClassAssignability;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail;
/**
* Tests of various features of native SQL queries.
*
* @author Steve Ebersole
*/
// todo (6.0): needs a composite user type mechanism e.g. by providing a custom ComponentTuplizer/Instantiator
@Ignore( "Missing support for composite user types" )
public class NativeSQLQueriesTest extends BaseCoreFunctionalTestCase {
@Override
protected String getBaseForMappings() {
return "org/hibernate/orm/test/";
}
@Override
public String[] getMappings() {
return new String[] { "sql/hand/query/NativeSQLQueries.hbm.xml" };
}
@Override
public void configure(Configuration cfg) {
super.configure( cfg );
cfg.setProperty( Environment.GENERATE_STATISTICS, "true" );
}
protected String getOrganizationFetchJoinEmploymentSQL() {
return "SELECT org.ORGID as {org.id}, " +
" org.NAME as {org.name}, " +
" emp.EMPLOYER as {emp.key}, " +
" emp.EMPID as {emp.element}, " +
" {emp.element.*} " +
"FROM ORGANIZATION org " +
" LEFT OUTER JOIN EMPLOYMENT emp ON org.ORGID = emp.EMPLOYER";
}
protected String getOrganizationJoinEmploymentSQL() {
return "SELECT org.ORGID as {org.id}, " +
" org.NAME as {org.name}, " +
" {emp.*} " +
"FROM ORGANIZATION org " +
" LEFT OUTER JOIN EMPLOYMENT emp ON org.ORGID = emp.EMPLOYER";
}
protected String getEmploymentSQL() {
return "SELECT * FROM EMPLOYMENT";
}
protected String getEmploymentSQLMixedScalarEntity() {
return "SELECT e.*, e.employer as employerid FROM EMPLOYMENT e" ;
}
protected String getOrgEmpRegionSQL() {
return "select {org.*}, {emp.*}, emp.REGIONCODE " +
"from ORGANIZATION org " +
" left outer join EMPLOYMENT emp on org.ORGID = emp.EMPLOYER";
}
protected String getOrgEmpPersonSQL() {
return "select {org.*}, {emp.*}, {pers.*} " +
"from ORGANIZATION org " +
" join EMPLOYMENT emp on org.ORGID = emp.EMPLOYER " +
" join PERSON pers on pers.PERID = emp.EMPLOYEE ";
}
protected String getDescriptionsSQL() {
return "select DESCRIPTION from TEXT_HOLDER";
}
protected String getPhotosSQL() {
return "select PHOTO from IMAGE_HOLDER";
}
@Test
@SkipForDialect( H2Dialect.class )
public void testFailOnNoAddEntityOrScalar() {
// Note: this passes, but for the wrong reason.
// there is actually an exception thrown, but it is the database
// throwing a sql exception because the SQL gets passed
// "un-processed"...
//
// Oddly, H2 accepts this query.
Session s = openSession();
s.beginTransaction();
try {
String sql = "select {org.*} " +
"from organization org";
s.createNativeQuery( sql ).list();
fail( "Should throw an exception since no addEntity nor addScalar has been performed." );
}
catch( PersistenceException pe) {
// expected behavior
}
finally {
s.getTransaction().rollback();
s.close();
}
}
@Test
public void testManualSynchronization() {
Session s = openSession();
s.beginTransaction();
sessionFactory().getStatistics().clear();
// create an Organization...
Organization jboss = new Organization( "JBoss" );
s.persist( jboss );
// now query on Employment, this should not cause an auto-flush
s.createNativeQuery( getEmploymentSQL() ).addSynchronizedQuerySpace( "ABC" ).list();
assertEquals( 0, sessionFactory().getStatistics().getEntityInsertCount() );
// now try to query on Employment but this time add Organization as a synchronized query space...
s.createNativeQuery( getEmploymentSQL() ).addSynchronizedEntityClass( Organization.class ).list();
assertEquals( 1, sessionFactory().getStatistics().getEntityInsertCount() );
// clean up
s.delete( jboss );
s.getTransaction().commit();
s.close();
}
@Test
public void testSQLQueryInterface() {
Session s = openSession();
Transaction t = s.beginTransaction();
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
s.persist(ifa);
s.persist(jboss);
s.persist(gavin);
s.persist(emp);
List l = s.createNativeQuery( getOrgEmpRegionSQL() )
.addEntity("org", Organization.class)
.addJoin("emp", "org.employments")
.addScalar("regionCode", StandardBasicTypes.STRING )
.list();
assertEquals( 2, l.size() );
l = s.createNativeQuery( getOrgEmpPersonSQL() )
.addEntity("org", Organization.class)
.addJoin("emp", "org.employments")
.addJoin("pers", "emp.employee")
.list();
assertEquals( l.size(), 1 );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
l = s.createNativeQuery( "select {org.*}, {emp.*} " +
"from ORGANIZATION org " +
" left outer join EMPLOYMENT emp on org.ORGID = emp.EMPLOYER, ORGANIZATION org2" )
.addEntity("org", Organization.class)
.addJoin("emp", "org.employments")
.setResultTransformer( DistinctRootEntityResultTransformer.INSTANCE )
.list();
assertEquals( l.size(), 2 );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
s.delete(emp);
s.delete(gavin);
s.delete(ifa);
s.delete(jboss);
t.commit();
s.close();
}
@Test
public void testResultSetMappingDefinition() {
Session s = openSession();
Transaction t = s.beginTransaction();
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
s.persist(ifa);
s.persist(jboss);
s.persist(gavin);
s.persist(emp);
List l = s.createNativeQuery( getOrgEmpRegionSQL(), "org-emp-regionCode" ).list();
assertEquals( l.size(), 2 );
l = s.createNativeQuery( getOrgEmpPersonSQL(), "org-emp-person" ).list();
assertEquals( l.size(), 1 );
s.delete(emp);
s.delete(gavin);
s.delete(ifa);
s.delete(jboss);
t.commit();
s.close();
}
@Test
public void testScalarValues() throws Exception {
Session s = openSession();
Transaction t = s.beginTransaction();
Organization ifa = new Organization( "IFA" );
Organization jboss = new Organization( "JBoss" );
Object idIfa = s.save( ifa );
Object idJBoss = s.save( jboss );
s.flush();
List result = s.getNamedQuery( "orgNamesOnly" ).list();
assertTrue( result.contains( "IFA" ) );
assertTrue( result.contains( "JBoss" ) );
result = s.getNamedQuery( "orgNamesOnly" ).setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP).list();
Map m = (Map) result.get(0);
assertEquals( 2, result.size() );
assertEquals( 1, m.size() );
assertTrue( m.containsKey("NAME") );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
Iterator iter = s.getNamedQuery( "orgNamesAndOrgs" ).list().iterator();
Object[] o = ( Object[] ) iter.next();
assertEquals( "expecting 2 values", 2, o.length );
assertEquals( o[0], "IFA" );
assertEquals( ( ( Organization ) o[1] ).getName(), "IFA" );
o = ( Object[] ) iter.next();
assertEquals( o[0], "JBoss" );
assertEquals( ( ( Organization ) o[1] ).getName(), "JBoss" );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
// test that the ordering of the results is truly based on the order in which they were defined
iter = s.getNamedQuery( "orgsAndOrgNames" ).list().iterator();
Object[] row = ( Object[] ) iter.next();
assertEquals( "expecting 2 values", 2, row.length );
assertEquals( "expecting non-scalar result first", Organization.class, row[0].getClass() );
assertEquals( "expecting scalar result second", String.class, row[1].getClass() );
assertEquals( ( ( Organization ) row[0] ).getName(), "IFA" );
assertEquals( row[1], "IFA" );
row = ( Object[] ) iter.next();
assertEquals( "expecting non-scalar result first", Organization.class, row[0].getClass() );
assertEquals( "expecting scalar result second", String.class, row[1].getClass() );
assertEquals( ( ( Organization ) row[0] ).getName(), "JBoss" );
assertEquals( row[1], "JBoss" );
assertFalse( iter.hasNext() );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
iter = s.getNamedQuery( "orgIdsAndOrgNames" ).list().iterator();
o = ( Object[] ) iter.next();
assertEquals( o[1], "IFA" );
assertEquals( o[0], idIfa );
o = ( Object[] ) iter.next();
assertEquals( o[1], "JBoss" );
assertEquals( o[0], idJBoss );
s.delete( ifa );
s.delete( jboss );
t.commit();
s.close();
}
@Test
@SuppressWarnings( {"deprecation", "UnusedDeclaration"})
public void testMappedAliasStrategy() {
Session s = openSession();
Transaction t = s.beginTransaction();
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
Object orgId = s.save(jboss);
Object orgId2 = s.save(ifa);
s.save(gavin);
s.save(emp);
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
Query namedQuery = s.getNamedQuery("AllEmploymentAsMapped");
List list = namedQuery.list();
assertEquals(1,list.size());
Employment emp2 = (Employment) list.get(0);
assertEquals(emp2.getEmploymentId(), emp.getEmploymentId() );
assertEquals(emp2.getStartDate().getDate(), emp.getStartDate().getDate() );
assertEquals(emp2.getEndDate(), emp.getEndDate() );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
Query sqlQuery = s.getNamedQuery("EmploymentAndPerson");
sqlQuery.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);
list = sqlQuery.list();
assertEquals(1,list.size() );
Object res = list.get(0);
assertClassAssignability( Map.class, res.getClass() );
Map m = (Map) res;
assertEquals(2,m.size());
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
sqlQuery = s.getNamedQuery( "organizationreturnproperty" );
sqlQuery.setResultTransformer( Transformers.ALIAS_TO_ENTITY_MAP );
list = sqlQuery.list();
assertEquals( 2,list.size() );
m = (Map) list.get(0);
assertEquals( 2, m.size() );
assertTrue( m.containsKey("org") );
assertTrue( m.containsKey("emp") );
assertClassAssignability( m.get("org").getClass(), Organization.class );
if ( jboss.getId() == ( (Organization) m.get("org") ).getId() ) {
assertClassAssignability( m.get("emp").getClass(), Employment.class );
}
Map m2 = (Map) list.get(1);
assertEquals( 2, m.size() );
assertTrue( m2.containsKey("org") );
assertTrue( m2.containsKey("emp") );
assertClassAssignability( m2.get("org").getClass(), Organization.class );
if ( jboss.getId() == ( (Organization) m2.get("org") ).getId() ) {
assertClassAssignability( m2.get("emp").getClass(), Employment.class );
}
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
namedQuery = s.getNamedQuery("EmploymentAndPerson");
list = namedQuery.list();
assertEquals(1,list.size() );
Object[] objs = (Object[]) list.get(0);
assertEquals(2, objs.length);
emp2 = (Employment) objs[0];
gavin = (Person) objs[1];
s.delete(emp2);
s.delete(jboss);
s.delete(gavin);
s.delete(ifa);
t.commit();
s.close();
}
@Test
@SuppressWarnings( {"unchecked"})
@FailureExpected( jiraKey = "unknown" )
public void testCompositeIdJoins() {
Session s = openSession();
Transaction t = s.beginTransaction();
Person person = new Person();
person.setName( "Noob" );
Product product = new Product();
product.setProductId( new Product.ProductId() );
product.getProductId().setOrgid( "x" );
product.getProductId().setProductnumber( "1234" );
product.setName( "Hibernate 3" );
Order order = new Order();
order.setOrderId( new Order.OrderId() );
order.getOrderId().setOrdernumber( "1" );
order.getOrderId().setOrgid( "y" );
product.getOrders().add( order );
order.setProduct( product );
order.setPerson( person );
s.save( product );
s.save( order);
s.save( person );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
Product p = (Product) s.createQuery( "from Product p join fetch p.orders" ).list().get(0);
assertTrue(Hibernate.isInitialized( p.getOrders()));
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
Object[] o = (Object[]) s.createNativeQuery( "select\r\n" +
" product.orgid as {product.id.orgid}," +
" product.productnumber as {product.id.productnumber}," +
" {prod_orders}.orgid as orgid3_1_,\r\n" +
" {prod_orders}.ordernumber as ordernum2_3_1_,\r\n" +
" product.name as {product.name}," +
" {prod_orders.element.*}" +
/*" orders.PROD_NO as PROD4_3_1_,\r\n" +
" orders.person as person3_1_,\r\n" +
" orders.PROD_ORGID as PROD3_0__,\r\n" +
" orders.PROD_NO as PROD4_0__,\r\n" +
" orders.orgid as orgid0__,\r\n" +
" orders.ordernumber as ordernum2_0__ \r\n" +*/
" from\r\n" +
" Product product \r\n" +
" inner join\r\n" +
" TBL_ORDER {prod_orders} \r\n" +
" on product.orgid={prod_orders}.PROD_ORGID \r\n" +
" and product.productnumber={prod_orders}.PROD_NO" )
.addEntity( "product", Product.class )
.addJoin( "prod_orders", "product.orders" )
.list().get(0);
p = (Product) o[0];
assertTrue(Hibernate.isInitialized( p.getOrders() ));
assertNotNull(p.getOrders().iterator().next());
t.commit();
s.close();
}
@Test
@SuppressWarnings( {"UnusedDeclaration", "deprecation", "UnusedAssignment"})
public void testAutoDetectAliasing() {
Session s = openSession();
Transaction t = s.beginTransaction();
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
Object orgId = s.save(jboss);
Object orgId2 = s.save(ifa);
s.save(gavin);
s.save(emp);
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
List list = s.createNativeQuery( getEmploymentSQL() )
.addEntity( Employment.class.getName() )
.list();
assertEquals( 1,list.size() );
Employment emp2 = (Employment) list.get(0);
assertEquals(emp2.getEmploymentId(), emp.getEmploymentId() );
assertEquals(emp2.getStartDate().getDate(), emp.getStartDate().getDate() );
assertEquals(emp2.getEndDate(), emp.getEndDate() );
s.clear();
list = s.createNativeQuery( getEmploymentSQL() )
.addEntity( Employment.class.getName() )
.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP)
.list();
assertEquals( 1,list.size() );
Map m = (Map) list.get(0);
assertTrue(m.containsKey("Employment"));
assertEquals(1,m.size());
list = s.createNativeQuery(getEmploymentSQL()).list();
assertEquals(1, list.size());
Object[] o = (Object[]) list.get(0);
assertEquals(8, o.length);
list = s.createNativeQuery( getEmploymentSQL() ).setResultTransformer( new UpperCasedAliasToEntityMapResultTransformer() ).list();
assertEquals(1, list.size());
m = (Map) list.get(0);
assertTrue(m.containsKey("EMPID"));
assertTrue(m.containsKey("AMOUNT"));
assertTrue(m.containsKey("ENDDATE"));
assertEquals(8, m.size());
list = s.createNativeQuery( getEmploymentSQLMixedScalarEntity() ).addScalar( "employerid" ).addEntity( Employment.class ).list();
assertEquals(1, list.size());
o = (Object[]) list.get(0);
assertEquals(2, o.length);
assertClassAssignability( Number.class, o[0].getClass() );
assertClassAssignability( Employment.class, o[1].getClass() );
Query queryWithCollection = s.getNamedQuery("organizationEmploymentsExplicitAliases");
queryWithCollection.setParameter("id", jboss.getId() );
list = queryWithCollection.list();
assertEquals(list.size(),1);
s.clear();
list = s.createNativeQuery( getOrganizationJoinEmploymentSQL() )
.addEntity( "org", Organization.class )
.addJoin( "emp", "org.employments" )
.list();
assertEquals( 2,list.size() );
s.clear();
list = s.createNativeQuery( getOrganizationFetchJoinEmploymentSQL() )
.addEntity( "org", Organization.class )
.addJoin( "emp", "org.employments" )
.list();
assertEquals( 2,list.size() );
s.clear();
// TODO : why twice?
s.getNamedQuery( "organizationreturnproperty" ).list();
list = s.getNamedQuery( "organizationreturnproperty" ).list();
assertEquals( 2,list.size() );
s.clear();
list = s.getNamedQuery( "organizationautodetect" ).list();
assertEquals( 2,list.size() );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
s.delete(emp2);
s.delete(jboss);
s.delete(gavin);
s.delete(ifa);
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
Dimension dim = new Dimension( 3, 30 );
s.save( dim );
list = s.createNativeQuery( "select d_len * d_width as surface, d_len * d_width * 10 as volume from Dimension" ).list();
s.delete( dim );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
SpaceShip enterprise = new SpaceShip();
enterprise.setModel( "USS" );
enterprise.setName( "Entreprise" );
enterprise.setSpeed( 50d );
Dimension d = new Dimension(45, 10);
enterprise.setDimensions( d );
s.save( enterprise );
Object[] result = (Object[]) s.getNamedQuery( "spaceship" ).uniqueResult();
assertEquals( "expecting 3 result values", 3, result.length );
enterprise = ( SpaceShip ) result[0];
assertTrue(50d == enterprise.getSpeed() );
assertTrue( 450d == extractDoubleValue( result[1] ) );
assertTrue( 4500d == extractDoubleValue( result[2] ) );
s.delete( enterprise );
t.commit();
s.close();
}
@Test
@SuppressWarnings( {"UnusedDeclaration"})
public void testExplicitReturnAPI() {
Session s = openSession();
s.beginTransaction();
Organization jboss = new Organization( "JBoss" );
Person me = new Person( "Steve" );
Employment emp = new Employment( me, jboss, "US" );
Object jbossId = s.save( jboss );
s.save( me );
s.save( emp );
s.getTransaction().commit();
s.close();
s = openSession();
s.beginTransaction();
String sql =
"SELECT org.ORGID as orgid," +
" org.NAME as name," +
" emp.EMPLOYER as employer," +
" emp.EMPID as empid," +
" emp.EMPLOYEE as employee," +
" emp.EMPLOYER as employer," +
" emp.STARTDATE as startDate," +
" emp.ENDDATE as endDate," +
" emp.REGIONCODE as regionCode," +
" emp.AMOUNT as AMOUNT," +
" emp.CURRENCY as CURRENCY" +
" FROM ORGANIZATION org" +
" LEFT OUTER JOIN EMPLOYMENT emp ON org.ORGID = emp.EMPLOYER";
// as a control, lets apply an existing rs mapping
NativeQuery sqlQuery = s.createNativeQuery( sql, "org-description" );
sqlQuery.list();
// next try a partial mapping def
sqlQuery.addRoot( "org", Organization.class );
sqlQuery.addFetch( "emp", "org", "employments" );
sqlQuery.list();
// now try full explicit mappings
sqlQuery.addRoot( "org", Organization.class )
.addProperty( "id", "orgid" )
.addProperty( "name" ).addColumnAlias( "name" );
sqlQuery.addFetch( "emp", "org", "employments" )
.addProperty( "key", "employer" )
.addProperty( "element", "empid" )
.addProperty( "element.employee", "employee" )
.addProperty( "element.employer", "employer" )
.addProperty( "element.startDate", "startDate" )
.addProperty( "element.endDate", "endDate" )
.addProperty( "element.regionCode", "regionCode" )
.addProperty( "element.employmentId", "empId" )
.addProperty( "element.salary" ).addColumnAlias( "AMOUNT" ).addColumnAlias( "CURRENCY" );
sqlQuery.list();
// lets try a totally different approach now and pull back scalars, first with explicit types
sqlQuery.addScalar( "orgid", StandardBasicTypes.LONG )
.addScalar( "name", StandardBasicTypes.STRING )
.addScalar( "empid", StandardBasicTypes.LONG )
.addScalar( "employee", StandardBasicTypes.LONG )
.addScalar( "startDate", StandardBasicTypes.TIMESTAMP )
.addScalar( "endDate", StandardBasicTypes.TIMESTAMP )
.addScalar( "regionCode", StandardBasicTypes.STRING )
.addScalar( "empId", StandardBasicTypes.LONG )
.addScalar( "AMOUNT", StandardBasicTypes.FLOAT )
.addScalar( "CURRENCY", StandardBasicTypes.STRING );
s.getTransaction().commit();
s.close();
s = openSession();
s.beginTransaction();
s.delete( emp );
s.delete( jboss );
s.delete( me );
s.getTransaction().commit();
s.close();
}
@Test
public void testMixAndMatchEntityScalar() {
Session s = openSession();
Transaction t = s.beginTransaction();
Speech speech = new Speech();
speech.setLength( new Double( 23d ) );
speech.setName( "Mine" );
s.persist( speech );
s.flush();
s.clear();
List l = s.createNativeQuery( "select name, id, flength, name as scalarName from Speech", "speech" ).list();
assertEquals( l.size(), 1 );
t.rollback();
s.close();
}
private double extractDoubleValue(Object value) {
if ( value instanceof BigInteger ) {
return ( ( BigInteger ) value ).doubleValue();
}
else if ( value instanceof BigDecimal ) {
return ( ( BigDecimal ) value ).doubleValue();
}
else {
return Double.valueOf( value.toString() ).doubleValue();
}
}
@Test
@SuppressWarnings( {"unchecked", "UnusedDeclaration"})
public void testAddJoinForManyToMany() {
Session s = openSession();
Transaction t = s.beginTransaction();
Person gavin = new Person( "Gavin" );
Person max = new Person( "Max" );
Person pete = new Person( "Pete" );
Group hibernate = new Group( "Hibernate" );
Group seam = new Group( "Seam" );
s.persist( gavin );
s.persist( max );
s.persist( pete );
s.persist( seam );
s.persist( hibernate );
hibernate.getPersons().add( gavin );
hibernate.getPersons().add( max );
seam.getPersons().add( gavin );
seam.getPersons().add( pete );
s.flush();
s.clear();
// todo : see http://opensource.atlassian.com/projects/hibernate/browse/HHH-3908
// String sqlStr = "SELECT {groupp.*} , {gp.*} " +
// "FROM GROUPP groupp, GROUP_PERSON gp, PERSON person WHERE groupp.ID = gp.GROUP_ID and person.PERID = gp.PERSON_ID";
//
// List l = s.createSQLQuery( sqlStr )
// .addEntity("groupp", Group.class)
// .addJoin("gp","groupp.persons")
// .list();
List l = s.getNamedQuery( "manyToManyFetch" ).list();
//assertEquals( 2, l.size() );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
seam.getPersons().remove( gavin );
seam.getPersons().remove( pete );
hibernate.getPersons().remove( gavin );
hibernate.getPersons().remove( max );
s.delete( seam );
s.delete( hibernate );
s.delete( gavin );
s.delete( max );
s.delete( pete );
t.commit();
s.close();
}
@SkipForDialect(value = AbstractHANADialect.class, comment = "On HANA, this returns an clob for the text column which doesn't get mapped to a String")
@Test
public void testTextTypeInSQLQuery() {
Session s = openSession();
Transaction t = s.beginTransaction();
String description = buildLongString( 15000, 'a' );
TextHolder holder = new TextHolder( description );
s.persist( holder );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
String descriptionRead = ( String ) s.createNativeQuery( getDescriptionsSQL() )
.uniqueResult();
assertEquals( description, descriptionRead );
s.delete( holder );
t.commit();
s.close();
}
@SkipForDialect(value = AbstractHANADialect.class, comment = "On HANA, this returns a blob for the image column which doesn't get mapped to a byte[]")
@Test
public void testImageTypeInSQLQuery() {
Session s = openSession();
Transaction t = s.beginTransaction();
byte[] photo = buildLongByteArray( 15000, true );
ImageHolder holder = new ImageHolder( photo );
s.persist( holder );
t.commit();
s.close();
s = openSession();
t = s.beginTransaction();
byte[] photoRead = ( byte[] ) s.createNativeQuery( getPhotosSQL() )
.uniqueResult();
assertTrue( Arrays.equals( photo, photoRead ) );
s.delete( holder );
t.commit();
s.close();
}
@Test
@RequiresDialect(MySQL5Dialect.class)
public void testEscapeColonInSQL() throws QueryException {
Session s = openSession();
Transaction t = s.beginTransaction();
NativeQuery query = s.createNativeQuery( "SELECT @row \\:= 1" );
List list = query.list();
assertTrue( list.get( 0 ).toString().equals( "1" ) );
t.commit();
s.close();
}
private String buildLongString(int size, char baseChar) {
StringBuilder buff = new StringBuilder();
for( int i = 0; i < size; i++ ) {
buff.append( baseChar );
}
return buff.toString();
}
private byte[] buildLongByteArray(int size, boolean on) {
byte[] data = new byte[size];
data[0] = mask( on );
for ( int i = 0; i < size; i++ ) {
data[i] = mask( on );
on = !on;
}
return data;
}
private byte mask(boolean on) {
return on ? ( byte ) 1 : ( byte ) 0;
}
@SuppressWarnings( {"unchecked"})
private static class UpperCasedAliasToEntityMapResultTransformer extends BasicTransformerAdapter implements Serializable {
public Object transformTuple(Object[] tuple, String[] aliases) {
Map result = new HashMap( tuple.length );
for ( int i = 0; i < tuple.length; i++ ) {
String alias = aliases[i];
if ( alias != null ) {
result.put( alias.toUpperCase(Locale.ROOT), tuple[i] );
}
}
return result;
}
}
}

View File

@ -0,0 +1,883 @@
/*
* 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.sql.hand.query;
import java.io.Serializable;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import org.hibernate.Hibernate;
import org.hibernate.QueryException;
import org.hibernate.Transaction;
import org.hibernate.cfg.Environment;
import org.hibernate.dialect.AbstractHANADialect;
import org.hibernate.dialect.H2Dialect;
import org.hibernate.dialect.MySQLDialect;
import org.hibernate.orm.test.sql.hand.Dimension;
import org.hibernate.orm.test.sql.hand.Employment;
import org.hibernate.orm.test.sql.hand.Group;
import org.hibernate.orm.test.sql.hand.ImageHolder;
import org.hibernate.orm.test.sql.hand.Order;
import org.hibernate.orm.test.sql.hand.Organization;
import org.hibernate.orm.test.sql.hand.Person;
import org.hibernate.orm.test.sql.hand.Product;
import org.hibernate.orm.test.sql.hand.SpaceShip;
import org.hibernate.orm.test.sql.hand.Speech;
import org.hibernate.orm.test.sql.hand.TextHolder;
import org.hibernate.query.NativeQuery;
import org.hibernate.query.Query;
import org.hibernate.transform.BasicTransformerAdapter;
import org.hibernate.transform.DistinctRootEntityResultTransformer;
import org.hibernate.transform.Transformers;
import org.hibernate.type.StandardBasicTypes;
import org.hibernate.testing.orm.junit.DomainModel;
import org.hibernate.testing.orm.junit.FailureExpected;
import org.hibernate.testing.orm.junit.RequiresDialect;
import org.hibernate.testing.orm.junit.ServiceRegistry;
import org.hibernate.testing.orm.junit.SessionFactory;
import org.hibernate.testing.orm.junit.SessionFactoryScope;
import org.hibernate.testing.orm.junit.Setting;
import org.hibernate.testing.orm.junit.SkipForDialect;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.Test;
import jakarta.persistence.PersistenceException;
import static org.hibernate.testing.orm.junit.ExtraAssertions.assertClassAssignability;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
/**
* Tests of various features of native SQL queries.
*
* @author Steve Ebersole
*/
@ServiceRegistry(
settings = {
@Setting( name = Environment.GENERATE_STATISTICS, value = "true" )
}
)
@DomainModel(
xmlMappings = {"org/hibernate/orm/test/sql/hand/query/NativeSQLQueries.hbm.xml"}
)
@SessionFactory
// todo (6.0): needs a composite user type mechanism e.g. by providing a custom ComponentTuplizer/Instantiator
@Disabled( value = "Missing support for composite user types" )
public class NativeSQLQueriesTest {
protected String getOrganizationFetchJoinEmploymentSQL() {
return "SELECT org.ORGID as {org.id}, " +
" org.NAME as {org.name}, " +
" emp.EMPLOYER as {emp.key}, " +
" emp.EMPID as {emp.element}, " +
" {emp.element.*} " +
"FROM ORGANIZATION org " +
" LEFT OUTER JOIN EMPLOYMENT emp ON org.ORGID = emp.EMPLOYER";
}
protected String getOrganizationJoinEmploymentSQL() {
return "SELECT org.ORGID as {org.id}, " +
" org.NAME as {org.name}, " +
" {emp.*} " +
"FROM ORGANIZATION org " +
" LEFT OUTER JOIN EMPLOYMENT emp ON org.ORGID = emp.EMPLOYER";
}
protected String getEmploymentSQL() {
return "SELECT * FROM EMPLOYMENT";
}
protected String getEmploymentSQLMixedScalarEntity() {
return "SELECT e.*, e.employer as employerid FROM EMPLOYMENT e" ;
}
protected String getOrgEmpRegionSQL() {
return "select {org.*}, {emp.*}, emp.REGIONCODE " +
"from ORGANIZATION org " +
" left outer join EMPLOYMENT emp on org.ORGID = emp.EMPLOYER";
}
protected String getOrgEmpPersonSQL() {
return "select {org.*}, {emp.*}, {pers.*} " +
"from ORGANIZATION org " +
" join EMPLOYMENT emp on org.ORGID = emp.EMPLOYER " +
" join PERSON pers on pers.PERID = emp.EMPLOYEE ";
}
protected String getDescriptionsSQL() {
return "select DESCRIPTION from TEXT_HOLDER";
}
protected String getPhotosSQL() {
return "select PHOTO from IMAGE_HOLDER";
}
@Test
@SkipForDialect( dialectClass = H2Dialect.class )
public void testFailOnNoAddEntityOrScalar(SessionFactoryScope scope) {
// Note: this passes, but for the wrong reason.
// there is actually an exception thrown, but it is the database
// throwing a sql exception because the SQL gets passed
// "un-processed"...
//
// Oddly, H2 accepts this query.
scope.inSession(
session -> {
session.beginTransaction();
try {
String sql = "select {org.*} " +
"from organization org";
session.createNativeQuery( sql ).list();
fail( "Should throw an exception since no addEntity nor addScalar has been performed." );
}
catch( PersistenceException pe) {
// expected behavior
}
finally {
session.getTransaction().rollback();
session.close();
}
}
);
}
@Test
public void testManualSynchronization(SessionFactoryScope scope) {
scope.inTransaction(
session -> {
session.getSessionFactory().getStatistics().clear();
// create an Organization...
Organization jboss = new Organization( "JBoss" );
session.persist( jboss );
// now query on Employment, this should not cause an auto-flush
session.createNativeQuery( getEmploymentSQL() ).addSynchronizedQuerySpace( "ABC" ).list();
assertEquals( 0, session.getSessionFactory().getStatistics().getEntityInsertCount() );
// now try to query on Employment but this time add Organization as a synchronized query space...
session.createNativeQuery( getEmploymentSQL() ).addSynchronizedEntityClass( Organization.class ).list();
assertEquals( 1, session.getSessionFactory().getStatistics().getEntityInsertCount() );
// clean up
session.delete( jboss );
}
);
}
@Test
public void testSQLQueryInterface(SessionFactoryScope scope) {
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
scope.inTransaction(
session -> {
session.persist(ifa);
session.persist(jboss);
session.persist(gavin);
session.persist(emp);
List l = session.createNativeQuery( getOrgEmpRegionSQL() )
.addEntity("org", Organization.class)
.addJoin("emp", "org.employments")
.addScalar("regionCode", StandardBasicTypes.STRING )
.list();
assertEquals( 2, l.size() );
l = session.createNativeQuery( getOrgEmpPersonSQL() )
.addEntity("org", Organization.class)
.addJoin("emp", "org.employments")
.addJoin("pers", "emp.employee")
.list();
assertEquals( l.size(), 1 );
}
);
scope.inTransaction(
session -> {
List l = session.createNativeQuery( "select {org.*}, {emp.*} " +
"from ORGANIZATION org " +
" left outer join EMPLOYMENT emp on org.ORGID = emp.EMPLOYER, ORGANIZATION org2" )
.addEntity("org", Organization.class)
.addJoin("emp", "org.employments")
.setResultTransformer( DistinctRootEntityResultTransformer.INSTANCE )
.list();
assertEquals( l.size(), 2 );
}
);
scope.inTransaction(
session -> {
session.delete(emp);
session.delete(gavin);
session.delete(ifa);
session.delete(jboss);
}
);
}
@Test
public void testResultSetMappingDefinition(SessionFactoryScope scope) {
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
scope.inTransaction(
session -> {
session.persist(ifa);
session.persist(jboss);
session.persist(gavin);
session.persist(emp);
List l = session.createNativeQuery( getOrgEmpRegionSQL(), "org-emp-regionCode" ).list();
assertEquals( l.size(), 2 );
l = session.createNativeQuery( getOrgEmpPersonSQL(), "org-emp-person" ).list();
assertEquals( l.size(), 1 );
session.delete(emp);
session.delete(gavin);
session.delete(ifa);
session.delete(jboss);
}
);
}
@Test
public void testScalarValues(SessionFactoryScope scope) throws Exception {
Organization ifa = new Organization( "IFA" );
Organization jboss = new Organization( "JBoss" );
Object idIfa = scope.fromTransaction( session -> session.save( ifa ) );
Object idJBoss = scope.fromTransaction( session -> session.save( jboss ) );
scope.inTransaction(
session -> {
List result = session.getNamedQuery( "orgNamesOnly" ).list();
assertTrue( result.contains( "IFA" ) );
assertTrue( result.contains( "JBoss" ) );
result = session.getNamedQuery( "orgNamesOnly" ).setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP).list();
Map m = (Map) result.get(0);
assertEquals( 2, result.size() );
assertEquals( 1, m.size() );
assertTrue( m.containsKey("NAME") );
}
);
scope.inTransaction(
session -> {
Iterator iter = session.getNamedQuery( "orgNamesAndOrgs" ).list().iterator();
Object[] o = ( Object[] ) iter.next();
assertEquals( 2, o.length, "expecting 2 values" );
assertEquals( o[0], "IFA" );
assertEquals( ( ( Organization ) o[1] ).getName(), "IFA" );
o = ( Object[] ) iter.next();
assertEquals( o[0], "JBoss" );
assertEquals( ( ( Organization ) o[1] ).getName(), "JBoss" );
}
);
scope.inTransaction(
session -> {
// test that the ordering of the results is truly based on the order in which they were defined
Iterator iter = session.getNamedQuery( "orgsAndOrgNames" ).list().iterator();
Object[] row = ( Object[] ) iter.next();
assertEquals( 2, row.length, "expecting 2 values" );
assertEquals( Organization.class, row[0].getClass(), "expecting non-scalar result first" );
assertEquals( String.class, row[1].getClass(), "expecting scalar result second" );
assertEquals( ( ( Organization ) row[0] ).getName(), "IFA" );
assertEquals( row[1], "IFA" );
row = ( Object[] ) iter.next();
assertEquals( Organization.class, row[0].getClass(), "expecting non-scalar result first" );
assertEquals( String.class, row[1].getClass(), "expecting scalar result second" );
assertEquals( ( ( Organization ) row[0] ).getName(), "JBoss" );
assertEquals( row[1], "JBoss" );
assertFalse( iter.hasNext() );
}
);
scope.inTransaction(
session -> {
Iterator iter = session.getNamedQuery( "orgIdsAndOrgNames" ).list().iterator();
Object[] o = ( Object[] ) iter.next();
assertEquals( o[1], "IFA" );
assertEquals( o[0], idIfa );
o = ( Object[] ) iter.next();
assertEquals( o[1], "JBoss" );
assertEquals( o[0], idJBoss );
session.delete( ifa );
session.delete( jboss );
}
);
}
@Test
@SuppressWarnings( {"deprecation", "UnusedDeclaration"})
public void testMappedAliasStrategy(SessionFactoryScope scope) {
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
scope.inTransaction(
session -> {
Object orgId = session.save(jboss);
Object orgId2 = session.save(ifa);
session.save(gavin);
session.save(emp);
}
);
scope.inTransaction(
session -> {
Query namedQuery = session.getNamedQuery("AllEmploymentAsMapped");
List list = namedQuery.list();
assertEquals(1,list.size());
Employment emp2 = (Employment) list.get(0);
assertEquals(emp2.getEmploymentId(), emp.getEmploymentId() );
assertEquals(emp2.getStartDate().getDate(), emp.getStartDate().getDate() );
assertEquals(emp2.getEndDate(), emp.getEndDate() );
}
);
scope.inTransaction(
session -> {
Query sqlQuery = session.getNamedQuery("EmploymentAndPerson");
sqlQuery.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP);
List list = sqlQuery.list();
assertEquals(1,list.size() );
Object res = list.get(0);
assertClassAssignability( Map.class, res.getClass() );
Map m = (Map) res;
assertEquals(2,m.size());
}
);
scope.inTransaction(
session -> {
Query sqlQuery = session.getNamedQuery( "organizationreturnproperty" );
sqlQuery.setResultTransformer( Transformers.ALIAS_TO_ENTITY_MAP );
List list = sqlQuery.list();
assertEquals( 2,list.size() );
Map m = (Map) list.get(0);
assertEquals( 2, m.size() );
assertTrue( m.containsKey("org") );
assertTrue( m.containsKey("emp") );
assertClassAssignability( m.get("org").getClass(), Organization.class );
if ( jboss.getId() == ( (Organization) m.get("org") ).getId() ) {
assertClassAssignability( m.get("emp").getClass(), Employment.class );
}
Map m2 = (Map) list.get(1);
assertEquals( 2, m.size() );
assertTrue( m2.containsKey("org") );
assertTrue( m2.containsKey("emp") );
assertClassAssignability( m2.get("org").getClass(), Organization.class );
if ( jboss.getId() == ( (Organization) m2.get("org") ).getId() ) {
assertClassAssignability( m2.get("emp").getClass(), Employment.class );
}
}
);
scope.inTransaction(
session -> {
Query namedQuery = session.getNamedQuery("EmploymentAndPerson");
List list = namedQuery.list();
assertEquals(1,list.size() );
Object[] objs = (Object[]) list.get(0);
assertEquals(2, objs.length);
Employment emp2 = (Employment) objs[0];
Person _gavin = (Person) objs[1];
session.delete(emp2);
session.delete(jboss);
session.delete(_gavin);
session.delete(ifa);
}
);
}
@Test
@SuppressWarnings( {"unchecked"})
@FailureExpected( jiraKey = "unknown" )
public void testCompositeIdJoins(SessionFactoryScope scope) {
scope.inTransaction(
session -> {
Person person = new Person();
person.setName( "Noob" );
Product product = new Product();
product.setProductId( new Product.ProductId() );
product.getProductId().setOrgid( "x" );
product.getProductId().setProductnumber( "1234" );
product.setName( "Hibernate 3" );
Order order = new Order();
order.setOrderId( new Order.OrderId() );
order.getOrderId().setOrdernumber( "1" );
order.getOrderId().setOrgid( "y" );
product.getOrders().add( order );
order.setProduct( product );
order.setPerson( person );
session.save( product );
session.save( order);
session.save( person );
}
);
scope.inTransaction(
session -> {
Product p = (Product) session.createQuery( "from Product p join fetch p.orders" ).list().get(0);
assertTrue(Hibernate.isInitialized( p.getOrders()));
}
);
scope.inTransaction(
session -> {
Object[] o = (Object[]) session.createNativeQuery( "select\r\n" +
" product.orgid as {product.id.orgid}," +
" product.productnumber as {product.id.productnumber}," +
" {prod_orders}.orgid as orgid3_1_,\r\n" +
" {prod_orders}.ordernumber as ordernum2_3_1_,\r\n" +
" product.name as {product.name}," +
" {prod_orders.element.*}" +
/*" orders.PROD_NO as PROD4_3_1_,\r\n" +
" orders.person as person3_1_,\r\n" +
" orders.PROD_ORGID as PROD3_0__,\r\n" +
" orders.PROD_NO as PROD4_0__,\r\n" +
" orders.orgid as orgid0__,\r\n" +
" orders.ordernumber as ordernum2_0__ \r\n" +*/
" from\r\n" +
" Product product \r\n" +
" inner join\r\n" +
" TBL_ORDER {prod_orders} \r\n" +
" on product.orgid={prod_orders}.PROD_ORGID \r\n" +
" and product.productnumber={prod_orders}.PROD_NO" )
.addEntity( "product", Product.class )
.addJoin( "prod_orders", "product.orders" )
.list().get(0);
Product p = (Product) o[0];
assertTrue(Hibernate.isInitialized( p.getOrders() ));
assertNotNull(p.getOrders().iterator().next());
}
);
}
@Test
@SuppressWarnings( {"UnusedDeclaration", "deprecation", "UnusedAssignment"})
public void testAutoDetectAliasing(SessionFactoryScope scope) {
Organization ifa = new Organization("IFA");
Organization jboss = new Organization("JBoss");
Person gavin = new Person("Gavin");
Employment emp = new Employment(gavin, jboss, "AU");
scope.inTransaction(
session -> {
Object orgId = session.save(jboss);
Object orgId2 = session.save(ifa);
session.save(gavin);
session.save(emp);
}
);
Employment emp2 = scope.fromTransaction(
session -> {
List list = session.createNativeQuery( getEmploymentSQL() )
.addEntity( Employment.class.getName() )
.list();
assertEquals( 1, list.size() );
Employment _emp2 = (Employment) list.get( 0 );
assertEquals( _emp2.getEmploymentId(), emp.getEmploymentId() );
assertEquals( _emp2.getStartDate().getDate(), emp.getStartDate().getDate() );
assertEquals( _emp2.getEndDate(), emp.getEndDate() );
return _emp2;
}
);
scope.inTransaction(
session -> {
List list = session.createNativeQuery( getEmploymentSQL() )
.addEntity( Employment.class.getName() )
.setResultTransformer(Transformers.ALIAS_TO_ENTITY_MAP)
.list();
assertEquals( 1,list.size() );
Map m = (Map) list.get(0);
assertTrue(m.containsKey("Employment"));
assertEquals(1,m.size());
list = session.createNativeQuery(getEmploymentSQL()).list();
assertEquals(1, list.size());
Object[] o = (Object[]) list.get(0);
assertEquals(8, o.length);
list = session.createNativeQuery( getEmploymentSQL() ).setResultTransformer( new UpperCasedAliasToEntityMapResultTransformer() ).list();
assertEquals(1, list.size());
m = (Map) list.get(0);
assertTrue(m.containsKey("EMPID"));
assertTrue(m.containsKey("AMOUNT"));
assertTrue(m.containsKey("ENDDATE"));
assertEquals(8, m.size());
list = session.createNativeQuery( getEmploymentSQLMixedScalarEntity() ).addScalar( "employerid" ).addEntity( Employment.class ).list();
assertEquals(1, list.size());
o = (Object[]) list.get(0);
assertEquals(2, o.length);
assertClassAssignability( Number.class, o[0].getClass() );
assertClassAssignability( Employment.class, o[1].getClass() );
Query queryWithCollection = session.getNamedQuery("organizationEmploymentsExplicitAliases");
queryWithCollection.setParameter("id", jboss.getId() );
list = queryWithCollection.list();
assertEquals(list.size(),1);
session.clear();
list = session.createNativeQuery( getOrganizationJoinEmploymentSQL() )
.addEntity( "org", Organization.class )
.addJoin( "emp", "org.employments" )
.list();
assertEquals( 2,list.size() );
session.clear();
list = session.createNativeQuery( getOrganizationFetchJoinEmploymentSQL() )
.addEntity( "org", Organization.class )
.addJoin( "emp", "org.employments" )
.list();
assertEquals( 2,list.size() );
session.clear();
// TODO : why twice?
session.getNamedQuery( "organizationreturnproperty" ).list();
list = session.getNamedQuery( "organizationreturnproperty" ).list();
assertEquals( 2,list.size() );
session.clear();
list = session.getNamedQuery( "organizationautodetect" ).list();
assertEquals( 2,list.size() );
}
);
scope.inTransaction(
session -> {
session.delete(emp2);
session.delete(jboss);
session.delete(gavin);
session.delete(ifa);
}
);
scope.inTransaction(
session -> {
Dimension dim = new Dimension( 3, 30 );
session.save( dim );
List list = session.createNativeQuery( "select d_len * d_width as surface, d_len * d_width * 10 as volume from Dimension" ).list();
session.delete( dim );
}
);
scope.inTransaction(
session -> {
SpaceShip enterprise = new SpaceShip();
enterprise.setModel( "USS" );
enterprise.setName( "Entreprise" );
enterprise.setSpeed( 50d );
Dimension d = new Dimension(45, 10);
enterprise.setDimensions( d );
session.save( enterprise );
Object[] result = (Object[]) session.getNamedQuery( "spaceship" ).uniqueResult();
assertEquals( 3, result.length, "expecting 3 result values" );
enterprise = ( SpaceShip ) result[0];
assertTrue(50d == enterprise.getSpeed() );
assertTrue( 450d == extractDoubleValue( result[1] ) );
assertTrue( 4500d == extractDoubleValue( result[2] ) );
session.delete( enterprise );
}
);
}
@Test
@SuppressWarnings( {"UnusedDeclaration"})
public void testExplicitReturnAPI(SessionFactoryScope scope) {
Organization jboss = new Organization( "JBoss" );
Person me = new Person( "Steve" );
Employment emp = new Employment( me, jboss, "US" );
scope.inTransaction(
session -> {
Object jbossId = session.save( jboss );
session.save( me );
session.save( emp );
}
);
scope.inTransaction(
session -> {
String sql =
"SELECT org.ORGID as orgid," +
" org.NAME as name," +
" emp.EMPLOYER as employer," +
" emp.EMPID as empid," +
" emp.EMPLOYEE as employee," +
" emp.EMPLOYER as employer," +
" emp.STARTDATE as startDate," +
" emp.ENDDATE as endDate," +
" emp.REGIONCODE as regionCode," +
" emp.AMOUNT as AMOUNT," +
" emp.CURRENCY as CURRENCY" +
" FROM ORGANIZATION org" +
" LEFT OUTER JOIN EMPLOYMENT emp ON org.ORGID = emp.EMPLOYER";
// as a control, lets apply an existing rs mapping
NativeQuery sqlQuery = session.createNativeQuery( sql, "org-description" );
sqlQuery.list();
// next try a partial mapping def
sqlQuery.addRoot( "org", Organization.class );
sqlQuery.addFetch( "emp", "org", "employments" );
sqlQuery.list();
// now try full explicit mappings
sqlQuery.addRoot( "org", Organization.class )
.addProperty( "id", "orgid" )
.addProperty( "name" ).addColumnAlias( "name" );
sqlQuery.addFetch( "emp", "org", "employments" )
.addProperty( "key", "employer" )
.addProperty( "element", "empid" )
.addProperty( "element.employee", "employee" )
.addProperty( "element.employer", "employer" )
.addProperty( "element.startDate", "startDate" )
.addProperty( "element.endDate", "endDate" )
.addProperty( "element.regionCode", "regionCode" )
.addProperty( "element.employmentId", "empId" )
.addProperty( "element.salary" ).addColumnAlias( "AMOUNT" ).addColumnAlias( "CURRENCY" );
sqlQuery.list();
// lets try a totally different approach now and pull back scalars, first with explicit types
sqlQuery.addScalar( "orgid", StandardBasicTypes.LONG )
.addScalar( "name", StandardBasicTypes.STRING )
.addScalar( "empid", StandardBasicTypes.LONG )
.addScalar( "employee", StandardBasicTypes.LONG )
.addScalar( "startDate", StandardBasicTypes.TIMESTAMP )
.addScalar( "endDate", StandardBasicTypes.TIMESTAMP )
.addScalar( "regionCode", StandardBasicTypes.STRING )
.addScalar( "empId", StandardBasicTypes.LONG )
.addScalar( "AMOUNT", StandardBasicTypes.FLOAT )
.addScalar( "CURRENCY", StandardBasicTypes.STRING );
}
);
scope.inTransaction(
session -> {
session.delete( emp );
session.delete( jboss );
session.delete( me );
}
);
}
@Test
public void testMixAndMatchEntityScalar(SessionFactoryScope scope) {
scope.inSession(
session -> {
Transaction t = session.beginTransaction();
Speech speech = new Speech();
speech.setLength( new Double( 23d ) );
speech.setName( "Mine" );
session.persist( speech );
session.flush();
session.clear();
List l = session.createNativeQuery( "select name, id, flength, name as scalarName from Speech", "speech" ).list();
assertEquals( l.size(), 1 );
t.rollback();
}
);
}
private double extractDoubleValue(Object value) {
if ( value instanceof BigInteger ) {
return ( ( BigInteger ) value ).doubleValue();
}
else if ( value instanceof BigDecimal ) {
return ( ( BigDecimal ) value ).doubleValue();
}
else {
return Double.valueOf( value.toString() ).doubleValue();
}
}
@Test
@SuppressWarnings( {"unchecked", "UnusedDeclaration"})
public void testAddJoinForManyToMany(SessionFactoryScope scope) {
Person gavin = new Person( "Gavin" );
Person max = new Person( "Max" );
Person pete = new Person( "Pete" );
Group hibernate = new Group( "Hibernate" );
Group seam = new Group( "Seam" );
scope.inTransaction(
session -> {
session.persist( gavin );
session.persist( max );
session.persist( pete );
session.persist( seam );
session.persist( hibernate );
hibernate.getPersons().add( gavin );
hibernate.getPersons().add( max );
seam.getPersons().add( gavin );
seam.getPersons().add( pete );
session.flush();
session.clear();
// todo : see http://opensource.atlassian.com/projects/hibernate/browse/HHH-3908
// String sqlStr = "SELECT {groupp.*} , {gp.*} " +
// "FROM GROUPP groupp, GROUP_PERSON gp, PERSON person WHERE groupp.ID = gp.GROUP_ID and person.PERID = gp.PERSON_ID";
//
// List l = session.createSQLQuery( sqlStr )
// .addEntity("groupp", Group.class)
// .addJoin("gp","groupp.persons")
// .list();
List l = session.getNamedQuery( "manyToManyFetch" ).list();
//assertEquals( 2, l.size() );
}
);
scope.inTransaction(
session -> {
seam.getPersons().remove( gavin );
seam.getPersons().remove( pete );
hibernate.getPersons().remove( gavin );
hibernate.getPersons().remove( max );
session.delete( seam );
session.delete( hibernate );
session.delete( gavin );
session.delete( max );
session.delete( pete );
}
);
}
@SkipForDialect(dialectClass = AbstractHANADialect.class, reason = "On HANA, this returns an clob for the text column which doesn't get mapped to a String")
@Test
public void testTextTypeInSQLQuery(SessionFactoryScope scope) {
String description = buildLongString( 15000, 'a' );
TextHolder holder = new TextHolder( description );
scope.inTransaction(
session -> session.persist( holder )
);
scope.inTransaction(
session -> {
String descriptionRead = ( String ) session.createNativeQuery( getDescriptionsSQL() )
.uniqueResult();
assertEquals( description, descriptionRead );
session.delete( holder );
}
);
}
@SkipForDialect(dialectClass = AbstractHANADialect.class, reason = "On HANA, this returns a blob for the image column which doesn't get mapped to a byte[]")
@Test
public void testImageTypeInSQLQuery(SessionFactoryScope scope) {
byte[] photo = buildLongByteArray( 15000, true );
ImageHolder holder = new ImageHolder( photo );
scope.inTransaction(
session -> session.persist( holder )
);
scope.inTransaction(
session -> {
byte[] photoRead = ( byte[] ) session.createNativeQuery( getPhotosSQL() )
.uniqueResult();
assertTrue( Arrays.equals( photo, photoRead ) );
session.delete( holder );
}
);
}
@Test
@RequiresDialect(value = MySQLDialect.class, version = 500)
public void testEscapeColonInSQL(SessionFactoryScope scope) throws QueryException {
scope.inTransaction(
session -> {
NativeQuery query = session.createNativeQuery( "SELECT @row \\:= 1" );
List list = query.list();
assertTrue( list.get( 0 ).toString().equals( "1" ) );
}
);
}
private String buildLongString(int size, char baseChar) {
StringBuilder buff = new StringBuilder();
for( int i = 0; i < size; i++ ) {
buff.append( baseChar );
}
return buff.toString();
}
private byte[] buildLongByteArray(int size, boolean on) {
byte[] data = new byte[size];
data[0] = mask( on );
for ( int i = 0; i < size; i++ ) {
data[i] = mask( on );
on = !on;
}
return data;
}
private byte mask(boolean on) {
return on ? ( byte ) 1 : ( byte ) 0;
}
@SuppressWarnings( {"unchecked"})
private static class UpperCasedAliasToEntityMapResultTransformer extends BasicTransformerAdapter implements Serializable {
public Object transformTuple(Object[] tuple, String[] aliases) {
Map result = new HashMap( tuple.length );
for ( int i = 0; i < tuple.length; i++ ) {
String alias = aliases[i];
if ( alias != null ) {
result.put( alias.toUpperCase(Locale.ROOT), tuple[i] );
}
}
return result;
}
}
}