From d1515a2911d52b54674219bbd2f629773a5cd75a Mon Sep 17 00:00:00 2001 From: Gail Badner Date: Fri, 21 Jan 2011 13:05:54 -0800 Subject: [PATCH] HHH-5126 : "in" expression and column-valued-input-parameter --- hibernate-core/src/main/antlr/hql.g | 9 ++- .../org/hibernate/impl/AbstractQueryImpl.java | 42 ++++++++-- .../java/org/hibernate/util/StringHelper.java | 80 ++++++++++++++++--- .../test/annotations/cid/CompositeIdTest.java | 2 +- .../test/hql/ASTParserLoadingTest.java | 10 +++ .../java/org/hibernate/test/hql/HQLTest.java | 5 ++ .../test/legacy/SQLFunctionsTest.java | 8 ++ .../hibernate/test/legacy/SQLLoaderTest.java | 16 +++- .../hibernate/ejb/test/query/QueryTest.java | 66 ++++++++++++++- 9 files changed, 216 insertions(+), 22 deletions(-) diff --git a/hibernate-core/src/main/antlr/hql.g b/hibernate-core/src/main/antlr/hql.g index 855715c9a4..5fe8917b39 100644 --- a/hibernate-core/src/main/antlr/hql.g +++ b/hibernate-core/src/main/antlr/hql.g @@ -612,10 +612,14 @@ atom primaryExpression : identPrimary ( options {greedy=true;} : DOT^ "class" )? | constant - | COLON^ identifier + | parameter // TODO: Add parens to the tree so the user can control the operator evaluation order. | OPEN! (expressionOrVector | subQuery) CLOSE! - | PARAM^ (NUM_INT)? + ; + +parameter + : COLON^ identifier + | PARAM^ (NUM_INT)? ; // This parses normal expression and a list of expressions separated by commas. If a comma is encountered @@ -676,6 +680,7 @@ compoundExpr : collectionExpr | path | (OPEN! ( (expression (COMMA! expression)*) | subQuery ) CLOSE!) + | parameter ; subQuery diff --git a/hibernate-core/src/main/java/org/hibernate/impl/AbstractQueryImpl.java b/hibernate-core/src/main/java/org/hibernate/impl/AbstractQueryImpl.java index cac26b4569..e822accfce 100644 --- a/hibernate-core/src/main/java/org/hibernate/impl/AbstractQueryImpl.java +++ b/hibernate-core/src/main/java/org/hibernate/impl/AbstractQueryImpl.java @@ -748,8 +748,35 @@ public abstract class AbstractQueryImpl implements Query { private String expandParameterList(String query, String name, TypedValue typedList, Map namedParamsCopy) { Collection vals = (Collection) typedList.getValue(); Type type = typedList.getType(); - if ( vals.size() == 1 ) { - // short-circuit for performance... + + boolean isJpaPositionalParam = parameterMetadata.getNamedParameterDescriptor( name ).isJpaStyle(); + String paramPrefix = isJpaPositionalParam ? "?" : ParserHelper.HQL_VARIABLE_PREFIX; + String placeholder = + new StringBuffer( paramPrefix.length() + name.length() ) + .append( paramPrefix ).append( name ) + .toString(); + + if ( query == null ) { + return query; + } + int loc = query.indexOf( placeholder ); + + if ( loc < 0 ) { + return query; + } + + String beforePlaceholder = query.substring( 0, loc ); + String afterPlaceholder = query.substring( loc + placeholder.length() ); + + // check if placeholder is already immediately enclosed in parentheses + // (ignoring whitespace) + boolean isEnclosedInParens = + StringHelper.getLastNonWhitespaceCharacter( beforePlaceholder ) == '(' && + StringHelper.getFirstNonWhitespaceCharacter( afterPlaceholder ) == ')'; + + if ( vals.size() == 1 && isEnclosedInParens ) { + // short-circuit for performance when only 1 value and the + // placeholder is already enclosed in parentheses... namedParamsCopy.put( name, new TypedValue( type, vals.iterator().next(), session.getEntityMode() ) ); return query; } @@ -757,7 +784,6 @@ public abstract class AbstractQueryImpl implements Query { StringBuffer list = new StringBuffer( 16 ); Iterator iter = vals.iterator(); int i = 0; - boolean isJpaPositionalParam = parameterMetadata.getNamedParameterDescriptor( name ).isJpaStyle(); while ( iter.hasNext() ) { String alias = ( isJpaPositionalParam ? 'x' + name : name ) + i++ + '_'; namedParamsCopy.put( alias, new TypedValue( type, iter.next(), session.getEntityMode() ) ); @@ -766,8 +792,14 @@ public abstract class AbstractQueryImpl implements Query { list.append( ", " ); } } - String paramPrefix = isJpaPositionalParam ? "?" : ParserHelper.HQL_VARIABLE_PREFIX; - return StringHelper.replace( query, paramPrefix + name, list.toString(), true ); + return StringHelper.replace( + beforePlaceholder, + afterPlaceholder, + placeholder.toString(), + list.toString(), + true, + true + ); } public Query setParameterList(String name, Collection vals) throws HibernateException { diff --git a/hibernate-core/src/main/java/org/hibernate/util/StringHelper.java b/hibernate-core/src/main/java/org/hibernate/util/StringHelper.java index e0146035ab..68f82230b7 100644 --- a/hibernate-core/src/main/java/org/hibernate/util/StringHelper.java +++ b/hibernate-core/src/main/java/org/hibernate/util/StringHelper.java @@ -106,6 +106,14 @@ public final class StringHelper { } public static String replace(String template, String placeholder, String replacement, boolean wholeWords) { + return replace( template, placeholder, replacement, wholeWords, false ); + } + + public static String replace(String template, + String placeholder, + String replacement, + boolean wholeWords, + boolean encloseInParensIfNecessary) { if ( template == null ) { return template; } @@ -114,20 +122,72 @@ public final class StringHelper { return template; } else { - final boolean actuallyReplace = !wholeWords || - loc + placeholder.length() == template.length() || - !Character.isJavaIdentifierPart( template.charAt( loc + placeholder.length() ) ); - String actualReplacement = actuallyReplace ? replacement : placeholder; - return new StringBuffer( template.substring( 0, loc ) ) - .append( actualReplacement ) - .append( replace( template.substring( loc + placeholder.length() ), - placeholder, - replacement, - wholeWords ) ).toString(); + String beforePlaceholder = template.substring( 0, loc ); + String afterPlaceholder = template.substring( loc + placeholder.length() ); + return replace( beforePlaceholder, afterPlaceholder, placeholder, replacement, wholeWords, encloseInParensIfNecessary ); } } + public static String replace(String beforePlaceholder, + String afterPlaceholder, + String placeholder, + String replacement, + boolean wholeWords, + boolean encloseInParensIfNecessary) { + final boolean actuallyReplace = + ! wholeWords || + afterPlaceholder.length() == 0 || + ! Character.isJavaIdentifierPart( afterPlaceholder.charAt( 0 ) ); + boolean encloseInParens = + actuallyReplace && + encloseInParensIfNecessary && + ! ( getLastNonWhitespaceCharacter( beforePlaceholder ) == '(' ) && + ! ( getFirstNonWhitespaceCharacter( afterPlaceholder ) == ')' ); + StringBuilder buf = new StringBuilder( beforePlaceholder ); + if ( encloseInParens ) { + buf.append( '(' ); + } + buf.append( actuallyReplace ? replacement : placeholder ); + if ( encloseInParens ) { + buf.append( ')' ); + } + buf.append( + replace( + afterPlaceholder, + placeholder, + replacement, + wholeWords, + encloseInParensIfNecessary + ) + ); + return buf.toString(); + } + + public static char getLastNonWhitespaceCharacter(String str) { + if ( str != null && str.length() > 0 ) { + for ( int i = str.length() - 1 ; i >= 0 ; i-- ) { + char ch = str.charAt( i ); + if ( ! Character.isWhitespace( ch ) ) { + return ch; + } + } + } + return '\0'; + } + + public static char getFirstNonWhitespaceCharacter(String str) { + if ( str != null && str.length() > 0 ) { + for ( int i = 0 ; i < str.length() ; i++ ) { + char ch = str.charAt( i ); + if ( ! Character.isWhitespace( ch ) ) { + return ch; + } + } + } + return '\0'; + } + public static String replaceOnce(String template, String placeholder, String replacement) { if ( template == null ) { return template; // returnign null! diff --git a/hibernate-core/src/test/java/org/hibernate/test/annotations/cid/CompositeIdTest.java b/hibernate-core/src/test/java/org/hibernate/test/annotations/cid/CompositeIdTest.java index 77f921bc66..eade82f698 100644 --- a/hibernate-core/src/test/java/org/hibernate/test/annotations/cid/CompositeIdTest.java +++ b/hibernate-core/src/test/java/org/hibernate/test/annotations/cid/CompositeIdTest.java @@ -280,7 +280,7 @@ public class CompositeIdTest extends TestCase { ids.add( new SomeEntityId(1,12) ); ids.add( new SomeEntityId(10,23) ); ids.add( new SomeEntityId(10,22) ); - Query query=s.createQuery( "from SomeEntity e where e.id in (:idList)" ); + Query query=s.createQuery( "from SomeEntity e where e.id in :idList" ); query.setParameterList( "idList", ids ); List list=query.list(); assertEquals( 3, list.size() ); diff --git a/hibernate-core/src/test/java/org/hibernate/test/hql/ASTParserLoadingTest.java b/hibernate-core/src/test/java/org/hibernate/test/hql/ASTParserLoadingTest.java index f6ec05a3d6..2e8b058a59 100644 --- a/hibernate-core/src/test/java/org/hibernate/test/hql/ASTParserLoadingTest.java +++ b/hibernate-core/src/test/java/org/hibernate/test/hql/ASTParserLoadingTest.java @@ -524,6 +524,11 @@ public class ASTParserLoadingTest extends FunctionalTestCase { list.add( new Id("123456789", order.getId().getOrderNumber(), "1234") ); query.setParameterList( "idList", list ); assertEquals( 2, query.list().size() ); + + query = s.createQuery( "from LineItem l where l.id in :idList" ); + query.setParameterList( "idList", list ); + assertEquals( 2, query.list().size() ); + s.getTransaction().rollback(); s.close(); @@ -690,6 +695,11 @@ public class ASTParserLoadingTest extends FunctionalTestCase { s.createQuery( "from Human where name.last in (?1)" ) .setParameterList( "1", params ) .list(); + + s.createQuery( "from Human where name.last in ?1" ) + .setParameterList( "1", params ) + .list(); + s.getTransaction().commit(); s.close(); } diff --git a/hibernate-core/src/test/java/org/hibernate/test/hql/HQLTest.java b/hibernate-core/src/test/java/org/hibernate/test/hql/HQLTest.java index 95528c7907..98c8b401b3 100644 --- a/hibernate-core/src/test/java/org/hibernate/test/hql/HQLTest.java +++ b/hibernate-core/src/test/java/org/hibernate/test/hql/HQLTest.java @@ -103,6 +103,7 @@ public class HQLTest extends QueryTranslatorTestCase { */ public void testRowValueConstructorSyntaxInInListFailureExpected() { assertTranslation( "from LineItem l where l.id in (:idList)" ); + assertTranslation( "from LineItem l where l.id in :idList" ); } public void testRowValueConstructorSyntaxInInList() { @@ -110,10 +111,14 @@ public class HQLTest extends QueryTranslatorTestCase { return; QueryTranslatorImpl translator = createNewQueryTranslator("from LineItem l where l.id in (?)"); assertInExist("'in' should be translated to 'and'", false, translator); + translator = createNewQueryTranslator("from LineItem l where l.id in ?"); + assertInExist("'in' should be translated to 'and'", false, translator); translator = createNewQueryTranslator("from LineItem l where l.id in (('a1',1,'b1'),('a2',2,'b2'))"); assertInExist("'in' should be translated to 'and'", false, translator); translator = createNewQueryTranslator("from Animal a where a.id in (?)"); assertInExist("only translate tuple with 'in' syntax", true, translator); + translator = createNewQueryTranslator("from Animal a where a.id in ?"); + assertInExist("only translate tuple with 'in' syntax", true, translator); translator = createNewQueryTranslator("from LineItem l where l.id in (select a1 from Animal a1 left join a1.offspring o where a1.id = 1)"); assertInExist("do not translate subqueries", true, translator); diff --git a/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLFunctionsTest.java b/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLFunctionsTest.java index ca1f58f8d5..6a50a120fb 100644 --- a/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLFunctionsTest.java +++ b/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLFunctionsTest.java @@ -163,10 +163,18 @@ public class SQLFunctionsTest extends LegacyTestCase { q.setProperties(single); assertTrue( q.list().get(0)==simple ); + q = s.createQuery("from Simple s where s.name in :several"); + q.setProperties(single); + assertTrue( q.list().get(0)==simple ); q = s.createQuery("from Simple s where s.name in (:stuff)"); q.setProperties(single); assertTrue( q.list().get(0)==simple ); + + q = s.createQuery("from Simple s where s.name in :stuff"); + q.setProperties(single); + assertTrue( q.list().get(0)==simple ); + s.delete(simple); t.commit(); s.close(); diff --git a/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLLoaderTest.java b/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLLoaderTest.java index 6269b0f074..bdb0ade782 100644 --- a/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLLoaderTest.java +++ b/hibernate-core/src/test/java/org/hibernate/test/legacy/SQLLoaderTest.java @@ -109,9 +109,21 @@ public class SQLLoaderTest extends LegacyTestCase { query = session.createSQLQuery("select {category.*} from category {category} where {category}.name in (:names)", "category", Category.class); String[] str = new String[] { "WannaBeFound", "NotThere" }; query.setParameterList("names", str); - query.uniqueResult(); - + + query = session.createSQLQuery("select {category.*} from category {category} where {category}.name in :names", "category", Category.class); + query.setParameterList("names", str); + query.uniqueResult(); + + query = session.createSQLQuery("select {category.*} from category {category} where {category}.name in (:names)", "category", Category.class); + str = new String[] { "WannaBeFound" }; + query.setParameterList("names", str); + query.uniqueResult(); + + query = session.createSQLQuery("select {category.*} from category {category} where {category}.name in :names", "category", Category.class); + query.setParameterList("names", str); + query.uniqueResult(); + session.connection().commit(); session.close(); diff --git a/hibernate-entitymanager/src/test/java/org/hibernate/ejb/test/query/QueryTest.java b/hibernate-entitymanager/src/test/java/org/hibernate/ejb/test/query/QueryTest.java index 0f79f970ec..f73c086312 100644 --- a/hibernate-entitymanager/src/test/java/org/hibernate/ejb/test/query/QueryTest.java +++ b/hibernate-entitymanager/src/test/java/org/hibernate/ejb/test/query/QueryTest.java @@ -3,6 +3,9 @@ package org.hibernate.ejb.test.query; import java.util.ArrayList; import java.util.Date; import java.util.List; +import java.util.HashSet; +import java.util.Set; + import javax.persistence.EntityManager; import javax.persistence.Query; import javax.persistence.TemporalType; @@ -12,7 +15,6 @@ import org.hibernate.ejb.test.Item; import org.hibernate.ejb.test.TestCase; import org.hibernate.ejb.test.Wallet; - /** * @author Emmanuel Bernard */ @@ -62,6 +64,54 @@ public class QueryTest extends TestCase { assertTrue( em.contains( item ) ); em.getTransaction().commit(); + em.getTransaction().begin(); + Query q = em.createQuery( "select item from Item item where item.name in :names" ); + //test hint in value and string + q.setHint( "org.hibernate.fetchSize", 10 ); + q.setHint( "org.hibernate.fetchSize", "10" ); + List params = new ArrayList(); + params.add( item.getName() ); + q.setParameter( "names", params ); + List result = q.getResultList(); + assertNotNull( result ); + assertEquals( 1, result.size() ); + + q = em.createQuery( "select item from Item item where item.name in :names" ); + //test hint in value and string + q.setHint( "org.hibernate.fetchSize", 10 ); + q.setHint( "org.hibernate.fetchSize", "10" ); + params.add( item2.getName() ); + q.setParameter( "names", params ); + result = q.getResultList(); + assertNotNull( result ); + assertEquals( 2, result.size() ); + + q = em.createQuery( "select item from Item item where item.name in ?1" ); + params = new ArrayList(); + params.add( item.getName() ); + params.add( item2.getName() ); + q.setParameter( "1", params ); + result = q.getResultList(); + assertNotNull( result ); + assertEquals( 2, result.size() ); + em.remove( result.get( 0 ) ); + em.remove( result.get( 1 ) ); + em.getTransaction().commit(); + + em.close(); + } + + public void testParameterListInExistingParens() throws Exception { + final Item item = new Item( "Mouse", "Micro$oft mouse" ); + final Item item2 = new Item( "Computer", "Dell computer" ); + + EntityManager em = getOrCreateEntityManager(); + em.getTransaction().begin(); + em.persist( item ); + em.persist( item2 ); + assertTrue( em.contains( item ) ); + em.getTransaction().commit(); + em.getTransaction().begin(); Query q = em.createQuery( "select item from Item item where item.name in (:names)" ); //test hint in value and string @@ -75,6 +125,18 @@ public class QueryTest extends TestCase { assertNotNull( result ); assertEquals( 2, result.size() ); + q = em.createQuery( "select item from Item item where item.name in ( \n :names \n)\n" ); + //test hint in value and string + q.setHint( "org.hibernate.fetchSize", 10 ); + q.setHint( "org.hibernate.fetchSize", "10" ); + params = new ArrayList(); + params.add( item.getName() ); + params.add( item2.getName() ); + q.setParameter( "names", params ); + result = q.getResultList(); + assertNotNull( result ); + assertEquals( 2, result.size() ); + q = em.createQuery( "select item from Item item where item.name in ( ?1 )" ); params = new ArrayList(); params.add( item.getName() ); @@ -199,7 +261,7 @@ public class QueryTest extends TestCase { em.persist( w ); em.getTransaction().commit(); em.getTransaction().begin(); - Query query = em.createQuery( "select w from " + Wallet.class.getName() + " w where w.brand in (?1)" ); + Query query = em.createQuery( "select w from " + Wallet.class.getName() + " w where w.brand in ?1" ); List brands = new ArrayList(); brands.add( "Lacoste" ); query.setParameter( 1, brands );