HHH-8699 - Multiselect with boolean predicate

Conflicts:
	hibernate-core/src/main/antlr/hql-sql.g
	hibernate-core/src/main/antlr/sql-gen.g
	hibernate-entitymanager/src/test/java/org/hibernate/jpa/test/criteria/QueryBuilderTest.java
	hibernate-entitymanager/src/test/java/org/hibernate/jpa/test/metamodel/Customer.java
This commit is contained in:
Lukasz Antoniak 2013-11-19 23:27:26 +01:00 committed by Brett Meyer
parent 4cccaa5d31
commit 1f2e4f2d0a
9 changed files with 222 additions and 6 deletions

View File

@ -409,6 +409,7 @@ selectExpr
| collectionFunction // elements() or indices() | collectionFunction // elements() or indices()
| literal | literal
| arithmeticExpr | arithmeticExpr
| logicalExpr
| query | query
; ;

View File

@ -247,6 +247,7 @@ selectExpr
| c:constant { out(c); } | c:constant { out(c); }
| arithmeticExpr | arithmeticExpr
| param:PARAM { out(param); } | param:PARAM { out(param); }
| selectBooleanExpr[false]
| sn:SQL_NODE { out(sn); } | sn:SQL_NODE { out(sn); }
| { out("("); } selectStatement { out(")"); } | { out("("); } selectStatement { out(")"); }
; ;
@ -306,6 +307,11 @@ booleanOp[ boolean parens ]
| #(NOT { out(" not ("); } booleanExpr[false] { out(")"); } ) | #(NOT { out(" not ("); } booleanExpr[false] { out(")"); } )
; ;
selectBooleanExpr[ boolean parens ]
: booleanOp [ parens ]
| comparisonExpr [ parens ]
;
booleanExpr[ boolean parens ] booleanExpr[ boolean parens ]
: booleanOp [ parens ] : booleanOp [ parens ]
| comparisonExpr [ parens ] | comparisonExpr [ parens ]
@ -387,6 +393,7 @@ simpleExpr
| count | count
| parameter | parameter
| arithmeticExpr | arithmeticExpr
| selectBooleanExpr[false]
; ;
constant constant

View File

@ -32,6 +32,7 @@ import org.hibernate.HibernateException;
import org.hibernate.TypeMismatchException; import org.hibernate.TypeMismatchException;
import org.hibernate.engine.spi.SessionFactoryImplementor; import org.hibernate.engine.spi.SessionFactoryImplementor;
import org.hibernate.hql.internal.antlr.HqlSqlTokenTypes; import org.hibernate.hql.internal.antlr.HqlSqlTokenTypes;
import org.hibernate.hql.internal.ast.util.ColumnHelper;
import org.hibernate.internal.util.StringHelper; import org.hibernate.internal.util.StringHelper;
import org.hibernate.param.ParameterSpecification; import org.hibernate.param.ParameterSpecification;
import org.hibernate.type.OneToOneType; import org.hibernate.type.OneToOneType;
@ -43,7 +44,7 @@ import org.hibernate.type.Type;
* *
* @author Steve Ebersole * @author Steve Ebersole
*/ */
public class BinaryLogicOperatorNode extends HqlSqlWalkerNode implements BinaryOperatorNode { public class BinaryLogicOperatorNode extends AbstractSelectExpression implements BinaryOperatorNode {
/** /**
* Performs the operator node initialization by seeking out any parameter * Performs the operator node initialization by seeking out any parameter
* nodes and setting their expected type, if possible. * nodes and setting their expected type, if possible.
@ -259,4 +260,8 @@ public class BinaryLogicOperatorNode extends HqlSqlWalkerNode implements BinaryO
public Node getRightHandOperand() { public Node getRightHandOperand() {
return ( Node ) getFirstChild().getNextSibling(); return ( Node ) getFirstChild().getNextSibling();
} }
public void setScalarColumnText(int i) throws SemanticException {
ColumnHelper.generateSingleScalarColumn( this, i );
}
} }

View File

@ -23,6 +23,9 @@
*/ */
package org.hibernate.hql.internal.ast.tree; package org.hibernate.hql.internal.ast.tree;
import antlr.SemanticException;
import org.hibernate.hql.internal.ast.util.ColumnHelper;
import org.hibernate.type.StandardBasicTypes; import org.hibernate.type.StandardBasicTypes;
import org.hibernate.type.Type; import org.hibernate.type.Type;
@ -31,7 +34,7 @@ import org.hibernate.type.Type;
* *
* @author Steve Ebersole * @author Steve Ebersole
*/ */
public class UnaryLogicOperatorNode extends HqlSqlWalkerNode implements UnaryOperatorNode { public class UnaryLogicOperatorNode extends AbstractSelectExpression implements UnaryOperatorNode {
public Node getOperand() { public Node getOperand() {
return ( Node ) getFirstChild(); return ( Node ) getFirstChild();
} }
@ -45,4 +48,8 @@ public class UnaryLogicOperatorNode extends HqlSqlWalkerNode implements UnaryOpe
// logic operators by definition resolve to booleans // logic operators by definition resolve to booleans
return StandardBasicTypes.BOOLEAN; return StandardBasicTypes.BOOLEAN;
} }
public void setScalarColumnText(int i) throws SemanticException {
ColumnHelper.generateSingleScalarColumn( this, i );
}
} }

View File

@ -124,6 +124,7 @@ public class ASTParserLoadingTest extends BaseCoreFunctionalTestCase {
"hql/Image.hbm.xml", "hql/Image.hbm.xml",
"hql/ComponentContainer.hbm.xml", "hql/ComponentContainer.hbm.xml",
"hql/VariousKeywordPropertyEntity.hbm.xml", "hql/VariousKeywordPropertyEntity.hbm.xml",
"hql/Constructor.hbm.xml",
"batchfetch/ProductLine.hbm.xml", "batchfetch/ProductLine.hbm.xml",
"cid/Customer.hbm.xml", "cid/Customer.hbm.xml",
"cid/Order.hbm.xml", "cid/Order.hbm.xml",
@ -203,6 +204,31 @@ public class ASTParserLoadingTest extends BaseCoreFunctionalTestCase {
session.close(); session.close();
} }
@Test
@TestForIssue( jiraKey = "HHH-8699" )
public void testBooleanPredicate() {
final Session session = openSession();
session.getTransaction().begin();
final Constructor constructor = new Constructor();
session.save( constructor );
session.getTransaction().commit();
session.clear();
Constructor.resetConstructorExecutionCount();
session.getTransaction().begin();
final Constructor result = (Constructor) session.createQuery(
"select new Constructor( c.id, c.id is not null, c.id = c.id, c.id + 1, concat( c.id, 'foo' ) ) from Constructor c where c.id = :id"
).setParameter( "id", constructor.getId() ).uniqueResult();
session.getTransaction().commit();
assertEquals( 1, Constructor.getConstructorExecutionCount() );
assertEquals( new Constructor( constructor.getId(), true, true, constructor.getId() + 1, constructor.getId() + "foo" ), result );
session.close();
}
@Test @Test
public void testJpaTypeOperator() { public void testJpaTypeOperator() {
// just checking syntax here... // just checking syntax here...

View File

@ -0,0 +1,16 @@
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="org.hibernate.test.hql">
<class name="Constructor">
<id name="id" column="ID" type="long">
<generator class="increment"/>
</id>
<property name="someString" type="string"/>
<property name="someNumber" type="int"/>
<property name="someBoolean" type="boolean"/>
<property name="anotherBoolean" type="boolean"/>
</class>
</hibernate-mapping>

View File

@ -0,0 +1,125 @@
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* Copyright (c) 2013, 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.test.hql;
import java.io.Serializable;
/**
* @author Lukasz Antoniak (lukasz dot antoniak at gmail dot com)
*/
public class Constructor implements Serializable {
private static int CONSTRUCTOR_EXECUTION_COUNT = 0;
private long id;
private String someString;
private Number someNumber;
private boolean someBoolean;
private boolean anotherBoolean;
public Constructor() {
}
public Constructor(long id, boolean someBoolean, boolean anotherBoolean, Number someNumber, String someString) {
this.id = id;
this.someBoolean = someBoolean;
this.anotherBoolean = anotherBoolean;
this.someNumber = someNumber;
this.someString = someString;
++CONSTRUCTOR_EXECUTION_COUNT;
}
@Override
public boolean equals(Object o) {
if ( this == o ) return true;
if ( !( o instanceof Constructor ) ) return false;
Constructor that = (Constructor) o;
if ( anotherBoolean != that.anotherBoolean ) return false;
if ( id != that.id ) return false;
if ( someBoolean != that.someBoolean ) return false;
if ( someNumber != null ? !someNumber.equals( that.someNumber ) : that.someNumber != null ) return false;
if ( someString != null ? !someString.equals( that.someString ) : that.someString != null ) return false;
return true;
}
@Override
public int hashCode() {
int result = (int) (id ^ (id >>> 32));
result = 31 * result + (someString != null ? someString.hashCode() : 0);
result = 31 * result + (someNumber != null ? someNumber.hashCode() : 0);
result = 31 * result + (someBoolean ? 1 : 0);
result = 31 * result + (anotherBoolean ? 1 : 0);
return result;
}
public boolean isSomeBoolean() {
return someBoolean;
}
public void setSomeBoolean(boolean someBoolean) {
this.someBoolean = someBoolean;
}
public Number getSomeNumber() {
return someNumber;
}
public void setSomeNumber(Number someNumber) {
this.someNumber = someNumber;
}
public String getSomeString() {
return someString;
}
public void setSomeString(String someString) {
this.someString = someString;
}
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public boolean isAnotherBoolean() {
return anotherBoolean;
}
public void setAnotherBoolean(boolean anotherBoolean) {
this.anotherBoolean = anotherBoolean;
}
public static int getConstructorExecutionCount() {
return CONSTRUCTOR_EXECUTION_COUNT;
}
public static void resetConstructorExecutionCount() {
CONSTRUCTOR_EXECUTION_COUNT = 0;
}
}

View File

@ -23,17 +23,18 @@
*/ */
package org.hibernate.ejb.criteria; package org.hibernate.ejb.criteria;
import static org.junit.Assert.assertEquals;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.List; import java.util.List;
import javax.persistence.EntityManager; import javax.persistence.EntityManager;
import javax.persistence.TypedQuery; import javax.persistence.TypedQuery;
import javax.persistence.criteria.CriteriaQuery; import javax.persistence.criteria.CriteriaQuery;
import javax.persistence.criteria.Root; import javax.persistence.criteria.Root;
import javax.persistence.metamodel.EntityType; import javax.persistence.metamodel.EntityType;
import org.junit.Test;
import org.hibernate.ejb.criteria.predicate.ComparisonPredicate; import org.hibernate.ejb.criteria.predicate.ComparisonPredicate;
import org.hibernate.ejb.metamodel.Address; import org.hibernate.ejb.metamodel.Address;
import org.hibernate.ejb.metamodel.Alias; import org.hibernate.ejb.metamodel.Alias;
@ -48,9 +49,10 @@ import org.hibernate.ejb.metamodel.Phone;
import org.hibernate.ejb.metamodel.Product; import org.hibernate.ejb.metamodel.Product;
import org.hibernate.ejb.metamodel.ShelfLife; import org.hibernate.ejb.metamodel.ShelfLife;
import org.hibernate.ejb.metamodel.Spouse; import org.hibernate.ejb.metamodel.Spouse;
import org.hibernate.ejb.metamodel.Customer_;
import org.hibernate.ejb.test.BaseEntityManagerFunctionalTestCase; import org.hibernate.ejb.test.BaseEntityManagerFunctionalTestCase;
import org.hibernate.testing.TestForIssue;
import static org.junit.Assert.assertEquals; import org.junit.Test;
/** /**
* @author Steve Ebersole * @author Steve Ebersole
@ -212,6 +214,26 @@ public class QueryBuilderTest extends BaseEntityManagerFunctionalTestCase {
em.close(); em.close();
} }
@Test
@TestForIssue( jiraKey = "HHH-8699" )
public void testMultiselectWithPredicates() {
EntityManager em = getOrCreateEntityManager();
em.getTransaction().begin();
CriteriaBuilderImpl cb = (CriteriaBuilderImpl) em.getCriteriaBuilder();
CriteriaQuery<Customer> cq = cb.createQuery( Customer.class );
Root<Customer> r = cq.from( Customer.class );
cq.multiselect(
r.get( Customer_.id ), r.get( Customer_.name ),
cb.concat( "Hello ", r.get( Customer_.name ) ), cb.isNotNull( r.get( Customer_.age ) )
);
TypedQuery<Customer> tq = em.createQuery( cq );
tq.getResultList();
em.getTransaction().commit();
em.close();
}
@Test @Test
public void testDateTimeFunctions() { public void testDateTimeFunctions() {
EntityManager em = getOrCreateEntityManager(); EntityManager em = getOrCreateEntityManager();

View File

@ -23,6 +23,7 @@
*/ */
package org.hibernate.ejb.metamodel; package org.hibernate.ejb.metamodel;
import java.util.Collection; import java.util.Collection;
import javax.persistence.CascadeType; import javax.persistence.CascadeType;
import javax.persistence.Column; import javax.persistence.Column;
import javax.persistence.Embedded; import javax.persistence.Embedded;
@ -62,6 +63,12 @@ public class Customer implements java.io.Serializable {
this.name = name; this.name = name;
} }
// Used by test case for HHH-8699.
public Customer(String id, String name, String greeting, Boolean something) {
this.id = id;
this.name = name;
}
public Customer(String id, String name, Country country) { public Customer(String id, String name, Country country) {
this.id = id; this.id = id;
this.name = name; this.name = name;