diff --git a/jpa/eclipselink.jpa.test.jse/src/org/eclipse/persistence/jpa/test/query/TestQueryHaving.java b/jpa/eclipselink.jpa.test.jse/src/org/eclipse/persistence/jpa/test/query/TestQueryHaving.java new file mode 100644 index 00000000000..309800791f1 --- /dev/null +++ b/jpa/eclipselink.jpa.test.jse/src/org/eclipse/persistence/jpa/test/query/TestQueryHaving.java @@ -0,0 +1,186 @@ +/* + * Copyright (c) 2021 IBM Corporation, Oracle, and/or affiliates. All rights reserved. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License v. 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0, + * or the Eclipse Distribution License v. 1.0 which is available at + * http://www.eclipse.org/org/documents/edl-v10.php. + * + * SPDX-License-Identifier: EPL-2.0 OR BSD-3-Clause + */ +// Contributors: +// IBM - Bug 417259: Add support for Parameters in CriteriaBuilder in HAVING clause +package org.eclipse.persistence.jpa.test.query; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; + +import java.util.List; +import javax.persistence.EntityManager; +import javax.persistence.EntityManagerFactory; +import javax.persistence.TypedQuery; +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.ParameterExpression; +import javax.persistence.criteria.Root; + +import org.eclipse.persistence.jpa.test.framework.DDLGen; +import org.eclipse.persistence.jpa.test.framework.Emf; +import org.eclipse.persistence.jpa.test.framework.EmfRunner; +import org.eclipse.persistence.jpa.test.framework.Property; +import org.eclipse.persistence.jpa.test.query.model.EntityTbl01; +import org.eclipse.persistence.jpa.test.query.model.EntityTbl01_; +import org.junit.Test; +import org.junit.runner.RunWith; + +@RunWith(EmfRunner.class) +public class TestQueryHaving { + @Emf(createTables = DDLGen.DROP_CREATE, classes = { EntityTbl01.class }, + properties = { @Property(name="eclipselink.logging.level", value="FINE")}) + private EntityManagerFactory emf; + + private static boolean POPULATED = false; + + @Test + public void testQueryHavingLiterals1() { + if (emf == null) + return; + + if(!POPULATED) + populate(); + + EntityManager em = emf.createEntityManager(); + + try { + TypedQuery query = em.createQuery("" + + "SELECT t.itemString1 FROM EntityTbl01 t " + + "GROUP BY t.itemString1 HAVING COUNT(t.itemString1) > 2", String.class); + + List dto01 = query.getResultList(); + assertNotNull(dto01); + assertEquals(1, dto01.size()); + assertEquals("A", dto01.get(0)); + + // equivalent CriteriaBuilder + CriteriaBuilder cb = em.getCriteriaBuilder(); + CriteriaQuery cquery = cb.createQuery(String.class); + Root root = cquery.from(EntityTbl01.class); + cquery.multiselect(root.get(EntityTbl01_.itemString1)); + + cquery.groupBy(root.get(EntityTbl01_.itemString1)); + cquery.having(cb.greaterThan(cb.count(root.get(EntityTbl01_.itemString1)), 2L)); + + query = em.createQuery(cquery); + dto01 = query.getResultList(); + assertNotNull(dto01); + assertEquals(1, dto01.size()); + assertEquals("A", dto01.get(0)); + } finally { + if (em.getTransaction().isActive()) { + em.getTransaction().rollback(); + } + if(em.isOpen()) { + em.close(); + } + } + } + + @Test + public void testQueryHavingParameters1() { + if (emf == null) + return; + + if(!POPULATED) + populate(); + + EntityManager em = emf.createEntityManager(); + + try { + TypedQuery query = em.createQuery("" + + "SELECT t.itemString1 FROM EntityTbl01 t " + + "GROUP BY t.itemString1 HAVING COUNT(t.itemString1) > ?1", String.class); + query.setParameter(1, 2); + + List dto01 = query.getResultList(); + assertNotNull(dto01); + assertEquals(1, dto01.size()); + assertEquals("A", dto01.get(0)); + + // equivalent CriteriaBuilder + CriteriaBuilder cb = em.getCriteriaBuilder(); + CriteriaQuery cquery = cb.createQuery(String.class); + Root root = cquery.from(EntityTbl01.class); + cquery.multiselect(root.get(EntityTbl01_.itemString1)); + + ParameterExpression checkParam1 = cb.parameter(Long.class); + cquery.groupBy(root.get(EntityTbl01_.itemString1)); + cquery.having(cb.greaterThan(cb.count(root.get(EntityTbl01_.itemString1)), checkParam1)); + + query = em.createQuery(cquery); + query.setParameter(checkParam1, 2L); + dto01 = query.getResultList(); + assertNotNull(dto01); + assertEquals(1, dto01.size()); + assertEquals("A", dto01.get(0)); + } finally { + if (em.getTransaction().isActive()) { + em.getTransaction().rollback(); + } + if(em.isOpen()) { + em.close(); + } + } + } + + private void populate() { + EntityManager em = emf.createEntityManager(); + try { + em.getTransaction().begin(); + + EntityTbl01 tbl1 = new EntityTbl01(); + tbl1.setKeyString("Key10"); + tbl1.setItemString1("A"); + tbl1.setItemString2(null); + tbl1.setItemString3("C"); + tbl1.setItemString4("D"); + tbl1.setItemInteger1(3); + em.persist(tbl1); + + EntityTbl01 tbl2 = new EntityTbl01(); + tbl2.setKeyString("Key11"); + tbl2.setItemString1("A"); + tbl2.setItemString2("B"); + tbl2.setItemString3("C"); + tbl2.setItemString4(null); + tbl2.setItemInteger1(4); + em.persist(tbl2); + + EntityTbl01 tbl3 = new EntityTbl01(); + tbl3.setKeyString("Key12"); + tbl3.setItemString1(null); + tbl3.setItemString2("B"); + tbl3.setItemString3("C"); + tbl3.setItemString4("D"); + tbl3.setItemInteger1(3); + em.persist(tbl3); + + EntityTbl01 tbl4 = new EntityTbl01(); + tbl4.setKeyString("Key13"); + tbl4.setItemString1("A"); + tbl4.setItemString2("B"); + tbl4.setItemString3("C"); + tbl4.setItemString4(null); + tbl4.setItemInteger1(4); + em.persist(tbl4); + + em.getTransaction().commit(); + + POPULATED = true; + } finally { + if(em.isOpen()) { + em.close(); + } + } + } +} \ No newline at end of file diff --git a/jpa/org.eclipse.persistence.jpa/src/org/eclipse/persistence/internal/jpa/querydef/AbstractQueryImpl.java b/jpa/org.eclipse.persistence.jpa/src/org/eclipse/persistence/internal/jpa/querydef/AbstractQueryImpl.java index 0b4b903c316..c5102258c2f 100644 --- a/jpa/org.eclipse.persistence.jpa/src/org/eclipse/persistence/internal/jpa/querydef/AbstractQueryImpl.java +++ b/jpa/org.eclipse.persistence.jpa/src/org/eclipse/persistence/internal/jpa/querydef/AbstractQueryImpl.java @@ -112,13 +112,13 @@ public AbstractQuery groupBy(Expression... grouping){ * @return the modified query */ @Override - public AbstractQuery having(Expression restriction){ - if (((InternalExpression)restriction).isCompoundExpression() || ((InternalExpression)restriction).isPredicate()){ + public AbstractQuery having(Expression restriction) { + findRootAndParameters(restriction); + if (((InternalExpression)restriction).isCompoundExpression() || ((InternalExpression)restriction).isPredicate()) { this.havingClause = (Predicate) restriction; - }else{ + } else { this.havingClause = queryBuilder.isTrue(restriction); } - return this; } @@ -139,6 +139,7 @@ public AbstractQuery having(Predicate... restrictions){ for (Predicate predicate : restrictions) { conjunction = this.queryBuilder.and(conjunction, predicate); } + findRootAndParameters(conjunction); this.havingClause = conjunction; } return this;