HHH-6394 fixing the check for resultClass in named native query
This commit is contained in:
parent
bcdefd10a6
commit
94aea008d3
|
@ -47,6 +47,7 @@ public class NamedSQLQueryDefinition extends NamedQueryDefinition {
|
||||||
* code when a the result-set mapping information is explicitly
|
* code when a the result-set mapping information is explicitly
|
||||||
* provided in the query definition (i.e., no resultset-mapping used)
|
* provided in the query definition (i.e., no resultset-mapping used)
|
||||||
*
|
*
|
||||||
|
* @param name The name of named query
|
||||||
* @param query The sql query string
|
* @param query The sql query string
|
||||||
* @param queryReturns The in-lined query return definitions
|
* @param queryReturns The in-lined query return definitions
|
||||||
* @param querySpaces Any specified query spaces (used for auto-flushing)
|
* @param querySpaces Any specified query spaces (used for auto-flushing)
|
||||||
|
@ -98,6 +99,7 @@ public class NamedSQLQueryDefinition extends NamedQueryDefinition {
|
||||||
* This form used to construct a NamedSQLQueryDefinition from the binder
|
* This form used to construct a NamedSQLQueryDefinition from the binder
|
||||||
* code when a resultset-mapping reference is used.
|
* code when a resultset-mapping reference is used.
|
||||||
*
|
*
|
||||||
|
* @param name The name of named query
|
||||||
* @param query The sql query string
|
* @param query The sql query string
|
||||||
* @param resultSetRef The resultset-mapping name
|
* @param resultSetRef The resultset-mapping name
|
||||||
* @param querySpaces Any specified query spaces (used for auto-flushing)
|
* @param querySpaces Any specified query spaces (used for auto-flushing)
|
||||||
|
|
|
@ -23,13 +23,14 @@
|
||||||
*/
|
*/
|
||||||
package org.hibernate.metamodel.source.annotations.global;
|
package org.hibernate.metamodel.source.annotations.global;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
import javax.persistence.NamedNativeQueries;
|
import javax.persistence.NamedNativeQueries;
|
||||||
import javax.persistence.NamedNativeQuery;
|
import javax.persistence.NamedNativeQuery;
|
||||||
import javax.persistence.NamedQueries;
|
import javax.persistence.NamedQueries;
|
||||||
import javax.persistence.NamedQuery;
|
import javax.persistence.NamedQuery;
|
||||||
import java.util.HashMap;
|
|
||||||
|
|
||||||
import org.jboss.jandex.AnnotationInstance;
|
import org.jboss.jandex.AnnotationInstance;
|
||||||
|
import org.jboss.jandex.AnnotationValue;
|
||||||
import org.jboss.jandex.Index;
|
import org.jboss.jandex.Index;
|
||||||
import org.jboss.logging.Logger;
|
import org.jboss.logging.Logger;
|
||||||
|
|
||||||
|
@ -56,11 +57,14 @@ public class QueryBinder {
|
||||||
QueryBinder.class.getName()
|
QueryBinder.class.getName()
|
||||||
);
|
);
|
||||||
|
|
||||||
|
private QueryBinder() {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Binds all {@link NamedQuery}, {@link NamedQueries}, {@link NamedNativeQuery}, {{@link NamedNativeQueries},
|
* Binds all {@link NamedQuery}, {@link NamedQueries}, {@link NamedNativeQuery}, {@link NamedNativeQueries},
|
||||||
* {@link org.hibernate.annotations.NamedQuery} , {@link org.hibernate.annotations.NamedQueries},
|
* {@link org.hibernate.annotations.NamedQuery}, {@link org.hibernate.annotations.NamedQueries},
|
||||||
* {@link org.hibernate.annotations.NamedNativeQuery}, and {@link org.hibernate.annotations.NamedNativeQueries} annotations to
|
* {@link org.hibernate.annotations.NamedNativeQuery}, and {@link org.hibernate.annotations.NamedNativeQueries}
|
||||||
* the supplied metadata.
|
* annotations to the supplied metadata.
|
||||||
*
|
*
|
||||||
* @param metadata the global metadata
|
* @param metadata the global metadata
|
||||||
* @param jandex the jandex index
|
* @param jandex the jandex index
|
||||||
|
@ -100,29 +104,42 @@ public class QueryBinder {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Binds {@link javax.persistence.NamedQuery} as well as {@link org.hibernate.annotations.NamedQuery}.
|
||||||
|
*
|
||||||
|
* @param metadata the current metadata
|
||||||
|
* @param annotation the named query annotation
|
||||||
|
*/
|
||||||
private static void bindNamedQuery(MetadataImplementor metadata, AnnotationInstance annotation) {
|
private static void bindNamedQuery(MetadataImplementor metadata, AnnotationInstance annotation) {
|
||||||
String name = JandexHelper.getValueAsString( annotation, "name" );
|
String name = JandexHelper.getValueAsString( annotation, "name" );
|
||||||
if ( StringHelper.isEmpty( name ) ) {
|
if ( StringHelper.isEmpty( name ) ) {
|
||||||
throw new AnnotationException( "A named query must have a name when used in class or package level" );
|
throw new AnnotationException( "A named query must have a name when used in class or package level" );
|
||||||
}
|
}
|
||||||
|
|
||||||
String query = JandexHelper.getValueAsString( annotation, "query" );
|
String query = JandexHelper.getValueAsString( annotation, "query" );
|
||||||
|
|
||||||
AnnotationInstance[] hints = JandexHelper.getValueAsArray( annotation, "hints" );
|
AnnotationInstance[] hints = JandexHelper.getValueAsArray( annotation, "hints" );
|
||||||
|
|
||||||
String cacheRegion = getString( hints, QueryHints.CACHE_REGION );
|
String cacheRegion = getString( hints, QueryHints.CACHE_REGION );
|
||||||
if ( StringHelper.isEmpty( cacheRegion ) ) {
|
if ( StringHelper.isEmpty( cacheRegion ) ) {
|
||||||
cacheRegion = null;
|
cacheRegion = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Integer timeout = getTimeout( hints, query );
|
Integer timeout = getTimeout( hints, query );
|
||||||
if ( timeout != null && timeout < 0 ) {
|
if ( timeout != null && timeout < 0 ) {
|
||||||
timeout = null;
|
timeout = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Integer fetchSize = getInteger( hints, QueryHints.FETCH_SIZE, name );
|
Integer fetchSize = getInteger( hints, QueryHints.FETCH_SIZE, name );
|
||||||
if ( fetchSize != null && fetchSize < 0 ) {
|
if ( fetchSize != null && fetchSize < 0 ) {
|
||||||
fetchSize = null;
|
fetchSize = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
String comment = getString( hints, QueryHints.COMMENT );
|
String comment = getString( hints, QueryHints.COMMENT );
|
||||||
if ( StringHelper.isEmpty( comment ) ) {
|
if ( StringHelper.isEmpty( comment ) ) {
|
||||||
comment = null;
|
comment = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
metadata.addNamedQuery(
|
metadata.addNamedQuery(
|
||||||
new NamedQueryDefinition(
|
new NamedQueryDefinition(
|
||||||
name,
|
name,
|
||||||
|
@ -140,29 +157,39 @@ public class QueryBinder {
|
||||||
if ( StringHelper.isEmpty( name ) ) {
|
if ( StringHelper.isEmpty( name ) ) {
|
||||||
throw new AnnotationException( "A named native query must have a name when used in class or package level" );
|
throw new AnnotationException( "A named native query must have a name when used in class or package level" );
|
||||||
}
|
}
|
||||||
|
|
||||||
String query = JandexHelper.getValueAsString( annotation, "query" );
|
String query = JandexHelper.getValueAsString( annotation, "query" );
|
||||||
|
|
||||||
String resultSetMapping = JandexHelper.getValueAsString( annotation, "resultSetMapping" );
|
String resultSetMapping = JandexHelper.getValueAsString( annotation, "resultSetMapping" );
|
||||||
|
|
||||||
AnnotationInstance[] hints = JandexHelper.getValueAsArray( annotation, "hints" );
|
AnnotationInstance[] hints = JandexHelper.getValueAsArray( annotation, "hints" );
|
||||||
|
|
||||||
boolean cacheable = getBoolean( hints, "org.hibernate.cacheable", name );
|
boolean cacheable = getBoolean( hints, "org.hibernate.cacheable", name );
|
||||||
String cacheRegion = getString( hints, QueryHints.CACHE_REGION );
|
String cacheRegion = getString( hints, QueryHints.CACHE_REGION );
|
||||||
if ( StringHelper.isEmpty( cacheRegion ) ) {
|
if ( StringHelper.isEmpty( cacheRegion ) ) {
|
||||||
cacheRegion = null;
|
cacheRegion = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Integer timeout = getTimeout( hints, query );
|
Integer timeout = getTimeout( hints, query );
|
||||||
if ( timeout != null && timeout < 0 ) {
|
if ( timeout != null && timeout < 0 ) {
|
||||||
timeout = null;
|
timeout = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Integer fetchSize = getInteger( hints, QueryHints.FETCH_SIZE, name );
|
Integer fetchSize = getInteger( hints, QueryHints.FETCH_SIZE, name );
|
||||||
if ( fetchSize != null && fetchSize < 0 ) {
|
if ( fetchSize != null && fetchSize < 0 ) {
|
||||||
fetchSize = null;
|
fetchSize = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
FlushMode flushMode = getFlushMode( hints, QueryHints.FLUSH_MODE, name );
|
FlushMode flushMode = getFlushMode( hints, QueryHints.FLUSH_MODE, name );
|
||||||
CacheMode cacheMode = getCacheMode( hints, QueryHints.CACHE_MODE, name );
|
CacheMode cacheMode = getCacheMode( hints, QueryHints.CACHE_MODE, name );
|
||||||
|
|
||||||
boolean readOnly = getBoolean( hints, QueryHints.READ_ONLY, name );
|
boolean readOnly = getBoolean( hints, QueryHints.READ_ONLY, name );
|
||||||
|
|
||||||
String comment = getString( hints, QueryHints.COMMENT );
|
String comment = getString( hints, QueryHints.COMMENT );
|
||||||
if ( StringHelper.isEmpty( comment ) ) {
|
if ( StringHelper.isEmpty( comment ) ) {
|
||||||
comment = null;
|
comment = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean callable = getBoolean( hints, QueryHints.CALLABLE, name );
|
boolean callable = getBoolean( hints, QueryHints.CALLABLE, name );
|
||||||
NamedSQLQueryDefinition def;
|
NamedSQLQueryDefinition def;
|
||||||
if ( StringHelper.isNotEmpty( resultSetMapping ) ) {
|
if ( StringHelper.isNotEmpty( resultSetMapping ) ) {
|
||||||
|
@ -175,32 +202,40 @@ public class QueryBinder {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
String resultClass = JandexHelper.getValueAsString( annotation, "resultClass" );
|
AnnotationValue annotationValue = annotation.value( "resultClass" );
|
||||||
if ( void.class.equals( resultClass ) ) {
|
if ( annotationValue == null ) {
|
||||||
throw new NotYetImplementedException( "Pure native scalar queries are not yet supported" );
|
throw new NotYetImplementedException( "Pure native scalar queries are not yet supported" );
|
||||||
}
|
}
|
||||||
def = new NamedSQLQueryDefinition(
|
NativeSQLQueryRootReturn queryRoots[] = new NativeSQLQueryRootReturn[] {
|
||||||
name,
|
|
||||||
query, new NativeSQLQueryRootReturn[] {
|
|
||||||
new NativeSQLQueryRootReturn(
|
new NativeSQLQueryRootReturn(
|
||||||
"alias1",
|
"alias1",
|
||||||
resultClass,
|
annotationValue.asString(),
|
||||||
new HashMap(),
|
new HashMap<String, String[]>(),
|
||||||
LockMode.READ
|
LockMode.READ
|
||||||
)
|
)
|
||||||
},
|
};
|
||||||
null, cacheable, cacheRegion, timeout, fetchSize, flushMode, cacheMode, readOnly,
|
def = new NamedSQLQueryDefinition(
|
||||||
comment, null, callable
|
name,
|
||||||
|
query,
|
||||||
|
queryRoots,
|
||||||
|
null,
|
||||||
|
cacheable,
|
||||||
|
cacheRegion,
|
||||||
|
timeout,
|
||||||
|
fetchSize,
|
||||||
|
flushMode,
|
||||||
|
cacheMode,
|
||||||
|
readOnly,
|
||||||
|
comment,
|
||||||
|
null,
|
||||||
|
callable
|
||||||
);
|
);
|
||||||
|
|
||||||
}
|
}
|
||||||
metadata.addNamedNativeQuery( def );
|
metadata.addNamedNativeQuery( def );
|
||||||
LOG.debugf( "Binding named native query: %s => %s", name, query );
|
LOG.debugf( "Binding named native query: %s => %s", name, query );
|
||||||
}
|
}
|
||||||
|
|
||||||
private static boolean getBoolean(AnnotationInstance[] hints,
|
private static boolean getBoolean(AnnotationInstance[] hints, String element, String query) {
|
||||||
String element,
|
|
||||||
String query) {
|
|
||||||
String val = getString( hints, element );
|
String val = getString( hints, element );
|
||||||
if ( val == null || val.equalsIgnoreCase( "false" ) ) {
|
if ( val == null || val.equalsIgnoreCase( "false" ) ) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -211,9 +246,7 @@ public class QueryBinder {
|
||||||
throw new AnnotationException( "Not a boolean in hint: " + query + ":" + element );
|
throw new AnnotationException( "Not a boolean in hint: " + query + ":" + element );
|
||||||
}
|
}
|
||||||
|
|
||||||
private static CacheMode getCacheMode(AnnotationInstance[] hints,
|
private static CacheMode getCacheMode(AnnotationInstance[] hints, String element, String query) {
|
||||||
String element,
|
|
||||||
String query) {
|
|
||||||
String val = getString( hints, element );
|
String val = getString( hints, element );
|
||||||
if ( val == null ) {
|
if ( val == null ) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -236,9 +269,7 @@ public class QueryBinder {
|
||||||
throw new AnnotationException( "Unknown CacheMode in hint: " + query + ":" + element );
|
throw new AnnotationException( "Unknown CacheMode in hint: " + query + ":" + element );
|
||||||
}
|
}
|
||||||
|
|
||||||
private static FlushMode getFlushMode(AnnotationInstance[] hints,
|
private static FlushMode getFlushMode(AnnotationInstance[] hints, String element, String query) {
|
||||||
String element,
|
|
||||||
String query) {
|
|
||||||
String val = getString( hints, element );
|
String val = getString( hints, element );
|
||||||
if ( val == null ) {
|
if ( val == null ) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -261,12 +292,9 @@ public class QueryBinder {
|
||||||
else {
|
else {
|
||||||
throw new AnnotationException( "Unknown FlushMode in hint: " + query + ":" + element );
|
throw new AnnotationException( "Unknown FlushMode in hint: " + query + ":" + element );
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Integer getInteger(AnnotationInstance[] hints,
|
private static Integer getInteger(AnnotationInstance[] hints, String element, String query) {
|
||||||
String element,
|
|
||||||
String query) {
|
|
||||||
String val = getString( hints, element );
|
String val = getString( hints, element );
|
||||||
if ( val == null ) {
|
if ( val == null ) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -279,8 +307,7 @@ public class QueryBinder {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static String getString(AnnotationInstance[] hints,
|
private static String getString(AnnotationInstance[] hints, String element) {
|
||||||
String element) {
|
|
||||||
for ( AnnotationInstance hint : hints ) {
|
for ( AnnotationInstance hint : hints ) {
|
||||||
if ( element.equals( JandexHelper.getValue( hint, "name" ) ) ) {
|
if ( element.equals( JandexHelper.getValue( hint, "name" ) ) ) {
|
||||||
return JandexHelper.getValueAsString( hint, "value" );
|
return JandexHelper.getValueAsString( hint, "value" );
|
||||||
|
@ -289,15 +316,11 @@ public class QueryBinder {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Integer getTimeout(AnnotationInstance[] hints,
|
private static Integer getTimeout(AnnotationInstance[] hints, String query) {
|
||||||
String query) {
|
|
||||||
Integer timeout = getInteger( hints, QueryHints.TIMEOUT_JPA, query );
|
Integer timeout = getInteger( hints, QueryHints.TIMEOUT_JPA, query );
|
||||||
if ( timeout == null ) {
|
if ( timeout == null ) {
|
||||||
return getInteger( hints, QueryHints.TIMEOUT_HIBERNATE, query ); // timeout is already in seconds
|
return getInteger( hints, QueryHints.TIMEOUT_HIBERNATE, query ); // timeout is already in seconds
|
||||||
}
|
}
|
||||||
return new Integer( (int) Math.round( timeout.doubleValue() / 1000.0 ) ); // convert milliseconds to seconds
|
return ( ( timeout + 500 ) / 1000 ); // convert milliseconds to seconds (rounded)
|
||||||
}
|
|
||||||
|
|
||||||
private QueryBinder() {
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,97 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* Copyright (c) 2011, Red Hat Inc. or third-party contributors as
|
||||||
|
* indicated by the @author tags or express copyright attribution
|
||||||
|
* statements applied by the authors. All third-party contributions are
|
||||||
|
* distributed under license by Red Hat Inc.
|
||||||
|
*
|
||||||
|
* This copyrighted material is made available to anyone wishing to use, modify,
|
||||||
|
* copy, or redistribute it subject to the terms and conditions of the GNU
|
||||||
|
* Lesser General Public License, as published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
|
||||||
|
* for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public License
|
||||||
|
* along with this distribution; if not, write to:
|
||||||
|
* Free Software Foundation, Inc.
|
||||||
|
* 51 Franklin Street, Fifth Floor
|
||||||
|
* Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
package org.hibernate.metamodel.source.annotations.global;
|
||||||
|
|
||||||
|
import javax.persistence.NamedNativeQuery;
|
||||||
|
|
||||||
|
import org.jboss.jandex.Index;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.hibernate.cfg.NotYetImplementedException;
|
||||||
|
import org.hibernate.engine.query.spi.sql.NativeSQLQueryReturn;
|
||||||
|
import org.hibernate.engine.query.spi.sql.NativeSQLQueryRootReturn;
|
||||||
|
import org.hibernate.engine.spi.NamedSQLQueryDefinition;
|
||||||
|
import org.hibernate.metamodel.MetadataSources;
|
||||||
|
import org.hibernate.metamodel.source.annotations.util.JandexHelper;
|
||||||
|
import org.hibernate.metamodel.source.internal.MetadataImpl;
|
||||||
|
import org.hibernate.service.ServiceRegistryBuilder;
|
||||||
|
import org.hibernate.service.classloading.spi.ClassLoaderService;
|
||||||
|
import org.hibernate.service.internal.BasicServiceRegistryImpl;
|
||||||
|
import org.hibernate.testing.junit4.BaseUnitTestCase;
|
||||||
|
|
||||||
|
import static junit.framework.Assert.assertEquals;
|
||||||
|
import static junit.framework.Assert.assertNotNull;
|
||||||
|
import static junit.framework.Assert.assertTrue;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Hardy Ferentschik
|
||||||
|
*/
|
||||||
|
public class QueryBinderTest extends BaseUnitTestCase {
|
||||||
|
|
||||||
|
private BasicServiceRegistryImpl serviceRegistry;
|
||||||
|
private ClassLoaderService service;
|
||||||
|
private MetadataImpl meta;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
serviceRegistry = (BasicServiceRegistryImpl) new ServiceRegistryBuilder().buildServiceRegistry();
|
||||||
|
service = serviceRegistry.getService( ClassLoaderService.class );
|
||||||
|
meta = (MetadataImpl) new MetadataSources( serviceRegistry ).buildMetadata();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() {
|
||||||
|
serviceRegistry.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(expected = NotYetImplementedException.class)
|
||||||
|
public void testNoResultClass() {
|
||||||
|
@NamedNativeQuery(name = "fubar", query = "SELECT * FROM FOO")
|
||||||
|
class Foo {
|
||||||
|
}
|
||||||
|
Index index = JandexHelper.indexForClass( service, Foo.class );
|
||||||
|
QueryBinder.bind( meta, index );
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testResultClass() {
|
||||||
|
@NamedNativeQuery(name = "fubar", query = "SELECT * FROM FOO", resultClass = Foo.class)
|
||||||
|
class Foo {
|
||||||
|
}
|
||||||
|
Index index = JandexHelper.indexForClass( service, Foo.class );
|
||||||
|
QueryBinder.bind( meta, index );
|
||||||
|
|
||||||
|
NamedSQLQueryDefinition namedQuery = meta.getNamedNativeQuery( "fubar" );
|
||||||
|
assertNotNull( namedQuery );
|
||||||
|
NativeSQLQueryReturn queryReturns[] = namedQuery.getQueryReturns();
|
||||||
|
assertTrue( "Wrong number of returns", queryReturns.length == 1 );
|
||||||
|
assertTrue( "Wrong query return type", queryReturns[0] instanceof NativeSQLQueryRootReturn );
|
||||||
|
NativeSQLQueryRootReturn rootReturn = (NativeSQLQueryRootReturn) queryReturns[0];
|
||||||
|
assertEquals( "Wrong result class", Foo.class.getName(), rootReturn.getReturnEntityName() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue