HHH-7165 - count() query on classes using EmbeddedId should not use id column tuple on Dialects which dont support non-distinct tuple counts
This commit is contained in:
parent
2c85cbefa3
commit
e532dc5e7d
|
@ -31,6 +31,7 @@ import antlr.collections.AST;
|
||||||
|
|
||||||
import org.hibernate.QueryException;
|
import org.hibernate.QueryException;
|
||||||
import org.hibernate.dialect.function.SQLFunction;
|
import org.hibernate.dialect.function.SQLFunction;
|
||||||
|
import org.hibernate.hql.internal.antlr.HqlSqlTokenTypes;
|
||||||
import org.hibernate.hql.internal.antlr.SqlTokenTypes;
|
import org.hibernate.hql.internal.antlr.SqlTokenTypes;
|
||||||
import org.hibernate.hql.internal.ast.util.ColumnHelper;
|
import org.hibernate.hql.internal.ast.util.ColumnHelper;
|
||||||
import org.hibernate.internal.util.StringHelper;
|
import org.hibernate.internal.util.StringHelper;
|
||||||
|
@ -155,18 +156,27 @@ public class IdentNode extends FromReferenceNode implements SelectExpression {
|
||||||
setFromElement( element );
|
setFromElement( element );
|
||||||
String[] columnExpressions = element.getIdentityColumns();
|
String[] columnExpressions = element.getIdentityColumns();
|
||||||
final boolean isInNonDistinctCount = getWalker().isInCount() && ! getWalker().isInCountDistinct();
|
final boolean isInNonDistinctCount = getWalker().isInCount() && ! getWalker().isInCountDistinct();
|
||||||
final boolean isCompositePk = columnExpressions.length > 1;
|
final boolean isCompositeValue = columnExpressions.length > 1;
|
||||||
if ( isCompositePk
|
if ( isCompositeValue ) {
|
||||||
&& isInNonDistinctCount
|
if ( isInNonDistinctCount && ! getWalker().getSessionFactoryHelper().getFactory().getDialect().supportsTupleCounts() ) {
|
||||||
&& ! getWalker().getSessionFactoryHelper().getFactory().getDialect().supportsTupleCounts() ) {
|
setText( columnExpressions[0] );
|
||||||
setText( columnExpressions[0] );
|
}
|
||||||
|
else {
|
||||||
|
String joinedFragment = StringHelper.join( ", ", columnExpressions );
|
||||||
|
// avoid wrapping in parenthesis (explicit tuple treatment) if possible due to varied support for
|
||||||
|
// tuple syntax across databases..
|
||||||
|
final boolean shouldSkipWrappingInParenthesis =
|
||||||
|
getWalker().isInCount()
|
||||||
|
|| getWalker().getCurrentTopLevelClauseType() == HqlSqlTokenTypes.ORDER
|
||||||
|
|| getWalker().getCurrentTopLevelClauseType() == HqlSqlTokenTypes.GROUP;
|
||||||
|
if ( ! shouldSkipWrappingInParenthesis ) {
|
||||||
|
joinedFragment = "(" + joinedFragment + ")";
|
||||||
|
}
|
||||||
|
setText( joinedFragment );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
String joinedFragment = StringHelper.join( ", ", columnExpressions );
|
setText( columnExpressions[0] );
|
||||||
if ( ! getWalker().isInCount() ) {
|
|
||||||
joinedFragment = "(" + joinedFragment + ")";
|
|
||||||
}
|
|
||||||
setText( joinedFragment );
|
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,41 +21,40 @@
|
||||||
* 51 Franklin Street, Fifth Floor
|
* 51 Franklin Street, Fifth Floor
|
||||||
* Boston, MA 02110-1301 USA\
|
* Boston, MA 02110-1301 USA\
|
||||||
*/
|
*/
|
||||||
package org.hibernate.ejb.test.query;
|
package org.hibernate.test.component.basic2;
|
||||||
|
|
||||||
import javax.persistence.EntityManager;
|
import org.hibernate.Session;
|
||||||
|
|
||||||
import org.hibernate.ejb.criteria.components.Client;
|
|
||||||
import org.hibernate.ejb.test.BaseEntityManagerFunctionalTestCase;
|
|
||||||
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.hibernate.testing.junit4.BaseCoreFunctionalTestCase;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests related to specifying joins on components (embedded values).
|
* Tests related to specifying joins on components (embedded values).
|
||||||
*
|
*
|
||||||
* @author Steve Ebersole
|
* @author Steve Ebersole
|
||||||
*/
|
*/
|
||||||
public class ComponentJoinsTest extends BaseEntityManagerFunctionalTestCase {
|
public class ComponentJoinsTest extends BaseCoreFunctionalTestCase {
|
||||||
@Override
|
@Override
|
||||||
public Class[] getAnnotatedClasses() {
|
public Class[] getAnnotatedClasses() {
|
||||||
return new Class[] { Client.class };
|
return new Class[] { Person.class };
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testComponentJoins() {
|
public void testComponentJoins() {
|
||||||
// Just checking proper query construction and syntax checking via database query parser...
|
// Just checking proper query construction and syntax checking via database query parser...
|
||||||
EntityManager em = getOrCreateEntityManager();
|
Session session = openSession();
|
||||||
em.getTransaction().begin();
|
session.beginTransaction();
|
||||||
// use it in WHERE
|
// use it in WHERE
|
||||||
em.createQuery( "select c from Client c join c.name as n where n.lastName like '%'" ).getResultList();
|
session.createQuery( "select p from Person p join p.name as n where n.lastName like '%'" ).list();
|
||||||
// use it in SELECT
|
// use it in SELECT
|
||||||
em.createQuery( "select n.lastName from Client c join c.name as n" ).getResultList();
|
session.createQuery( "select n.lastName from Person p join p.name as n" ).list();
|
||||||
em.createQuery( "select n from Client c join c.name as n" ).getResultList();
|
session.createQuery( "select n from Person p join p.name as n" ).list();
|
||||||
// use it in ORDER BY
|
// use it in ORDER BY
|
||||||
em.createQuery( "select n from Client c join c.name as n order by n.lastName" ).getResultList();
|
session.createQuery( "select n from Person p join p.name as n order by n.lastName" ).list();
|
||||||
em.createQuery( "select n from Client c join c.name as n order by c" ).getResultList();
|
session.createQuery( "select n from Person p join p.name as n order by p" ).list();
|
||||||
em.createQuery( "select n from Client c join c.name as n order by n" ).getResultList();
|
session.createQuery( "select n from Person p join p.name as n order by n" ).list();
|
||||||
em.getTransaction().commit();
|
session.getTransaction().commit();
|
||||||
em.close();
|
session.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -0,0 +1,63 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* Copyright (c) 2012, 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.component.basic2;
|
||||||
|
|
||||||
|
import javax.persistence.Column;
|
||||||
|
import javax.persistence.Embeddable;
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Steve Ebersole
|
||||||
|
*/
|
||||||
|
@Embeddable
|
||||||
|
public class Name implements Serializable {
|
||||||
|
private String firstName;
|
||||||
|
private String lastName;
|
||||||
|
|
||||||
|
public Name() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public Name(String firstName, String lastName) {
|
||||||
|
this.firstName = firstName;
|
||||||
|
this.lastName = lastName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Column(name = "FIRST_NAME", nullable = false)
|
||||||
|
public String getFirstName() {
|
||||||
|
return firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFirstName(String firstName) {
|
||||||
|
this.firstName = firstName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Column(name = "LAST_NAME", nullable = false)
|
||||||
|
public String getLastName() {
|
||||||
|
return lastName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setLastName(String lastName) {
|
||||||
|
this.lastName = lastName;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,68 @@
|
||||||
|
/*
|
||||||
|
* Hibernate, Relational Persistence for Idiomatic Java
|
||||||
|
*
|
||||||
|
* Copyright (c) 2012, 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.component.basic2;
|
||||||
|
|
||||||
|
import javax.persistence.Embedded;
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.Id;
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Steve Ebersole
|
||||||
|
*/
|
||||||
|
@Entity
|
||||||
|
public class Person implements Serializable {
|
||||||
|
private int id;
|
||||||
|
private Name name;
|
||||||
|
|
||||||
|
public Person() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public Person(int id, Name name) {
|
||||||
|
this.id = id;
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Person(int id, String firstName, String lastName) {
|
||||||
|
this( id, new Name( firstName, lastName ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
@Id
|
||||||
|
public int getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(int id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Embedded
|
||||||
|
public Name getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(Name name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue